原版的:Understanding Widgets in Ext JS 5

在Ext JS 5,引入了新的“widgetcolumn”,支持在网格的单元格中放置组件。

同一时候,还在Ext JS 5引入了一种新的被称为“小部件”的轻量级组件。

在Ext JS 5中,已包括了几个小部件,在本文将告诉你怎样轻松的去打造自己的小部件。

为了说明当中的关键概念。在文中将创建一个简单的名为“ratings”的例如以下图所看到的的小部件:

入门

与经常使用的从Ext.Component派生的组件不同。小部件派生于新的基类Ext.Widget。并且,Ext.Widget的派生类差点儿全然是由配置系统来定义的(后面会谈到)。

Ext.Widget还要定义了DOM元素是怎样产生且怎样与DOM事件连接的。

渲染

对于小部件,首先要考虑的是怎样去定义它的DOM树,典型的方式是例如以下面代码哪样在类指定element属性来实现:

Ext.define('Ext.ux.rating.Picker', {
extend: 'Ext.Widget', //... element: {
cls: 'ux-rating-picker',
reference: 'element', children: [{
reference: 'innerEl',
cls: 'ux-rating-picker-inner', listeners: {
click: 'onClick',
mousemove: 'onMouseMove',
mouseenter: 'onMouseEnter',
mouseleave: 'onMouseLeave'
}, children: [{
reference: 'valueEl',
cls: 'ux-rating-picker-value'
},{
reference: 'trackerEl',
cls: 'ux-rating-picker-tracker'
}]
}]
}, //...
});

对象element是基于Ext.dom.Helper规范来创建DOM元素的,基本的新增功能是reference和listeners属性。

这些名字在视图控制器上已经司空见惯了。而他们在小部件上所要做的事情也相似。在Ext.Widget,全部元素的reference属性都会以名称作为属性值被缓存在小部件实例中(比如:element、innerEl等等)。

事件

在上述的element中。在innerEl对象中还定义了listeners对象。

这些监听会附加到从规范块产生的元素上。方法会依据名称在小部件类中寻找。比如:

Ext.define('Ext.ux.rating.Picker', {
extend: 'Ext.Widget', //... onClick: function (event) {
var value = this.valueFromEvent(event);
this.setValue(value);
}, onMouseEnter: function () {
this.element.addCls(this.overCls);
}, onMouseLeave: function () {
this.element.removeCls(this.overCls);
}, onMouseMove: function (event) {
var value = this.valueFromEvent(event);
this.setTrackingValue(value);
},

尽管这看起来有点相似编写传统的组件类。但比較明显的是缺少初始化代码和清理代码。Ext.Widget的构造函数会处理元素的创建、跟踪他们的引用并设置监听。

除了这些动作(和对应的销毁方法)外,Ext.Widget再没有额外的生命周期或相关开销。

作为替代方法,派生类可通过配置系统提供的config属性来定义它的行为。对于那些不了解配置系统的,下面将简介一下。

配置系统101

Ext JS的核心理念之中的一个就是“配置(config)”属性的概念。

他们从一開始就是Ext JS的一部分,而不单是在Ext JS 5(或Sencha Touch 2.x)才有。框架已经将这些属性的机制规范化。常见的配置是这样声明的:

Ext.define('Ext.ux.rating.Picker', {
//...
config: {
family: 'monospace'
}
//...
});

上述声明等同于下面的手写代码:

Ext.define('Ext.ux.rating.Picker', {
//... getFamily: function () {
return this._family;
}, setFamily: function (newValue) {
var oldValue = this._family; if (this.applyTitle) {
newValue = this.applyFamily(newValue, oldValue); // #1
if (newValue === undefined) {
return this;
}
} if (newValue !== oldValue) {
this._family = newValue; if (this.updateFamily) {
this.updateFamily(newValue, oldValue); // #2
}
} return this;
}, //...
});

这样的自己主动化处理主要有下面显著的长处:

  • 清晰:更少的代码。类更易读
  • 一致性:全部配置都有同样的行为
  • 灵活性:当实现正确时,配置属性能够在不论什么时候改变,而不是仅仅能在创建时改变(在Ext JS中很多旧的配置属性的常见的局限)

