除了我们已经讨论过的核心Javascrīpt库,现在的Ext亦包括了一系列的最前端的JavaScirptUI组件库。文本以一个最常用的widget为例子,作简单的介绍。

MessageBox

比起略为沉闷的“HelloWolrd”消息窗口,我们做少许变化,前面我们写的代码是,单击某个段落便会高亮显示,现在是单击段落,在消息窗口中显示段落内容出来。
在上面的paragraphClicked的function中,将这行代码:

1.Ext.get(e.target).highlight();

替换为:

1.var paragraph = Ext.get(e.target);
2.paragraph.highlight();
3.Ext.MessageBox.show({
4.    title: 'Paragraph Clicked',
5.    msg: paragraph.dom.innerHTML,
6.    width:400,
7.    buttons: Ext.MessageBox.OK,
8.    animEl: paragraph});

这里有些新的概念讨论一下。在第一行中我们创建了一个局部变量(Local Variable)来保存某个元素的引用,即被单击的那个DOM节点(本例中,我们总是段落paragrah,事因我们已经定义该事件与<p>标签发生关联的了)。为什么要这样做呢?嗯...观察上面的代码,我们需要引用同一元素来高亮显示,在MessageBox中也是引用同一元素作为参数使用。
一般来说,多次重复使用同一值(Value)或对象,是一个不好的方式,所以,作为一个好的OOP开发者,应该是将其分配到一个局部变量中,反复使用这变量!

现在,观察MessageBox的调用,准备作为阐述新概念的演示用。乍一看,这像一连串的参数传入到方法中,但仔细看,这是一个非常特别的语法。实际上,传入到MessageBox.show的只有一个参数:一个Object literal,包含一组属性和属性值。在Javascrīpt中,Object Literal是动态的,你可在任何时候用{和}创建一个典型的对象(object)。其中的字符由一系列的name/value组成的属性,属性的格式是[property name]:[property value]。在整个Ext中,你将会经常遇到这种模式的语法,因此你应该消耗掉这知识!
使用Object Literal的原因是什么呢?主要的原因是“韧性(flexibility)",随时可新增、删除属性,亦可不管顺序地插入。而方法不需要改变。这也是多个参数的情况下,为最终开发者带来不少的方便(本例中的MessageBox.show())。例如,我们说这儿的foo.action方法,有四个参数,而只有一个是你必须传入的。本例中,你想像中的代码可能会是这样的foo.action(null, null, null, 'hello').,若果那方法用Object Literal来写,却是这样, foo.action({ param4: 'hello' }),这更易用和易读。

Gird

Gird是Ext中人们最想先睹为快的Widgets之一,也是最流行之一。好,让我们看看怎么轻松地创建一个Gird并运行。用下列代码替换ExtStart.js中全部语句:

01.Ext.onReady(function() {
02.    var myData = [
03.    ['Apple',29.89,0.24,0.81,'9/1 12:00am'],
04.    ['Ext',83.81,0.28,0.34,'9/12 12:00am'], 
05.    ['Google',71.72,0.02,0.03,'10/1 12:00am'],  
06.    ['Microsoft',52.55,0.01,0.02,'7/4 12:00am'],    
07.    ['Yahoo!',29.01,0.42,1.47,'5/22 12:00am']
08.    ];  
09.var ds = new Ext.data.Store({
10.        proxy: new Ext.data.MemoryProxy(myData),
11.        reader: new Ext.data.ArrayReader({id: 0}, [
12.            {name: 'company'},
13.            {name: 'price', type: 'float'},
14.            {name: 'change', type: 'float'},
15.            {name: 'pctChange', type: 'float'},
16.            {name: 'lastChange', type: 'date', dateFormat: 'n/j h:ia'}
17.    ])
18.    });
19.    ds.load();
20.    var colModel = new Ext.grid.ColumnModel([
21.        {header: "Company", width: 120, sortable: true, dataIndex: 'company'},
22.        {header: "Price", width: 90, sortable: true, dataIndex: 'price'},
23.        {header: "Change", width: 90, sortable: true, dataIndex: 'change'},
24.        {header: "% Change", width: 90, sortable: true, dataIndex: 'pctChange'},
25.        {header: "Last Updated", width: 120, sortable: true,
26.        renderer: Ext.util.Format.dateRenderer('m/d/Y'),
27. dataIndex: 'lastChange'}
28.    ]);
29.    var grid = new Ext.grid.Grid('grid-example', {ds: ds, cm: colModel});
30.        grid.render();
31.        grid.getSelectionModel().selectFirstRow();
32.    });

