控件系统主要用来解决系统复杂性的问题,使得系统不会因为变得复杂而不可控,同时保证其维护性和扩展性
NEJ框架提供了基于常规面向对象的思想构建的控件系统,主要用于:
- 提供通用解决方案的封装支持
- 提供核心功能、分析设计的重用
- 提供跨平台控件及API的支持
因为JavaScript本身没有提供类的概念,在控件系统中提供了一套类模型的解决方案,用以模拟常规面向对象语言中的“类”的概念。
类模型的实现见NEJ框架的base/klass模块
控件提供使用统一的类定义接口 _$klass 来定义一个类,通过此接口定义的类才具备以下的继承、初始化等特性
NEJ.define([
'base/klass'
],function(_k,_p){
// 定义一个类
_p._$$Klass = _k._$klass();
// TODO
return _p;
});
使用 _$klass 定义的类可以使用 _$extend 接口来继承其他父类
NEJ.define([
'base/klass'
],function(_k,_p){
// 定义一个类
_p._$$Klass = _k._$klass();
// 继承其他类
var pro = _p._$$Klass._$extend(Super);
// TODO
return _p;
});
使用 _$klass 定义的类统一使用 __init 接口来初始化类,所有子类的接口均可以使用 this.__super 方式调用父类同名接口
NEJ.define([
'base/klass'
],function(_k,_p){
// 定义一个类
_p._$$Klass = _k._$klass();
// 继承其他类
var pro = _p._$$Klass._$extend(Super);
// 初始化
pro.__init = function(){
// 调用父类的__init
this.__super();
// TODO something
};
// TODO
return _p;
});
类的所有方法均定义在类函数的 prototype 对象上
NEJ.define([
'base/klass'
],function(_k,_p){
// 定义一个类
_p._$$Klass = _k._$klass();
// 继承其他类
var pro = _p._$$Klass._$extend(Super);
// 初始化
pro.__init = function(){
// 调用父类的__init
this.__super();
// TODO something
};
// private 方法
pro._privateMethod = function(){
};
// protected 方法
pro.__protectedMethod = function(){
// TODO
};
// public 方法
pro._$publicMethod = function(){
// TODO
};
// TODO
return _p;
});
-
类命名
类命名使用前缀_$$标识,首字母大写,驼峰式,如_$$Klass,_$$OneKlass
-
类方法
方法分为私有、保护、公共方法三类,各类方法的前缀标识如下
- 私有方法使用_(单个下划线)作为前缀,如_privateMethod
- 受保护的方法使用__(两个下划线)作为前缀,如__protectedMethod
- 公共的方法使用_$(下划线+美元符)作为前缀,如_$publicMethod
方法的命名首字母小写,驼峰式,如_$myApi,__myApi,_myApi
控件模型使用类模型来实现,基于类模型的基础做扩展,主要在util/event模块中实现;控件采用分配回收机制,因此控件的生命周期包括以下三个阶段:
- 控件创建:首次使用时控件创建阶段,主要用于构建控件相关结构、数据等
- 控件重置:重复使用时控件重置阶段,主要用于处理外部输入数据、事件侦测等
- 控件回收:回收不用时控件销毁阶段,主要用于销毁重置阶段产生的结构、数据等
所有的控件均继承自 util/event 模块的 _$$EventTarget 类
NEJ.define([
'base/klass',
'util/event'
],function(_k,_t,_p){
var _pro;
// 定义控件
_p._$$Widget = _k._$klass();
// 继承_$$EventTarget
_pro = _p._$$Widget._$extend(_t._$$EventTarget);
// TODO
return _p;
});
控件采用分配回收重用机制,因此控件需实现__init、__reset、__destroy接口
NEJ.define([
'base/klass',
'util/event'
],function(_k,_t,_p){
var _pro;
// 定义控件
_p._$$Widget = _k._$klass();
// 继承_$$EventTarget
_pro = _p._$$Widget._$extend(_t._$$EventTarget);
// 控件首次创建构造过程
_pro.__init = function(){
this.__super();
// TODO
};
// 控件重复使用重置过程
_pro.__reset = function(_options){
this.__super(_options);
// TODO
};
// 控件回收销毁过程
_pro.__destroy = function(){
this.__super();
// TODO
};
// TODO
return _p;
});
其他扩展的业务逻辑根据控件实际需求实现
NEJ.define([
'base/klass',
'util/event'
],function(_k,_t,_p){
var _pro;
// 定义控件
_p._$$Widget = _k._$klass();
// 继承_$$EventTarget
_pro = _p._$$Widget._$extend(_t._$$EventTarget);
// 控件首次创建构造过程
_pro.__init = function(){
this.__super();
// TODO
};
// 控件重复使用重置过程
// 重置过程可以接受到分配控件时输入的配置信息
_pro.__reset = function(_options){
this.__super(_options);
// TODO
};
// 控件回收销毁过程
_pro.__destroy = function(){
this.__super();
// TODO
};
// 扩展私有接口
_pro._myPrivateMethod = function(){
// TODO
};
// 扩展保护接口
_pro.__myProtectedMethod = function(){
// TODO
};
// 扩展对外接口
_pro._$myPublicMethod = function(){
// TODO
};
// TODO
return _p;
});
控件支持自定义事件的触发,在控件的业务逻辑中可根据实际需求通过 _$dispatchEvent 接口触发自定义事件来与外界进行交互
NEJ.define([
'base/klass',
'util/event'
],function(_k,_t,_p){
var _pro;
// 定义控件
_p._$$Widget = _k._$klass();
// 继承_$$EventTarget
_pro = _p._$$Widget._$extend(_t._$$EventTarget);
// 控件首次创建构造过程
_pro.__init = function(){
this.__super();
// TODO
};
// 控件重复使用重置过程
// 重置过程可以接受到分配控件时输入的配置信息
_pro.__reset = function(_options){
this.__super(_options);
// TODO
};
// 控件回收销毁过程
_pro.__destroy = function(){
this.__super();
// TODO
};
// 扩展私有接口
_pro._myPrivateMethod = function(){
// TODO
// 触发自定义的onchange事件
this._$dispatchEvent(
'onchange',{
x:'xxxxx',
y:'yyyyyyy'
}
);
};
// 扩展保护接口
_pro.__myProtectedMethod = function(){
// TODO
// 触发自定义的onupdate事件
this._$dispatchEvent(
'onupdate',{
a:'aaaa',
b:'bbbbbbb'
}
);
};
// 扩展对外接口
_pro._$myPublicMethod = function(){
// TODO
};
// TODO
return _p;
});
控件的平台适配规则遵循《平台适配系统》的规范,可以按照以下步骤实现:
-
在控件实现文件处构建平台适配目录platform,可以通过nej工具集中nej-widget指令来自动生成控件目录结构,或者使用nej-patch指令来自动生成platform目录结构,如
widget | - widget.js | - platform | - widget.js | - widget.patch.js
-
提取控件涉及的存在平台差异的API,在platform/widget.js中根据W3C/ES规范实现API
NEJ.define([ 'base/platform' ],function(_m,_p){ // 存在平台差异的API _p.__api1 = function(){ // TODO }; // 存在平台差异的API _p.__api2 = function(){ // TODO } // 返回平台差异API集合 return _p; });
-
根据平台差异,在platform/widget.patch.js文件中实现各平台的差异化逻辑
NEJ.define([ './widget.js' // 这里注入标准API集合 ],function(_h){ // 根据平台特点重写API实现 NEJ.patch('TR<=2.0',function(){ // for ie6- _h.__api1 = function(){ // TODO }; }); // 根据平台特点采用AOP方式切入平台逻辑 NEJ.patch('WR',function(){ // for webkit _h.__api2 = _h.__api2._$aop( function(_event){ // 标准逻辑之前处理业务逻辑 // _event.args // _event.value // _event.stopped // TODO }, function(_event){ // 标准逻辑之后处理业务逻辑 // _event.args // _event.value // _event.stopped // TODO } ); }); // 这里必须返回注入的标准API集合 return _h; });
-
控件中使用{platform}注入平台适配API使用
NEJ.define([ 'base/klass', 'util/event', '{platform}widget.js' ],function(_k,_t,_h,_p){ var _pro; // 定义控件 _p._$$Widget = _k._$klass(); // 继承_$$EventTarget _pro = _p._$$Widget._$extend(_t._$$EventTarget); // 控件首次创建构造过程 _pro.__init = function(){ this.__super(); // TODO }; // 控件重复使用重置过程 // 重置过程可以接受到分配控件时输入的配置信息 _pro.__reset = function(_options){ this.__super(_options); // TODO }; // 控件回收销毁过程 _pro.__destroy = function(){ this.__super(); // TODO }; // 扩展私有接口 _pro._myPrivateMethod = function(){ // 使用平台适配接口 _h.__api1(); // TODO // 触发自定义的onchange事件 this._$dispatchEvent( 'onchange',{ x:'xxxxx', y:'yyyyyyy' } ); }; // 扩展保护接口 _pro.__myProtectedMethod = function(){ // TODO // 触发自定义的onupdate事件 this._$dispatchEvent( 'onupdate',{ a:'aaaa', b:'bbbbbbb' } ); }; // 扩展对外接口 _pro._$myPublicMethod = function(){ // 使用平台适配接口 _h.__api2(); // TODO }; // TODO return _p; });
控件使用分配回收机制而非 new 的方式使用
NEJ.define([
'/path/to/widget.js'
],function(_t){
// 分配控件
var _widget = _t._$$Widget._$allocate({
a:'aaaaaaaa',
b:'bbbbbbbbbbb',
c:'ccccccccccccc',
onchange:function(_event){
// 控件支持的事件
// _event.x
// _event.y
// TODO
},
onupdate:function(_event){
// 控件支持的事件
// _event.a
// _event.b
// TODO
}
});
// 外界可以调用控件的public方法
_widget._$myPublicMethod();
// 回收控件
// 注意这里必须将原控件持有的引用置空
_widget = _widget._$recycle();
// 或者
_widget._$recycle();
_widget = null;
});
控件根据其封装元素的差异可以分为通用控件和UI控件两类
- 通用控件:此类控件关注功能业务逻辑的实现,不关注视觉效果
- UI控件:此类控件会构建一套默认的视觉效果,具体功能逻辑由与之匹配的通用控件来实现
由于UI控件在实际项目中差异性比较大,因此NEJ框架会主要关注通用控件的支持,项目中可以根据通用控件结合实际项目视觉效果来实现项目相关的UI控件
通用控件只需遵循控件模型实现即可
UI控件基于控件模型扩展而来,其抽象实现在 ui/base 模块中的 _$$Abstract 类,UI控件的主要元素包括:
- 样式:控件展示效果样式,独立在控件对应的css文件中
- 结构:控件组成结构,独立在控件对应的html文件中
- 逻辑:控件逻辑实现,独立在控件对应的javascript文件中
一个UI控件典型的目录结构为
widget
| - widget.css
| - widget.html
| - widget.js
每个UI控件都使用一个唯一的样式标识,以防止与其他控件样式冲突,样式文件范例如下:
.#<uispace>-parent{position:relative;}
.#<uispace>{position:absolute;border:1px solid #aaa;background:#fff;text-align:left;visibility:hidden;}
.#<uispace> .zitm{height:20px;line-height:20px;cursor:default;}
.#<uispace> .js-selected{background:#1257F9;}
这里可以使用 #<KEY> 格式的简单模板来做数据占位,其中
- #<uispace> - 表示自动生成的样式标识名称
- #<uispace>-parent - 表示控件节点的父容器节点的样式
- 其他参数可以使用#<KEY>来占位,后续使用时输入{KEY:'XXXXX'}的数据即可
每个UI控件可以关联若干的结构模板,模板规则遵循NEJ的模板系统规范
单个模板文件范例
<div>
<div class="zbar">
<div class="zttl">标题</div>
</div>
<div class="zcnt"></div>
<span class="zcls" title="关闭窗体">×</span>
</div>
多个模板文件范例,模板的ID支持使用 #<KEY> 形式的简单模板做ID占位
<textarea name='jst' id='#<icmd>'>
{list xlist as x}
<div class="zitm zbg ${'js-'|seed}" data-command="${x.cmd}" title="${x.txt}">
<div class="zicn zbg ${x.icn}"> </div>
<div class="ztxt">${x.txt}</div>
</div>
{/list}
{if defined("hr")&&!!hr}
<div class="zbg zisp"> </div>
{/if}
</textarea>
<textarea name='jst' id='#<ifnt>'>
<div class="zsel ${icn} ${'js-'|seed}" data-command="${cmd}">
<span class="${'js-t-'|seed}">${txt}</span>
<span class="zarw zbg"> </span>
</div>
</textarea>
<textarea name='jst' id='#<iedt>'>
<div>
<div class="ztbar">${toolbar}</div>
<div class="zarea"></div>
</div>
</textarea>
逻辑部分主要用来实现UI控件的核心逻辑,主要分以下几部分功能
- 注入样式处理
- 注入结构处理
- 控件初始化
根据依赖系统规则,UI控件使用 text! 注入样式,注入的样式通过 base/element 模块中的 _$pushCSSText 接口做预处理,并返回自动生成的控件样式标识
NEJ.define([
'base/element',
'ui/base',
'text!./widget.css'
],function(_e,_i,_css,_p){
// 将注入的样式做预处理后缓存
var _seed_css = _e._$pushCSSText(_css);
// TODO
});
如果样式中已做了样式标识无需自动生成则只需缓存样式即可,如
.ui-suggest-parent{position:relative;}
.ui-suggest{position:absolute;border:1px solid #aaa;background:#fff;text-align:left;visibility:hidden;}
.ui-suggest .zitm{height:20px;line-height:20px;cursor:default;}
.ui-suggest .js-selected{background:#1257F9;}
NEJ.define([
'base/element',
'ui/base',
'text!./widget.css'
],function(_e,_i,_css,_p){
// 将注入的样式缓存
_e._$pushCSSText(_css);
// TODO
});
根据依赖系统规则,UI控件使用 text! 注入结构,注入的结构符合模板系统规则,后续使用 util/template/tpl 模块中的模板处理接口做处理
单个模板结构注入
NEJ.define([
'base/element',
'util/template/tpl',
'ui/base',
'text!./widget.css',
'text!./widget.html'
],function(_e,_t,_i,_css,_html,_p){
// 将注入的样式做预处理后缓存
var _seed_css = _e._$pushCSSText(_css),
_seed_html = _t._$addNodeTemplate(_html);
// TODO
});
多个模板结构注入
NEJ.define([
'base/element',
'util/template/tpl',
'ui/base',
'text!./widget.css',
'text!./widget.html'
],function(_e,_t,_i,_css,_html,_p){
// 将注入的样式做预处理后缓存
var _seed_css = _e._$pushCSSText(_css);
// 这里可以自动生成模板ID
// 返回 {icmd:'tpl-127363653',ifnt:'tpl-5985857444',iedt:'tpl-48763635374'}
var _seed = _t._$parseUITemplate(_html);
// 这里也可以自己指定模板ID
// 可以指定全部的ID,也可以指定某几个,未指定的ID自动生成
var _seed = _t._$parseUITemplate(_html,{
icmd:'abc',
ifnt:'def',
iedt:'ghi'
});
// TODO
});
UI控件的逻辑实现主要扩展自 ui/base 模块中的 _$$Abstract 类,需要实现外观的设置和结构的初始化
-
初始化外观
NEJ.define([ 'base/klass', 'base/element', 'util/template/tpl', 'ui/base', 'text!./widget.css', 'text!./widget.html' ],function(_k,_e,_t,_i,_css,_html,_p){ var _pro; // 定义UI控件 _p._$$UIWidget = _k._$klass(); _pro = _p._$$UIWidget._$extend(_i._$$Abstract); // 按需完成通用控件接口重写 // _pro.__init ... // _pro.__reset ... // _pro.__destroy ... // 初始化外观 // 此过程只会在控件第一次创建时进入 _pro.__initXGui = (function(){ // 将注入的样式/结构做预处理后缓存 var _seed_css = _e._$pushCSSText(_css), _seed_html = _t._$addNodeTemplate(_html); return function(){ this.__seed_css = _seed_css; this.__seed_html = _seed_html; }; })(); // TODO return _p; });
-
初始化结构
NEJ.define([ 'base/klass', 'base/element', 'util/template/tpl', 'ui/base', 'text!./widget.css', 'text!./widget.html' ],function(_k,_e,_t,_i,_css,_html,_p){ var _pro; // 定义UI控件 _p._$$UIWidget = _k._$klass(); _pro = _p._$$UIWidget._$extend(_i._$$Abstract); // 按需完成通用控件接口重写 // _pro.__init ... // _pro.__reset ... // _pro.__destroy ... // 初始化外观 // 此过程只会在控件第一次创建时进入 _pro.__initXGui = (function(){ // 将注入的样式/结构做预处理后缓存 var _seed_css = _e._$pushCSSText(_css), _seed_html = _t._$addNodeTemplate(_html); return function(){ this.__seed_css = _seed_css; this.__seed_html = _seed_html; }; })(); // 初始化结构 // 此过程只会在控件第一次创建时进入 _pro.__initNode = function(){ // 调用父类接口通过提供的__seed_html构建控件结构 // 构建好的控件结构可以通过this.__body访问 this.__super(); // TODO }; // TODO return _p; });
-
功能实现
NEJ.define([ 'base/klass', 'base/element', 'util/template/tpl', 'ui/base', 'text!./widget.css', 'text!./widget.html' ],function(_k,_e,_t,_i,_css,_html,_p){ var _pro; // 定义UI控件 _p._$$UIWidget = _k._$klass(); _pro = _p._$$UIWidget._$extend(_i._$$Abstract); // 按需完成通用控件接口重写 // _pro.__init ... // _pro.__reset ... // _pro.__destroy ... // 初始化外观 // 此过程只会在控件第一次创建时进入 _pro.__initXGui = (function(){ // 将注入的样式/结构做预处理后缓存 var _seed_css = _e._$pushCSSText(_css), _seed_html = _t._$addNodeTemplate(_html); return function(){ this.__seed_css = _seed_css; this.__seed_html = _seed_html; }; })(); // 初始化结构 // 此过程只会在控件第一次创建时进入 _pro.__initNode = function(){ // 调用父类接口通过提供的__seed_html构建控件结构 // 构建好的控件结构可以通过this.__body访问 this.__super(); // TODO }; // 实现控件核心功能 _pro._myPrivateMethod = function(){ // TODO }; _pro.__myProtectedMethod = function(){ // TODO }; _pro._$myPublicMethod = function(){ // TODO }; // TODO return _p; });
控件的使用同通用控件,这里需要注意的是UI控件需要输入parent配置参数才能在页面上渲染出来,否则构建的控件只存在于内存中,页面上无法看到
NEJ.define([
'/path/to/ui/widget.js'
],function(_i){
// 分配控件
var _uiwidget = _i._$$UIWidget._$allocate({
parent:document.body, // 注意这里输入parent
clazz:'m-ui-widget'
});
// 回收控件
_uiwidget = _uiwidget._$recycle();
});
项目过程中如果觉得有些控件可以通用,分享给其他项目使用,可以将控件提交到NEJ控件仓库,提交的控件遵循以下规范
提交的通用控件目录结构如下所示(注:目录及文件命名中不得出现"."等特殊字符)
widget
| - test
| - demo
| - platform
| - widget.js
提交的UI控件的目录结构为
widget
| - test
| - demo
| - platform
| - widget.js
| - widget.css
| - widget.html
用于自动化测试控件的代码,后期会统一规范控件的测试方式 (TODO:测试规范)
用于放置当前控件的使用场景及使用范例
根据NEJ平台适配系统规则,如果控件需要做平台适配则在此目录下实现适配接口,如果无需平台适配则可以不提交此目录
控件关联的样式文件,如无关联样式可不提交此文件
控件关联的结构文件,遵循NEJ模板系统规范,如无关联结构可不提交此文件
控件核心业务逻辑实现文件
所有注释遵循JSDOC3规范,注释描述支持markerdown语法
文件起始位置注释文件的描述信息、作者、版本等
/*
* ------------------------------------------
* 控件描述内容
*
* @version 1.0
* @author genify(caijf@corp.netease.com)
* ------------------------------------------
*/
使用@module标记注释当前文件的模块,模块名称可被依赖系统直接引入使用
/** @module util/event */
使用@class、@extends标记注释类及继承关系
/**
* 标签切换控件封装
*
* 结构举例
*
* ```html
* <div id="box">
* <a>1</a>
* <a>2</a>
* <a class="js-disabled">3</a>
* <a>4</a>
* </div>
* ```
*
* 脚本举例
*
* ```javascript
* NEJ.define([
* 'util/tab/tab'
* ],function(_t){
* // 实例化控件
* var _tab = _t._$$Tab._$allocate({
* list:_e._$getChildren('box'),
* index:1,
* onchange:function(_event){
* // TODO
* }
* });
* // 使用控件
* _tab._$go(2);
* });
* ```
*
* @class module:util/tab/tab._$$Tab
* @extends module:util/event._$$EventTarget
*
* @param {Object} config - 可选配置参数
* @property {Array} list - 标签项列表
* @property {Number} index - 初始选中项索引值,默认为0
* @property {String} event - 触发选择事件名称,默认为click
* @property {Boolean} inverse - 是否反过程,true表示选中时删除选中样式,否则选中时添加样式
* @property {String} disabled - 选项禁用样式,默认为js-disabled
* @property {String} selected - 选中样式名,默认为js-selected
*/
使用@event标记注释控件支持的事件
/**
* 标签切换事件,输入{last:1,index:5}
*
* ```javascript
* NEJ.define([
* 'util/tab/tab'
* ],function(_t){
* // 实例化控件
* var _tab = _t._$$Tab._$allocate({
* list:_e._$getChildren(_e._$get('box')),
* index:1,
* onchange:function(_event){
* // _event.last 上一次的tab索引
* // _event.index 需要切换到的tab索引
* // _event.list 节点列表
* // _event.data 节点上通过data-value设置的内容
* // TODO
* }
* });
* });
* ```
*
* @event module:util/tab/tab._$$Tab#onchange
* @param {Object} event - tab信息
* @property {Number} last - 上一次的tab索引
* @property {Number} index - 需要切换到的tab索引
* @property {Array} list - 节点列表
* @property {String} data - 节点上通过data-value设置的内容
* @property {Boolean} stopped - 是否阻止触发节点的默认事件,回调过程中如果设置为false则后续继续触发节点的默认事件
*/
使用@method标记注释控件接口,使用@private、@protected标记注释私有和受保护的方法
/**
* 设置标签选中状态
*
* @protected
* @method module:util/tab/tab._$$Tab#__doTabItemSelect
* @param {Node} arg0 - 标签节点
* @param {Boolean} arg1 - 是否选中
* @return {Void}
*/
/**
* 切换到指定索引位置
*
* ```javascript
* // 切换到索引为2的位置,如果当前索引为2则不触发回调
* _tab._$go(2);
* // 切换索引为2,如果当前索引为2也触发onchange回调
* _tab._$go(2,true);
* ```
*
* @method module:util/tab/tab._$$Tab#_$go
* @param {Number} arg0 - 索引值
* @param {Boolean} arg1 - 是否强行触发onchange事件
* @return {Void}
*/
控件编码使用前缀标识变量使用范围
前缀 | 说明 |
---|---|
_ | 私有属性、方法,局部变量,仅限于当前控件范围内使用 |
__ | 受保护的属性、方法,控件范围及所有子类可使用 |
_$ | 对外属性、方法,控件外可直接调用 |
_$$ | 类名前缀,控件外可直接使用 |
on | 事件名前缀,控件外可直接使用 |
控件命名遵循以下规则便于识别
- 类名首字母大写,驼峰形式,如_$$MyClassName等
- 属性、方法名首字母小写,驼峰形式,如 _myMethod、__myProtectedMethod、_$doSomething等
- 事件名称全小写,采用名称+动词形式,如 onchange、onlistload等