Flex AJAX Bridge Source

Save this into a new S-Control named FlexAjaxBridge, it will be included by the main S-Control mentioned in the article This source came from the Adobe Labs site, under the name FAB, Flex Ajax Bridge. You will still need the Flex portion which you can get from here (as part of LiveCycle Data Services ES), this code is only the javascript portion. There is more information on the Flex Ajax Bridge here, including path information.

<script language="javascript">
// Flash Player Version Detection - Rev 1.5
// Detect Client Browser type
// Copyright(c) 2005-2006 Adobe Macromedia Software, LLC. All rights reserved.
var isIE  = (navigator.appVersion.indexOf("MSIE") != -1) ? true : false;
var isWin = (navigator.appVersion.toLowerCase().indexOf("win") != -1) ? true : false;
var isOpera = (navigator.userAgent.indexOf("Opera") != -1) ? true : false;

function ControlVersion()
{
	var version;
	var axo;
	var e;

	// NOTE : new ActiveXObject(strFoo) throws an exception if strFoo isn't in the registry

	try {
		// version will be set for 7.X or greater players
		axo = new ActiveXObject("ShockwaveFlash.ShockwaveFlash.7");
		version = axo.GetVariable("$version");
	} catch (e) {
	}

	if (!version)
	{
		try {
			// version will be set for 6.X players only
			axo = new ActiveXObject("ShockwaveFlash.ShockwaveFlash.6");
			
			// installed player is some revision of 6.0
			// GetVariable("$version") crashes for versions 6.0.22 through 6.0.29,
			// so we have to be careful. 
			
			// default to the first public version
			version = "WIN 6,0,21,0";

			// throws if AllowScripAccess does not exist (introduced in 6.0r47)		
			axo.AllowScriptAccess = "always";

			// safe to call for 6.0r47 or greater
			version = axo.GetVariable("$version");

		} catch (e) {
		}
	}

	if (!version)
	{
		try {
			// version will be set for 4.X or 5.X player
			axo = new ActiveXObject("ShockwaveFlash.ShockwaveFlash.3");
			version = axo.GetVariable("$version");
		} catch (e) {
		}
	}

	if (!version)
	{
		try {
			// version will be set for 3.X player
			axo = new ActiveXObject("ShockwaveFlash.ShockwaveFlash.3");
			version = "WIN 3,0,18,0";
		} catch (e) {
		}
	}

	if (!version)
	{
		try {
			// version will be set for 2.X player
			axo = new ActiveXObject("ShockwaveFlash.ShockwaveFlash");
			version = "WIN 2,0,0,11";
		} catch (e) {
			version = -1;
		}
	}
	
	return version;
}

// JavaScript helper required to detect Flash Player PlugIn version information
function GetSwfVer(){
	// NS/Opera version >= 3 check for Flash plugin in plugin array
	var flashVer = -1;
	
	if (navigator.plugins != null && navigator.plugins.length > 0) {
		if (navigator.plugins["Shockwave Flash 2.0"] || navigator.plugins["Shockwave Flash"]) {
			var swVer2 = navigator.plugins["Shockwave Flash 2.0"] ? " 2.0" : "";
			var flashDescription = navigator.plugins["Shockwave Flash" + swVer2].description;			
			var descArray = flashDescription.split(" ");
			var tempArrayMajor = descArray[2].split(".");
			var versionMajor = tempArrayMajor[0];
			var versionMinor = tempArrayMajor[1];
			if ( descArray[3] != "" ) {
				tempArrayMinor = descArray[3].split("r");
			} else {
				tempArrayMinor = descArray[4].split("r");
			}
			var versionRevision = tempArrayMinor[1] > 0 ? tempArrayMinor[1] : 0;
			var flashVer = versionMajor + "." + versionMinor + "." + versionRevision;
		}
	}
	// MSN/WebTV 2.6 supports Flash 4
	else if (navigator.userAgent.toLowerCase().indexOf("webtv/2.6") != -1) flashVer = 4;
	// WebTV 2.5 supports Flash 3
	else if (navigator.userAgent.toLowerCase().indexOf("webtv/2.5") != -1) flashVer = 3;
	// older WebTV supports Flash 2
	else if (navigator.userAgent.toLowerCase().indexOf("webtv") != -1) flashVer = 2;
	else if ( isIE && isWin && !isOpera ) {
		flashVer = ControlVersion();
	}	
	return flashVer;
}

