Flex4使用Cairngorm2框架

Flex4使用Cairngorm2框架

虽然说Cairngorm框架已经出到3.0版本,但是3.0的Cairngorm已经不能被称为框架了,而称为软件开发的一种指导原则,其中包括了“指导原则”、“第三方类库”、“工具”组成,因此如果使用Cairngorm还是应该整合Cairngorm2版本的。

Cairngorm2有两个版本,即普通版和企业版,区别是企业版的数据服务使用的是高收费的 LiveCycle Data Services,而普通版的使用的是Blazeds。因此我们只要集成普通版的就可以了。

首先下载Cairngorm2普通版的SWC包,点击这里。

将下载下的SWF包引入到项目中,建立Cairngorm需要的包,Cairngorm需要7个包,分别是:

1. Business –》业务逻辑 Delegate类和ServiceLocator文件

2.Command -》命令逻辑 Command类

3. Event -》事件类

4. Vo -》放置ValueObject类

5. Model -》 放置Model类

6. View -》放置视图文件

7. Contorller -》放置Control类

Cairngorm 是事件驱动的,具体原理请参考官网文档,或者我以后有时间了也许会总结一下。

本文只介绍Flex整合Cairngorm框架,后台实现不再介绍。下面就开始写具体的代码了,首先我们先把视图创建出来,确定我们要做个什么东西。在view包里面创建一个组件文件,拖进来一个datagrid,几个按钮和label,最后创建出来的样式应该如下:

 

当然现在的视图只有一个样子,什么功能都没有,接下来我们要创建跟这个视图绑定的数据,即Model,在Model包里创建一个as文件,因为我们需要用到数据的分页,因此将这些数据都放到同一个Model里面。代码如下:

01 [Bindable]
02     public class UserPageModelLocator implements ModelLocator
03     {
04         private static var _instance:UserPageModelLocator;
05  
06         public var pageSize:int;
07         public var pageNo:int;
08         public var pageCount:String;
09         public var prevPage:int;
10         public var nextPage:int;
11         public var totalCount:String;
12         public var order:String;
13         public var orderBy:String;
14         public var hasPrev:Boolean;
15         public var hasNext:Boolean;
16         public var result:IList;
17  
18         public function UserPageModelLocator()
19         {
20             if (_instance != null ) {
21                 throw new CairngormError(CairngormMessageCodes.SINGLETON_EXCEPTION,
22                     "EmployeesModelLocator");
23             }
24  
25             _instance = this;
26         }
27  
28         public static function getInstance():UserPageModelLocator{
29             if(_instance == null){
30                 _instance = new UserPageModelLocator();
31             }
32             return _instance;
33         }
34     }

在Cairngorm里面,Model都是单例对象,所以这点要注意。

与视图绑定的数据已经创建好了,接下来要让视图能够工作了,又得提一下Cairngorm的事件驱动了,在Cairngorm里面,所有的操作数据传递等都是由事件进行驱动的,因此我们要创建一个要进行的操作的事件类,首先我希望能够在程序一打开的时候就把第一页的数据全部都取出,所以创建一个获取所有用户的事件类,代码如下:

1 public class GetAllUserEvent extends CairngormEvent
2     {
3         public static const EVENT_GETALLUSER:String = "getAllUser";
4  
5         public function GetAllUserEvent(){
6             super(GetAllUserControl.EVENT_GETALLUSER);
7         }
8     }

Cairngorm里的自定义事件都应该继承CairngormEvent类。这个事件被触发了之后要进行什么样的操作呢?这个时候我们要创建控制器了,在Control包里创建一个Contoller类,代码如下:

1 public class GetAllUserControl extends FrontController
2     {
3         public static const EVENT_GETALLUSER = "getAllUser";
4  
5         public function GetAllUserControl()
6         {
7             addCommand(GetAllUserControl.EVENT_GETALLUSER, GetAllUserCommand);
8         }
9     }

继承了Cairngorm的前端控制器,意思是监听这事件,如果这个事件被触发了,我们要执行一向操作,也就是Cairngorm的Command,这个时候我们可以创建Command类了,代码如下:

01 public class GetAllUserCommand implements ICommand, IResponder
02     {
03  
04         public function execute(event:CairngormEvent):void
05         {
06             var delegate:GetAllUserDelegate = new GetAllUserDelegate(this);
07             var getAllUserEvent:GetAllUserEvent = GetAllUserEvent(event);
08             delegate.getAllUser();
09         }
10  
11         public function result(event:Object):void
12         {
13             var model:UserPageModelLocator = UserPageModelLocator.getInstance();
14             model.pageSize = event.result.pageSize;
15             model.pageCount = event.result.pageCount;
16             model.result = event.result.result;
17             model.pageNo = event.result.pageNo;
18             model.hasNext = event.result.hasNext;
19             model.hasPrev = event.result.hasPrev;
20             model.nextPage = event.result.nextPage;
21             model.prevPage = event.result.prevPage;
22             model.order = event.result.order;
23             model.orderBy = event.result.orderBy;
24             model.result = event.result.result;
25             model.totalCount = event.result.totalCount;
26         }
27  
28         public function fault(event:Object):void
29         {
30             Alert.show("Somthing is wrong!");
31         }
32     }

