`
全能骑士
  • 浏览: 68905 次
  • 性别: Icon_minigender_1
  • 来自: 北京
社区版块
存档分类
最新评论

Flex Module基础

    博客分类:
  • FLEX
阅读更多
Module加载方式概览
在理解这种加载方式之前,我们应该首先知道什么是Module模块。Module 实际上是一个预编译的SWF文件。虽然是SWF格式的文件,但是这个文件不能独立运行,并且只能被ModuleLoader加载后才能显示。逻辑上它是一个容器,可以像一般的容器一样包含别的容器,组件,甚至是别的Module模块。根据需要,预编译的Module模块可以被应用加载和卸载。不同的应用可以共享这些Module模块。Flex应用可以被分割为若干个预编译的Module模块,可以在需要时分别加载这些模块,避免在系统初始化时加载全部子容器。采用这种方式的Flex应用从设计上分隔了逻辑相对独立的模块,减少了系统初始化时的加载时间。



Module的创建、编译、加载和卸载方法
我们有两种方式创建一个Module模块,一是利用MXML标签<mx:Module>创建Module类;令一种方式采用ModuleManager类在ActionScript中创建Module模块类。模块类创建后将被编译成SWF文件。如前面章节所述我们可以利用mxmlc编译器手动编译或者在Flex Builder3集成开发环境提供的工具自动编译Module类为SWF文件。这里介绍采用Flex Builder3集成开发环境提供的方式创建一个Module类。

在FlexBuilder3中创建基于MXML标签的Module模块

创建基于MXML标签的Module模块,需要扩展mx.modules.Module.

1.        第一步,在Flex Builder3集成开发环境中创建一个Flex项目Moduler。



2.        第二步,选择FileàNewàMXML Module



3.        第三步,输入Module文件名字为MXMLDemoModule,设置Module的高度和宽度,选择Module容器的布局方式为absolute.,选择默认的预编译后优化的SWF选项,单击Finish按钮。

4.        Module文件被创建,编辑Module文件,在此例中我们可以假定是一个登陆Module界面,

其源代码为,

<?xml version="1.0" encoding="utf-8"?>

<mx:Module xmlns:mx="http://www.adobe.com/2006/mxml" layout="absolute" width="400" height="300">

        <mx:Panel width="300" height="220" layout="absolute" horizontalCenter="0" 

                               verticalCenter="0" backgroundAlpha="0.5">

                               <mx:Form horizontalCenter="0" verticalCenter="0"  paddingBottom="0" paddingLeft="0" paddingRight="0" 

                                      paddingTop="0">

                                      <mx:FormItem label="Account"  indicatorGap="5">

                                              <mx:TextInput borderStyle="solid" id="account" text="admin"/>

                                      </mx:FormItem>

                                      <mx:FormItem label="Password"  indicatorGap="5">

                                              <mx:TextInput borderStyle="solid" id="password" text="admin" displayAsPassword="true"/>

                                      </mx:FormItem>

                                      <mx:FormItem direction="horizontal" >

                                              <mx:Button id="logined" label="Login" styleName="blueButton"/>

                                              <mx:Button id="reset" label="Reset" styleName="greenButton"/>

                                      </mx:FormItem>

                               </mx:Form>

                       </mx:Panel>

                       <mx:Label text="CopyRight 2008 demo Solutions" horizontalCenter="0" verticalCenter="252.5"/>

</mx:Module>



集成登陆Module模块到主应用文件ModulerApp.mxml,

<?xml version="1.0" encoding="utf-8"?>

<mx:Application xmlns:mx="http://www.adobe.com/2006/mxml"

layout="absolute">

<mx:ModuleLoader url="MXMLDemoModule.swf"/>

</mx:Application>



创建基于ActionScript的Module模块

在ActionScript中创建的Module模块类大都继承了mx.modules.Module或者mx.modules.ModuleBase基础类。我们都知道,MXML标签<mx:Module>实际上是mx.modules.Module类的另一种表现方式,这也就不难理解无论在MXML标签中还是在ActionScript中创建Module模块其实都是相通的。自定义Module类继承mx.modules.Module和继承mx.modules.ModuleBase两个基类的不同是,继承自前者的或者基于<mx:Module>标签的自定义Module组件将被加入到框架可视化显示列表,后者则没有。参考示例

package {

import mx.modules.ModuleBase;

public class MyModule extends ModuleBase {

public function MyModule() {

trace("MyModule was created!");

}

}

}





  编译Module模块

前面已经提及我们有两种方式来编译Module模块文件。像编译主应用文件一样,一种是在命令行手动编译,最简单的情形 可以利用这个命令

                Mxmlc MyModule.mxml

另一种是在Flex Builder3中提供的自动编译工具编译。编译的结果是一个可以被装载的SWF文件,与编译后的SWF主应用文件最大的不同就是,模块SWF文件不能独立运行,只能在被装载后才能够和其宿主共同运行。



   怎样装载和卸载Module模块



总体来说,有几种方式可以实现Module模块的装载和卸载。

利用ModuleLoader:ModuleLoader类提供了一系列高层处理Module的编程接口。
利用ModuleManager: ModuleManager类提供了低层次的处理Module的装载卸载以及事件响应等的变成接口。
下面将详细地讲解这两种处理方式。

我们可以利用ModuleLoader类在主应用文件或者别的Module模块中加载任意预编译的Module对象。最简单的方式是采用ModuleLoader类的标签形式<mx:ModuleLoader>在主应用的MXML文件中显式地加载Module,然后只需要设置这个标签的url属性为预编译Module的SWF文件位置,可以是相对路径,也可以是绝对路径。参考实例,

<?xml version="1.0" encoding="utf-8"?>

<mx:Application xmlns:mx="http://www.adobe.com/2006/mxml"

layout="absolute">

<mx:ModuleLoader url="MXMLDemoModule.swf"/>

</mx:Application>

编程时我们可以根据需要替换此url所指向的模块位置,实现不同模块的更迭显示。当ModuleLoader类初始化时或者每次更改这个url属性值时,这个ModuleLoader类的loadModule()方法被触发。如果设定url属性为空字符串,ModuleLoader卸载当前加载的Module模块。

ModuleLoader其实是一种特殊的导航式容器。和一般导航式容器如ViewStack不同的是,ModuleLoader不必在初始化时携带加载所有的孩子组件。了解这一点,我们可以猜想到,在一个Flex主应用中可以包含甚至嵌套包含多个ModuleLoader实例,以下示例展示在一个ViewStack容器中包含多个ModuleLoader的示例。

<?xml version="1.0"?>

<mx:Application xmlns:mx="http://www.adobe.com/2006/mxml">

<mx:Panel

title="Multiple Modules Demo "

height="100%"

width="100%"

paddingTop="0"

paddingLeft="0"

paddingRight="0"

paddingBottom="0"

> 

<mx:VBox id="v1" label="Module1">

<mx:Label id="l1" text="Module1.swf"/>

<mx:ModuleLoader url="Module1.swf"/>

</mx:VBox>

 

<mx:VBox id="v2" label="Module2">

<mx:Label id="l2" text="Module2.swf"/>

<mx:ModuleLoader url="Module2.swf"/>

</mx:VBox>

 

<mx:VBox id="v3" label="Module3">

<mx:Label id="l3" text="Module3.swf"/>

<mx:ModuleLoader url="Module3.swf"/>

</mx:VBox>

</mx:TabNavigator>

</mx:Panel>

</mx:Application>



同时我们可以利用ModuleLoader类提供的loadModule()和unloadModule()方法动态地指定被加载的Module模块。这两个方法没有输入形参,调用时分别加载和卸载当前ModuleLoader实例url属性所指向的Module模块。下面示例展示了在TabNavigator容器中,按钮点击后分别加载卸载不同的Module模块

<?xml version="1.0"?>

<mx:Application xmlns:mx="http://www.adobe.com/2006/mxml">

<mx:Script>

<![CDATA[

import mx.modules.*;

public function moduleLoader(m:ModuleLoader, s:String):void {

if (!m.url) {

m.url = s;

return;

}

m.loadModule();

}

public function moduleUnloader(m:ModuleLoader):void {

m.unloadModule();

}

]]>

</mx:Script>

<mx:Panel title="Dynamically load/unload multiple Modules"

height="100%"

width="100%"

paddingTop="0"

paddingLeft="0"

paddingRight="0"

paddingBottom="0"

> 

<mx:TabNavigator id="navigator"

width="100%"

height="100%"

creationPolicy="all"

> 

<mx:VBox id="vb1" label="Module1">

<mx:Button label="Load" click="moduleLoader(moduleLoader, l1.text)" />

<mx:Button label="Unload" click="moduleUnloader(moduleLoader)" />

<mx:Label id="l1" text="Module1.swf"/>

<mx:ModuleLoader id="moduleLoader"/>

</mx:VBox>

 

<mx:VBox id="vb2" label="Module2">

<mx:Button label="Load" click="moduleLoader(moduleLoader2, l2.text)"/>

<mx:Button label="Unload" click="removeModule(moduleLoader2)" />

<mx:Label id="l2" text="Module2.swf"/>

<mx:ModuleLoader id="moduleLoader2"/>

</mx:VBox>

 

<mx:VBox id="vb3" label="Module3">

<mx:Button label="Load" click="moduleLoader(moduleLoader3, l3.text)"/>

<mx:Button label="Unload" click="removeModule(moduleLoader3)" />

<mx:Label id="l3" text="Module3.swf"/>

<mx:ModuleLoader id="moduleLoader3"/>

</mx:VBox>

 

</mx:TabNavigator>

</mx:Panel>

</mx:Application>



我们当然也可以利用ModuleManager类来加载Module模块。这种方式比起纯粹的ModuleLoader方式稍微复杂一点,但是ModuleManager提供了比ModuleLoader更加强大的能力来管理Module模块的加载过程。具体操作可以分成以下几步

通过ModuleManager实例的getModule()方法拿到Module模块的一个索引,索引类型为IModuleInfo。
调用这个索引的load()方法。
利用这个接口的factory属性拿到它相关连的Module工厂,调用此工厂的create()方法,并将返回值强制转换成当前的Module类型。
参考示例

<?xml version="1.0"?>

<mx:Application xmlns:mx="http://www.adobe.com/2006/mxml" creationComplete="init()">

<mx:Script>

<![CDATA[

import mx.events.ModuleEvent;

import mx.modules.ModuleManager;

import mx.modules.IModuleInfo;

public var info:IModuleInfo;

private function init():void {

info = ModuleManager.getModule("Module1.swf");

info.addEventListener(ModuleEvent.READY, moduleEventHandler);

info.load();

}

private function moduleEventHandler(e:ModuleEvent):void {

vb.addChild(info.factory.create() as Module1);

}

]]>

</mx:Script>

<mx:VBox id="vb"/>

</mx:Application>



理论上说,Flex应用第一次启动时初始化下载的大小,集成了Module模块的应用相对比没有集成Module模块的相似应用要小一些。这减少了初始化页面的等待时间。甚至在Module模块的SWF文件还没有下载完毕的时候,主应用文件也可以顺利显示。Module模块第一次加载时将被缓存在客户端IE, 已缓存的Module模块被再次加载时,FlashPlayer将直接在缓存中加载Module实例,减少了加载时间,提高了用户体验。当然这有一个前提就是,客户端IE没有被清空。所以在实际操作中,你可以在SWF模块真正的被使用之前将其预加载到客户端缓存。方法是调用ModuleManager.getModule("Module1.swf").load()语句。在调用create()方法之前并没有创建这个模块实例,而只是纯粹地将模块SWF文件加载到客户端内存。下面实例展示了在系统初始化时预先加载一个名字叫做Module1.swf的模块,当此模块需要显示时才创建这个Module模块实例。

<?xml version="1.0"?>

<mx:Application xmlns:mx="http://www.adobe.com/2006/mxml"

creationComplete="modulePreloader()">

<mx:Script>

<![CDATA[

import mx.events.ModuleEvent;

import mx.modules.ModuleManager;

import mx.modules.IModuleInfo;

private function modulePreloader():void {

var info:IModuleInfo = ModuleManager.getModule("Module1.swf");

info.addEventListener(ModuleEvent.READY, moduleEventHandler);

info.load();

}

private function moduleEventHandler(e:ModuleEvent):void {

trace('' + e.type); 

}

]]>

</mx:Script>

<mx:Panel title="Module Preloader" height="100%" width="100%" >

<mx:TabNavigator id="tn" width="100%" height="100%" creationPolicy="all" >

<mx:VBox id="vb1" label="Module1">

<mx:ModuleLoader url="Module1.swf"/>

</mx:VBox>

</mx:TabNavigator>

</mx:Panel>

</mx:Application>



  怎样处理ModuleLoader事件
我们不仅可以以上述的方式加载卸载预编译Module模块,而且更进一步,也可以捕捉和处理模块在加载卸载过程中可能会触发的各种事件。这些事件由ModuleLoader触发,分别有setup,ready,loading,unload,progress,error和urlChanged等。很好的利用这些内置的事件可以使我们更加精细地跟踪模块加载卸载过程,实现更加强大的功能。下述示例展示了一个自定义的ModuleLoader组件CustomModuleLoader,这个组件会跟踪当模块实例被主应用加载时所有触发的事件。

<?xml version="1.0" encoding="iso-8859-1"?>

<mx:ModuleLoader xmlns:mx="http://www.adobe.com/2006/mxml" xmlns="*"

creationComplete="init()">

<mx:Script>

<![CDATA[

public function init():void {

addEventListener("urlChanged", onUrlChanged);

addEventListener("loading", onLoading);

addEventListener("progress", onProgress);

addEventListener("setup", onSetup);

addEventListener("ready", onReady);

addEventListener("error", onError);

addEventListener("unload", onUnload);

standin = panel;

removeChild(standin);

}

public function onUrlChanged(event:Event):void {

if (url == null) {

if (contains(standin))

removeChild(standin);

} else {

if (!contains(standin))

addChild(standin);

}

progress.indeterminate=true;

unload.enabled=false;

reload.enabled=false;

}

public function onLoading(event:Event):void {

progress.label="Loading module " + url;

if (!contains(standin))

addChild(standin);

progress.indeterminate=true;

unload.enabled=false;

reload.enabled=false;

}

public function onProgress(event:Event):void {

progress.label="Loaded %1 of %2 bytes...";

progress.indeterminate=false;

unload.enabled=true;

reload.enabled=false;

}

 

public function onSetup(event:Event):void {

progress.label="Module " + url + " initialized!";

progress.indeterminate=false;

unload.enabled=true;

reload.enabled=true;

}

public function onReady(event:Event):void {

progress.label="Module " + url + " successfully loaded!";

unload.enabled=true;

reload.enabled=true;

if (contains(standin))

removeChild(standin);

}

public function onError(event:Event):void {

progress.label="Error loading module " + url;

unload.enabled=false;

reload.enabled=true;

}

public function onUnload(event:Event):void {

if (url == null) {

if (contains(standin))

removeChild(standin);

} else {

if (!contains(standin))

addChild(standin);

}

progress.indeterminate=true;

progress.label="Module " + url + " was unloaded!";

unload.enabled=false;

reload.enabled=true;

}

public var standin:DisplayObject;

]]>

</mx:Script>

<mx:Panel id="panel" width="100%">

<mx:ProgressBar width="100%" id="progress" source="{this}"/>

<mx:HBox width="100%">

<mx:Button id="unload"

label="Unload Module"

click="unloadModule()"

/>

<mx:Button id="reload"

label="Reload Module"

click="unloadModule();loadModule()"

/>

</mx:HBox>

</mx:Panel>

</mx:ModuleLoader>

 



<?xml version="1.0"?>

<mx:Application xmlns="*" xmlns:mx="http://www.adobe.com/2006/mxml">

<mx:Script>

<![CDATA[

[Bindable]

public var selectedItem:Object;

]]>

</mx:Script>

<mx:ComboBox

width="215"

labelField="label"

close="selectedItem=ComboBox(event.target).selectedItem"

> 

<mx:dataProvider>

<mx:Object label="Select Coverage"/>

<mx:Object

label="Life Insurance"

module="insurancemodules/LifeInsurance.swf"

/>

<mx:Object

label="Auto Insurance"

module="insurancemodules/AutoInsurance.swf"

/>

<mx:Object

label="Home Insurance"

module="insurancemodules/HomeInsurance.swf"

/>

</mx:dataProvider>

</mx:ComboBox>

<mx:Panel width="100%" height="100%">

<CustomModuleLoader id="mod"

width="100%"

url="{selectedItem.module}"

/>

</mx:Panel>

<mx:HBox>

<mx:Button label="Unload" click="mod.unloadModule()"/>

<mx:Button label="Nullify" click="mod.url = null"/>

</mx:HBox>

</mx:Application>



应用error事件

使用ModuleLoader的error事件可以允许开发者在当不知什么原因Module模块没有被加载或卸载成功的时候,做一些必要的动作。参考以下示例

<?xml version="1.0"?>

<mx:Application xmlns:mx="http://www.adobe.com/2006/mxml">

<mx:Script>

<![CDATA[

import mx.events.ModuleEvent;

import mx.modules.*;

import mx.controls.Alert;

private function errorHandler(e:ModuleEvent):void {

Alert.show("There was an error loading the module." +

" Please contact the Help Desk.");

trace(e.errorText);

}

public function createModule():void {

if (chartModuleLoader.url == ti1.text) {

// If they are the same, call loadModule.

chartModuleLoader.loadModule();

} else {

// If they are not the same, then change the url,

// which triggers a call to the loadModule() method.

chartModuleLoader.url = ti1.text;

}

}

public function removeModule():void {

chartModuleLoader.unloadModule();

}

]]>

</mx:Script>

<mx:Panel title="Module Example"

height="90%"

width="90%"

paddingTop="10"

paddingLeft="10"

paddingRight="10"

paddingBottom="10"

> 

<mx:HBox>

<mx:Label text="URL:"/>

<mx:TextInput width="200" id="ti1" text="ColumnChartModule.swf"/

> 

<mx:Button label="Load" click="createModule()"/>

<mx:Button label="Unload" click="removeModule()"/>

</mx:HBox>

<mx:ModuleLoader id="chartModuleLoader" error="errorHandler(event)"/

> 

</mx:Panel>

</mx:Application>



  应用progress事件

使用ModuleLoader的progress事件可以让你跟踪模块实例加载的进度信息。参考示例,

<?xml version="1.0"?>

<!-- modules/SimpleProgressEventHandler.mxml -->

<mx:Application xmlns:mx="http://www.adobe.com/2006/mxml">

<mx:Script>

<![CDATA[

import mx.events.ModuleEvent;

import flash.events.ProgressEvent;

import mx.modules.*;

[Bindable]

public var progBar:String = "";

[Bindable]

public var progMessage:String = "";

private function progressEventHandler(e:ProgressEvent):void {

progBar += ".";

progMessage =

"Module " +

Math.round((e.bytesLoaded/e.bytesTotal) * 100) +

"% loaded";

}

public function createModule():void {

chartModuleLoader.loadModule();

}

public function removeModule():void {

chartModuleLoader.unloadModule();

progBar = "";

progMessage = "";

}

]]>

</mx:Script>

<mx:Panel title="Module Example"

height="90%"

width="90%"

paddingTop="10"

paddingLeft="10"

paddingRight="10"

paddingBottom="10"

> 

<mx:HBox>

<mx:Label id="l2" text="{progMessage}"/>

<mx:Label id="l1" text="{progBar}"/>

</mx:HBox>

<mx:Button label="Load" click="createModule()"/>

<mx:Button label="Unload" click="removeModule()"/>

<mx:ModuleLoader

id="chartModuleLoader"

url="ColumnChartModule.swf"

progress="progressEventHandler(event)"

/>

</mx:Panel>

</mx:Application>



  怎样共享和传输Module模块间数据
Module模块是一个容器,每个独立的模块对象都相当于一个自定义组件。有以下几种方式可以实现模块-模块、模块-主应用、主应用-模块、模块-一般自定义组件间的数据传输和通信。

利用ModuleLoader的child、ModuleManager的factory、以及Application的parentApplication属性存取Module模块和主应用文件对象索引。
因为Module模块在ModuleLoader中通常用url属性来指定,所以我们可以通过在url上面拼GET参数,然后在Module模块中解析这些拼上去的参数的方式来传输数据。
通过ActionScript接口方式。你可以定义一个ActionScript接口,这个接口定义了一系列方法和属性。Module模块和主应用Application都可以访问这些属性。从而实现数据共享。
  在主应用Application中访问Module模块对象



我们可以在主应用Application中访问子Module模块对象中定义的方法和属性。下面示例展示了通过ModuleLoader的child属性拿到子Module模块对象索引,然后调用其定义的公共方法getTitle().

<?xml version="1.0"?>

<!-- modules/ParentApplication.mxml -->

<mx:Application xmlns:mx="http://www.adobe.com/2006/mxml">

<mx:Script><![CDATA[

[Bindable]

private var s:String;

private function getTitle():void {

s = (m1.child as ChildModule1).getModTitle();

}

]]></mx:Script>

<mx:Label id="l1" text="{s}"/>

<mx:ModuleLoader url="ChildModule1.swf" id="m1" ready="getTitle()"/>

</mx:Application>

Module模块文件,

<?xml version="1.0"?>

<!-- modules/ChildModule1.mxml -->

<mx:Module xmlns:mx="http://www.adobe.com/2006/mxml" width="100%"

height="100%">

<mx:Script><![CDATA[

// Defines the method that the application calls.

public function getModTitle():String {

return "Child Module 1";

}

]]></mx:Script>

</mx:Module>



通过这种方式得到子Module模块的索引导致了主应用Application和Module模块之间的紧耦合,不利于模块逻辑的重用。同样地,利用ModuleManager的factory属性也可以拿到当前ModuleLoader的子模块对象的索引。参考以下示例,模块文件为,

<?xml version="1.0"?>

<!-- modules/mxmlmodules/SimpleModule.mxml -->

<mx:Module xmlns:mx="http://www.adobe.com/2006/mxml">

<mx:Script>

<![CDATA[

public function computeAnswer(a:Number, b:Number):Number {

return a + b;

}

]]>

</mx:Script>

</mx:Module>



主应用Application为,

<?xml version="1.0"?>

<!-- modules/mxmlmodules/SimpleMXMLApp.mxml -->

<mx:Application xmlns:mx="http://www.adobe.com/2006/mxml"

creationComplete="initApp()">

<mx:Script>

<![CDATA[

import mx.modules.IModuleInfo;

import mx.modules.ModuleManager;

public var assetModule:IModuleInfo;

public var sm:Object;

[Bindable]

public var answer:Number = 0;

public function initApp():void {

// Get the IModuleInfo interface for the specified URL.

assetModule = ModuleManager.getModule("SimpleModule.swf");

assetModule.addEventListener("ready", getModuleInstance);

assetModule.load();

}

public function getModuleInstance(e:Event):void {

// Get an instance of the module.

sm = assetModule.factory.create() as SimpleModule;

}

public function addNumbers():void {

var a:Number = Number(ti1.text);

var b:Number = Number(ti2.text);

// Call a method on the module.

answer = sm.computeAnswer(a, b).toString();

}

]]>

</mx:Script>

<mx:Form>

<mx:FormHeading label="Enter values to sum."/>

<mx:FormItem label="First Number">

<mx:TextInput id="ti1" width="50"/>

</mx:FormItem>

<mx:FormItem label="Second Number">

<mx:TextInput id="ti2" width="50"/>

</mx:FormItem>

<mx:FormItem label="Result">

<mx:Label id="ti3" width="100" text="{answer}"/>

</mx:FormItem>

<mx:Button id="b1" label="Compute" click="addNumbers()"/>

</mx:Form>

</mx:Application>



  在Module模块对象中存取主应用Application

Module模块可以通过其parentApplication属性拿到主应用Application对象的索引,通过这个索引可以访问主应用对象的公共方法和属性。参考以下示例,Module模块文件为,

<?xml version="1.0"?>

<!-- modules/ChartChildModule.mxml -->

<mx:Module xmlns:mx="http://www.adobe.com/2006/mxml" width="100%"

height="100%" creationComplete="getDataFromParent()">

<mx:Script><![CDATA[

import mx.collections.ArrayCollection;

[Bindable]

private var expenses:ArrayCollection;

// Access properties of the parent application.

private function getDataFromParent():void {

expenses = parentApplication.expenses;

}

]]></mx:Script>

<mx:ColumnChart id="myChart" dataProvider="{expenses}">

<mx:horizontalAxis>

<mx:CategoryAxis

dataProvider="{expenses}"

categoryField="Month"

/>

</mx:horizontalAxis>

<mx:series>

<mx:ColumnSeries

xField="Month"

yField="Profit"

displayName="Profit"

/>

<mx:ColumnSeries

xField="Month"

yField="Expenses"

displayName="Expenses"

/>

</mx:series>

</mx:ColumnChart>

<mx:Legend dataProvider="{myChart}"/>

<mx:Button id="b1" click="expenses = parentApplication.getNewData();"

label="Get New Data"/>

</mx:Module>



主应用文件为,

<?xml version="1.0"?>

<!-- modules/ChartChildModuleLoader.mxml -->

<mx:Application xmlns:mx="http://www.adobe.com/2006/mxml">

<mx:Script><![CDATA[

import mx.collections.ArrayCollection;

[Bindable]

public var expenses:ArrayCollection = new ArrayCollection([

{Month:"Jan", Profit:2000, Expenses:1500},

{Month:"Feb", Profit:1000, Expenses:200},

{Month:"Mar", Profit:1500, Expenses:500}

]);

public function getNewData():ArrayCollection {

return new ArrayCollection([

{Month:"Apr", Profit:1000, Expenses:1100},

{Month:"May", Profit:1300, Expenses:500},

{Month:"Jun", Profit:1200, Expenses:600}

]);

}

]]></mx:Script>

<mx:ModuleLoader url="ChartChildModule.swf" id="m1"/>

</mx:Application>



不过这种方式导致的一个缺点是,自定义的Module模块文件的可移植特性将大打折扣。

在一个Module模块对象中存取另一个Module模块对象

同样地,我们也可以利用ModuleLoader的child属性,在一个Module模块对象中拿到另一个Module模块对象的索引,通过这个索引访问当前Module模块对象的公共方法和属性。参考以下示例,主应用文件,

<?xml version="1.0"?>

<!-- modules/TitleModuleLoader.mxml -->

<mx:Application xmlns:mx="http://www.adobe.com/2006/mxml">

<mx:Script><![CDATA[

]]></mx:Script>

<mx:ModuleLoader url="InterModule1.swf" id="m1"/>

<mx:ModuleLoader url="InterModule2.swf" id="m2"/>

</mx:Application>



Module1文件,

<?xml version="1.0"?>

<!-- modules/InterModule1.mxml -->

<mx:Module xmlns:mx="http://www.adobe.com/2006/mxml" width="100%"

height="100%">

<mx:Script><![CDATA[

// Defines the method that the other module calls.

public function getNewTitle():String {

return "New Module Title";

}

]]></mx:Script>

</mx:Module>

Module2文件,

<?xml version="1.0"?>

<!-- modules/InterModule2.mxml -->

<mx:Module xmlns:mx="http://www.adobe.com/2006/mxml" width="100%"

height="100%">

<mx:Script><![CDATA[

[Bindable]

private var title:String;

// Call method of another module.

private function changeTitle():void {

title = parentApplication.m1.child.getNewTitle();

}

]]></mx:Script>

<mx:HBox>

<mx:Label id="l1" text="Title: "/>

<mx:Label id="myTitle" text="{title}"/>

</mx:HBox>

<mx:Button id="b1" label="Change Title" click="changeTitle()"/>

</mx:Module>



  通过拼ModuleLoader的url参数方式实现数据传输

在url上面拼GET参数基本上是这种格式,url=module1.swf?param1=value1&param2=value2

比如在主应用文件Application中拼一系列GET参数到ModuleLoader的url属性上,在Module模块文件中解析并处理这些参数,参考以下示例,主应用文件,

<?xml version="1.0"?>

<!-- modules/QueryStringApp.mxml -->

<mx:Application xmlns:mx="http://www.adobe.com/2006/mxml" height="500"

width="400">

<mx:Script><![CDATA[

public function initModule():void {

// Build query string so that it looks something like this:

// "QueryStringModule.swf?firstName=Nick&lastName=Danger"

var s:String = "QueryStringModule.swf?" + "firstName=" +

ti1.text + "&lastName=" + ti2.text;

// Changing the url property of the ModuleLoader causes

// the ModuleLoader to load a new module.

m1.url = s;

}

]]></mx:Script>

<mx:Form>

<mx:FormItem id="fi1" label="First Name:">

<mx:TextInput id="ti1"/>

</mx:FormItem>

<mx:FormItem id="fi2" label="Last Name:">

<mx:TextInput id="ti2"/>

</mx:FormItem>

</mx:Form>

<mx:ModuleLoader id="m1"/>

<mx:Button id="b1" label="Submit" click="initModule()"/>

</mx:Application>



模块文件,



<?xml version="1.0"?>

<!-- modules/QueryStringModule.mxml -->

<mx:Module xmlns:mx="http://www.adobe.com/2006/mxml"

creationComplete="parseString()">

<mx:Script>

<![CDATA[

import mx.utils.*;

[Bindable]

private var salutation:String;

public var o:Object = {};

public function parseString():void {

try {

// Remove everything before the question mark, including

// the question mark.

var myPattern:RegExp = /.*\?/;

var s:String = this.loaderInfo.url.toString();

s = s.replace(myPattern, "");

// Create an Array of name=value Strings.

var params:Array = s.split("&");

// Print the params that are in the Array.

var keyStr:String;

var valueStr:String;

var paramObj:Object = params;

for (keyStr in paramObj) {

valueStr = String(paramObj[keyStr]);

ta1.text += keyStr + ":" + valueStr + "\n";

}

// Set the values of the salutation.

for (var i:int = 0; i < params.length; i++) {

var tempA:Array = params[i].split("=");

if (tempA[0] == "firstName") {

o.firstName = tempA[1];

}

if (tempA[0] == "lastName") {

o.lastName = tempA[1];

}

}

if (StringUtil.trim(o.firstName) != "" &&

StringUtil.trim(o.lastName) != "") {

salutation = "Welcome " +

o.firstName + " " + o.lastName + "!";

} else {

salutation = "Full name not entered."

}

} catch (e:Error) {

trace(e);

}

// Show some of the information available through loaderInfo:

trace("AS version: " + this.loaderInfo.actionScriptVersion);

trace("App height: " + this.loaderInfo.height);

trace("App width: " + this.loaderInfo.width);

trace("App bytes: " + this.loaderInfo.bytesTotal);

}

]]>

</mx:Script>

<mx:Label text="{salutation}"/>

<mx:TextArea height="100" width="300" id="ta1"/>

</mx:Module>

  利用ActionScript接口实现Module模块间的数据通信

在面向对象的编程中,我们讲要面向接口编程。面向接口的编程方式从一定程度上解决了相互关联的模块间的紧密耦合问题。以上提到的所有数据传输和共享方式都在不同程度上导致了模块间的紧耦合。不过,Flex提供了一种利用标准的ActionScript接口实现Module模块间数据通信的方式。具体地说,对于Module模块对象和主应用Application对象间的通信,我们可以定义一个ActionScript接口,Module模块对象实现了这个接口中定义的方法和属性,那么主应用Application就可以访问这个接口中定义的属性和方法。接口中定义了Module模块对象和主应用Application需要共享的数据和方法,是两者间共同的一个契约,同时也实现了接口和实现的分离,达到了松耦合的目的。参考以下示例,主应用Application,

<?xml version="1.0"?>

<!-- modules/interfaceexample/Main.mxml -->

<mx:Application xmlns="*" xmlns:mx="http://www.adobe.com/2006/mxml">

<mx:Script>

<![CDATA[

import mx.events.ModuleEvent;

import mx.modules.ModuleManager;

[Bindable]

public var selectedItem:Object;

[Bindable]

public var currentModuleName:String;

private function applyModuleSettings(e:Event):void {

// Cast the ModuleLoader's child to the interface.

// This child is an instance of the module.

// You can now call methods on that instance.

var ichild:* = mod.child as IModuleInterface;

if (mod.child != null) {

// Call setters in the module to adjust its

// appearance when it loads.

ichild.setAdjusterID(myId.text);

ichild.setBackgroundColor(myColor.selectedColor);

} else {

trace("Uh oh. The mod.child property is null");

}

// Set the value of a local variable by calling a method

// on the interface.

currentModuleName = ichild.getModuleName();

}

private function reloadModule():void {

mod.unloadModule();

mod.loadModule();

}

]]>

</mx:Script>

<mx:Form>

<mx:FormItem label="Current Module:">

<mx:Label id="l1" text="{currentModuleName}"/>

</mx:FormItem>

<mx:FormItem label="Adjuster ID:">

<mx:TextInput id="myId" text="Enter your ID"/>

</mx:FormItem>

<mx:FormItem label="Background Color:">

<mx:ColorPicker id="myColor"

selectedColor="0xFFFFFF"

change="reloadModule()"

/>

</mx:FormItem>

</mx:Form>

<mx:Label text="Long Shot Insurance" fontSize="24"/>

<mx:ComboBox

labelField="label"

close="selectedItem=ComboBox(event.target).selectedItem"

> 

<mx:dataProvider>

<mx:Object label="Select Module"/>

<mx:Object label="Auto Insurance" module="AutoInsurance.swf"/>

</mx:dataProvider>

</mx:ComboBox>

<mx:Panel width="100%" height="100%">

<mx:ModuleLoader id="mod"

width="100%"

url="{selectedItem.module}"

ready="applyModuleSettings(event)"

/>

</mx:Panel>

<mx:Button id="b1" label="Reload Module" click="reloadModule()"/>

</mx:Application>

接口文件,

// modules/interfaceexample/IModuleInterface

package

{

import flash.events.IEventDispatcher;

public interface IModuleInterface extends IEventDispatcher {

function getModuleName():String;

function setAdjusterID(s:String):void;

function setBackgroundColor(n:Number):void;

}

}

Module模块文件,

<?xml version="1.0"?>

<!-- modules/interfaceexample/AutoInsurance.mxml -->

<mx:Module xmlns:mx="http://www.adobe.com/2006/mxml" width="100%"

height="100%" implements="IModuleInterface">

<mx:Panel id="p1"

title="Auto Insurance"

width="100%"

height="100%"

backgroundColor="{bgcolor}"

> 

<mx:Label id="myLabel" text="ID: {adjuster}"/>

</mx:Panel>

<mx:Script>

<![CDATA[

[Bindable]

private var adjuster:String;

[Bindable]

private var bgcolor:Number;

public function setAdjusterID(s:String):void {

adjuster = s;

}

public function setBackgroundColor(n:Number):void {

// Use a bindable property to set values of controls

// in the module. This ensures that the property will be set

// even if Flex applies the property after the module is

// loaded but before it is rendered by the player.

bgcolor = n;

// Don't do this. The backgroundColor style might not be set

// by the time the ModuleLoader triggers the READY

// event:

// p1.setStyle("backgroundColor", n);

}

public function getModuleName():String {

return "Auto Insurance";

}

]]>

</mx:Script>

</mx:Module>
1
0
分享到:
评论
1 楼 _MyCould 2014-09-04  
    

相关推荐

    flex 面试题flex

    【Flex面试题】Flex面试题主要涵盖Flex的基础概念、开发框架、MVC模式的应用、内存管理、垃圾回收机制、前端性能优化以及与后端通信等多个方面。以下是对这些知识点的详细解析: 1. AS2与AS3的区别: AS2...

    flex嵌入jsp心得

    开发者在个人文件中偶然发现了FlexModule_j2ee.zip,这是一个关键的资源包,包含了将Flex嵌入到JSP中的必要组件和文档。通过解压并阅读其中的readme.txt文件,开发者获取了初步的指导思路,了解到该模块能够简化Flex...

    Flex与ActionScript3程序开发

    第2章 Flex基础知识 第3章 语言基础 第4章 数据基础 第5章 Flex事件驱动 第2篇 视觉设计篇 第6章 Flex皮肤设计 第7章 特效effect和状态state 第8章 Flex与Flash无缝衔接 第9章 综合演练:实现仿Office 2007...

    flex布局布局篇

    Flex布局,全称CSS Flexible Box Layout Module,简称Flexbox,是CSS3中一种新的布局模式。这种布局模式专门用于更高效地对齐、分布和排序容器中的项目,即使它们的大小未知或是动态变化的。Flexbox的目的是提供一种...

    Flex入门学习文档

    注意,`FlexModule_j2ee.zip`实际上是一个war文件,需要重命名为zip并再次解压,以便获取其内部内容。 2. 将解压后的`flex`文件夹复制到你的Java Web工程的`WEB-INF`目录下,同时将`lib`目录中的`flex-bootstrap....

    Flex之模块化

    Flex模块化主要通过Flex Module Framework(模块框架)实现,这是一个内置于Flex SDK中的组件,支持在运行时动态加载和卸载模块。这种动态加载能力使得Flex应用程序能够根据用户需求或网络状况来决定加载哪些模块,...

    IBM Flex System产品介绍

    - **管理节点**:通过CMM (Chassis Management Module) 和FSM (Flex System Manager) 实现对系统的集中管理和监控。 - **机箱设计**:提供14个半宽节点的布局,支持灵活的配置选项。 #### 六、IBM Flex System的...

    flex4 皮肤

    基础的皮肤类是Skin,它提供了一种将样式、图形和子组件组合在一起的方法。在Flex4中,每个可视组件(如Button、Label等)都有一个或多个关联的皮肤类。这些皮肤类通常是根据组件的状态(如鼠标悬停、按下、选中等)...

    FLEX弹出框特效,很好很强大

    1. **Flexbox基础**:Flexbox,全称为“Flexible Box Layout Module”,是CSS3的一个模块,设计用于处理容器内元素的一维布局,如行或列。通过Flexbox,我们可以轻松实现弹性布局,包括自适应大小、对齐方式以及在...

    FLEX面试题

    1. **Flex Project**:这是最基础的Flex项目类型,主要用于开发基于Flex的应用程序或组件。 2. **AS Project**:即ActionScript项目,用于纯ActionScript编程,不包含任何Flex框架元素。 3. **Flex Library Project*...

    FLEX企业应用开发实战.part1

    第2章 Flex企业应用开发基础  2.1 MXML语言  2.1.1 用MXML表示ActionScript对象  2.1.2 查看由MXML文件所翻译的ActionScript代码  2.1.3 IMXMLObject接口  2.2 客户端保持状态  2.3 客户端MVC  2.4 ...

    FLEX企业应用开发实战.part2

    第2章 Flex企业应用开发基础  2.1 MXML语言  2.1.1 用MXML表示ActionScript对象  2.1.2 查看由MXML文件所翻译的ActionScript代码  2.1.3 IMXMLObject接口  2.2 客户端保持状态  2.3 客户端MVC  2.4 ...

    Flex精通_66013.rar

    1. **Flash AS3 Programming**:Flash ActionScript 3.0是Flex开发的基础,AS3的语法与之前版本有显著提升,包括面向对象编程的支持和性能优化。通过`flash_as3_programming.pdf`,你可以了解AS3的基本语法、类和...

    IBM Flex System V7000 Storage Node Introduction and Implementation Guide

    - **IBMPureFlexSystem**:一种集成化的基础架构解决方案,集成了计算、存储和网络资源,可以快速部署并自动调整以适应不断变化的业务需求。 - **亮点**:易于部署、自动化管理和优化资源分配。 - **组件**:包括...

    Flex企业应用开发实战源代码

    3.3 Flex Module 101 3.3.1 创建模块 102 3.3.2 模块的编辑与编译 104 3.3.3 模块文件的加载 104 3.3.4 主应用和模块的交互 107 3.4 Flex库文件SWC 107 3.5 Flex编译模式、链接模式与RSL 109 3.5.1 使用系统...

    我的flex资料1

    6. **Flex应用结构**:了解Flex应用的基本结构,包括Application、Module、State等,有助于组织和管理大型项目。 7. **Adobe AIR**:如果你打算开发桌面应用,需要了解Adobe AIR,它是将Flex应用部署到桌面环境的...

    《Flex第一步》第二章PDF下载

    7. **Flex应用程序结构**:这一章可能会讲解Flex应用程序的基本结构,包括Application、Module、Document Class等,以及它们在项目中的作用。 8. **Flex项目部署**:了解如何将Flex应用程序发布为SWF文件,并将其...

    Flex 开发类的反射

    #### Flex中的反射基础 Flex的反射功能与Java类似,但考虑到Flex的编译模型和运行环境,其实现方式有所不同。在Flex中,反射主要通过`Class`和`Reflect`类来实现。`Class`类用于表示特定类型的类信息,而`Reflect`...

    Flex--used-in-the-example-module.zip_FlashMX/Flex源码_Flex_

    1. **Flex框架**:理解Flex的基础架构,包括ActionScript和MXML的使用。 2. **模块化编程**:学习如何创建和使用`mx:Module`,以及其对项目结构的影响。 3. **代码组织**:掌握如何通过模块化来优化代码结构,提高...

Global site tag (gtag.js) - Google Analytics