Skip to content

Latest commit

 

History

History
1086 lines (876 loc) · 27.6 KB

WIDGET.md

File metadata and controls

1086 lines (876 loc) · 27.6 KB

控件系统

概述

控件系统主要用来解决系统复杂性的问题,使得系统不会因为变得复杂而不可控,同时保证其维护性和扩展性

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;
});

类规范

  1. 类命名

    类命名使用前缀_$$标识,首字母大写,驼峰式,如_$$Klass,_$$OneKlass

  2. 类方法

    方法分为私有、保护、公共方法三类,各类方法的前缀标识如下

    • 私有方法使用_(单个下划线)作为前缀,如_privateMethod
    • 受保护的方法使用__(两个下划线)作为前缀,如__protectedMethod
    • 公共的方法使用_$(下划线+美元符)作为前缀,如_$publicMethod

    方法的命名首字母小写,驼峰式,如_$myApi,__myApi,_myApi

控件模型

控件模型使用类模型来实现,基于类模型的基础做扩展,主要在util/event模块中实现;控件采用分配回收机制,因此控件的生命周期包括以下三个阶段:

  1. 控件创建:首次使用时控件创建阶段,主要用于构建控件相关结构、数据等
  2. 控件重置:重复使用时控件重置阶段,主要用于处理外部输入数据、事件侦测等
  3. 控件回收:回收不用时控件销毁阶段,主要用于销毁重置阶段产生的结构、数据等

控件定义

所有的控件均继承自 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;
});

平台适配

控件的平台适配规则遵循《平台适配系统》的规范,可以按照以下步骤实现:

  1. 在控件实现文件处构建平台适配目录platform,可以通过nej工具集中nej-widget指令来自动生成控件目录结构,或者使用nej-patch指令来自动生成platform目录结构,如

      widget
        | - widget.js
        | - platform
                | - widget.js
                | - widget.patch.js
    
  2. 提取控件涉及的存在平台差异的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;
    });
  3. 根据平台差异,在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;
    });
  4. 控件中使用{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控件基于控件模型扩展而来,其抽象实现在 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}">&nbsp;</div>
    <div class="ztxt">${x.txt}</div>
  </div>
{/list}
{if defined("hr")&&!!hr}
  <div class="zbg zisp">&nbsp;</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">&nbsp;</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 类,需要实现外观的设置和结构的初始化

  1. 初始化外观

    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;
    });
  2. 初始化结构

    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;
    });
  3. 功能实现

    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

test

用于自动化测试控件的代码,后期会统一规范控件的测试方式 (TODO:测试规范)

demo

用于放置当前控件的使用场景及使用范例

platform

根据NEJ平台适配系统规则,如果控件需要做平台适配则在此目录下实现适配接口,如果无需平台适配则可以不提交此目录

widget.css

控件关联的样式文件,如无关联样式可不提交此文件

widget.html

控件关联的结构文件,遵循NEJ模板系统规范,如无关联结构可不提交此文件

widget.js

控件核心业务逻辑实现文件

注释规范

所有注释遵循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等