可以看出,Command类我们实现了两个接口,分别是ICommand和IResponder,前面一个是Cairngorm的接口,后面一个是Flex中的异同通讯接口。ICommand定义了一个方法execute,就是要执行的内容,这里我们可以看到我们执行的逻辑是什么,当然里面的业务逻辑类我们还没有创建,但是可以看出调用了远程的方法,IResponder有两个接口,分别是result和fault,分别是返回成功和失败后腰调用的方法。在result里,我们将后台返回的结果放入了model里面,这样通过Flex的数据绑定,就可以自动的更新视图的内容了。

现在就只差业务逻辑还没有创建了,在Business包里面包括两方面的内容,一个是要进行的业务逻辑操作即Delegate,一个是ServiceLocator,即远程访问的地址。首先创建ServiceLocator,创建一个mxml文件,内容如下:

01 <?xml version="1.0" encoding="utf-8"?>
02 <cairngorm:ServiceLocator
03     xmlns:mx="http://www.adobe.com/2006/mxml"
04     xmlns:cairngorm="http://www.adobe.com/2006/cairngorm">
05  
06     <mx:RemoteObject id="flexSev" destination="flexServlet">
07  
08     </mx:RemoteObject>
09  
10 </cairngorm:ServiceLocator>

这里我们创建了一个id为flexServ的RemoteObject,他的Destination为flexServlet。然后我们要创建它的逻辑了,代码:

01 public class GetAllUserDelegate
02     {
03         private var responder:IResponder
04         private var services:Object;
05  
06         public function GetAllUserDelegate(responder:IResponder)
07         {
08             this.responder = responder;
09             this.services = ServiceLocator.getInstance().getRemoteObject("flexSev");
10         }
11  
12         public function getAllUser():void{
13             var token:AsyncToken = services.findAll();
14             token.addResponder(responder);
15         }
16  
17         public function getUserByPageNo(pageNo:int):void{
18             var token:AsyncToken = services.findPrev(pageNo);
19             token.addResponder(responder);
20         }
21     }

在构造函数里,我们就调用了这个service,这里面有两个方法,一个是获取所有用户,一个是根据页面查找用户,我们现在用的是前面一个方法,后面一个方法在后面的翻页功能时才能用到。

 

到现在为止,Cairngorm所需的所有类都已经创建完毕,但是不是就能够使用了呢?当然不是,没说嘛,刚开始创建视图的时候我们只拉好了界面,其余什么都没做,现在我们要把这些类串联起来,让Cairngorm能够将数据取出来,首先我们想要在程序一打开的时候就把第一页的所有用户取出,则要在createComplete里添加如下方法:

1 public function init():void{
2                 var getAllUserEvent:GetAllUserEvent = new GetAllUserEvent( );
3                 CairngormEventDispatcher.getInstance().dispatchEvent(getAllUserEvent);
4             }

我们直接在程序打开的的时候就触发了获取所有用户的事件,这是Controller监听了这个事件,发现被这个事件被触发了,马上调用了获取所有对象的command,command就去调用获取所有对象的业务逻辑Delegate,Delegate返回了结果,把结果放入了Model,这个时候,我们就需要把视图与Model进行绑定,让数据自动更新到视图中,我们需要在datagrid中添加:

1 dataProvider="{UserPageModelLocator.getInstance().result}"

这样应该可以了吧?其实还不行,为虾米呢?虽然我们在后台调用业务逻辑的时候调用了id为flexServ的remoteObject,但是他怎么知道这个flexServ在哪里呢?所以我们还要把他引入才行,在组件里添加如下:

1 <fx:Declarations>
2             <rds:Services xmlns:rds="com.boco.wb.cairngorm.business.demo.*"/>
3             <router:GetAllUserControl xmlns:router="com.boco.wb.cairngorm.control.demo.*"/>
4             <router:FindPageUserContol xmlns:router="com.boco.wb.cairngorm.control.demo.*"/>
5     </fx:Declarations>

现在程序已经完全串联起来,可以使用了。其中还有分页的内容现在就不贴代码了。


现在是不是觉得使用Cairngorm太麻烦了,要建这么多的类行,这么个小功能自己写很简单就写好了,当然,这只是一个小demo,小的演示程序根本不必使用Cairngorm,真正使用时要在大型的项目中,这样才能使程序开发起来更加清晰明了,加快开发速度。


你可能感兴趣的:(框架,datagrid,String,Flex,command,Class)