开发者能够为不论什么属性提供两个关键的。也是可选的方法。如family,则为applyFamily和updateFamily(上面代码中的#1和#2处)。

这些方法差点儿总是会比重写而不是get或set方法。

应用方法(applier)

应用方法执行开发者去将接收值转换为存储的实际值。对于很多应用方法来说,这可能意味着基于接收到的配置对象来创建一些类的实例,又或者可能是应用方法在有一个地方标准化内部表示形式以避免在全部使用该属性的地方对它进行检查。

更新方法(updater)

当配置属性的值发生了改变,就会调用更新方法。更新方法的作用就是将旧值转换为新值。

initConfig——把它捎上

最后要说的是,要将一个类增加熬配置系统中。就必须在某个点调用initConfig方法。在Ext.Widget,它会在构造函数中执行。方法initConfig会接收config对象并会处理它的每个属性以便这些在类中的声明能适当的去调用set、apply和update方法。

该方法还提供了一个“适时(just in time)”的设置机制来解决配置属性之间的顺序问题,比如,假设一个配置属性的更新方法须要还有一个配置属性的值。它就要调用还有一个配置属性的get方法。在底层。initConfig会依据所请求属性之前返回的结果正确依照顺序调用set/apply/update方法。

使用cachedConfig进行优化

对于小部件,在有些时候须要很多配置来维护DOM。

因为不论什么给定的小部件实例不太可能重写全部的默认配置值。假设能够将这些要处理的默认值缓存起来。是最理想只是了。对于这些配置,能够对类进行下面改动:

Ext.define('Ext.panel.Panel', {
//...
cachedConfig: {
family: 'monospace'
}
//...
});

在大多数情况。这些配置与经常使用的配置是一样的。只是,在缓存这些配置的时候。配置系统会在类创建第一个实例的时候执行一些额外处理。

第一次实例化

在第一个实例的配置对象处理之前,配置系统仅仅会从类的默认值进行初始化。该处理会调用各种apply和update方法。这反过来会依据元素规范更新DOM元素最初的生成。

考虑下family配置,它带有下面更新方法:

updateFamily: function (family) {
this.element.setStyle('fontFamily', "'" + family + "'");
},

全部的更新方法有助于为小部件设置DOM的默认状态。

一旦配置被设置为他们的默认值。就会调用afterCachedConfig方法。该方法,仅仅会在第一次实例化的时候才起作用。Ext.Widget会深度克隆所得到的DOM树(使用cloneNode(true)DOM API)。

第二个实例(及以后)

在创建同样的小部件类的还有一个实例的时候。Ext.Widget会使用缓存的DOM树克隆并深度克隆它来创建性能的小部件DOM树。这可避免又一次处理元素规范和执行默认值的更新方法的开销。假设配置的更新方法书写正确,该处理过程非常大程度上是透明的。

当然。Ext.Widget在复制DOM树后,还有一些工作须要去做,比如检索元素的引用、封装监听并将不论什么非默认值的配置属性设置到实例。

只是,这时的开销就直接与赋予实例的配置值的数量有关而不是类的配置属性了。

重用,循环

下面来研究下怎样创建并初始化一个单一的小部件,有几个重要的概念是与在widgetcolumn中使用小部件有关的。

因为限制创建的实例一直是重点,因而缓冲渲染是关键。使用该方法。网格将渲染比记录少得多的小部件。并须要在行移出滚动区域“之后”且新行渲染“之前”循环使用。

当这些转变发生时,widgetcolumn将会将DOM中的小部件移动到新行,通过dataIndex从对应的记录读取字段数据并调用小部件的setConfig来设置它的defaultBind属性,这将会调用apply和update方法,因而,仅仅要编码正确。小部件如今就能够被又一次配置来显示新的字段值。

在当前演示样例的小部件,因为它仅仅表示一个可编辑的值,因而须要在updateValue方法做检查以了解小部件是否已经使用在网格单元格:

column = me.getWidgetColumn && me.getWidgetColumn();
record = column && me.getWidgetRecord && me.getWidgetRecord();
if (record && column.dataIndex) {
record.set(column.dataIndex, value);
}

方法getWidgetColumn和getWidgetRecord通过widgetcolumn被放置到小部件上,因此它知道它在网格中的上下文(context)。

小结

尽管大部分小部件的讨论与网格有关。但小部件也可用于传统部件的不论什么地方。

评分小部件作为迷你小部件引入Ext JS 5已经成为事实。

下面是演示样例应用程序的主面板截图。显示了它的items数组中的4个实例。

假设对上面的看起来非常熟悉,预计是你已经了解了Sencha Touch的模式。尽管这些是Ext JS 5的扩展,Ext.Widget本质上是最初在Sencha Touch中的Ext.AbstractComponent的最后版本号。

因此。你会使用小部件来取代组件?在很多方面,编写小部件是比编写组件简单。假设纯粹仅仅使用CSS来处理布局需求,尤其如此。

另外,随着我们继续将Sencha移动和桌面框架结合在一起。小部件在未来将会有跨界的可能。

能够在这里找到新的小部件的完整代码演示样例。享受它并让我们知道你的想法。

作者:Don Griffin
Don Griffin is a member of the Ext JS core team. He was an Ext JS user for 2 years before joining Sencha and has over 20 years of software engineering experience on a broad range of platforms. His experience includes designing web application front-ends and back-ends, native GUI applications, network protocols and device drivers. Don’s passion is to build world class products that people love to use.

【翻译】要理解Ext JS 5小工具的更多相关文章

  1. 【翻译】了解Ext JS 5的小部件

    原文:Understanding Widgets in Ext JS 5 在Ext JS 5,引入了新的"widgetcolumn",支持在网格的单元格中放置组件.同时,还在Ext ...

  2. 【翻译】为Ext JS和Sencha Touch开发人员准备的应用程序监测(App Inspector)

    和其他的Sencha开发人员一样,我会花费大约半天的时间在我喜欢的IDE工具上编写JavaScript,而另一半时间则是在浏览器上测试和调试我的应用程序.在过去几年,每一个主要的浏览器都已大为改善.现 ...

  3. 撸一个JS正则小工具

    写完正则在浏览器上检测自己写得对不对实在是不方便,于是就撸了一个JS正则小demo出来. demo demo展示 项目地址 代码部分 首先把布局样式先写好. <!DOCTYPE html> ...

  4. 【翻译】Sencha Ext JS 5公布

    原文:Announcing Sencha Ext JS 5 简单介绍 我代表Sencha和整个Ext JS团队,非常自豪的宣布,在今天,Sencha Ext JS 5公布了.Ext JS 5已经迈出了 ...

  5. 【翻译】使用Ext JS设计响应式应用程序

    原文:Designing Responsive Applications with Ext JS 在当今这个时代,用户都希望Web应用程序无论在形状还是大小上,既能在桌面电脑,也能在移动设备上使用.使 ...

  6. 【翻译】Sencha Ext JS 5发布

    原文:Announcing Sencha Ext JS 5 简介 我代表Sencha和整个Ext JS团队,很自豪的宣布,在今天,Sencha Ext JS 5发布了.Ext JS 5已经迈出了一大步 ...

  7. 【翻译】对于Ext JS 5,你准备好了吗?

    原文:Are You Ready for Ext JS 5? Ext JS 5:准备升级 对于Ext JS 5加入Sencha的大家庭,我们感到非常高兴!作为一个主要版本,在Ext JS 5引入了一堆 ...

  8. 【翻译】在Ext JS集成第三方库

    原文地址:http://www.sencha.com/blog/integrating-ext-js-with-3rd-party-libraries/ 作者:Kevin Kazmierczak Ke ...

  9. 【翻译】在Ext JS应用程序中使用自定义图标

    原文:Using Custom Icons in Your Ext JS App 作者:Lee BoonstraLee is a technical trainer at Sencha. She's ...

随机推荐

  1. 10 个迅速提升你 Git 水平的提示(转)

    最近我们推出了两个教程:熟悉Git的基本功能和 让你在开发团队中熟练的使用Git . 我们所讨论的命令足够一个开发者在Git使用方面游刃有余.在这篇文章中,我们试图探索怎样有效的管理你的时间和充分的使 ...

  2. hdu1695(莫比乌斯反演)

    传送门:GCD 题意:求[1,n],[1,m]gcd为k的对数. 分析:莫比乌斯入反演门题,gcd(x,y)==k等价于gcd(x/k,y/k)==1,求出[1,n][1,m]互质的对数,在减去[1, ...

  3. hdu1532(最大流)

    传送门:Drainage Ditches 题意:给出n个河流,m个点,以及每个河流的流量,求从1到m点的最大流量. 分析:网络流入门题,第一次写按照白书上毫无优化的Ford_fulkerson算法,先 ...

  4. poj2253(最短路小变形)

    题目连接:http://poj.org/problem?id=2253 题意:给出一个无向图,求一条1~2的路径使得路径上的最大边权最小. 分析:dij将距离更新改成取最大值即可,即dp[i]表示到达 ...

  5. 使用注解实现 bean 转 csv

    csv 文件是 aaa,bbb,ccc aaa,bbb,ccc 保存 这里的要求是 List<T> 线性表的类型 转换成 类别似 html 中 table的格式,即第一行是 head 后面 ...

  6. 【mysql】关于子查询的一个例子

    假设表my_tbl包含三个字段a,b,c:现在需要查询表中列a的每个不同值下的列b为最小值的记录量. 比如表记录为: a  b  c 1  3  'cd' 2  3  'nhd' 1  5  'bg' ...

  7. 使用windows-SQLyog连接linux-mysql

          嘿嘿,最近又清闲了一点,重新安装了mysql去学习.   -----博客园-邦邦酱好 系统环境: 1. 主机为windows系统,安装了SQLyog. 2. 主机上还安装了虚拟机,系统为c ...

  8. Ajax—初识

    看DRP的过程.又一次学习了一遍Ajax.更深刻的体会到了Ajax所具有的魅力.Ajax是一种技术或者方法,更是一 种艺术.它让我们的程序以一种更完美的姿态呈如今用户面前.以下就让我们一起走进Ajax ...

  9. windows phone 7 通过麦克风录音,并且播放

    原文:windows phone 7 通过麦克风录音,并且播放 //模拟XNA的框架(凡是在wp7中应用xna的都必须先模拟此类) public class XNAAsyncDispatcher : ...

  10. Jetty开发指导:Jetty Websocket API

    Jetty WebSocket API使用 Jetty提供了功能更强的WebSocket API,使用一个公共的核心API供WebSockets的服务端和client使用. 他是一个基于WebSock ...