除了我们已经讨论过的核心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请求有不同的方案。而这里仅演示最简单的那种。
最后一个关于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/