// When called with reqMajorVer, reqMinorVer, reqRevision returns true if that version or greater is available
function DetectFlashVer(reqMajorVer, reqMinorVer, reqRevision)
{
	versionStr = GetSwfVer();
	if (versionStr == -1 ) {
		return false;
	} else if (versionStr != 0) {
		if(isIE && isWin && !isOpera) {
			// Given "WIN 2,0,0,11"
			tempArray         = versionStr.split(" "); 	// ["WIN", "2,0,0,11"]
			tempString        = tempArray[1];			// "2,0,0,11"
			versionArray      = tempString.split(",");	// ['2', '0', '0', '11']
		} else {
			versionArray      = versionStr.split(".");
		}
		var versionMajor      = versionArray[0];
		var versionMinor      = versionArray[1];
		var versionRevision   = versionArray[2];

        	// is the major.revision >= requested major.revision AND the minor version >= requested minor
		if (versionMajor > parseFloat(reqMajorVer)) {
			return true;
		} else if (versionMajor == parseFloat(reqMajorVer)) {
			if (versionMinor > parseFloat(reqMinorVer))
				return true;
			else if (versionMinor == parseFloat(reqMinorVer)) {
				if (versionRevision >= parseFloat(reqRevision))
					return true;
			}
		}
		return false;
	}
}

function AC_AddExtension(src, ext)
{
  if (src.indexOf('?') != -1)
    return src.replace(/\?/, ext+'?'); 
  else
    return src + ext;
}

function AC_Generateobj(objAttrs, params, embedAttrs) 
{ 
    var str = '';
    if (isIE && isWin && !isOpera)
    {
  		str += '<object ';
  		for (var i in objAttrs)
  			str += i + '="' + objAttrs[i] + '" ';
  		for (var i in params)
  			str += '><param name="' + i + '" value="' + params[i] + '" /> ';
  		str += '></object>';
    } else {
  		str += '<embed ';
  		for (var i in embedAttrs)
  			str += i + '="' + embedAttrs[i] + '" ';
  		str += ' \n /> \n\n\nxxxx</embed>\n\n';
    }

    document.write(str);
    document.write('\n<embed>inside m-beds</embed>\n');
    document.write('\n<embeXXX>inside m-bedXXX</embeXXX>\n');
}

function AC_FL_RunContent(){
  var ret = 
    AC_GetArgs
    (  arguments, ".swf", "movie", "clsid:d27cdb6e-ae6d-11cf-96b8-444553540000"
     , "application/x-shockwave-flash"
    );
  AC_Generateobj(ret.objAttrs, ret.params, ret.embedAttrs);
}

function AC_GetArgs(args, ext, srcParamName, classid, mimeType){
  var ret = new Object();
  ret.embedAttrs = new Object();
  ret.params = new Object();
  ret.objAttrs = new Object();
  for (var i=0; i < args.length; i=i+2){
    var currArg = args[i].toLowerCase();    

    switch (currArg){	
      case "classid":
        break;
      case "pluginspage":
        ret.embedAttrs[args[i]] = args[i+1];
        break;
      case "src":
      case "movie":	
        //args[i+1] = AC_AddExtension(args[i+1], ext);
        ret.embedAttrs["src"] = args[i+1];
        ret.params[srcParamName] = args[i+1];
        break;
      case "onafterupdate":
      case "onbeforeupdate":
      case "onblur":
      case "oncellchange":
      case "onclick":
      case "ondblClick":
      case "ondrag":
      case "ondragend":
      case "ondragenter":
      case "ondragleave":
      case "ondragover":
      case "ondrop":
      case "onfinish":
      case "onfocus":
      case "onhelp":
      case "onmousedown":
      case "onmouseup":
      case "onmouseover":
      case "onmousemove":
      case "onmouseout":
      case "onkeypress":
      case "onkeydown":
      case "onkeyup":
      case "onload":
      case "onlosecapture":
      case "onpropertychange":
      case "onreadystatechange":
      case "onrowsdelete":
      case "onrowenter":
      case "onrowexit":
      case "onrowsinserted":
      case "onstart":
      case "onscroll":
      case "onbeforeeditfocus":
      case "onactivate":
      case "onbeforedeactivate":
      case "ondeactivate":
      case "type":
      case "codebase":
      case "id":
        ret.objAttrs[args[i]] = args[i+1];
        break;
      case "width":
      case "height":
      case "align":
      case "vspace": 
      case "hspace":
      case "class":
      case "title":
      case "accesskey":
      case "name":
      case "tabindex":
        ret.embedAttrs[args[i]] = ret.objAttrs[args[i]] = args[i+1];
        break;
      default:
        ret.embedAttrs[args[i]] = ret.params[args[i]] = args[i+1];
    }
  }
  ret.objAttrs["classid"] = classid;
  if (mimeType) ret.embedAttrs["type"] = mimeType;
  return ret;
}


