• EXTJS组件化(三)----组件之间的暧昧关系



    我忽然发现,菜鸟更愿意与人分享他的学习成果.

        在开发过程中,当许多小组件拼合成一个大组件之后,最先遇到的问题就是组件与组件之间的通信和数据交互.

        如果你的组件封装的比较死(即在创建的时候不需要配置属性).则可以通过组建的自定义事件来完成组件与组件之间的项目调用.下面的界面其实没有必要这么做,这么做的目的只是为了解释一下组件与组件之间相互交互数据:
    Js代码  收藏代码
    1. //Panel1和Panel2为视图组件,Main为容器组件  
    2. Ext.namespace("Lesson2.Panel1");  
    3. /** 
    4.  * @author andy_ghg 
    5.  * @version 2009年10月17日1:36:26 
    6.  * @description 组件之间的数据交互(Grid) 
    7.  * @class Lesson2.Panel1 
    8.  * @extends Ext.Panel 
    9.  */  
    10. Lesson2.Panel1 = Ext.extend(Ext.Panel,{  
    11.     layout:"fit",  
    12.     height:200,  
    13.     //初始化函数  
    14.     initComponent : function(){  
    15.         Lesson2.Panel1.superclass.initComponent.call(this,arguments);  
    16.         this.addEvents("gridRowSelected");  
    17.         this.gridStore = new Ext.data.JsonStore({  
    18.             url:"",  
    19.             fields:["xx","yy"],  
    20.             totalPropery:"results",  
    21.             root:"items"  
    22.         });  
    23.         this.gridSm = new Ext.grid.CheckboxSelectionModel();  
    24.           
    25.         this.gridCm = new Ext.grid.ColumnModel([this.gridSm,{  
    26.             header:"列一",  
    27.             dataIndex:"xx"  
    28.         },{  
    29.             header:"列二",  
    30.             dataIndex:"yy"  
    31.         }]);  
    32.           
    33.         this.gridPanel = new Ext.grid.GridPanel({  
    34.             sm:this.gridSm,  
    35.             cm:this.gridCm,  
    36.             store:this.gridStore,  
    37.             viewConfig:{  
    38.                 autoFill:true,  
    39.                 forceFit:true  
    40.             }  
    41.         });  
    42.         this.gridPanel.on("rowclick",this.rowSelect,this);  
    43.         this.add(this.gridPanel);  
    44.     },  
    45.     //提供给外部调用的函数,返回其内部的store  
    46.     getStore:function(){  
    47.         return this.gridPanel.getStore();  
    48.     },  
    49.     rowSelect:function(grid,index,e){  
    50.         var record = grid.getStore().getAt(index);  
    51.         this.fireEvent("gridRowSelected",record);  
    52.     }  
    53. });  
    54. Ext.namespace("Lesson2.Panel2");  
    55. /** 
    56.  * @description 组件之间的相互交互(formPanel) 
    57.  * @class Lesson2.Panel2 
    58.  * @extends Ext.Panel 
    59.  */  
    60. Lesson2.Panel2 = Ext.extend(Ext.Panel,{  
    61.     layout:"fit",  
    62.     frame:true,  
    63.     initComponent:function(){  
    64.         Lesson2.Panel2.superclass.initComponent.call(this,arguments);  
    65.         this.addEvents("addRecord");  
    66.         this.formPanel = new Ext.FormPanel({  
    67.             defaults:{anchor:"95%"},  
    68.             defaultType:"textfield",  
    69.             labelWidth:55,  
    70.             items:[{  
    71.                 fieldLabel:"XXXXX",  
    72.                 name:"xx"  
    73.             },{  
    74.                 fieldLabel:"YYYYY",  
    75.                 name:"yy"  
    76.             }]  
    77.         });  
    78.         this.add(this.formPanel);  
    79.         this.addButton("加入",this.addRecord,this);  
    80.     },  
    81.     //触发自定义事件,并向事件中传递一个参数values  
    82.     addRecord:function(){  
    83.         var values = this.formPanel.getForm().getValues();  
    84.         this.fireEvent("addRecord",values);  
    85.     }  
    86. });  
    87.   
    88. Ext.namespace("Lesson2.Main");  
    89. /** 
    90.  * @description 用于将两个子组件拼合在一起的容器 
    91.  * @class Lesson2.Main 
    92.  * @extends Ext.Panel 
    93.  */  
    94. Lesson2.Main = Ext.extend(Ext.Panel,{  
    95.     renderTo:Ext.getBody(),  
    96.     layout:"form",  
    97.     initComponent:function(){  
    98.         Lesson2.Main.superclass.initComponent.call(this,arguments);  
    99.         this.panel1 = new Lesson2.Panel1();  
    100.         this.panel2 = new Lesson2.Panel2();  
    101.         //在这里捕获panel2的自定义事件  
    102.         this.panel2.on("addRecord",this.addRecordToGrid,this);  
    103.         this.panel1.on("gridRowSelected",this.addRecordToForm,this);  
    104.         //将两个组件加入到视图中去  
    105.         this.add(this.panel1);  
    106.         this.add(this.panel2);  
    107.     },  
    108.     //TODO panel2的事件处理函数,在这里的this代表Lesson2.Main  
    109.     //这里通过this获取panel1的实例,再通过panel1的实例调用panel1的方法getStore()  
    110.     //panel1的getStore()函数会返回其内部的gridPanel的Store  
    111.     //参数的values就是panel2触发了自定义事件后传递进来的  
    112.     addRecordToGrid:function(values){  
    113.         var record = new Ext.data.Record(values);  
    114.         this.panel1.getStore().add(record);  
    115.     },  
    116.     //TODO 第二种方法,直接获取panel1里的store,效果是一样的  
    117.     addRecordToGrid_2:function(values){  
    118.         var record = new Ext.data.Record(values);  
    119.         this.panel1.gridStore.add(record);  
    120.     },  
    121.     //TODO panel1的事件处理函数,在这里会获取到panel2的实例,并通过该实例获取其内部的formPanel并调用  
    122.     //formPanel的相应方法来达到读取数据的目的  
    123.     addRecordToForm:function(record){  
    124.         this.panel2.formPanel.getForm().loadRecord(record);  
    125.     }  
    126. });  
    127. Ext.onReady(function(){  
    128.     var ls = new Lesson2.Main({  
    129.         title:"测试",  
    130.         400  
    131.     });  
    132. });  


        如果你的组件封装的比较灵活,则可以在容器内就直接调用容器中的方法进行操作,比如上面的代码稍微修改一下(注意panel2的Button):
    Js代码  收藏代码
    1. Ext.namespace("Lesson2.Panel1");  
    2. /** 
    3.  * @author andy_ghg 
    4.  * @version 2009年10月17日1:36:26 
    5.  * @description 组件之间的数据交互(Grid) 
    6.  * @class Lesson2.Panel1 
    7.  * @extends Ext.Panel 
    8.  */  
    9. Lesson2.Panel1 = Ext.extend(Ext.Panel,{  
    10.     layout:"fit",  
    11.     height:200,  
    12.     //初始化函数  
    13.     initComponent : function(){  
    14.         Lesson2.Panel1.superclass.initComponent.call(this,arguments);  
    15.         this.addEvents("gridRowSelected");  
    16.         this.gridStore = new Ext.data.JsonStore({  
    17.             url:"",  
    18.             fields:["xx","yy"],  
    19.             totalPropery:"results",  
    20.             root:"items"  
    21.         });  
    22.         this.gridSm = new Ext.grid.CheckboxSelectionModel();  
    23.           
    24.         this.gridCm = new Ext.grid.ColumnModel([this.gridSm,{  
    25.             header:"列一",  
    26.             dataIndex:"xx"  
    27.         },{  
    28.             header:"列二",  
    29.             dataIndex:"yy"  
    30.         }]);  
    31.           
    32.         this.gridPanel = new Ext.grid.GridPanel({  
    33.             sm:this.gridSm,  
    34.             cm:this.gridCm,  
    35.             store:this.gridStore,  
    36.             viewConfig:{  
    37.                 autoFill:true,  
    38.                 forceFit:true  
    39.             }  
    40.         });  
    41.         this.gridPanel.on("rowclick",this.rowSelect,this);  
    42.         this.add(this.gridPanel);  
    43.     },  
    44.     //提供给外部调用的函数,返回其内部的store  
    45.     getStore:function(){  
    46.         return this.gridPanel.getStore();  
    47.     },  
    48.     rowSelect:function(grid,index,e){  
    49.         var record = grid.getStore().getAt(index);  
    50.         this.fireEvent("gridRowSelected",record);  
    51.     }  
    52. });  
    53. Ext.namespace("Lesson2.Panel2");  
    54. /** 
    55.  * @description 组件之间的相互交互(formPanel) 
    56.  * @class Lesson2.Panel2 
    57.  * @extends Ext.Panel 
    58.  */  
    59. Lesson2.Panel2 = Ext.extend(Ext.Panel,{  
    60.     layout:"fit",  
    61.     frame:true,  
    62.     initComponent:function(){  
    63.         Lesson2.Panel2.superclass.initComponent.call(this,arguments);  
    64.         this.formPanel = new Ext.FormPanel({  
    65.             defaults:{anchor:"95%"},  
    66.             defaultType:"textfield",  
    67.             labelWidth:55,  
    68.             items:[{  
    69.                 fieldLabel:"XXXXX",  
    70.                 name:"xx"  
    71.             },{  
    72.                 fieldLabel:"YYYYY",  
    73.                 name:"yy"  
    74.             }]  
    75.         });  
    76.         this.add(this.formPanel);  
    77.     }  
    78. });  
    79.   
    80. Ext.namespace("Lesson2.Main");  
    81. /** 
    82.  * @description 用于将两个子组件拼合在一起的容器 
    83.  * @class Lesson2.Main 
    84.  * @extends Ext.Panel 
    85.  */  
    86. Lesson2.Main = Ext.extend(Ext.Panel,{  
    87.     renderTo:Ext.getBody(),  
    88.     layout:"form",  
    89.     initComponent:function(){  
    90.         Lesson2.Main.superclass.initComponent.call(this,arguments);  
    91.         this.panel1 = new Lesson2.Panel1();  
    92.         this.panel2 = new Lesson2.Panel2({  
    93.             buttons:[{  
    94.                 text:"确定",  
    95.                 handler:this.addRecordToGrid_2,  
    96.                 scope:this  
    97.             }]  
    98.         });  
    99.         //在这里捕获panel2的自定义事件  
    100.         this.panel1.on("gridRowSelected",this.addRecordToForm,this);  
    101.         //将两个组件加入到视图中去  
    102.         this.add(this.panel1);  
    103.         this.add(this.panel2);  
    104.     },  
    105.     //这里直接就获取当前容器的子组件panel2并获取panel2中的formPanel进行操作  
    106.     addRecordToGrid_2:function(){  
    107.         var values = this.panel2.formPanel.getForm().getValues();  
    108.         var record = new Ext.data.Record(values);  
    109.         this.panel1.gridStore.add(record);  
    110.     },  
    111.     //TODO panel1的事件处理函数,在这里会获取到panel2的实例,并通过该实例获取其内部的formPanel并调用  
    112.     //formPanel的相应方法来达到读取数据的目的  
    113.     addRecordToForm:function(record){  
    114.         this.panel2.formPanel.getForm().loadRecord(record);  
    115.     }  
    116. });  
    117. Ext.onReady(function(){  
    118.     var ls = new Lesson2.Main({  
    119.         title:"测试",  
    120.         400  
    121.     });  
    122. });  


        两种写法有各自的好处.看大家怎么取舍了,当然,组件和组件之间的交互远不只这些简单的操作,还包括当一个组件还未被创建,而另外一个组件已经向这个组件发送数据等等,这些就要考虑使用一个第三方的数据组件来做中转.Extjs的StoreMgr可以做到,你也可以自己写一个符合你自己要求的数据组件,可以模仿StoreMgr写一个.
  • 相关阅读:
    设计模式总结
    内存模型
    运行时内存
    网络
    iOS安全攻防(十)dump自己的app
    iOS安全攻防(九)使用Theos开发SpringBoard的Tweat
    iOS安全攻防(八)Thoes的Logos简介
    iOS安全攻防(七)使用iOSOpenDev开发SpringBoard的Tweat
    iOS安全攻防(六)使用class-dump导出Frameworks头文件
    iOS安全攻防(五)使用dpkg安装deb到iOS设备
  • 原文地址:https://www.cnblogs.com/sunscheung/p/4839406.html
Copyright © 2020-2023  润新知