/*!	SWFObject v2.2 <http://code.google.com/p/swfobject/> 
	is released under the MIT License <http://www.opensource.org/licenses/mit-license.php> 
*/

var swfobject = function() {
	
	var UNDEF = "undefined",
		OBJECT = "object",
		SHOCKWAVE_FLASH = "Shockwave Flash",
		SHOCKWAVE_FLASH_AX = "ShockwaveFlash.ShockwaveFlash",
		FLASH_MIME_TYPE = "application/x-shockwave-flash",
		EXPRESS_INSTALL_ID = "SWFObjectExprInst",
		ON_READY_STATE_CHANGE = "onreadystatechange",
		
		win = window,
		doc = document,
		nav = navigator,
		
		plugin = false,
		domLoadFnArr = [main],
		regObjArr = [],
		objIdArr = [],
		listenersArr = [],
		storedAltContent,
		storedAltContentId,
		storedCallbackFn,
		storedCallbackObj,
		isDomLoaded = false,
		isExpressInstallActive = false,
		dynamicStylesheet,
		dynamicStylesheetMedia,
		autoHideShow = true,
	
	/* Centralized function for browser feature detection
		- User agent string detection is only used when no good alternative is possible
		- Is executed directly for optimal performance
	*/	
	ua = function() {
		var w3cdom = typeof doc.getElementById != UNDEF && typeof doc.getElementsByTagName != UNDEF && typeof doc.createElement != UNDEF,
			u = nav.userAgent.toLowerCase(),
			p = nav.platform.toLowerCase(),
			windows = p ? /win/.test(p) : /win/.test(u),
			mac = p ? /mac/.test(p) : /mac/.test(u),
			webkit = /webkit/.test(u) ? parseFloat(u.replace(/^.*webkit\/(\d+(\.\d+)?).*$/, "$1")) : false, // returns either the webkit version or false if not webkit
			ie = !+"\v1", // feature detection based on Andrea Giammarchi's solution: http://webreflection.blogspot.com/2009/01/32-bytes-to-know-if-your-browser-is-ie.html
			playerVersion = [0,0,0],
			d = null;
		if (typeof nav.plugins != UNDEF && typeof nav.plugins[SHOCKWAVE_FLASH] == OBJECT) {
			d = nav.plugins[SHOCKWAVE_FLASH].description;
			if (d && !(typeof nav.mimeTypes != UNDEF && nav.mimeTypes[FLASH_MIME_TYPE] && !nav.mimeTypes[FLASH_MIME_TYPE].enabledPlugin)) { // navigator.mimeTypes["application/x-shockwave-flash"].enabledPlugin indicates whether plug-ins are enabled or disabled in Safari 3+
				plugin = true;
				ie = false; // cascaded feature detection for Internet Explorer
				d = d.replace(/^.*\s+(\S+\s+\S+$)/, "$1");
				playerVersion[0] = parseInt(d.replace(/^(.*)\..*$/, "$1"), 10);
				playerVersion[1] = parseInt(d.replace(/^.*\.(.*)\s.*$/, "$1"), 10);
				playerVersion[2] = /[a-zA-Z]/.test(d) ? parseInt(d.replace(/^.*[a-zA-Z]+(.*)$/, "$1"), 10) : 0;
			}
		}
		else if (typeof win.ActiveXObject != UNDEF) {
			try {
				var a = new ActiveXObject(SHOCKWAVE_FLASH_AX);
				if (a) { // a will return null when ActiveX is disabled
					d = a.GetVariable("$version");
					if (d) {
						ie = true; // cascaded feature detection for Internet Explorer
						d = d.split(" ")[1].split(",");
						playerVersion = [parseInt(d[0], 10), parseInt(d[1], 10), parseInt(d[2], 10)];
					}
				}
			}
			catch(e) {}
		}
		return { w3:w3cdom, pv:playerVersion, wk:webkit, ie:ie, win:windows, mac:mac };
	}(),
	
	/* Cross-browser onDomLoad
		- Will fire an event as soon as the DOM of a web page is loaded
		- Internet Explorer workaround based on Diego Perini's solution: http://javascript.nwbox.com/IEContentLoaded/
		- Regular onload serves as fallback
	*/ 
	onDomLoad = function() {
		if (!ua.w3) { return; }
		if ((typeof doc.readyState != UNDEF && doc.readyState == "complete") || (typeof doc.readyState == UNDEF && (doc.getElementsByTagName("body")[0] || doc.body))) { // function is fired after onload, e.g. when script is inserted dynamically 
			callDomLoadFunctions();
		}
		if (!isDomLoaded) {
			if (typeof doc.addEventListener != UNDEF) {
				doc.addEventListener("DOMContentLoaded", callDomLoadFunctions, false);
			}		
			if (ua.ie && ua.win) {
				doc.attachEvent(ON_READY_STATE_CHANGE, function() {
					if (doc.readyState == "complete") {
						doc.detachEvent(ON_READY_STATE_CHANGE, arguments.callee);
						callDomLoadFunctions();
					}
				});
				if (win == top) { // if not inside an iframe
					(function(){
						if (isDomLoaded) { return; }
						try {
							doc.documentElement.doScroll("left");
						}
						catch(e) {
							setTimeout(arguments.callee, 0);
							return;
						}
						callDomLoadFunctions();
					})();
				}
			}
			if (ua.wk) {
				(function(){
					if (isDomLoaded) { return; }
					if (!/loaded|complete/.test(doc.readyState)) {
						setTimeout(arguments.callee, 0);
						return;
					}
					callDomLoadFunctions();
				})();
			}
			addLoadEvent(callDomLoadFunctions);
		}
	}();
	
	function callDomLoadFunctions() {
		if (isDomLoaded) { return; }
		try { // test if we can really add/remove elements to/from the DOM; we don't want to fire it too early
			var t = doc.getElementsByTagName("body")[0].appendChild(createElement("span"));
			t.parentNode.removeChild(t);
		}
		catch (e) { return; }
		isDomLoaded = true;
		var dl = domLoadFnArr.length;
		for (var i = 0; i < dl; i++) {
			domLoadFnArr[i]();
		}
	}
	
	function addDomLoadEvent(fn) {
		if (isDomLoaded) {
			fn();
		}
		else { 
			domLoadFnArr[domLoadFnArr.length] = fn; // Array.push() is only available in IE5.5+
		}
	}
	
	/* Cross-browser onload
		- Based on James Edwards' solution: http://brothercake.com/site/resources/scripts/onload/
		- Will fire an event as soon as a web page including all of its assets are loaded 
	 */
	function addLoadEvent(fn) {
		if (typeof win.addEventListener != UNDEF) {
			win.addEventListener("load", fn, false);
		}
		else if (typeof doc.addEventListener != UNDEF) {
			doc.addEventListener("load", fn, false);
		}
		else if (typeof win.attachEvent != UNDEF) {
			addListener(win, "onload", fn);
		}
		else if (typeof win.onload == "function") {
			var fnOld = win.onload;
			win.onload = function() {
				fnOld();
				fn();
			};
		}
		else {
			win.onload = fn;
		}
	}
	
	/* Main function
		- Will preferably execute onDomLoad, otherwise onload (as a fallback)
	*/
	function main() { 
		if (plugin) {
			testPlayerVersion();
		}
		else {
			matchVersions();
		}
	}
	
	/* Detect the Flash Player version for non-Internet Explorer browsers
		- Detecting the plug-in version via the object element is more precise than using the plugins collection item's description:
		  a. Both release and build numbers can be detected
		  b. Avoid wrong descriptions by corrupt installers provided by Adobe
		  c. Avoid wrong descriptions by multiple Flash Player entries in the plugin Array, caused by incorrect browser imports
		- Disadvantage of this method is that it depends on the availability of the DOM, while the plugins collection is immediately available
	*/
	function testPlayerVersion() {
		var b = doc.getElementsByTagName("body")[0];
		var o = createElement(OBJECT);
		o.setAttribute("type", FLASH_MIME_TYPE);
		var t = b.appendChild(o);
		if (t) {
			var counter = 0;
			(function(){
				if (typeof t.GetVariable != UNDEF) {
					var d = t.GetVariable("$version");
					if (d) {
						d = d.split(" ")[1].split(",");
						ua.pv = [parseInt(d[0], 10), parseInt(d[1], 10), parseInt(d[2], 10)];
					}
				}
				else if (counter < 10) {
					counter++;
					setTimeout(arguments.callee, 10);
					return;
				}
				b.removeChild(o);
				t = null;
				matchVersions();
			})();
		}
		else {
			matchVersions();
		}
	}
	
	/* Perform Flash Player and SWF version matching; static publishing only
	*/
	function matchVersions() {
		var rl = regObjArr.length;
		if (rl > 0) {
			for (var i = 0; i < rl; i++) { // for each registered object element
				var id = regObjArr[i].id;
				var cb = regObjArr[i].callbackFn;
				var cbObj = {success:false, id:id};
				if (ua.pv[0] > 0) {
					var obj = getElementById(id);
					if (obj) {
						if (hasPlayerVersion(regObjArr[i].swfVersion) && !(ua.wk && ua.wk < 312)) { // Flash Player version >= published SWF version: Houston, we have a match!
							setVisibility(id, true);
							if (cb) {
								cbObj.success = true;
								cbObj.ref = getObjectById(id);
								cb(cbObj);
							}
						}
						else if (regObjArr[i].expressInstall && canExpressInstall()) { // show the Adobe Express Install dialog if set by the web page author and if supported
							var att = {};
							att.data = regObjArr[i].expressInstall;
							att.width = obj.getAttribute("width") || "0";
							att.height = obj.getAttribute("height") || "0";
							if (obj.getAttribute("class")) { att.styleclass = obj.getAttribute("class"); }
							if (obj.getAttribute("align")) { att.align = obj.getAttribute("align"); }
							// parse HTML object param element's name-value pairs
							var par = {};
							var p = obj.getElementsByTagName("param");
							var pl = p.length;
							for (var j = 0; j < pl; j++) {
								if (p[j].getAttribute("name").toLowerCase() != "movie") {
									par[p[j].getAttribute("name")] = p[j].getAttribute("value");
								}
							}
							showExpressInstall(att, par, id, cb);
						}
						else { // Flash Player and SWF version mismatch or an older Webkit engine that ignores the HTML object element's nested param elements: display alternative content instead of SWF
							displayAltContent(obj);
							if (cb) { cb(cbObj); }
						}
					}
				}
				else {	// if no Flash Player is installed or the fp version cannot be detected we let the HTML object element do its job (either show a SWF or alternative content)
					setVisibility(id, true);
					if (cb) {
						var o = getObjectById(id); // test whether there is an HTML object element or not
						if (o && typeof o.SetVariable != UNDEF) { 
							cbObj.success = true;
							cbObj.ref = o;
						}
						cb(cbObj);
					}
				}
			}
		}
	}
	
	function getObjectById(objectIdStr) {
		var r = null;
		var o = getElementById(objectIdStr);
		if (o && o.nodeName == "OBJECT") {
			if (typeof o.SetVariable != UNDEF) {
				r = o;
			}
			else {
				var n = o.getElementsByTagName(OBJECT)[0];
				if (n) {
					r = n;
				}
			}
		}
		return r;
	}
	
	/* Requirements for Adobe Express Install
		- only one instance can be active at a time
		- fp 6.0.65 or higher
		- Win/Mac OS only
		- no Webkit engines older than version 312
	*/
	function canExpressInstall() {
		return !isExpressInstallActive && hasPlayerVersion("6.0.65") && (ua.win || ua.mac) && !(ua.wk && ua.wk < 312);
	}
	
	/* Show the Adobe Express Install dialog
		- Reference: http://www.adobe.com/cfusion/knowledgebase/index.cfm?id=6a253b75
	*/
	function showExpressInstall(att, par, replaceElemIdStr, callbackFn, jObj) {
		isExpressInstallActive = true;
		storedCallbackFn = callbackFn || null;
		storedCallbackObj = {success:false, id:replaceElemIdStr};
		var obj = getElementById(replaceElemIdStr);
		if (obj) {
			if (obj.nodeName == "OBJECT") { // static publishing
				storedAltContent = abstractAltContent(obj);
				storedAltContentId = null;
			}
			else { // dynamic publishing
				storedAltContent = obj;
				storedAltContentId = replaceElemIdStr;
			}
			att.id = EXPRESS_INSTALL_ID;
			if (typeof att.width == UNDEF || (!/%$/.test(att.width) && parseInt(att.width, 10) < 310)) { att.width = "310"; }
			if (typeof att.height == UNDEF || (!/%$/.test(att.height) && parseInt(att.height, 10) < 137)) { att.height = "137"; }
			doc.title = doc.title.slice(0, 47) + " - Flash Player Installation";
			var pt = ua.ie && ua.win ? "ActiveX" : "PlugIn",
				fv = "MMredirectURL=" + win.location.toString().replace(/&/g,"%26") + "&MMplayerType=" + pt + "&MMdoctitle=" + doc.title;
			if (typeof par.flashvars != UNDEF) {
				par.flashvars += "&" + fv;
			}
			else {
				par.flashvars = fv;
			}
			// IE only: when a SWF is loading (AND: not available in cache) wait for the readyState of the object element to become 4 before removing it,
			// because you cannot properly cancel a loading SWF file without breaking browser load references, also obj.onreadystatechange doesn't work
			if (ua.ie && ua.win && obj.readyState != 4) {
				var newObj = createElement("div");
				replaceElemIdStr += "SWFObjectNew";
				newObj.setAttribute("id", replaceElemIdStr);
				obj.parentNode.insertBefore(newObj, obj); // insert placeholder div that will be replaced by the object element that loads expressinstall.swf
				obj.style.display = "none";
				(function(){
					if (obj.readyState == 4) {
						obj.parentNode.removeChild(obj);
					}
					else {
						setTimeout(arguments.callee, 10);
					}
				})();
			}
			createSWF(att, par, replaceElemIdStr, jObj);
		}
	}
	
	/* Functions to abstract and display alternative content
	*/
	function displayAltContent(obj) {
		if (ua.ie && ua.win && obj.readyState != 4) {
			// IE only: when a SWF is loading (AND: not available in cache) wait for the readyState of the object element to become 4 before removing it,
			// because you cannot properly cancel a loading SWF file without breaking browser load references, also obj.onreadystatechange doesn't work
			var el = createElement("div");
			obj.parentNode.insertBefore(el, obj); // insert placeholder div that will be replaced by the alternative content
			el.parentNode.replaceChild(abstractAltContent(obj), el);
			obj.style.display = "none";
			(function(){
				if (obj.readyState == 4) {
					obj.parentNode.removeChild(obj);
				}
				else {
					setTimeout(arguments.callee, 10);
				}
			})();
		}
		else {
			obj.parentNode.replaceChild(abstractAltContent(obj), obj);
		}
	} 

	function abstractAltContent(obj) {
		var ac = createElement("div");
		if (ua.win && ua.ie) {
			ac.innerHTML = obj.innerHTML;
		}
		else {
			var nestedObj = obj.getElementsByTagName(OBJECT)[0];
			if (nestedObj) {
				var c = nestedObj.childNodes;
				if (c) {
					var cl = c.length;
					for (var i = 0; i < cl; i++) {
						if (!(c[i].nodeType == 1 && c[i].nodeName == "PARAM") && !(c[i].nodeType == 8)) {
							ac.appendChild(c[i].cloneNode(true));
						}
					}
				}
			}
		}
		return ac;
	}
	
	/* Cross-browser dynamic SWF creation
	*/
	function createSWF(attObj, parObj, id, jObj) {
		var r,el = jObj||getElementById(id);
		if (ua.wk && ua.wk < 312) { return r; }
		if (el) {
			if (typeof attObj.id == UNDEF) { // if no 'id' is defined for the object element, it will inherit the 'id' from the alternative content
				attObj.id = id;
			}
			if (ua.ie && ua.win) { // Internet Explorer + the HTML object element + W3C DOM methods do not combine: fall back to outerHTML
				var att = "";
				for (var i in attObj) {
					if (attObj[i] != Object.prototype[i]) { // filter out prototype additions from other potential libraries
						if (i.toLowerCase() == "data") {
							parObj.movie = attObj[i];
						}
						else if (i.toLowerCase() == "styleclass") { // 'class' is an ECMA4 reserved keyword
							att += ' class="' + attObj[i] + '"';
						}
						else if (i.toLowerCase() != "classid") {
							att += ' ' + i + '="' + attObj[i] + '"';
						}
					}
				}
				var par = "";
				for (var j in parObj) {
					if (parObj[j] != Object.prototype[j]) { // filter out prototype additions from other potential libraries
						par += '<param name="' + j + '" value="' + parObj[j] + '" />';
					}
				}
				el.outerHTML = '<object classid="clsid:D27CDB6E-AE6D-11cf-96B8-444553540000"' + att + '>' + par + '</object>';
				objIdArr[objIdArr.length] = attObj.id; // stored to fix object 'leaks' on unload (dynamic publishing only)
				r = getElementById(attObj.id);	
			}
			else { // well-behaving browsers
				var o = createElement(OBJECT);
				o.setAttribute("type", FLASH_MIME_TYPE);
				for (var m in attObj) {
					if (attObj[m] != Object.prototype[m]) { // filter out prototype additions from other potential libraries
						if (m.toLowerCase() == "styleclass") { // 'class' is an ECMA4 reserved keyword
							o.setAttribute("class", attObj[m]);
						}
						else if (m.toLowerCase() != "classid") { // filter out IE specific attribute
							o.setAttribute(m, attObj[m]);
						}
					}
				}
				for (var n in parObj) {
					if (parObj[n] != Object.prototype[n] && n.toLowerCase() != "movie") { // filter out prototype additions from other potential libraries and IE specific param element
						createObjParam(o, n, parObj[n]);
					}
				}
				el.parentNode.replaceChild(o, el);
				r = o;
			}
		}
		return r;
	}
	
	function createObjParam(el, pName, pValue) {
		var p = createElement("param");
		p.setAttribute("name", pName);	
		p.setAttribute("value", pValue);
		el.appendChild(p);
	}
	
	/* Cross-browser SWF removal
		- Especially needed to safely and completely remove a SWF in Internet Explorer
	*/
	function removeSWF(id) {
		var obj = getElementById(id);
		if (obj && obj.nodeName == "OBJECT") {
			if (ua.ie && ua.win) {
				obj.style.display = "none";
				(function(){
					if (obj.readyState == 4) {
						removeObjectInIE(id);
					}
					else {
						setTimeout(arguments.callee, 10);
					}
				})();
			}
			else {
				obj.parentNode.removeChild(obj);
			}
		}
	}
	
	function removeObjectInIE(id) {
		var obj = getElementById(id);
		if (obj) {
			for (var i in obj) {
				if (typeof obj[i] == "function") {
					obj[i] = null;
				}
			}
			obj.parentNode.removeChild(obj);
		}
	}
	
	/* Functions to optimize JavaScript compression
	*/
	function getElementById(id) {
		var el = null;
		try {
			el = doc.getElementById(id);
		}
		catch (e) {}
		return el;
	}
	
	function createElement(el) {
		return doc.createElement(el);
	}
	
	/* Updated attachEvent function for Internet Explorer
		- Stores attachEvent information in an Array, so on unload the detachEvent functions can be called to avoid memory leaks
	*/	
	function addListener(target, eventType, fn) {
		target.attachEvent(eventType, fn);
		listenersArr[listenersArr.length] = [target, eventType, fn];
	}
	
	/* Flash Player and SWF content version matching
	*/
	function hasPlayerVersion(rv) {
		var pv = ua.pv, v = rv.split(".");
		v[0] = parseInt(v[0], 10);
		v[1] = parseInt(v[1], 10) || 0; // supports short notation, e.g. "9" instead of "9.0.0"
		v[2] = parseInt(v[2], 10) || 0;
		return (pv[0] > v[0] || (pv[0] == v[0] && pv[1] > v[1]) || (pv[0] == v[0] && pv[1] == v[1] && pv[2] >= v[2])) ? true : false;
	}
	
	/* Cross-browser dynamic CSS creation
		- Based on Bobby van der Sluis' solution: http://www.bobbyvandersluis.com/articles/dynamicCSS.php
	*/	
	function createCSS(sel, decl, media, newStyle) {
		if (ua.ie && ua.mac) { return; }
		var h = doc.getElementsByTagName("head")[0];
		if (!h) { return; } // to also support badly authored HTML pages that lack a head element
		var m = (media && typeof media == "string") ? media : "screen";
		if (newStyle) {
			dynamicStylesheet = null;
			dynamicStylesheetMedia = null;
		}
		if (!dynamicStylesheet || dynamicStylesheetMedia != m) { 
			// create dynamic stylesheet + get a global reference to it
			var s = createElement("style");
			s.setAttribute("type", "text/css");
			s.setAttribute("media", m);
			dynamicStylesheet = h.appendChild(s);
			if (ua.ie && ua.win && typeof doc.styleSheets != UNDEF && doc.styleSheets.length > 0) {
				dynamicStylesheet = doc.styleSheets[doc.styleSheets.length - 1];
			}
			dynamicStylesheetMedia = m;
		}
		// add style rule
		if (ua.ie && ua.win) {
			if (dynamicStylesheet && typeof dynamicStylesheet.addRule == OBJECT) {
				dynamicStylesheet.addRule(sel, decl);
			}
		}
		else {
			if (dynamicStylesheet && typeof doc.createTextNode != UNDEF) {
				dynamicStylesheet.appendChild(doc.createTextNode(sel + " {" + decl + "}"));
			}
		}
	}
	
	function setVisibility(id, isVisible) {
		if (!autoHideShow) { return; }
		var v = isVisible ? "visible" : "hidden";
		if (isDomLoaded && getElementById(id)) {
			getElementById(id).style.visibility = v;
		}
		else {
			createCSS("#" + id, "visibility:" + v);
		}
	}

	/* Filter to avoid XSS attacks
	*/
	function urlEncodeIfNecessary(s) {
		var regex = /[\\\"<>\.;]/;
		var hasBadChars = regex.exec(s) != null;
		return hasBadChars && typeof encodeURIComponent != UNDEF ? encodeURIComponent(s) : s;
	}
	
	/* Release memory to avoid memory leaks caused by closures, fix hanging audio/video threads and force open sockets/NetConnections to disconnect (Internet Explorer only)
	*/
	var cleanup = function() {
		if (ua.ie && ua.win) {
			window.attachEvent("onunload", function() {
				// remove listeners to avoid memory leaks
				var ll = listenersArr.length;
				for (var i = 0; i < ll; i++) {
					listenersArr[i][0].detachEvent(listenersArr[i][1], listenersArr[i][2]);
				}
				// cleanup dynamically embedded objects to fix audio/video threads and force open sockets and NetConnections to disconnect
				var il = objIdArr.length;
				for (var j = 0; j < il; j++) {
					removeSWF(objIdArr[j]);
				}
				// cleanup library's main closures to avoid memory leaks
				for (var k in ua) {
					ua[k] = null;
				}
				ua = null;
				for (var l in swfobject) {
					swfobject[l] = null;
				}
				swfobject = null;
			});
		}
	}();
	
	return {
		/* Public API
			- Reference: http://code.google.com/p/swfobject/wiki/documentation
		*/ 
		registerObject: function(objectIdStr, swfVersionStr, xiSwfUrlStr, callbackFn) {
			if (ua.w3 && objectIdStr && swfVersionStr) {
				var regObj = {};
				regObj.id = objectIdStr;
				regObj.swfVersion = swfVersionStr;
				regObj.expressInstall = xiSwfUrlStr;
				regObj.callbackFn = callbackFn;
				regObjArr[regObjArr.length] = regObj;
				setVisibility(objectIdStr, false);
			}
			else if (callbackFn) {
				callbackFn({success:false, id:objectIdStr});
			}
		},
		
		getObjectById: function(objectIdStr) {
			if (ua.w3) {
				return getObjectById(objectIdStr);
			}
		},

		embedSWF: function(swfUrlStr, replaceElemIdStr, widthStr, heightStr, swfVersionStr, xiSwfUrlStr, flashvarsObj, parObj, attObj, callbackFn, jObj, alternateHtml) {
			var callbackObj = {success:false, id:replaceElemIdStr};
			if (ua.w3 && !(ua.wk && ua.wk < 312) && swfUrlStr && replaceElemIdStr && widthStr && heightStr && swfVersionStr) {
				setVisibility(replaceElemIdStr, false);
				addDomLoadEvent(function() {
					widthStr += ""; // auto-convert to string
					heightStr += "";
					var att = {};
					if (attObj && typeof attObj === OBJECT) {
						for (var i in attObj) { // copy object to avoid the use of references, because web authors often reuse attObj for multiple SWFs
							att[i] = attObj[i];
						}
					}
					att.data = swfUrlStr;
					att.width = widthStr;
					att.height = heightStr;
					var par = {}; 
					if (parObj && typeof parObj === OBJECT) {
						for (var j in parObj) { // copy object to avoid the use of references, because web authors often reuse parObj for multiple SWFs
							par[j] = parObj[j];
						}
					}
					if (flashvarsObj && typeof flashvarsObj === OBJECT) {
						for (var k in flashvarsObj) { // copy object to avoid the use of references, because web authors often reuse flashvarsObj for multiple SWFs
							if (typeof par.flashvars != UNDEF) {
								par.flashvars += "&" + k + "=" + flashvarsObj[k];
							}
							else {
								par.flashvars = k + "=" + flashvarsObj[k];
							}
						}
					}
					if (hasPlayerVersion(swfVersionStr)) { // create SWF
						var obj = createSWF(att, par, replaceElemIdStr, jObj);
						if (att.id == replaceElemIdStr) {
							setVisibility(replaceElemIdStr, true);
						}
						callbackObj.success = true;
						callbackObj.ref = obj;
					}
					else if (xiSwfUrlStr && canExpressInstall()) { // show Adobe Express Install
						att.data = xiSwfUrlStr;
						showExpressInstall(att, par, replaceElemIdStr, callbackFn);
						return;
					}
					else if ( alternateHtml ) { // show alternative content
						var el = jObj||getElementById(id);
						if ( el ) {
							el.parentNode.innerHTML = alternateHtml;
						}
					}
					else {
					
						// Actually, we're going to render the original one anyway, see what happens.
						att = att||{};
						att.codebase = 'http://download.macromedia.com/pub/shockwave/cabs/flash/swflash.cab#version=9,0,0,0';
						var obj = createSWF(att, par, replaceElemIdStr, jObj);
						if (att.id == replaceElemIdStr) {
							setVisibility(replaceElemIdStr, true);
						}
						callbackObj.success = true;
						callbackObj.ref = obj;
						/*
						setVisibility(replaceElemIdStr, true);
						*/
					}
					if (callbackFn) { callbackFn(callbackObj); }
				});
			}
			else if (callbackFn) { callbackFn(callbackObj);	}
		},
		
		switchOffAutoHideShow: function() {
			autoHideShow = false;
		},
		
		ua: ua,
		
		getFlashPlayerVersion: function() {
			return { major:ua.pv[0], minor:ua.pv[1], release:ua.pv[2] };
		},
		
		hasFlashPlayerVersion: hasPlayerVersion,
		
		createSWF: function(attObj, parObj, replaceElemIdStr, jObj) {
			if (ua.w3) {
				return createSWF(attObj, parObj, replaceElemIdStr, jObj);
			}
			else {
				return undefined;
			}
		},
		
		showExpressInstall: function(att, par, replaceElemIdStr, callbackFn) {
			if (ua.w3 && canExpressInstall()) {
				showExpressInstall(att, par, replaceElemIdStr, callbackFn);
			}
		},
		
		removeSWF: function(objElemIdStr) {
			if (ua.w3) {
				removeSWF(objElemIdStr);
			}
		},
		
		createCSS: function(selStr, declStr, mediaStr, newStyleBoolean) {
			if (ua.w3) {
				createCSS(selStr, declStr, mediaStr, newStyleBoolean);
			}
		},
		
		addDomLoadEvent: addDomLoadEvent,
		
		addLoadEvent: addLoadEvent,
		
		getQueryParamValue: function(param) {
			var q = doc.location.search || doc.location.hash;
			if (q) {
				if (/\?/.test(q)) { q = q.split("?")[1]; } // strip question mark
				if (param == null) {
					return urlEncodeIfNecessary(q);
				}
				var pairs = q.split("&");
				for (var i = 0; i < pairs.length; i++) {
					if (pairs[i].substring(0, pairs[i].indexOf("=")) == param) {
						return urlEncodeIfNecessary(pairs[i].substring((pairs[i].indexOf("=") + 1)));
					}
				}
			}
			return "";
		},
		
		// For internal usage only
		expressInstallCallback: function() {
			if (isExpressInstallActive) {
				var obj = getElementById(EXPRESS_INSTALL_ID);
				if (obj && storedAltContent) {
					obj.parentNode.replaceChild(storedAltContent, obj);
					if (storedAltContentId) {
						setVisibility(storedAltContentId, true);
						if (ua.ie && ua.win) { storedAltContent.style.display = "block"; }
					}
					if (storedCallbackFn) { storedCallbackFn(storedCallbackObj); }
				}
				isExpressInstallActive = false;
			} 
		}
	};
}();

// If the current browser doesn't support flash, we need to load the jwplayer script.
if ( !swfobject.ua || !swfobject.ua.pv || !swfobject.ua.pv.length || swfobject.ua.pv[0] == 0 ) {
	window._jwscript = false;
	$.ajax({
		url: '/Shared/js2/j/jwplayer.min.js',
		dataType: 'script',
		success: function(){
			window._jwscript === true;
		}
	});
}

// Setup a specific flash movie.
$.fn.embedswf = function(options,reload) {
	return this.each(function(i) {
		// If we're doing a reload of the flash movie, clear out any existing remnants.
		var el = $(this);
		if ( reload ) {
			el.removeswf();
		}
			
		if ( el.children('object').length == 0 ) {
			var movie,width,height,version,flashvars,attributes;
				
			// Set the default parameters.
			var params = {
				allowfullscreen:true,
				allowscriptaccess:'sameDomain',
				wmode:'transparent',
				pluginspage:'http://www.macromedia.com/go/getflashplayer'
			};
			if (options) {
				// Conver the options to SWFObject parameters.
				if ( options.nocache ) {
					movie = $.getAjaxUrl(options.swf, {RND: Math.random()});
				} else {
					movie = options.swf;
				}
				width = options.width;
				height = options.height;
				flashvars = {};
				if (options.params) {
					for (var p in options.params) {
						if (p=='flashvars') {
							flashvars = options.params[p];
						} else {
							params[p] = options.params[p];
						}
					}
				}
				if (!options.id) {
					options.id = ('Player'+Math.random()).replace('.','');
				}
				attributes = {};
				attributes.id = options.id;
				attributes.name = options.id;
			} else {
				// Get the standard options.
				var movie = el.attr('_movie'),
					w = $.curCSS(el[0],'width');
					width = el.attr('_width') || $.toInt( w ) || el.width(),
					height = el.attr('_height') || $.toInt( $.curCSS(el[0],'height') ) || el.height(),
					loop = $.toBool(el.attr('_loop')),
					vars = el.attr('_flashvars'),
					id = el.attr('_id'),
					version = el.attr('_version');

				if ( w === 'auto' ) {
					if ( !width ) {
						width = el.parent().width();
					}
					height = ( width / 16 * 9 ) + $.toInt( el.attr('_skin') );
				}

				if ( !movie || !width || !height ) {
					return;
				}

				// Add random number as needed.
				if ( $.toBool(el.attr('_nocache')) ) {
					movie = $.getAjaxUrl(movie, {RND: Math.random()});
				}

				// Make sure we have an id.
				if (!id) {
					id = ('Player'+Math.random()).replace('.','');
				}

				// We need to assign both the id and name for cross-platform support.
				attributes = {};
				attributes.id = id;
				attributes.name = id;

				// Add the loop
				params.loop = loop.toString();

				// Only override the default wmode if it is specifically set for opaque.
				if (el.attr('_wmode')=='opaque') {
					params.wmode = 'opaque';
				}

				// If we have flash variables.
				var flashvars = {};
				if (vars) {
					try {
						// Split the parameters and iterate through each one.
						var v = vars.replace(/&(?:amp;)+/g,'&').split('&');
						for (var i=0;i<v.length;i++) {
							// Split the name/value pair and add each one as a named property.
							var pair = v[i].split('=');
							flashvars[pair[0]] = pair[1];
						}
					}
					catch(ex){
						flashvars = {};
					}
				}
			}

			if ( movie && width && height ) {
				// Record the id of the flash movie.
				el.attr('_id', attributes.id);

				// Create a container for the SWF movie inside the current element.
				var control = ('FlashSWF'+Math.random()).replace('.',''),
					alternate = el.children("[_alternate]").attr('_alternate');
				if ( alternate ) {
					alternate = $.decode(alternate);
				} else if ( !el.is('.icobalt') ) {
					alternate = $.cleanHtml( $.trim(el.html()||"") );
				}
				el.html('<div id="'+control+'"></div>');

				// If this is the jw player.
				if ( movie.endsWith('/longtail/player.swf') || movie.endsWith('/lt/player.swf') ) {
					if ( $.toBool($.getQueryString('bandwidth')) ) {
						if ( flashvars.plugins ) {
							flashvars.plugins += ',qualitymonitor';
						} else {
							flashvars.plugins = 'qualitymonitor';
						}
					}

					// Get the jw options.
					var jwoptions = $.extend({
						flashplayer: movie,
						width: width,
						height: height
					}, flashvars);

					// Look for levels.
					var levels = null;
					if ( levels = el.attr('_levels') ) {
						try {
							levels = eval( '('+levels+')' );
						} catch(ex) {
							levels = null;
						}
					}
					// For mobile HTML5 players, try to pick the 400k if available.
					if ( levels ) {
						flashvars.levels = $.encode( "[[JSON]]"+el.attr('_levels') );
						jwoptions.levels = levels;
						if ( levels.length >= 3 ) {
							jwoptions.file = levels[2].file;
						}
					}

					// HTML5 doesn't support RTMP.
					if ( jwoptions.streamer ) {
						delete jwoptions.streamer;
						if ( typeof jwoptions.file === 'string' ) {
							jwoptions.file = jwoptions.file.replace(/^\d+/,'');
						}
						if ( $.isArray(jwoptions.levels) ) {
							for ( var i=0; i<jwoptions.levels.length; i++ ) {
								jwoptions.levels[i].file = jwoptions.levels[i].file.replace(/^\d+/,'');
							}
						}
					}

					// Zip skins have to be referenced as unpacked xml skins.
					if ( jwoptions.skin ) {
						jwoptions.skin = jwoptions.skin.replace(/\/(\w+).zip/,'/$1/$1.xml');
					}

					if ( typeof jwplayer !== 'undefined' ) {
						// If we've loaded the JW script.
						if ( !jwoptions["controlbar.position"] ) {
							jwoptions["controlbar.position"] = "bottom";
						}
						jwplayer(control).setup(jwoptions);
						return;
					} else if ( window._jwscript === false ) {
						// If we're still in the process of loading the jw script, initialize the jwplayer on a timeout.
						var fn = function(_control,_options){
							return function(){
								if ( typeof jwplayer !== 'undefined' ) {
									// If we've loaded the JW script.
									jwplayer(_control).setup(_options);
								}
								_control = null;
								_options = null;
							};
						}(control,jwoptions);
						setTimeout(fn,100);
						return;
					} else if ( levels && levels.length > 1 ) {
						// If we're going to use the standard swf jwplayer, and we have multiple levels, remove the
						// starting file and image, 
					}
				}

				// Save a reference to the alternate html.
				if ( alternate ) {
					el.data('alternate',alternate);
				} else {
					el.removeData('alternate');
				}

				// Embed the movie.
				swfobject.embedSWF(movie, control, width, height, version||'9.0.0', "Shared/js/expressInstall.swf", flashvars, params, attributes, null, el[0].childNodes[0], alternate);
			}
		}
	});
};

// Remove any existing SWF objects.
$.fn.removeswf = function(){
	this.find('object').each(function(i) {
		var id = $(this).attr('id');
		if ( id ) {
			swfobject.removeSWF(id);
		}
	});
	return this;
};


// Setup swf uploader
$.cobalt = $.cobalt || {};
$.cobalt.swfUploader = function(el,url,options)
{
	if (!el || !url) return;

	// Configure the button.
	var btn = el.find('span.swfu_button');
	if (!btn.length) return;
	var id = ('swfub_'+Math.random()).replace(/\./g,'');
	btn.attr('id',id);

	var o = options||{};
	var swfu = new SWFUpload({
		// Backend Settings
		upload_url: url,
		post_params : {
			"ASPSESSID" : $(document.body).attr('_session'),
			"AUTHID" : $(document.body).attr('_auth')
		},

		// File Upload Settings
		file_size_limit : o.size||"256 KB",
		file_types : o.types||"*.jpg;*.jpeg;*.jpe;*.gif;*.png",
		file_types_description : o.typeDescription||"Images",
		file_upload_limit : o.limit||"1",    // Zero means unlimited

		// Alert the user when there is an error.
		file_queue_error_handler : function(file, errorCode, message) {
			switch (errorCode)
			{
				case SWFUpload.QUEUE_ERROR.FILE_EXCEEDS_SIZE_LIMIT:
					alert('File too large, maximum size is '+(o.limit||"256 KB"));
					break;
				case SWFUpload.QUEUE_ERROR.QUEUE_LIMIT_EXCEEDED:
					if ((o.limit||"1")=="1")
						alert('Please upload one file at a time.');
					else
						alert('Too many files.');
					break;
				case SWFUpload.QUEUE_ERROR.ZERO_BYTE_FILE:
					alert('The selected file is empty.');
					break;
				case SWFUpload.QUEUE_ERROR.INVALID_FILETYPE:
					alert('Invalid file type.');
					break;
				default:
					alert(message||'The file could not be uploaded.  Tech services has been notified.');
					break;
			}
		},
		// Start the upload if files were selected.
		file_dialog_complete_handler : function(numFilesSelected, numFilesQueued) {
			try {
				if (numFilesQueued > 0) {
					this.startUpload();
				}
			} catch (ex) {
				this.debug(ex);
			}
		},
		// Update the progress bar.
		upload_progress_handler : function(file, bytesLoaded) {
			if (!this.customSettings.progress_bar)
			{
				this.customSettings.progress_bar = this.customSettings.upload_target.children('div.progressBar');
				this.customSettings.progress_status = this.customSettings.upload_target.children('div.progressStatus');
			}
			var w = parseInt(bytesLoaded / file.size) * 250;
			this.customSettings.progress_bar.css({width:w});
			this.customSettings.progress_status.html('Uploading - '+$.kbFormat(bytesLoaded)+' out of '+$.kbFormat(file.size));
		},
		// Any server-side error.
		upload_error_handler : function(file, errorCode, message) {
			switch (errorCode) {
				case SWFUpload.UPLOAD_ERROR.FILE_CANCELLED:
				case SWFUpload.UPLOAD_ERROR.UPLOAD_STOPPED:
				case SWFUpload.UPLOAD_ERROR.UPLOAD_LIMIT_EXCEEDED:
				default:
					alert(message);
					break;
			}
		},
		// After the file has successfully uploaded.
		upload_success_handler : function(file, serverData) {
			if (!this.customSettings.progress_status)
				this.customSettings.progress_status = this.customSettings.upload_target.children('div.progressStatus');
			this.customSettings.progress_status.html('Success - '+$.kbFormat(file.size));
		},
		// Once the upload is complete.
		upload_complete_handler : function(file) {
			/*  I want the next upload to continue automatically so I'll call startUpload here */
			if (this.getStats().files_queued > 0) {
				this.startUpload();
			}
			else if ($.isFunction(o.oncomplete))
			{
				// The file uploader replaces spaces with dashes, so make sure that updated filename is passed to the complete function.
				file.name = file.name.replace(/%20/g,'-').replace(/ /g,'-');
				o.oncomplete(this,file);
			}
		},

		// Button settings
		button_image_url : o.uploadBtn||"Shared/js/swfupload/uploadbtn.gif",
		button_placeholder_id : id,
		button_width: 79,
		button_height: 18,
		button_text_top_padding: 0,
		button_text_left_padding: 0,

		// Flash Settings
		flash_url : "Shared/js/swfupload/swfupload.swf",	// Relative to this file

		// Custom settings.
		custom_settings : {
			container : el,
			upload_target : el.find('div.progressContainer')
		},

		// Debug Settings
		debug: false
	});

	// Safe the swf uploader object.
	el.data('swfu',swfu);
};

// When the JWPlayer is ready.
function playerReady(p) {
	// Grab the swf.
	var swf = window.document[p.id];
	if ( swf ) {
		// Record a reference to the swf.
		var div = $(swf).parent('div');
		div.data('swf',swf);

		// Fire the swfready event if it hasn't been done already.
		if ( swf.sendEvent ) {
			$.log('playerReady - '+p.id);
			div.data('swfready',true);
			div.trigger('swfready');
		}
	} else {
		$.log('player could not be found');
		$.log(p);
	}
}

// When the JWPlayer is ready.
function playerFire(p,evt) {
	// Grab the swf.
	var swf = window.document[p.id];
	if (swf) {
		// Get the parent div and trigger the event.
		$.log('playerFire - '+p.id+' - '+evt);
		var div = $(swf).parent('div');
		var args = Array.prototype.slice.call(arguments);
		args.shift();
		args.shift();
		$.log(args);
		div.trigger(evt, args);
	}
}

// When the Flash Text Editor is ready.
function fteReady(p) {
	// Grab the swf.
	var swf = window.document[p.id];
	if ( swf ) {
		// Record a reference to the swf.
		var div = $(swf).parent('div');
		div.data('swf',swf);

		// Fire the swfready event if it hasn't been done already.
		if ( !div.data('swfready') ) {
			$.log('fteReady - '+p.id);
			div.data('swfready',true);
			div.trigger('swfready');
		}
	} else {
		$.log('fte could not be found');
		$.log(p);
	}
}

// When the Flash Text Editor contents have been changed.
function fteChanged(p) {
	// Grab the swf.
	var swf = window.document[p.id];
	if (swf) {
		// Fire the change event.
		var div = $(swf).parent('div');
		div.trigger('swfchange');
	}
}

// When the Flash Text Editor needs to be saved.
function fteSave(p) {
	// Grab the swf.
	var swf = window.document[p.id];
	if (swf) {
		// Fire the change event.
		var div = $(swf).parent('div');
		div.trigger('swfsave');
	}
}

// Fire an event when the player is ready.
$.fn.swfready = function(fn){
	return this.each(function(i){
		var div = $(this);
		if ( div.data('swfready') ) {
			// If the swf has already been set up, fire the event now.
			div.trigger('swfready');
		} else {
			// Otherwise, bind a custom event.
			div.bind('swfready',fn);
		}
	});
}