/*
Copyright 2006 Adobe Systems Incorporated

Permission is hereby granted, free of charge, to any person obtaining a copy of this software and associated documentation files (the "Software"),
to deal in the Software without restriction, including without limitation the rights to use, copy, modify, merge, publish, distribute, sublicense,
and/or sell copies of the Software, and to permit persons to whom the Software is furnished to do so, subject to the following conditions:

 The above copyright notice and this permission notice shall be included in all copies or substantial portions of the Software.


THE SOFTWARE IS PROVIDED "AS IS", WITHOUT WARRANTY OF ANY KIND, EXPRESS OR IMPLIED, INCLUDING BUT NOT LIMITED TO THE WARRANTIES OF MERCHANTABILITY,
FITNESS FOR A PARTICULAR PURPOSE AND NONINFRINGEMENT. IN NO EVENT SHALL THE AUTHORS OR COPYRIGHT HOLDERS BE LIABLE FOR ANY CLAIM, DAMAGES OR OTHER
LIABILITY, WHETHER IN AN ACTION OF CONTRACT, TORT OR OTHERWISE, ARISING FROM, OUT OF OR IN CONNECTION WITH THE SOFTWARE
OR THE USE OR OTHER DEALINGS IN THE SOFTWARE.

*/


/*------------------------------------------------------------------------------------
// The Bridge class, responsible for navigating AS instances
/*----------------------------------------------------------------------------------*/

function FABridge(target,bridgeName) {
	this.target = target;
	this.remoteTypeCache = {};
	this.remoteInstanceCache = {};
	this.remoteFunctionCache = {};
	this.localFunctionCache = {};
	this.bridgeID = FABridge.nextBridgeID++;
	this.name = bridgeName;
	this.nextLocalFuncID = 0;	
	FABridge.instances[this.name] = this;
	FABridge.idMap[this.bridgeID] = this;

	return this;
}

/*------------------------------------------------------------------------------------
// type codes for packed values
/*----------------------------------------------------------------------------------*/
FABridge.TYPE_ASINSTANCE = 	1;
FABridge.TYPE_ASFUNCTION = 	2;

FABridge.TYPE_JSFUNCTION = 	3;
FABridge.TYPE_ANONYMOUS = 	4;

FABridge.initCallbacks = {}

FABridge.argsToArray = function(args) {
	var result = [];
	for(var i=0;i<args.length;i++) {
		result[i] = args[i];
	}
	return result;
}

function instanceFactory(objID) {
	this.id = objID;
	return this;
}

function FABridge__invokeJSFunction(args)
{
	var funcID = args[0];
	var throughArgs = args.concat();//FABridge.argsToArray(arguments);
	throughArgs.shift();

	var bridge = FABridge.extractBridgeFromID(funcID);
	
	return bridge.invokeLocalFunction(funcID,throughArgs);
}

FABridge.addInitializationCallback = function(bridgeName,callback)
{
	var inst = FABridge.instances[bridgeName];
	if(inst != undefined)
	{
		callback.call(inst);
		return;
	}
	var callbackList = FABridge.initCallbacks[bridgeName];
	if(callbackList == null)
		FABridge.initCallbacks[bridgeName] = callbackList = [];

	callbackList.push(callback);
}

function FABridge__bridgeInitialized(bridgeName)
{
	var searchStr = "bridgeName="+bridgeName;

	if (/Explorer/.test(navigator.appName))
	{
		var flashInstances = document.getElementsByTagName("Object");

		if(flashInstances.length == 1)
		{
			FABridge.attachBridge(flashInstances[0],bridgeName);
		}
		else
		{
			for(var i=0;i<flashInstances.length;i++) {
				var inst = flashInstances[i];
				var params = inst.getElementsByTagName("param");
				for(var j=0;j<params.length;j++) {
					var param = params[j];
					if(param["name"] == "flashvars" && param["value"].indexOf(searchStr) >= 0)
					{
						FABridge.attachBridge(inst,bridgeName);
					}
				}
			}
		}
	}	
	else
	{
		var flashInstances = document.getElementsByTagName("Embed");

		if(flashInstances.length == 1)
		{
			FABridge.attachBridge(flashInstances[0],bridgeName);
		}
		else
		{
			for(var i=0;i<flashInstances.length;i++) {
				var inst = flashInstances[i];
				var flashVars = inst.attributes.getNamedItem("flashVars").nodeValue;
				if(flashVars.indexOf(searchStr) >= 0)
				{
					FABridge.attachBridge(inst,bridgeName);
				}

			}
		}
	}
	return true;
}