这看上去很复杂,但实际上加起来,只有七行代码。第一行创建数组并作为数据源。实际案例中,你很可能从数据库、或者WebService那里得到动态的数据。接着,我们创建并加载data store, data store将会告诉Ext的底层库接手处理和格式化这些数据。接着,我们定义一个column模型,用来轻松地调配Gird的每一列参数。最后我们生成这个Gird,传入data store和column模型两个对象,进行渲染并选好第一行。不是太困难吧?如果一切顺利,搞掂之后你会看到像这样的:

当然,你可能对这段代码的某些细节,并不完全掌握其中的含义(像MemoryProxy究竟是什么?)但先不要紧,这个例子的目的是告诉你,用少量的代码,创建一个富界面的多功能的UI组件而已——这是完全可能的,更多细节的内容,留给读者你自己学习吧。这儿有许多学习Grid的资源。Ext Grid教程交叉Gird演示Gird API文档

还有更多的..

这只是冰山一角。还有一打的UI Widgets可以供调用,如 layouts, tabs, menus, toolbars, dialogs, tree view等等。请参阅API文档中范例演示

使用Ajax

在弄好一些页面后,你已经懂得在页面和脚本之间的控制原理(interact)。接下来,你想知道的是,怎样与后台服务器(remote server)交换数据,常见的是从数据库加载数据(load)或是保存数据(save)到数据库中。通过Javascrīpt异步无刷新交换数据的这种方式,就是所谓的Ajax。Ext内建卓越的Ajax支持,例如,一个普遍的用户操作就是,异步发送一些东西到服务器,然后,UI元素根据回应(Response)作出更新。这是一个包含text input的表单,一个div用于显示消息(注意,你可以在ExtStart.html中加入下列代码,但这必须要访问服务器):

1.<DIV id=msg style="VISIBILITY: hidden"></DIV>
2. Name: <INPUT id=name>
3.<BR> 
4.<INPUT id=oKButton type=button value=OK>

接着,我们加入这些处理交换数据的Javascrīpt代码到文件ExtStart.js中(用下面的代码覆盖):

01.Ext.onReady(function(){
02.    Ext.get('oKButton').on('click', function(){
03.        var msg = Ext.get("msg");
04.            msg.load({
05.            url: [server url], //换成你的URL
06.            params: "name=" + Ext.get('name').dom.value,
07.            text: "Updating..."
08.        });
09.        msg.show();
10.    });});

这种模式看起来已经比较熟悉了吧!先获取按钮元素,加入单击事件的监听。在事件处理器中(event handler),我们使用一个负责处理Ajax请求、接受响应(Response)和更新另一个元素的Ext内建类,称作UpdateManager。UpdateManager可以直接使用,或者和我们现在的做法一样,通过Element的load方法来访问(本例中该元素是“msg“的div)。当使用Element.load方法,请求(request)会被加工处理后发送,等待服务器的响应(Response),来自动替换元素的innerHTML。简单传入服务器url地址,加上字符串参数,便可以处理这个请求(本例中,参数值来自“name”元素的value),而text值就是请求发送时提示的文本,完毕后显示那个msg的div(因为开始时默认隐藏)。当然,和大多数Ext组件一样,UpdateManager有许多的参数可选,不同的Ajax请求有不同的方案。而这里仅演示最简单的那种。

PHP
1.<? if(isset($_GET['name'])) { echo 'From Server: '.$_GET['name']; } ?>
ASP.Net
1.protected void Page_Load(object sender, EventArgs e)
2.{
3.if (Request["name"] != null)
4.{
5.Response.Write("From Server: " + Request["name"]);
6.Response.End();
7.}
8.}
Cold Fusion
1.<cfif StructKeyExists(url, "name")>
2.<cfoutput>From Server: #url.name#</cfoutput>
3.</cfif>

最后一个关于Ajax的谜题就是,服务器实际处理请求和返回(Resposne)的具体过程。这个过程会是一个服务端页面,一个Servlet,一个Http处理器,一个WebService,甚至是Perl或CGI脚本,即任何一个服务器都可以处理http请求。无法预料的是,服务器返回什么是服务器的事情,无法给一个标准的例子来覆盖所有的可能性。(这段代码输出刚才我们传入'name'的那个值到客户端,即发送什么,返回什么)。

使用Ajax的真正挑战,是需经过适当处理过的手工代码,并相应格式化为服务端可用接受的数据结构。有几种格式供人们选择(最常用为JSON/XML)。正因Ext是一种与服务器中立的语言,使得其它特定语言的库亦可用于Ext处理Ajax服务。只要页面接受到结果是合适的数据格式,Ext绝不会过问服务器的事情!要全面讨论这个问题,已超出本文的范围。推荐正在使用Ajax的您,深入阅读Ext Ajax教程

译者姓名:Frank
译者博客:http://www.ajaxjs.com/blog/

posted on 2010-05-25 13:34  朱胜  阅读(338)  评论(0编辑  收藏  举报