<rt id="bn8ez"></rt>
<label id="bn8ez"></label>

  • <span id="bn8ez"></span>

    <label id="bn8ez"><meter id="bn8ez"></meter></label>

    TWaver - 專注UI技術

    http://twaver.servasoft.com/
    posts - 171, comments - 191, trackbacks - 0, articles - 2
      BlogJava :: 首頁 :: 新隨筆 :: 聯系 :: 聚合  :: 管理

    打造模塊化的TWaver Flex應用

    Posted on 2012-05-10 11:55 TWaver 閱讀(1694) 評論(1)  編輯  收藏
           當Flex應用越來越龐大時,問題會越來越多:
    1. SWF文件的大小也會越來越大;
    2. 下載SWF文件的時間也會越來越長;
    3. 如果有多個Flex應用,如何復用相同的代碼(包括Flex Framework、自定義組件庫和第三方包,比如TWaver Flex);
    4. 每次升級后,用戶都需要重新下載新的SWF文件;
    5. 如何在不修改并編譯舊SWF文件的情況下,增加新功能;

           如何解決這些問題?
    一、 使用動態共享庫(Runtime Shared Libraries)。Flex Framework、自定義組件庫和第三方包都是獨立的swf文件,Flex應用程序中將不包含Flex Framework、自定義組件庫和第三方包的代碼,所以:
    1. 程序修改后,只要更新用戶程序代碼即可,不用更新Flex Framework、自定義組件庫和第三方包;
    2. RSL被瀏覽器緩存了,更新用戶程序后,RSL不用重新下載;
    3. 多個Flex應用可以共享一份RSL,不用重復下載;
    二、使用模塊化(Modular)。各個功能模塊被拆分成不同的SWF文件,所以:
    1. 可以單獨更新某個功能模塊;
    2. 可以動態添加模塊;

           不過單單使用RSL或者Modular,并不能解決所有問題:
    1. 僅僅使用RSL時,無法將功能模塊拆分,無法動態加載功能模塊;
    2. 僅僅使用Modular時,無法去除所有模塊用到的公共的Flex Framework、自定義組件庫和第三方包,導致子模塊文件過于龐大;

          本文詳細介紹了RSL和Modular的結合,幫您打造模塊化的、可擴展的、強健的TWaver Flex應用。
    一、為了避免主程序和子模塊之間的高度耦合,需要創建一個Flex Library工程,此工程定義了所有子模塊用到的公共類,以及子模塊和主程序之間通訊的接口:

    1. IModule接口:子模塊實現此接口,用于主程序顯示子模塊的名稱(get title),以及子模塊加載完畢后,回調子模塊(ready)
    1 package demo {
    2 public interface IModule {
    3 function get title():String;
    4 function ready(app:IApplication):void;
    5 }
    6 }

    2. IApplication接口,主程序實現此接口,用于子模塊和主程序交互,目前此接口無任何方法,可自行根據需要添加
    1 package demo {
    2 public interface IApplication {
    3 
    4 }
    5 } 

          另外此Library工程還自定義了Network組件、Node,供子模塊使用,這里不一一列出,具體參考附件的源代碼。不過需要注意的是,工程選項里,framework linkage和twaver.swc的link type必須改成external,以減小Library工程生成的swc文件的大小。



    二、創建主程序Flex工程,此工程引用上面的Library工程以及twaver.swc。實現的功能為左邊顯示Tree,點擊樹節點后,右邊加載相應的子模塊。

    1. 添加組件標簽,初始化界面,首先是一HDividedBox組件,左邊為FastTree,右邊為VBox;VBox里上面為Label,顯示子模塊名稱,下面為子模塊容器:
    1 &amp;lt;mx:HDividedBox width="100%" height="100%"&amp;gt;
    2 &amp;lt;tw:FastTree id="tree" width="300" height="100%"/&amp;gt;
    3 &amp;lt;mx:VBox width="100%" height="100%"&amp;gt;
    4 &amp;lt;mx:Label id="title" width="100%" textAlign="center"/&amp;gt;
    5 &amp;lt;mx:Canvas id="content" width="100%" height="100%"/&amp;gt;
    6 &amp;lt;/mx:VBox&amp;gt;
    7 &amp;lt;/mx:HDividedBox&amp;gt;

    2. 初始化樹節點:為了實現動態添加模塊,這里從xml文件讀取模塊信息。以后添加新模塊時,直接修改xml文件即可,不用修改主程序。
    xml文件包含模塊名稱和模塊url:
    1 &amp;lt;modules&amp;gt;
    2 &amp;lt;module name="PSTN" url="ModulePSTN.swf"/&amp;gt;
    3 &amp;lt;module name="Alarm" url="ModuleAlarm.swf"/&amp;gt;
    4 &amp;lt;/modules&amp;gt;

          加載xml文件代碼如下,主要是將url信息存到client屬性中,用于點擊該節點時用此url加載子模塊:
     1 private function initTreeBox():void {
     2 var httpService:HTTPService = new HTTPService();
     3 httpService.resultFormat = "e4x";
     4 httpService.addEventListener(ResultEvent.RESULT, this.addModules);
     5 httpService.url = "modules.xml";
     6 httpService.send();
     7 }
     8 
     9 private function addModules(e:ResultEvent):void {
    10 for each(var module:XML in e.result.module){
    11 this.addModule(module.@name, module.@url);
    12 }
    13 }
    14 
    15 private function addModule(name:String, url:String):void {
    16 var data:Data = new Data();
    17 data.name = name;
    18 data.setClient("url", url);
    19 this.tree.dataBox.add(data);
    20 }

    3. 添加Tree的選中監聽:當樹節點被選中時,先判斷對應的子模塊是否加載過,如果未加載過,則動態加載之,并將加載的模塊存入client屬性中,否則直接將之前存儲在client屬性中的子模塊加入右邊容器中:
     1 this.tree.selectionModel.addSelectionChangeListener(this.handleSelectionChangeEvent);
     2 
     3 private function handleSelectionChangeEvent(e:SelectionChangeEvent):void {
     4 var selectedData:IData = this.tree.selectionModel.lastData;
     5 if(selectedData){
     6 var moduleLoader:ModuleLoader = selectedData.getClient("module");
     7 if(moduleLoader){
     8 this.content.removeAllChildren();
     9 this.content.addChild(moduleLoader);
    10 this.title.text = (moduleLoader.child as IModule).title;
    11 }else{
    12 moduleLoader = new ModuleLoader();
    13 moduleLoader.percentWidth = 100;
    14 moduleLoader.percentHeight = 100;
    15 moduleLoader.addEventListener(ModuleEvent.READY, this.moduleReady);
    16 moduleLoader.loadModule(selectedData.getClient("url"));
    17 selectedData.setClient("module", moduleLoader);
    18 }
    19 }
    20 }
    21 
    22 private function moduleReady(event:ModuleEvent):void {
    23 var moduleLoader:ModuleLoader = event.target as ModuleLoader;
    24 var module:IModule = moduleLoader.child as IModule;
    25 content.removeAllChildren();
    26 content.addChild(moduleLoader);
    27 this.title.text = module.title;
    28 module.ready(this);
    29 }

          注意,引用twaver.swc和Library工程時,twaver.swc必須在Library工程的上面,否則會報找不到twaver.network::Network類,而且framework linkage,twaver.swc以及上面的Library工程的link type必須為Runtime shared library(RSL),具體設置見下面第三步。

    三、創建子模塊Flex工程,這里以Demo里的PSTNDemo和AlarmPropagationDemo為例,創建2個子模塊工程,子模塊工程為Flex工程,編譯選項里,framework linkage,twaver.swc以及上面的Library工程的link type必須為Runtime shared library(RSL)。不過需要注意的是,如果twaver.swc是通過“Add SWC Folder"添加的話,link type就沒有Runtime shared library(RSL)這個選項,這或許是Flash Builder的bug,但如果是用"Add SWC"添加的,就沒這個問題,見下圖:
           
            另外,子模塊的編譯路徑可以修改為主程序工程的bin-debug目錄,免得每次修改子模塊后,需要復制子模塊swf到主程序的bin-debug中:


             還有,不用生成HTML Wrapper,因為子模塊不能獨立運行,只能從主程序中加載,所以沒有必要生成包裝子模塊的html文件:

         
    最后,要注意的是,修改link type為RSL時,如果沒有添加RSL路徑,OK按鈕是不能點的,只能點擊”Add“按鈕,添加RSL路徑后,才能點擊OK按鈕,這點很坑爹:
            子模塊的代碼比較簡單,需要注意的是mxml文件的根標簽要改為Module,還有要實現IModule接口: 
     1 &amp;lt;?xml version="1.0" encoding="utf-8"?&amp;gt;
     2 &amp;lt;mx:Module xmlns:mx="http://www.adobe.com/2006/mxml"
     3 xmlns:demo="http://www.demo.com/demo" implements="demo.IModule"
     4 xmlns:tw="http://www.servasoftware.com/2009/twaver/flex"
     5 creationComplete="init()" width="100%" height="100%"&amp;gt;
     6 &amp;lt;mx:Script&amp;gt;
     7 &amp;lt;![CDATA[
     8 import demo.*;
     9 
    10 private var _app:IApplication = null;
    11 
    12 public function get title():String {
    13 return "Alarm Demo";
    14 }
    15 
    16 public function ready(app:IApplication):void {
    17 this._app = app;
    18 }
    19 ]]&amp;gt;
    20 &amp;lt;/mx:Script&amp;gt;
    21 &amp;lt;/mx:Module&amp;gt;

          經過了這么多繁瑣的步驟,終于可以測試一下程序了:

           再看看子模塊、Library工程以及主程序的包大小(子模塊只有50K不到,的確夠小了):
          關于更多Modular和RSL的內容,請參考Adobe官方文檔:
    Creating Modular Applications
    Using Runtime Shared Libraries
    本文完整Demo見附件:ModuleDemo

    評論

    # re: 打造模塊化的TWaver Flex應用  回復  更多評論   

    2012-05-11 08:33 by tb
    很好的

    只有注冊用戶登錄后才能發表評論。


    網站導航:
     
    主站蜘蛛池模板: 久久亚洲精品国产精品| 青娱乐在线视频免费观看| 人禽杂交18禁网站免费| 最新亚洲人成网站在线观看| 亚洲人成影院在线无码观看| 全免费a级毛片免费看| 亚洲一区二区三区高清不卡| 免费a级毛片无码av| 无码专区AAAAAA免费视频| 亚洲日韩乱码中文字幕| 91麻豆精品国产自产在线观看亚洲 | 四虎永久精品免费观看| 久久久国产精品无码免费专区| 亚洲高清视频在线| 亚洲国产精品乱码一区二区| 在线观看视频免费国语| 免费观看91视频| 国产精品亚洲综合| 亚洲精品国产成人| 久久国产成人亚洲精品影院| 一二三四免费观看在线电影 | 中文字幕乱理片免费完整的| 33333在线亚洲| 亚洲AV无码一区二区三区DV| 永久黄网站色视频免费观看| 久久精品无码精品免费专区| 免费一级毛suv好看的国产网站| 亚洲国产成人久久三区| 亚洲情综合五月天| 免费不卡中文字幕在线| 国产一卡2卡3卡4卡无卡免费视频 国产一卡二卡3卡四卡免费 | 日本一区二区在线免费观看| 久久亚洲精精品中文字幕| 亚洲日韩在线观看| 国产一卡二卡≡卡四卡免费乱码| 6080午夜一级毛片免费看6080夜福利| 国产成人无码免费看片软件 | 一区二区三区免费高清视频| 亚洲熟妇丰满xxxxx| 亚洲成A∨人片在线观看无码| 亚洲欧洲成人精品香蕉网|