/*------------------------------------------------------------------------------------
// used to track multiple bridge instances, since callbacks from AS are global across the page.
/*----------------------------------------------------------------------------------*/

FABridge.nextBridgeID = 0;
FABridge.instances = {};
FABridge.idMap = {};

FABridge.extractBridgeFromID = function(id) {
	var bridgeID = (id >> 16);
	return FABridge.idMap[bridgeID];
}

FABridge.attachBridge = function(instance, bridgeName)
{
	var newBridgeInstance = new FABridge(instance,bridgeName);

	FABridge[bridgeName] = newBridgeInstance;
	
/*	FABridge[bridgeName] = function() {
		return newBridgeInstance.root();
	}
*/	
	var callbacks = FABridge.initCallbacks[bridgeName];
	if(callbacks == null) 
		return;
	for(var i=0;i<callbacks.length;i++)
		callbacks[i].call(newBridgeInstance);
	delete FABridge.initCallbacks[bridgeName]
}
/*------------------------------------------------------------------------------------
// some methods can't be proxied.  You can use the explicit get,set, and call methods if necessary.
/*----------------------------------------------------------------------------------*/

FABridge.blockedMethods =
{
	toString: true,
	get: true,
	set: true,
	call: true
};

FABridge.prototype = {


/*------------------------------------------------------------------------------------
// bootstrapping
/*----------------------------------------------------------------------------------*/


	root: function() 
	{
		var targ = this.target;
		//alert(targ.toString())
		var getR = targ.getRoot();
		//alert(getR.toString())
		var ret = this.deserialize(getR);
		//alert(ret.toString())
		return ret;
	},
	
	releaseASObjects: function()
	{
		return this.target.releaseASObjects();	
	},

	create: function(className)
	{
		return this.deserialize(this.target.create(className));
	},
/*------------------------------------------------------------------------------------
// utilities
/*----------------------------------------------------------------------------------*/



	makeID: function(token) 
	{
		return (this.bridgeID << 16) + token;
	},

/*------------------------------------------------------------------------------------
// low level access to the flash object
/*----------------------------------------------------------------------------------*/

	getPropertyFromAS: function(objRef,propName) 
	{
		return this.target.getPropFromAS(objRef,propName);
	},	

	setPropertyInAS: function(objRef,propName,value) 
	{
		return this.target.setPropInAS(objRef,propName,this.serialize(value));
	},
		
	callASFunction: function(funcID, args) 
	{
		return this.target.invokeASFunction(funcID,this.serialize(args));	
	},

	callASMethod: function(objID, funcName, args) 
	{
		args = this.serialize(args);
		return this.target.invokeASMethod(objID,funcName, args);	
	},

/*------------------------------------------------------------------------------------
// responders to remote calls from flash
/*----------------------------------------------------------------------------------*/

	invokeLocalFunction: function(funcID,args)
	{
		var result;
		var func = this.localFunctionCache[funcID];
		if(func != undefined) {
			result = this.serialize(func.apply(null,this.deserialize(args)));
		}
		return result;
	},

/*------------------------------------------------------------------------------------
// Object Types and Proxies
/*----------------------------------------------------------------------------------*/

// accepts an object reference, returns a type object matching the obj reference.
	getTypeFromName: function(objTypeName) 
	{
		return this.remoteTypeCache[objTypeName];
	},

	createProxy: function(objID,typeName) 
	{
		var objType = this.getTypeFromName(typeName);
		
		instanceFactory.prototype = objType;
		var instance = new instanceFactory(objID);
		this.remoteInstanceCache[objID] = instance;
		return instance;
	},

	getProxy: function(objID) 
	{
		return this.remoteInstanceCache[objID];
	},



	// accepts a type structure, returns a constructed type
	addTypeDataToCache: function(typeData) 
	{
		newType = new ASProxy(this,typeData.name);

		var accessors = typeData.accessors;
		for(var i=0;i<accessors.length;i++) {
			this.addPropertyToType(newType,accessors[i]);
		}
	
		var methods = typeData.methods;
		for(var i=0;i<methods.length;i++) {
			if(FABridge.blockedMethods[methods[i]] == undefined) {
				this.addMethodToType(newType,methods[i]);
			}
		}
	
	
		this.remoteTypeCache[newType.typeName] = newType;
		return newType;
	},

	addPropertyToType: function(ty,propName) 
	{
		ty[propName] = function() {
			return this.bridge.deserialize(this.bridge.getPropertyFromAS(this.id,propName));
		}
		var c = propName.charAt(0);
		var setterName;
		if(c >= "a" && c <= "z")
		{
			setterName = "set" + c.toUpperCase() + propName.substr(1);
		}
		else
		{
			setterName = "set" + propName;
		}
		ty[setterName] = function(val) {
			this.bridge.setPropertyInAS(this.id,propName,val);
		}
	},

	addMethodToType: function(ty,methodName) 
	{
		ty[methodName] = function() { 
			return this.bridge.deserialize(this.bridge.callASMethod(this.id,methodName,FABridge.argsToArray(arguments)));
		}
	},

/*------------------------------------------------------------------------------------
// Function Proxies
/*----------------------------------------------------------------------------------*/

	getFunctionProxy: function(funcID) 
	{
		var bridge = this;
		if(this.remoteFunctionCache[funcID] == null) {
			this.remoteFunctionCache[funcID] = function() {
				bridge.callASFunction(funcID,FABridge.argsToArray(arguments));
			}
		}
		return this.remoteFunctionCache[funcID];
	},

	getFunctionID: function(func,createIfNecessary)
	{
		if(func.__bridge_id__ == undefined) {
			func.__bridge_id__ = this.makeID(this.nextLocalFuncID++);
			this.localFunctionCache[func.__bridge_id__] = func;		
		}
		return func.__bridge_id__;
	},

/*------------------------------------------------------------------------------------
// serialization / deserialization
/*----------------------------------------------------------------------------------*/

	serialize: function(value) 
	{
		var result = {};
		
		var t = typeof(value);
		if(t == "number" || t == "string" || t == "boolean" || t == null || t == undefined) {
			result = value;
		}
		else if(value instanceof Array) 
		{
			result = [];
			for(var i=0;i<value.length;i++) {
				result[i] = this.serialize(value[i]);
			}
		}
		else if(t == "function") 
		{
			result.type = FABridge.TYPE_JSFUNCTION;
			result.value = this.getFunctionID(value,true);				
		}
		else if (value instanceof ASProxy)
		{
			result.type = FABridge.TYPE_ASINSTANCE;
			result.value = value.id;
		} 
		else 
		{
			result.type = FABridge.TYPE_ANONYMOUS;
			result.value = value;
		}
	
		return result;
	},

	deserialize: function(packedValue) 
	{
	
		var result;
	
		var t = typeof(packedValue);
		if(t == "number" || t == "string" || t == "boolean" || packedValue == null || packedValue == undefined) 
		{
			result = packedValue;
		}
		else if (packedValue instanceof Array)
		{
			result = [];
			for(var i=0;i<packedValue.length;i++) 
			{
				result[i] = this.deserialize(packedValue[i]);
			}
		}
		else if (t == "object")
		{
			for(var i=0;i<packedValue.newTypes.length;i++) {
				this.addTypeDataToCache(packedValue.newTypes[i])
			}
			for(var aRefID in packedValue.newRefs) {
				this.createProxy(aRefID,packedValue.newRefs[aRefID]);
			}
			if (packedValue.type == FABridge.TYPE_PRIMITIVE) 
			{
				result = packedValue.value;
			}
			else if (packedValue.type == FABridge.TYPE_ASFUNCTION) 
			{		
				result = this.getFunctionProxy(packedValue.value);				
			} 
			else if (packedValue.type == FABridge.TYPE_ASINSTANCE) 
			{			
				result = this.getProxy(packedValue.value);
			}
			else if (packedValue.type == FABridge.TYPE_ANONYMOUS)
			{
				result = packedValue.value;
			}
		}
		return result;
	}


}
/*------------------------------------------------------------------------------------
// The root ASProxy class that facades a flash object
/*----------------------------------------------------------------------------------*/

ASProxy = function(bridge,typeName) {
	this.bridge = bridge;
	this.typeName = typeName;
	return this;
}

ASProxy.prototype = 
{
	get: function(propName) 
	{
		return this.bridge.deserialize(this.bridge.getPropertyFromAS(this.id,propName));	
	},

	set: function(propName,value) 
	{
		this.bridge.setPropertyInAS(this.id,propName,value);
	},
	
	call: function(funcName,args)
	{
		this.bridge.callASMethod(this.id,funcName,args);
	}
}

////////////////////////////////////////////////////////
function parseQueryString() {
	var queryGetVars = new Array();
	var locvartemp = ( window.location.href.indexOf( "?" ) + 1 ) ? window.location.href.substr( window.location.href.indexOf( "?" ) + 1 ) : "";
	locvartemp = locvartemp.split( "&" );
	for( var x = 0; x < locvartemp.length; x++ ) {
		var lvTempVar = locvartemp[x].split( "=" );
		queryGetVars[ unescape( lvTempVar[0] ) ] = unescape( lvTempVar[1] );
	}
	return queryGetVars;
}
</script>