O2OA API

source

o2_core/o2/xScript/ViewEnvironment.js

/**
 * @readonly
 * @enum {String} ViewFilterDataLogic
 * @property {String} or color for a white square or piece.
 * @property {String} and color for a black square or piece.
 */

/**
 * ViewFilter 视图过滤条件
 * @typedef {Object} ViewFilter
 * @property {String} logic - 可选值:“and”或者“or”,表示和前一个条件的逻辑运算关系。
 * @property {String} path - 要过滤的data数据的路径。
 * @property {String} comparison - 比较运算符,可选值:<br/>
 * <div style='padding-left:150px;'>
 * <b>equals</b> 或 <b>==</b> 或:表示等于。<br/>
 * <b>notEquals</b> 或 <b>!=</b> :表示不等于。<br/>
 * <b>greaterThan</b> 或 <b>></b> :表示大于。<br/>
 * <b>greaterThanOrEqualTo</b> 或 <b>>=</b> :表示大于或等于。<br/>
 * <b>lessThan</b> 或 <b><</b> :表示小于。<br/>
 * <b>lessThanOrEqualTo</b> 或 <b><=</b> :表示小于等于。<br/>
 * <b>like</b> :表示部分匹配。<br/>
 * <b>notLike</b> :表示不匹配。<br/>
 * <b>range</b> :表示一定的范围。<br/>
 * <b>in</b> :表示在某几个特定的值当中。<br/>
 * </div>
 * @property {String} formatType  - 过滤数据的数据类型,可选值:
 * <div style='padding-left:150px;'>
 * <b>textValue</b> :文本。<br/>
 * <b>numberValue</b> :数字。<br/>
 * <b>dateTimeValue</b> :日期时间。<br/>
 * <b>dateValue</b> :日期。<br/>
 * <b>booleanValue</b> :布尔值。<br/>
 * </div>
 * @property {(String|Number|Boolean)} value - 过滤的值,根据formatType提供匹配的数据类型的值.
 * <br/>如果是dateTimeValue数据类型,则提供日期时间格式的字符串,格式如“YYYY-MM-DD HH:MM:SS”。
 * <br/>如果是dateValue数据类型,则提供日期格式的字符串,格式如“YYYY-MM-DD”。
 * <br/>当comparison值为“range”时,此值表示范围中的第一个值。当comparison值为“in”时,多个值用半角逗号","分开。
 * @property {(String|Number|Boolean)} otherValue  - 当comparison值为“range”时,此值表示范围中的第二个值。当comparison值不为“range”时,忽略此值。
 * @example
 *  {
 *    "logic":"and",
 *    "path":"$work.title",
 *    "comparison":"like",
 *    "value":"关于",
 *    "formatType":"textValue"
 * }
 * @example
 *  {
 *    "logic":"and",
 *    "path":"month",
 *    "comparison":"in",
 *    "value":"一月,二月,三月",
 *    "formatType":"textValue"
 * }
 */

/**
 * StatementFilter 查询视图的过滤条件
 * @typedef {Object} StatementFilter
 * @property {String} path - 要过滤的data数据的路径。
 * 如果语句格式是JPQL,形式为查询语句中的"表别名.字段名",如"o.title"。
 * 在v8.0以后,系统增加了原生SQL,则使用数据库的字段名,如"xtitle"。
 * @property {String} comparison - 比较运算符,可选值:<br/>
 * <div style='padding-left:150px;'>
 * <b>equals</b> 或 <b>==</b> 或:表示等于。<br/>
 * <b>notEquals</b> 或 <b>!=</b> :表示不等于。<br/>
 * <b>greaterThan</b> 或 <b>></b> :表示大于。<br/>
 * <b>greaterThanOrEqualTo</b> 或 <b>=></b> :表示大于或等于。<br/>
 * <b>lessThan</b> 或 <b><</b> :表示小于。<br/>
 * <b>lessThanOrEqualTo</b> 或 <b><=</b> :表示小于等于。<br/>
 * <b>like</b> :表示部分匹配。<br/>
 * <b>notLike</b> :表示不匹配。<br/>
 * </div>
 * @property {String} formatType  - 过滤数据的数据类型,可选值:
 * <div style='padding-left:150px;'>
 * <b>textValue</b> :文本。<br/>
 * <b>numberValue</b> :数字。<br/>
 * <b>dateTimeValue</b> :日期时间。<br/>
 * <b>dateValue</b> :仅日期。<br/>
 * <b>timeValue</b> :仅时间。<br/>
 * <b>booleanValue</b> :布尔值。<br/>
 * </div>
 * @property {(String|Number|Boolean)} value - 过滤的值,根据formatType提供匹配的数据类型的值。如果是dateTimeValue数据类型,则提供日期格式的字符串,格式如“YYYY-MM-DD HH:MM:SS”。
 * 如果是dateValue数据类型,则提供日期格式的字符串,格式如“YYYY-MM-DD”。如果是timeValue数据类型,则提供时间格式的字符串,格式如“HH:MM:SS”。
 * @example
 *{
 *    "path":"o.title", //查询语句格式为jpql使用o.title,为原生sql中使用xtitle
 *    "comparison":"like",
 *    "value":"关于",
 *    "formatType":"textValue"
 *}
 */

/**
 * StatementParameter  查询视图的过滤条件值参数,对查询语句where语句的形如":person"的参数部分进行赋值<br/>
 * 在v8.0以后,系统还新增了问号加数字的传值参数,如"?1",用法和 ":field"一致。
 * 有以下规则:<br/>
 * 1、参数名称为下列值时,后台自动赋值:person(当前人),identityList(当前人身份列表),unitList(当前人所在直接组织), unitAllList(当前人所在所有组织), groupList(当前人所在群组)。v8.0以后系统自动解析,不需要再传这类参数。<br/>
 * 2、如果对比的是日期,需要传入 Date 类型。<br/>
 * 3、如果运算符用的是 like, noLike,模糊查询,值为 "%{value}%"。
 * 4、
 * @typedef {Object} StatementParameter
 * @example
 * {
 *    "person" : "", //v8.0以后系统自动解析,不需要再传这类参数。
 *    "startTime" : (new Date("2020-01-01")), //日期格式
 *    "applicationName" : "%test%", //like或notlike
 *    "processName" : "test流程", //其他写确定的值
 *    "?1": "关于" //v8.0后查询语句支持问号加数字的传参
 * }
 */

/*********this.org 的 typedef 开始********/

/**
 * 在本API的this.org模块中,当使用群组作为检索参数的时候,允许使用群组的distinguishedName、name、id、unique属性值,或者包含这些属性值的对象。
 * @typedef {(String|Object)} GroupFlag
 * @example
 * //以下均为合法参数
 *
 * //群组属性值
 * "工作汇报读者@ReportReader@G" //群组识别名
 * "cce8bc22-225a-4f85-8132-7374d546886e" //id
 * "ReportReader" //群组unique
 * "工作汇报读者" //群组名称
 *
 * //包含下列属性的对象
 * { "distinguishedName": "工作汇报读者@ReportReader@G" } //群组识别名
 * { "unique": "cce8bc22-225a-4f85-8132-7374d546886e" } //群组id
 * { "unique": "ReportReader" } //群组unique
 * { "name": "工作汇报读者" } //群组名称
 */


/**
 * 在本API的this.org模块中,当使用人员(个人)作为检索参数的时候,允许使用人员的distinguishedName、name、id、unique属性值或包含这些属性的对象。
 * @typedef {(String|Object)} PersonFlag
 * @example
 * //以下均为合法参数
 *
 * //人员属性值
 * "李四@lisi@P" //人员识别名
 * "cce8bc22-225a-4f85-8132-7374d546886e" //人员id
 * "lisi" //人员unique
 *
 * //包含下列属性的对象
 * { "distinguishedName": "李四@lisi@P" } //人员识别名
 * { "id": "cce8bc22-225a-4f85-8132-7374d546886e" } //人员id
 * { "unique": "lisi" } //人员unique
 */

/**
 * 在本API的this.org模块中,当使用角色作为检索参数的时候,允许使用角色的distinguishedName、name、id、unique属性值或包含这些属性的对象。
 * @typedef {(String|Object)} RoleFlag
 * @example
 * //以下均为合法参数

 * //角色属性值
 * "PersonManager@PersonManagerSystemRole@R" //角色识别名
 * "85874fbc-9af0-4a07-bc2f-0ab899cc4c7d" //角色id
 * "PersonManagerSystemRole" //角色unique
 * "PersonManager" //角色名称
 *
 * //包含下列属性的对象
 * { "distinguishedName": "PersonManager@PersonManagerSystemRole@R" } //角色识别名
 * { "id": "85874fbc-9af0-4a07-bc2f-0ab899cc4c7d" } //角色id
 * { "unique": "PersonManagerSystemRole" } //角色unique
 * { "name": "PersonManager" } //角色名称
 */

/**
 * 在本API的this.org模块中,当使用身份作为检索参数的时候,允许使用身份的distinguishedName、id、unique属性值或包含这些属性的对象。
 * @typedef {(String|Object)} IdentityFlag
 * @example
 * //以下均为合法参数

 * //角色属性值
 * "张三@903dbb04-085c-4bb1-9c24-3009bd5f7372@I" //身份识别名
 * "903dbb04-085c-4bb1-9c24-3009bd5f7372" //身份id
 * "903dbb04-085c-4bb1-9c24-3009bd5f7372" //身份unique,系统默认的unique和id一致
 *
 * //包含下列属性的对象
 * { "distinguishedName": "张三@903dbb04-085c-4bb1-9c24-3009bd5f7372@I" } //身份识别名
 * { "id": "903dbb04-085c-4bb1-9c24-3009bd5f7372" } //身份id
 * { "unique": "903dbb04-085c-4bb1-9c24-3009bd5f7372" } //身份unique,系统默认的unique和id一致
 */

/**
 * 在本API的this.org模块中,当使用组织作为检索参数的时候,允许使用组织的distinguishedName、id、unique属性值或包含这些属性的对象。
 * @typedef {(String|Object)} UnitFlag
 * @example
 * //以下均为合法参数

 * //组织属性值
 * "开发部@kfb@U" //组织标识名
 * "cce8bc22-225a-4f85-8132-7374d546886e" //组织id
 * "kfb" //组织unique
 *
 * //包含下列属性的对象
 * { "distinguishedName": "开发部@kfb@U" } //组织标识名
 * { "id": "cce8bc22-225a-4f85-8132-7374d546886e" } //组织id
 * { "unique": "kfb" } //组织unique
 */

MWF.xScript = MWF.xScript || {};
MWF.xScript.ViewEnvironment = function (ev) {
    var _form = ev.view;

    /**
     * 字符串,当前应用类型。<br/><br/>
     * <table>
     *   <tr>
     *      <th>值</th>
     *      <th>应用类型</th>
     *   </tr>
     *   <tr>
     *      <td>cms</td>
     *      <td>内容管理</td>
     *   </tr>
     *   <tr>
     *      <td>process</td>
     *      <td>流程管理</td>
     *   </tr>
     *   <tr>
     *      <td>portal</td>
     *      <td>门户管理</td>
     *   </tr>
     *   <tr>
     *      <td>query</td>
     *      <td>数据中心</td>
     *   </tr>
     * </table>
     * @module appType
     * @o2cn 应用类型
     * @o2category web
     * @o2ordernumber 170
     * @instance
     * @example
     * var title;
     * if( this.appType === "cms"  ){
     *     title = this.documentContext.getDocument().title
     * }else if( this.appType === "process"  ){
     *     title = this.workContext.getWork().title
     * }
     **/
    this.appType = "query";

    this.library = COMMON;
    //this.library.version = "4.0";

    //data
    // var getJSONData = function (jData) {
    //     return new MWF.xScript.JSONData(jData, function (data, key, _self) {
    //         var p = { "getKey": function () { return key; }, "getParent": function () { return _self; } };
    //         while (p && !_forms[p.getKey()]) p = p.getParent();
    //         if (p) if (p.getKey()) if (_forms[p.getKey()]) _forms[p.getKey()].resetData();
    //     });
    // };
    // this.setData = function (data) {
    //     this.data = getJSONData(data);
    //     this.data.save = function (callback) {
    //         var formData = {
    //             "data": data,
    //             "sectionList": _form.getSectionList()
    //         };
    //         form.workAction.saveData(function (json) { if (callback) callback(); }.bind(this), null, work.id, jData);
    //     }
    // };
    // this.setData(_data);

    //dict
    /**
     * this.Dict是一个工具类,如果您在流程、内容管理、门户和服务管理中创建了数据字典,可以使用this.Dict类对数据字典进行增删改查操作。<br/>
     * 从v8.0版本开始,支持在门户和服务管理中创建数据字典。
     * @module Dict
     * @o2cn 数据字典
     * @o2category web
     * @o2ordernumber 120
     * @param {(String|Object)} optionsOrName 数据字典标识字符串或者是对象。
     * <div>如果对本应用的数据字典操作,将optionsOrName设置为string。</div>
     * <pre><code class='language-js'>var dict = new this.Dict("bulletinDictionary"); //数据字典的名称、别名或id
     * </code></pre>
     * <div>如果需要对其他应用的数据字典进行操作,将options设置为JsonObject</div>
     * <pre><code class='language-js'>var dict = new this.Dict({
     *     //type: 应用类型。可以为process  cms  portal service。
     *     //在流程和内容管理中如果没有该选项或者值为空字符串,则表示应用脚本和被应用的脚本配置类型相同。
     *     //比如在流程的A应用脚本中引用流程B应用的脚本配置,则type可以省略。
     *     //为了兼容老版本,在门户中使用需要指定type,否则默认值为process
     *    type : "cms",
     *    application : "bulletin", //数据字典所在的流程、门户、CMS的名称、别名、id, 默认为当前应用,服务管理中忽略该参数
     *    name : "bulletinDictionary", // 数据字典的名称、别名、id
     *    anonymous : true //允许用户在未登录的情况下读取cms的数据字典, type为cms的时候改参数才有效,默认为false,该参数名也可以是 enableAnonymous
     * });
     *
     * //引用服务管理中的数据字典
     * var dict = new this.Dict({
     *    "type": "service",
     *    "name": "dictName"
     * });
     *
     * //引用流程管理中的数据字典
     * var dict = new this.Dict({
     *    "type": "process",
     *    "application": "appName",
     *    "name": "dictName"
     * });
     *
     * //引用内容管理中的数据字典
     * var dict = new this.Dict({
     *     "type": "cms",
     *     "application": "appName",
     *    "name": "dictName"
     * });
     *
     * //引用门户管理中的数据字典
     * var dict = new this.Dict({
     *    "type": "portal",
     *    "application": "appName",
     *    "name": "dictName"
     * });
     *
     * </code></pre>
     * @return {Object} Dict对象
     * @o2syntax
     * //您可以在页面、表单、流程各个嵌入脚本中,通过this.Dict()对本应用或其他应用的数据字典中的数据进行增删改查,如下:
     * var dict = new this.Dict( options )
     */

    /**
     * 根据路径获取数据字典中的数据。
     * @method get
     * @methodOf module:Dict
     * @static
     * @param {String} [path] 数据字典中的数据路径,允许使用中文。当路径为多级时,用点号(.)分隔。当值为空的时候,表示获取数据字典中的所有数据。
     * @param {Function} [success] 获取数据成功时的回调函数。
     * @param {Function} [failure] 获取数据失败时的回调。
     * @param {Boolean} [async] 是否异步执行,默认为false。
     * @param {Boolean} [refresh] 是否忽略本地缓存直接从服务器获取,默认为false。如果需要在脚本中该数据字典是变化的(比如编号),需设置为true。
     * @return {(Promise|Object|Array|String|Number|Boolean)}
     * 当async为true时返回Promise({@link https://developer.mozilla.org/zh-CN/docs/Web/JavaScript/Reference/Global_Objects/Promise|Promise说明});
     * 否则返回数据字典的数据,类型和配置数据字典时候指定的一致。
     * @o2syntax
     * var data = dict.get( path, success, failure, async, refresh )
     * @example
     * var dict = new this.Dict("bulletinDictionary");
     *
     * //没有参数的时候,表示同步获取获取所有数据
     *  var data = dict.get()
     *
     *  //同步执行,获取category下key为subCategory的数据
     *  var data = dict.get("category.subCategory");
     *
     * //异步执行,使用回调处理数据,如果category为数组,获取第0项数据
     * dict.get("category.0", function(data){
     *      //data 是数据字典的数据
     *  }, function(xhr){
     *      //xhr 为 xmlHttpRequest
     *  }, true //异步执行
     * )
     *
     * //异步执行,使用Promise处理结果
     * var promise = dict.get("category", null, null, true);
     * promise.then( function(data){
     *     //data 是数据字典的数据
     * })
     * @example
     * <caption>
     *     已经配置好了如下图所示的数据字典
     * <img src='img/module/Dict/dict.png' />
     * </caption>
     * var dict = new this.Dict({
     *     //type: 应用类型。可以为process  cms。
     *     //如果没有该选项或者值为空字符串,则表示应用脚本和被应用的脚本配置类型相同。
     *     //比如在流程的A应用脚本中引用流程B应用的脚本配置,则type可以省略。
     *    type : "cms",
     *    application : "bulletin", //流程、CMS的名称、别名、id, 默认为当前应用
     *    name : "bulletinDictionary", // 数据字典的名称、别名、id
     *    anonymous : true //允许用户在未登录的情况下读取cms的数据字典, type为process的时候此参数无效,默认为false,该参数名也可以是enableAnonymous
     * });
     *
     * var data = dict.get();
     * //data的值为
     * {
     *    "category": [
     *        {
     *            "enable": true,
     *            "sequence": 1.0,
     *            "text": "公司公告",
     *            "value": "company"
     *        },
     *        {
     *            "enable": "false",
     *            "sequence": 2.0,
     *            "text": "部门公告",
     *            "value": "department"
     *        }
     *    ]
     * }
     *
     *  var category = dict.get("category");
     *  //category的值为
     *  [
     *     {
     *        "enable": true,
     *        "sequence": 1.0,
     *        "text": "公司公告",
     *        "value": "company"
     *    },
     *     {
     *       "enable": "false",
     *       "sequence": 2.0,
     *        "text": "部门公告",
     *        "value": "department"
     *    }
     *  ]
     *
     *  var array0 = dict.get("category.0");
     *  //array0 的值为
     *  {
     *    "enable": true,
     *    "sequence": 1.0,
     *    "text": "公司公告",
     *    "value": "company"
     * }
     *
     * var enable = dict.get("category.0.eanble");
     * //enable 的值为 true
     */

    /**
     * 根据路径新增数据字典的数据。
     * @method add
     * @methodOf module:Dict
     * @instance
     * @param {String} path 数据字典中的数据路径,允许使用中文。当路径为多级时,用点号(.)分隔。如果path在数据字典中已有数据,且原有数据是数组,则数组添加一项;如果原有数据不是数组,则报错。
     * @param {(Object|Array|String|Number|Boolean)} data 需要新增的数据
     * @param {Function} [success] 增加数据成功时的回调函数。
     * @param {Function} [failure] 增加数据错误时的回调函数。
     * @o2syntax
     * dict.add( path, data, success, failure )
     * @example
     * var dict = new this.Dict("bulletinDictionary");
     *
     * dict.add( "category", { text : "系统公告", value : "system" }, function(data){
     *    //data 形如
     *    //{
     *    //    "id": "80ed5f60-500f-4358-8bbc-b7e81f77aa39" //id为数据字典ID
     *    //}
     * }, function(xhr){
     *    //xhr 为 xmlHttpRequest
     * });
     * @example
     * <caption>
     *     对get方法样例的数据字典进行赋值,如下:
     * </caption>
     * var dict = new this.Dict("bulletinDictionary");
     *
     * dict.add( "category", { text : "系统公告", value : "system" }, function(data){
     *    //data 形如
     *    //{
     *    //    "id": "80ed5f60-500f-4358-8bbc-b7e81f77aa39" //id为数据字典ID
     *    //}
     * }, function(xhr, text, error){
     *    //xhr 为 xmlHttpRequest, text 为错误文本, error为Error对象
     * });
     *     //数据字典的值变为
     * {
     *    "category": [
     *        {
     *            "enable": true,
     *            "sequence": 1.0,
     *            "text": "公司公告",
     *            "value": "company"
     *        },
     *        {
     *            "enable": "false",
     *            "sequence": 2.0,
     *            "text": "部门公告",
     *            "value": "department"
     *        },
     *        {
     *            "text": "系统公告",
     *            "value": "system"
     *        }
     *    ]
     * }
     *
     *  dict.add( "category.2.sequence", 3 );
     *     //数据字典的值变为
     * {
     *    "category": [
     *        {
     *            "enable": true,
     *            "sequence": 1.0,
     *            "text": "公司公告",
     *            "value": "company"
     *        },
     *        {
     *            "enable": "false",
     *            "sequence": 2.0,
     *            "text": "部门公告",
     *            "value": "department"
     *        },
     *        {
     *            "sequence" : 3.0,
     *            "text": "系统公告",
     *            "value": "system"
     *        }
     *    ]
     * }

     * dict.add( "archiveOptions", {
     *    "yes" : "是",
     *    "no" : "否"
     * });
     *     //数据字典的值变为
     * {
     *    "category": [
     *        {
     *            "enable": true,
     *            "sequence": 1.0,
     *            "text": "公司公告",
     *            "value": "company"
     *        },
     *        {
     *            "enable": "false",
     *            "sequence": 2.0,
     *            "text": "部门公告",
     *            "value": "department"
     *        },
     *        {
     *            "sequence" : 3.0,
     *            "text": "系统公告",
     *            "value": "system"
     *        }
     *
     *    ],
     *    "archiveOptions" : {
     *        "yes" : "是",
     *        "no" : "否"
     *    }
     * }
     * @example
     * <caption>下面是错误的赋值,如下:</caption>
     * dict.add( "category.3", { text : "系统公告", value : "system" }); //出错,因为不能对数组下标直接赋值
     *
     * dict.add( "category.1.value", { text : "系统公告" } ); //出错,因为不能对已经存在的非数组路径赋值
     */

    /**
     * 根据路径修改数据字典的数据。
     * @method set
     * @methodOf module:Dict
     * @instance
     * @param {String} path 数据字典中的数据路径,允许使用中文。当路径为多级时,用点号(.)分隔。如果数据路径不存在,则报错。
     * @param {(Object|Array|String|Number|Boolean)} data 修改后的数据
     * @param {Function} [success] 设置数据成功时的回调函数。
     * @param {Function} [failure] 设置数据错误时的回调函数。
     * @o2syntax
     * dict.set( path, data, success, failure )
     * @example
     * var dict = new this.Dict("bulletinDictionary");
     *
     * dict.set( "category", { text : "系统公告", value : "system" }, function(data){
     *    //data 形如
     *    //{
     *    //    "id": "80ed5f60-500f-4358-8bbc-b7e81f77aa39" //id为数据字典ID
     *    //}
     * }, function(xhr){
     *    //xhr 为 xmlHttpRequest
     * });
     * @example
     * <caption>
     *     对Example add的数据字典进行赋值,如下:
     * </caption>
     * var dict = new this.Dict("bulletinDictionary");
     *
     * dict.set( "archiveOptions", [ { text : "是" }, { text : "否" } ]);
     *      //数据字典的值变为
     *  {
     *     "category": [
     *         {
     *             "enable": true,
     *             "sequence": 1.0,
     *             "text": "公司公告",
     *             "value": "company"
     *         },
     *         {
     *             "enable": "false",
     *             "sequence": 2.0,
     *             "text": "部门公告",
     *             "value": "department"
     *         },
     *         {
     *             "sequence" : 3.0,
     *             "text": "系统公告",
     *             "value": "system"
     *         }
     *
     *     ],
     *     "archiveOptions" : [ { text : "是" }, { text : "否" } ]
     * }
     *
     * dict.set( "category.2", { text : "县级公告", value : "county" }, function(data){
     *     //data 形如
     *     //{
     *     //    "id": "80ed5f60-500f-4358-8bbc-b7e81f77aa39" //id为数据字典ID
     *     //}
     *  }, function(xhr){
     *     //xhr 为 xmlHttpRequest
     *  });
     *
     *   /数据字典的值变为
     *  {
     *     "category": [
     *         {
     *             "enable": true,
     *             "sequence": 1.0,
     *             "text": "公司公告",
     *             "value": "company"
     *         },
     *         {
     *             "enable": "false",
     *             "sequence": 2.0,
     *             "text": "部门公告",
     *             "value": "department"
     *         },
     *         {
     *             "text": "县级公告",
     *             "value": "county"
     *         }
     *     ],
     *     "archiveOptions" : [ { text : "是" }, { text : "否" } ]
     * }
     *
     * dict.set( "category.1.sequence", 3 );
     * dict.set( "category.2.sequence", 2 );
     *      //数据字典的值变为
     *      {
     *     "category": [
     *         {
     *             "enable": true,
     *             "sequence": 1.0,
     *             "text": "公司公告",
     *             "value": "company"
     *         },
     *         {
     *             "enable": "false",
     *             "sequence": 3.0,
     *             "text": "部门公告",
     *             "value": "department"
     *         },
     *         {
     *             "sequence": 2.0,
     *             "text": "县级公告",
     *             "value": "county"
     *         }
     *     ],
     *     "archiveOptions" : [ { text : "是" }, { text : "否" } ]
     * }
     * @example
     * <caption>
     *     下面是错误的赋值:
     * </caption>
     * dict.set( "category_1", { text : "公司公告" } ); //出错,因为category_1在数据字典中不存在
     */

    /**
     * 根据路径删除数据字典的数据。
     * @method delete
     * @methodOf module:Dict
     * @instance
     * @param {String} path 数据字典中的数据路径,允许使用中文。当路径为多级时,用点号(.)分隔。如果数据路径不存在,则报错。如果删除数组中的某一项,只能删除最后一项。
     * @param {Function} [success] 删除数据成功时的回调函数。
     * @param {Function} [failure] 删除数据错误时的回调函数。
     * @o2syntax
     * dict.delete( path, success, failure )
     * @example
     * var dict = new this.Dict("bulletinDictionary");
     *
     * dict.delete( "category", function(){
     * }, function(xhr){
     *    //xhr 为 xmlHttpRequest
     * });
     * @example
     * <caption>
     *     对Example set的数据字典进行赋值,如下:
     * </caption>
     * var dict = new this.Dict("bulletinDictionary");
     *
     * dict.delete( "archiveOptions");
     * //数据字典的值变为
     * {
     *    "category": [
     *        {
     *            "enable": true,
     *            "sequence": 1.0,
     *            "text": "公司公告",
     *     *            "value": "company"
     *        },
     *        {
     *            "enable": "false",
     *            "sequence": 3.0,
     *            "text": "部门公告",
     *            "value": "department"
     *        },
     *        {
     *            "sequence": 2.0,
     *            "text": "县级公告",
     *            "value": "county"
     *        }
     *    ]
     * }
     *
     * dict.delete( "category.2.sequence", function(data){
     *    //data 形如
     *    //{
     *    //    "id": "80ed5f60-500f-4358-8bbc-b7e81f77aa39" //id为数据字典ID
     *    //}
     * }, function(xhr){
     *    //xhr 为 xmlHttpRequest
     * });
     * //数据字典的值变为
     * {
     *    "category": [
     *        {
     *            "enable": true,
     *            "sequence": 1.0,
     *            "text": "公司公告",
     *            "value": "company"
     *        },
     *        {
     *            "enable": "false",
     *            "sequence": 3.0,
     *            "text": "部门公告",
     *            "value": "department"
     *        },
     *        {
     *            "text": "县级公告",
     *            "value": "county"
     *        }
     *    ]
     * }
     *
     * dict.delete( "category.2");
     * //category是数组,只能删除最后一项也就是下标2
     * //数据字典的值变为
     * {
     *    "category": [
     *        {
     *            "enable": true,
     *            "sequence": 1.0,
     *            "text": "公司公告",
     *            "value": "company"
     *        },
     *        {
     *            "enable": "false",
     *            "sequence": 3.0,
     *            "text": "部门公告",
     *            "value": "department"
     *        }
     *    ]
     * }
     * @example
     * <caption>
     *     下面是错误的删除:
     * </caption>
     * dict.delete( "category_1" ); //出错,因为category_1在数据字典中不存在
     */

    this.Dict = MWF.xScript.createDict();
    //org
    var orgActions = null;
    var getOrgActions = function () {
        if (!orgActions) {
            MWF.require("MWF.xScript.Actions.UnitActions", null, false);
            orgActions = new MWF.xScript.Actions.UnitActions();
        }
    };
    var getNameFlag = function (name) {
        var t = typeOf(name);
        if (t === "array") {
            var v = [];
            name.each(function (id) {
                v.push((typeOf(id) === "object") ? (id.distinguishedName || id.id || id.unique || id.name) : id);
            });
            return v;
        } else {
            return [(t === "object") ? (name.distinguishedName || name.id || name.unique || name.name) : name];
        }
    };

    /**
     * 您可以通过this.org获取组织中的人员、人员属性、组织、组织属性、身份、群组和角色。
     * @module org
     * @o2cn 组织查询
     * @o2category web
     * @o2ordernumber 100
     * @o2syntax
     * //您可以在流程表单、内容管理表单、门户页面、视图和查询视图中,通过this来获取当前实例的org对象,如下:
     * var org = this.org;
     */
    this.org = {
        //身份**********
        //获取身份
        /**
         根据身份标识获取对应的身份对象或数组
         * @method getIdentity
         * @o2membercategory identity
         * @methodOf module:org
         * @static
         * @param {IdentityFlag|IdentityFlag[]} name - 身份的distinguishedName、id、unique属性值,身份对象,或上述属性值和对象的数组。
         * @param {(Boolean|Function)} [asyncOrCallback] 当参数为boolean,表示是否异步执行,默认为false。当参数为function,表示回调方法。
         * @return {Promise|IdentityData|IdentityData[]} 当async为true时,返回
         * {@link https://developer.mozilla.org/zh-CN/docs/Web/JavaScript/Reference/Global_Objects/Promise|Promise}。
         * 否则返回身份,单个是Object,多个是Array。
         * @o2ActionOut x_organization_assemble_express.IdentityAction.listObject|example=Identity|ignoreNoDescr=true|ignoreProps=[woUnitDutyList,woUnit,woGroupList]
         * @o2syntax
         * //同步执行,返回身份,单个是对象,多个是数组。
         * var identityList = this.org.getIdentity( name );
         *
         * //异步执行,返回Promise对象
         * var promise = this.org.getIdentity( name, true);
         * promise.then(function(identityList){
         *     //identityList 为返回的身份,单个是对象,多个是数组。
         * })
         *
         * //异步执行,在回调方法中获取身份
         * this.org.getIdentity( name, function(identityList){
         *     //identityList 为返回的身份,单个是对象,多个是数组。
         * })
         */
        getIdentity: function(name, async){
            getOrgActions();
            var data = {"identityList":getNameFlag(name)};
            var v = null;
            var cb = function(json){
                v = json.data;
                v = (v && v.length===1) ? v[0] : v;
                if (async && o2.typeOf(async)=="function") return async(v);
                return v;
            };

            var promise = orgActions.listIdentity(data, cb, null, !!async);
            return (!!async) ? promise : v;
        },
        //列出人员的身份
        /**
         * 根据人员标识获取对应的身份对象数组。
         * @method listIdentityWithPerson
         * @o2membercategory identity
         * @methodOf module:org
         * @static
         * @param {PersonFlag|PersonFlag[]} name - 人员的distinguishedName、id、unique属性值,人员对象,或上述属性值和对象的数组。
         * @param {(Boolean|Function)} [asyncOrCallback] 当参数为boolean,表示是否异步执行,默认为false。当参数为function,表示回调方法。
         * @param {(Boolean)} [findCN] 是否需要额外查找中文名称(如张三),默认false。如果为true,除匹配unique和distingiushedName外,还会在身份的第一段中查找所有匹配到的身份(精确匹配)。
         * @return {Promise|IdentityData[]} 当async为true时,返回
         * {@link https://developer.mozilla.org/zh-CN/docs/Web/JavaScript/Reference/Global_Objects/Promise|Promise}。
         * 否则返回身份对象数组。
         * @o2ActionOut x_organization_assemble_express.IdentityAction.listWithPersonObject|example=Identity
         * @o2syntax
         * //同步执行,返回身份对象数组。
         * var identityList = this.org.listIdentityWithPerson( person );
         *
         * //异步执行,返回Promise对象
         * var promise = this.org.listIdentityWithPerson( person, true);
         * promise.then(function(identityList){
         *     //identityList 返回的身份对象数组。
         * })
         *
         * //异步执行,在回调方法中获取
         * this.org.listIdentityWithPerson( person, function(identityList){
         *     //identityList 返回的身份对象数组。
         * })
         */
        listIdentityWithPerson: function(name, async, findCN){
            getOrgActions();
            var data = {"personList":getNameFlag(name)};
            if( o2.typeOf(findCN) === "boolean"){
                data.useNameFind = findCN;
            }
            var v = null;
            var cb = function(json){
                v = json.data;
                if (async && o2.typeOf(async)=="function") return async(v);
                return v;
            };

            var promise = orgActions.listIdentityWithPerson(data, cb, null, !!async);
            return (!!async) ? promise : v;
        },
        //查询组织成员身份--返回身份的对象数组
        //nested  布尔  true嵌套的所有成员;false直接成员;默认false;
        /**
         * 根据组织标识获取对应的身份对象数组:identity对象数组。
         * @method listIdentityWithUnit
         * @o2membercategory identity
         * @methodOf module:org
         * @static
         * @param {UnitFlag|UnitFlag[]} name - 组织的distinguishedName、id、unique属性值,组织对象,或上述属性值和对象的数组。
         * @param {Boolean} [nested] true嵌套的所有身份成员;false直接身份成员;默认false。
         * @param {(Boolean|Function)} [asyncOrCallback] 当参数为boolean,表示是否异步执行,默认为false。当参数为function,表示回调方法。
         * @return {Promise|IdentityData[]} 当async为true时,返回
         * {@link https://developer.mozilla.org/zh-CN/docs/Web/JavaScript/Reference/Global_Objects/Promise|Promise}。
         * 否则返回身份对象数组。
         * @o2ActionOut x_organization_assemble_express.IdentityAction.listWithUnitSubNestedObject|example=Identity
         * @o2syntax
         * //同步执行,返回直接组织身份对象数组。
         * var identityList = this.org.listIdentityWithUnit( unit );
         *
         *
         * //同步执行,返回嵌套组织身份对象数组。
         * var identityList = this.org.listIdentityWithUnit( unit, true );
         *
         * //异步执行,返回Promise对象
         * var promise = this.org.listIdentityWithUnit( unit, false, true);
         * promise.then(function(identityList){
         *     //identityList 返回直接组织身份对象数组。
         * })
         *
         * //异步执行,在回调方法中获取
         * this.org.listIdentityWithUnit( unit, false, function(identityList){
         *     //identityList 返回直接组织身份对象数组。
         * })
         */
        listIdentityWithUnit: function(name, nested, async){
            getOrgActions();
            var data = {"unitList": getNameFlag(name)};
            var v = null;

            // var cb = function(json){
            //     v = json.data;
            //     if (async && o2.typeOf(async)=="function") return async(v);
            //     return v;
            // }.ag().catch(function(json){ return json; });

            var cb = function(json){
                v = json.data;
                if (async && o2.typeOf(async)=="function") return async(v);
                return v;
            };

            var method = (nested) ? "listIdentityWithUnitNested" : "listIdentityWithUnitDirect";
            var promise = orgActions[method](data, cb, null, !!async);
            promise.name = "org";

            //
            // if (nested){
            //     orgActions.listIdentityWithUnitNested(data, cb, null, !!async);
            // }else{
            //     orgActions.listIdentityWithUnitDirect(data, cb, null, !!async);
            // }
            return (!!async) ? promise : v;
        },

        //组织**********
        //获取组织
        /**
         根据组织标识获取对应的组织:unit对象或数组
         * @method getUnit
         * @o2membercategory unit
         * @methodOf module:org
         * @static
         * @param {UnitFlag|UnitFlag[]} name - 组织的distinguishedName、id、unique属性值,组织对象,或上述属性值和对象的数组。
         * @param {(Boolean|Function)} [asyncOrCallback] 当参数为boolean,表示是否异步执行,默认为false。当参数为function,表示回调方法。
         * @param {(Boolean)} [findCN] 是否需要额外查找中文名称(如综合部),默认false。如果为true,除匹配unique和distingiushedName外,还会在名称的第一段中查找所有匹配到的部门(精确匹配)。
         * @return {Promise|UnitData|UnitData[]} 当async为true时,返回
         * {@link https://developer.mozilla.org/zh-CN/docs/Web/JavaScript/Reference/Global_Objects/Promise|Promise}。
         * 否则返回组织,单个是Object,多个是Array。
         * @o2ActionOut x_organization_assemble_express.UnitAction.listObject|example=Unit
         * @o2syntax
         * //同步执行,返回组织,单个是对象,多个是数组。
         * var unitList = this.org.getUnit( name );
         *
         * //异步执行,返回Promise对象
         * var promise = this.org.getUnit( name, true);
         * promise.then(function(unitList){
         *     //unitList 为返回的组织,单个是对象,多个是数组。
         * })
         *
         * //异步执行,在回调方法中获取组织
         * this.org.getUnit( name, function(unitList){
         *     //unitList 为返回的组织,单个是对象,多个是数组。
         * })
         */
        getUnit: function(name, async, findCN){
            getOrgActions();
            var data = {"unitList":getNameFlag(name)};
            if( o2.typeOf(findCN) === "boolean"){
                data.useNameFind = findCN;
            }
            var v = null;
            var cb = function(json){
                v = json.data;
                v = (v && v.length===1) ? v[0] : v;
                if (async && o2.typeOf(async)=="function") return async(v);
                return v;
            };

            var promise = orgActions.listUnit(data, cb, null, !!async);
            return (!!async) ? promise : v;
        },
        //查询组织的下级--返回组织的对象数组
        //nested  布尔  true嵌套下级;false直接下级;默认false;
        /**
         根据组织标识获取下级组织的对象数组:unit对象数组。
         * @method listSubUnit
         * @o2membercategory unit
         * @methodOf module:org
         * @static
         * @param {UnitFlag|UnitFlag[]} name - 组织的distinguishedName、id、unique属性值,组织对象,或上述属性值和对象的数组。
         * @param {Boolean} [nested]  true嵌套的所有下级组织;false直接下级组织;默认false。
         * @param {(Boolean|Function)} [asyncOrCallback] 当参数为boolean,表示是否异步执行,默认为false。当参数为function,表示回调方法。
         * @return {Promise|UnitData[]} 当async为true时,返回
         * {@link https://developer.mozilla.org/zh-CN/docs/Web/JavaScript/Reference/Global_Objects/Promise|Promise}。
         * 否则返回组织数组。
         * @o2ActionOut x_organization_assemble_express.UnitAction.listWithUnitSubNestedObject|example=Unit
         * @o2syntax
         * //同步执行,返回嵌套下级组织数组。
         * var unitList = this.org.listSubUnit( name, true );
         *
         * //异步执行,返回Promise对象
         * var promise = this.org.listSubUnit( name, false, true);
         * promise.then(function(unitList){
         *     //unitList 为返回的直接下级组织数组。
         * })
         *
         * //异步执行,在回调方法中获取
         * this.org.listSubUnit( name, true, function(unitList){
         *     //unitList 为返回嵌套下级组织数组。
         * })
         */
        listSubUnit: function(name, nested, async){
            getOrgActions();
            var data = {"unitList": getNameFlag(name)};
            var v = null;
            var cb = function(json){
                v = json.data;
                if (async && o2.typeOf(async)=="function") return async(v);
                return v;
            };

            var promise;
            if (nested){
                promise = orgActions.listUnitSubNested(data, cb, null, !!async);
            }else{
                promise = orgActions.listUnitSubDirect(data, cb, null, !!async);
            }
            return (!!async) ? promise : v;
        },
        //查询组织的上级--返回组织的对象数组
        //nested  布尔  true嵌套上级;false直接上级;默认false;
        //async 布尔 true异步请求
        /**
         根据组织标识批量获取上级组织的对象数组:unit对象数组。
         * @method listSupUnit
         * @o2membercategory unit
         * @methodOf module:org
         * @static
         * @param {UnitFlag|UnitFlag[]} name - 组织的distinguishedName、id、unique属性值,组织对象,或上述属性值和对象的数组。
         * @param {Boolean} [nested]  true嵌套的所有上级组织;false直接上级组织;默认false。
         * @param {(Boolean|Function)} [asyncOrCallback] 当参数为boolean,表示是否异步执行,默认为false。当参数为function,表示回调方法。
         * @return {Promise|UnitData[]} 当async为true时,返回
         * {@link https://developer.mozilla.org/zh-CN/docs/Web/JavaScript/Reference/Global_Objects/Promise|Promise}。
         * 否则返回组织数组。
         * @o2ActionOut x_organization_assemble_express.UnitAction.listWithUnitSupNestedObject|example=Unit
         * @o2syntax
         * //同步执行,返回嵌套上级组织数组。
         * var unitList = this.org.listSupUnit( name, true );
         *
         * //异步执行,返回Promise对象
         * var promise = this.org.listSupUnit( name, false, true);
         * promise.then(function(unitList){
         *     //unitList 为返回的直接上级组织数组。
         * })
         *
         * //异步执行,在回调方法中获取
         * this.org.listSupUnit( name, true, function(unitList){
         *     //unitList 为返回嵌套上级组织数组。
         * })
         */
        listSupUnit: function(name, nested, async){
            getOrgActions();
            var data = {"unitList": getNameFlag(name)};

            var v = null;
            var cb = function(json){
                v = json.data;
                if (async && o2.typeOf(async)=="function") return async(v);
                return v;
            };

            var promise;
            if (nested){
                promise = orgActions.listUnitSupNested(data, cb, null, !!async);
            }else{
                promise = orgActions.listUnitSupDirect(data, cb, null, !!async);
            }
            return (!!async) ? promise : v;

            // if (callback){
            //     if (nested){
            //         orgActions.listUnitSupNested(data, function(json){v = json.data; o2.runCallback(callback, "success", [v], this);});
            //     }else{
            //         orgActions.listUnitSupDirect(data, function(json){v = json.data; o2.runCallback(callback, "success", [v], this);});
            //     }
            // }else{
            //     var v = null;
            //     if (nested){
            //         orgActions.listUnitSupNested(data, function(json){v = json.data;}, null, false);
            //     }else{
            //         orgActions.listUnitSupDirect(data, function(json){v = json.data;}, null, false);
            //     }
            //     return v;
            // }
        },


        /**
         根据组织标识和指定层级批量获取上级组织的对象数组(包括传入的数组):unit对象数组。
         * @method listSupUnitWithLevel
         * @o2membercategory unit
         * @methodOf module:org
         * @static
         * @param {UnitFlag|UnitFlag[]} name - 组织的distinguishedName、id、unique属性值,组织对象,或上述属性值和对象的数组。
         * @param {Number} level  指定的层级
         * @param {(Boolean|Function)} [asyncOrCallback] 当参数为boolean,表示是否异步执行,默认为false。当参数为function,表示回调方法。
         * @return {Promise|UnitData[]} 当async为true时,返回
         * {@link https://developer.mozilla.org/zh-CN/docs/Web/JavaScript/Reference/Global_Objects/Promise|Promise}。
         * 否则返回组织数组。
         * @o2ActionOut x_organization_assemble_express.UnitAction.listWithUnitSupNestedObject|example=Unit
         * @o2syntax
         * //同步执行,返回嵌套上级组织数组。
         * var unitList = this.org.listSupUnitWithLevel( name, level );
         *
         * //异步执行,返回Promise对象
         * var promise = this.org.listSupUnitWithLevel( name, level, true);
         * promise.then(function(unitList){
         *     //unitList 为返回的直接上级组织数组。
         * })
         *
         * //异步执行,在回调方法中获取
         * this.org.listSupUnitWithLevel( name, level, function(unitList){
         *     //unitList 为返回嵌套上级组织数组。
         * })
         */
        listSupUnitWithLevel: function(name, level, async){
            var v;
            var supUnitList = this.listSupUnit( name, true, !!async);
            var unitList = this.getUnit( name, false, !!async );

            var cb = function(sups, units){
                v = [].concat(
                    sups,
                    typeOf( units ) === "object" ? [units] : units
                ).filter(function (u){
                    return u.level === level;
                });
                if (async && o2.typeOf(async)==="function") return async(v);
                return v;
            };

            if( typeof supUnitList.then === 'function' ){
                return Promise.all([supUnitList, unitList]).then(function( result){
                    return cb(result[0], result[1]);
                });
            }else{
                return cb(supUnitList, unitList);
            }
        },


        /**
         根据组织标识和指定组织类型批量获取上级组织的对象数组(包括传入的数组):unit对象数组。
         * @method listSupUnitWithType
         * @o2membercategory unit
         * @methodOf module:org
         * @static
         * @param {UnitFlag|UnitFlag[]} name - 组织的distinguishedName、id、unique属性值,组织对象,或上述属性值和对象的数组。
         * @param {String} type  指定的组织类型
         * @param {(Boolean|Function)} [asyncOrCallback] 当参数为boolean,表示是否异步执行,默认为false。当参数为function,表示回调方法。
         * @return {Promise|UnitData[]} 当async为true时,返回
         * {@link https://developer.mozilla.org/zh-CN/docs/Web/JavaScript/Reference/Global_Objects/Promise|Promise}。
         * 否则返回组织数组。
         * @o2ActionOut x_organization_assemble_express.UnitAction.listWithUnitSupNestedObject|example=Unit
         * @o2syntax
         * //同步执行,返回嵌套上级组织数组。
         * var unitList = this.org.listSupUnitWithType( name, level );
         *
         * //异步执行,返回Promise对象
         * var promise = this.org.listSupUnitWithType( name, level, true);
         * promise.then(function(unitList){
         *     //unitList 为返回的直接上级组织数组。
         * })
         *
         * //异步执行,在回调方法中获取
         * this.org.listSupUnitWithType( name, level, function(unitList){
         *     //unitList 为返回嵌套上级组织数组。
         * })
         */
        listSupUnitWithType: function(name, type, async){
            var v;
            var supUnitList = this.listSupUnit( name, true, !!async);
            var unitList = this.getUnit( name, false, !!async );

            var cb = function(sups, units){
                v = [].concat(
                    sups,
                    typeOf( units ) === "object" ? [units] : units
                ).filter(function (u){
                    return (u.typeList || []).contains( type );
                });
                if (async && o2.typeOf(async)==="function") return async(v);
                return v;
            };

            if( typeof supUnitList.then === 'function' ){
                return Promise.all([supUnitList, unitList]).then(function( result){
                    return cb(result[0], result[1]);
                });
            }else{
                return cb(supUnitList, unitList);
            }
        },



        //根据个人身份获取组织
        //flag 数字    表示获取第几层的组织
        //     字符串  表示获取指定类型的组织
        //     空     表示获取直接所在的组织
        /**
         根据个人身份获取组织:unit对象或数组。
         * @method getUnitByIdentity
         * @o2membercategory unit
         * @methodOf module:org
         * @static
         * @param {IdentityFlag} name - 身份的distinguishedName、id、unique属性值,身份对象。
         * @param {String|Number} [flag]  当值为数字的时候, 表示获取第几层的组织。<br/> 当值为字符串的时候,表示获取指定类型的组织。<br/> 当值为空的时候,表示获取直接所在组织。
         * @param {(Boolean|Function)} [asyncOrCallback] 当参数为boolean,表示是否异步执行,默认为false。当参数为function,表示回调方法。
         * @return {Promise|UnitData|UnitData[]} 当async为true时,返回
         * {@link https://developer.mozilla.org/zh-CN/docs/Web/JavaScript/Reference/Global_Objects/Promise|Promise}。
         * 否则返回对应组织,单个为对象,多个为数组。
         * @o2ActionOut x_organization_assemble_express.UnitAction.getWithIdentityWithLevelObject|example=Unit
         * @o2syntax
         * //同步执行,返回直接所在组织,单个为对象,多个为数组。
         * var unitList = this.org.getUnitByIdentity( name );
         *
         * //同步执行,返回第一层组织,单个为对象,多个为数组。
         * var unitList = this.org.getUnitByIdentity( name, 1 );
         *
         * * //同步执行,返回类型为company的组织,单个为对象,多个为数组。
         * var unitList = this.org.getUnitByIdentity( name, "company" );
         *
         * //异步执行,返回Promise对象
         * var promise = this.org.getUnitByIdentity( name, null, true);
         * promise.then(function(unitList){
         *     //unitList 返回直接所在组织,单个为对象,多个为数组。
         * })
         *
         * //异步执行,在回调方法中获取
         * this.org.getUnitByIdentity( name, 1, function(unitList){
         *     //unitList 返回第一层组织,单个为对象,多个为数组。
         * })
         */
        getUnitByIdentity: function(name, flag, async){
            getOrgActions();
            var getUnitMethod = "current";
            var v;
            if (flag){
                if (typeOf(flag)==="string") getUnitMethod = "type";
                if (typeOf(flag)==="number") getUnitMethod = "level";
            }

            var cb;
            var promise;
            switch (getUnitMethod){
                case "current":
                    var data = {"identityList":getNameFlag(name)};

                    // var cb = ((async && o2.typeOf(async)=="function") ? (async.isAG ? async : async.ag()) : null) || function(json){
                    //     v = json.data;  v=(v&&v.length===1) ? v[0] : v; return v;
                    // }.ag().catch(function(json){ return json; });


                    cb = function(json){
                        v = json.data;  v=(v&&v.length===1) ? v[0] : v;
                        if (async && o2.typeOf(async)=="function") return async(v);
                        return v;
                    };


                    promise = orgActions.listUnitWithIdentity(data, cb, null, !!async);
                    break;
                case "type":
                    var data = {"identity":(typeOf(name)==="object") ? (name.distinguishedName || name.id || name.unique || name.name) : name,"type":flag};

                    cb = function(json){
                        v = json.data;
                        if (async && o2.typeOf(async)=="function") return async(v);
                        return v;
                    };

                    // var cb = ((async && o2.typeOf(async)=="function") ? (async.isAG ? async : async.ag()) : null) || function(json){
                    //     v = json.data;  return v;
                    // }.ag().catch(function(json){ return json; });

                    promise = orgActions.getUnitWithIdentityAndType(data, cb, null, !!async);
                    break;
                case "level":
                    var data = {"identity":(typeOf(name)==="object") ? (name.distinguishedName || name.id || name.unique || name.name) : name,"level":flag};

                    cb = function(json){
                        v = json.data;  v=(v&&v.length===1) ? v[0] : v;
                        if (async && o2.typeOf(async)=="function") return async(v);
                        return v;
                    };

                    // var cb = ((async && o2.typeOf(async)=="function") ? (async.isAG ? async : async.ag()) : null) || function(json){
                    //     v = json.data;  return v;
                    // }.ag().catch(function(json){ return json; });

                    promise = orgActions.getUnitWithIdentityAndLevel(data, cb, null, !!async);
                    break;
            }
            return (!!async) ? promise : v;
        },
        //列出身份所在组织的所有上级组织
        /**
         * 批量查询身份所在的组织,并递归查找其上级组织对象.
         * @method listAllSupUnitWithIdentity
         * @o2membercategory unit
         * @methodOf module:org
         * @static
         * @param {IdentityFlag|IdentityFlag[]} name - 身份的distinguishedName、id、unique属性值,身份对象,或上述属性值和对象的数组。
         * @param {(Boolean|Function)} [asyncOrCallback] 当参数为boolean,表示是否异步执行,默认为false。当参数为function,表示回调方法。
         * @return {Promise|UnitData[]} 当async为true时,返回
         * {@link https://developer.mozilla.org/zh-CN/docs/Web/JavaScript/Reference/Global_Objects/Promise|Promise}。
         * 否则返回组织数组。
         * @o2ActionOut x_organization_assemble_express.UnitAction.listWithIdentitySupNestedObject|example=Unit
         * @o2syntax
         * //同步执行,返回组织数组。
         * var unitList = this.org.listAllSupUnitWithIdentity( name );
         *
         * //异步执行,返回Promise对象
         * var promise = this.org.listAllSupUnitWithIdentity( name, true);
         * promise.then(function(unitList){
         *     //unitList 返回组织数组。
         * })
         *
         * //异步执行,在回调方法中获取
         * this.org.listAllSupUnitWithIdentity( name, function(unitList){
         *     //unitList 返回组织数组。
         * })
         */
        listAllSupUnitWithIdentity: function(name, async){
            getOrgActions();
            var data = {"identityList":getNameFlag(name)};
            var v = null;
            var cb = function(json){
                v = json.data;
                if (async && o2.typeOf(async)=="function") return async(v);
                return v;
            };

            var promise = orgActions.listUnitSupNestedWithIdentity(data, cb, null, !!async);
            return (!!async) ? promise : v;
        },
        //获取人员所在的所有组织
        /**
         * 根据个人标识批量获取组织对象成员:Unit对象数组。
         * @method listUnitWithPerson
         * @o2membercategory unit
         * @methodOf module:org
         * @static
         * @param {PersonFlag|PersonFlag[]} name - 人员的distinguishedName、id、unique属性值,人员对象,或上述属性值和对象的数组。
         * @param {(Boolean|Function)} [asyncOrCallback] 当参数为boolean,表示是否异步执行,默认为false。当参数为function,表示回调方法。
         * @return {Promise|UnitData[]} 当async为true时,返回
         * {@link https://developer.mozilla.org/zh-CN/docs/Web/JavaScript/Reference/Global_Objects/Promise|Promise}。
         * 否则返回组织数组。
         * @o2ActionOut x_organization_assemble_express.UnitAction.listWithPersonObject|example=Unit
         * @o2syntax
         * //同步执行,返回组织数组。
         * var unitList = this.org.listUnitWithPerson( name );
         *
         * //异步执行,返回Promise对象
         * var promise = this.org.listUnitWithPerson( name, true);
         * promise.then(function(unitList){
         *     //unitList 返回组织数组。
         * })
         *
         * //异步执行,在回调方法中获取
         * this.org.listUnitWithPerson( name, function(unitList){
         *     //unitList 返回组织数组。
         * })
         */
        listUnitWithPerson: function(name, async){
            getOrgActions();
            var data = {"personList":getNameFlag(name)};
            var v = null;
            var cb = function(json){
                v = json.data;
                if (async && o2.typeOf(async)=="function") return async(v);
                return v;
            };

            var promise = orgActions.listUnitWithPerson(data, cb, null, !!async);
            return (!!async) ? promise : v;
        },
        //列出人员所在组织的所有上级组织
        /**
         * 根据个人标识批量查询所在组织及所有上级组织:Unit对象数组。
         * @method listAllSupUnitWithPerson
         * @o2membercategory unit
         * @methodOf module:org
         * @static
         * @param {PersonFlag|PersonFlag[]} name - 人员的distinguishedName、id、unique属性值,人员对象,或上述属性值和对象的数组。
         * @param {(Boolean|Function)} [asyncOrCallback] 当参数为boolean,表示是否异步执行,默认为false。当参数为function,表示回调方法。
         * @return {Promise|UnitData[]} 当async为true时,返回
         * {@link https://developer.mozilla.org/zh-CN/docs/Web/JavaScript/Reference/Global_Objects/Promise|Promise}。
         * 否则返回个人所在组织及所有上级组织。
         * @o2ActionOut x_organization_assemble_express.UnitAction.listWithPersonSupNestedObject|example=Unit
         * @o2syntax
         * //同步执行,返回组织数组。
         * var unitList = this.org.listAllSupUnitWithPerson( name );
         *
         * //异步执行,返回Promise对象
         * var promise = this.org.listAllSupUnitWithPerson( name, true);
         * promise.then(function(unitList){
         *     //unitList 返回组织数组。
         * })
         *
         * //异步执行,在回调方法中获取
         * this.org.listAllSupUnitWithPerson( name, function(unitList){
         *     //unitList 返回组织数组。
         * })
         */
        listAllSupUnitWithPerson: function(name, async){
            getOrgActions();
            var data = {"personList":getNameFlag(name)};
            var v = null;
            var cb = function(json){
                v = json.data;
                if (async && o2.typeOf(async)=="function") return async(v);
                return v;
            };

            var promise = orgActions.listUnitSupNestedWithPerson(data, cb, null, !!async);
            return (!!async) ? promise : v;
        },
        //根据组织属性,获取所有符合的组织
        /**
         * 根据组织属性,获取所有符合的组织。
         * @method listUnitWithAttribute
         * @o2membercategory unit
         * @methodOf module:org
         * @static
         * @param {String} attributeName 组织属性名称。
         * @param {String} attributeValue 组织属性值。
         * @param {(Boolean|Function)} [asyncOrCallback] 当参数为boolean,表示是否异步执行,默认为false。当参数为function,表示回调方法。
         * @return {Promise|UnitData[]} 当async为true时,返回
         * {@link https://developer.mozilla.org/zh-CN/docs/Web/JavaScript/Reference/Global_Objects/Promise|Promise}。
         * 否则返回组织数组。
         * @o2ActionOut x_organization_assemble_express.UnitAction.listWithUnitAttributeObject|example=Unit
         * @o2syntax
         * //同步执行,返回组织数组。
         * var unitList = this.org.listUnitWithAttribute( attributeName, attributeName );
         *
         * //异步执行,返回Promise对象
         * var promise = this.org.listUnitWithAttribute( attributeName, attributeName, true);
         * promise.then(function(unitList){
         *     //unitList 返回组织数组。
         * })
         *
         * //异步执行,在回调方法中获取
         * this.org.listUnitWithAttribute( attributeName, attributeName, function(unitList){
         *     //unitList 返回组织数组。
         * })
         */
        listUnitWithAttribute: function(name, attribute, async){
            getOrgActions();
            var data = {"name":name,"attribute":attribute};
            var v = null;
            var cb = function(json){
                v = json.data;
                if (async && o2.typeOf(async)=="function") return async(v);
                return v;
            };

            promise = orgActions.listUnitWithAttribute(data, cb, null, !!async);
            return (!!async) ? promise : v;
        },
        //根据组织职务,获取所有符合的组织
        /**
         * 根据组织职务,获取所有符合的组织。
         * @method listUnitWithDuty
         * @o2membercategory unit
         * @methodOf module:org
         * @static
         * @param {String} dutyName 组织职务名称。
         * @param {IdentityFlag} identity 身份的distinguishedName、id、unique属性值,身份对象。
         * @param {(Boolean|Function)} [asyncOrCallback] 当参数为boolean,表示是否异步执行,默认为false。当参数为function,表示回调方法。
         * @return {Promise|UnitData[]} 当async为true时,返回
         * {@link https://developer.mozilla.org/zh-CN/docs/Web/JavaScript/Reference/Global_Objects/Promise|Promise}。
         * 否则返回组织数组。
         * @o2ActionOut x_organization_assemble_express.UnitAction.listWithUnitDutyObject|example=Unit
         * @o2syntax
         * //同步执行,返回组织数组。
         * var unitList = this.org.listUnitWithDuty( dutyName, identity );
         *
         * //异步执行,返回Promise对象
         * var promise = this.org.listUnitWithDuty( dutyName, identity, true);
         * promise.then(function(unitList){
         *     //unitList 返回组织数组。
         * })
         *
         * //异步执行,在回调方法中获取
         * this.org.listUnitWithDuty( dutyName, identity, function(unitList){
         *     //unitList 返回组织数组。
         * })
         */
        listUnitWithDuty: function(name, id, async){
            getOrgActions();
            var data = {"name":name,"identity":(typeOf(id)==="object") ? (id.distinguishedName || id.id || id.unique || id.name) : id};
            var v = null;
            var cb = function(json){
                v = json.data;
                if (async && o2.typeOf(async)=="function") return async(v);
                return v;
            };

            var promise = orgActions.listUnitWithDuty(data, cb, null, !!async);
            return (!!async) ? promise : v;
        },

        /**
         * 列式所有顶层组织。
         * @method listTopUnit
         * @o2membercategory unit
         * @methodOf module:org
         * @static
         * @param {(Boolean|Function)} [asyncOrCallback] 当参数为boolean,表示是否异步执行,默认为false。当参数为function,表示回调方法。
         * @return {Promise|UnitData[]} 当async为true时,返回
         * {@link https://developer.mozilla.org/zh-CN/docs/Web/JavaScript/Reference/Global_Objects/Promise|Promise}。
         * 否则返回顶层组织数组。
         * @o2ActionOut x_organization_assemble_express.UnitAction.listObject|example=Unit
         * @o2syntax
         * //同步执行,返回顶层组织数组。
         * var unitList = this.org.listTopUnit();
         *
         * //异步执行,返回Promise对象
         * var promise = this.org.listTopUnit(true);
         * promise.then(function(unitList){
         *     //unitList 返回顶层组织数组。
         * })
         *
         * //异步执行,在回调方法中获取
         * this.org.listTopUnit(function(unitList){
         *     //unitList 返回顶层组织数组。
         * })
         */
        listTopUnit: function(async){
            var action = MWF.Actions.get("x_organization_assemble_control");
            var v = null;
            var cb = function(json){
                v = json.data;
                if (async && o2.typeOf(async)=="function") return async(v);
                return v;
            };

            var promise = action.listTopUnit(cb, null, !!async);
            return (!!async) ? promise : v;
        },

        //人员
        //获取人员,附带身份,身份所在的组织,个人所在群组,个人拥有角色.
        /**
         根据人员标识获取对应的人员对象,附带身份,身份所在的组织,个人所在群组,个人拥有角色.
         * @method getPersonData
         * @o2membercategory person
         * @methodOf module:org
         * @static
         * @param {String} name - 人员的distinguishedName、id、unique属性值,人员名称。
         * @param {(Boolean|Function)} [asyncOrCallback] 当参数为boolean,表示是否异步执行,默认为false。当参数为function,表示回调方法。
         * @return {Promise|PersonData} 当async为true时,返回
         * {@link https://developer.mozilla.org/zh-CN/docs/Web/JavaScript/Reference/Global_Objects/Promise|Promise}。
         * 否则返回人员对象。
         * @o2ActionOut x_organization_assemble_express.PersonAction.get|example=PersonData
         * @o2syntax
         * //同步执行,返回人员对象。
         * var person = this.org.getPersonData( name );
         *
         * //异步执行,返回Promise对象
         * var promise = this.org.getPersonData( name, true);
         * promise.then(function(person){
         *     //personList 为返回的人员对象。
         * })
         *
         * //异步执行,在回调方法中获取人员
         * this.org.getPersonData( name, function(person){
         *     //personList 为返回的人员对象。
         * })
         */
        getPersonData: function(name, async){
            getOrgActions();
            var v = null;
            var cb = function(json){
                v = json.data;
                if (async && o2.typeOf(async)=="function") return async(v);
                return v;
            };
            var promise = orgActions.getPerson(null, cb, null, !!async, {"flag": name});
            return (!!async) ? promise : v;
        },
        //获取人员--返回人员的对象数组
        /**
         根据人员标识获取对应的人员对象或数组:person对象或数组
         * @method getPerson
         * @o2membercategory person
         * @methodOf module:org
         * @static
         * @param {PersonFlag|PersonFlag[]} name - 人员的distinguishedName、id、unique属性值,人员对象,或上述属性值和对象的数组。
         * @param {(Boolean|Function)} [asyncOrCallback] 当参数为boolean,表示是否异步执行,默认为false。当参数为function,表示回调方法。
         * @param {(Boolean)} [findCN] 是否需要额外查找中文名称(如张三),默认false。如果为true,除匹配unique和distingiushedName外,还会在名称的第一段中查找所有匹配到的人(精确匹配)。
         * @return {Promise|PersonData|PersonData[]} 当async为true时,返回
         * {@link https://developer.mozilla.org/zh-CN/docs/Web/JavaScript/Reference/Global_Objects/Promise|Promise}。
         * 否则返回人员,单个是Object,多个是Array。
         * @o2ActionOut x_organization_assemble_express.PersonAction.listObject|example=Person
         * @o2syntax
         * //同步执行,返回人员,单个是对象,多个是数组。
         * var personList = this.org.getPerson( name );
         *
         * //异步执行,返回Promise对象
         * var promise = this.org.getPerson( name, true);
         * promise.then(function(personList){
         *     //personList 为返回的人员,单个是对象,多个是数组。
         * })
         *
         * //异步执行,在回调方法中获取人员
         * this.org.getPerson( name, function(personList){
         *     //personList 为返回的人员,单个是对象,多个是数组。
         * })
         */
        getPerson: function(name, async, findCN){
            getOrgActions();
            var data = {"personList": getNameFlag(name)};
            if( o2.typeOf(findCN) === "boolean"){
                data.useNameFind = findCN;
            }
            var v = null;
            var cb = function(json){
                v = json.data;
                v = (v && v.length===1) ? v[0] : v;
                if (async && o2.typeOf(async)=="function") return async(v);
                return v;
            };

            var promise = orgActions.listPerson(data, cb, null, !!async);
            return (!!async) ? promise : v;
            // var v = null;
            // orgActions.listPerson(data, function(json){v = json.data;}, null, false);
            // return (v && v.length===1) ? v[0] : v;
        },
        //查询下级人员--返回人员的对象数组
        //nested  布尔  true嵌套下级;false直接下级;默认false;
        /**
         根据人员标识获取下级人员的对象数组:person对象数组。该上下级关系被人员的汇报对象值(superior)决定。
         * @method listSubPerson
         * @o2membercategory person
         * @methodOf module:org
         * @static
         * @param {PersonFlag|PersonFlag[]} name - 人员的distinguishedName、id、unique属性值,人员对象,或上述属性值和对象的数组。
         * @param {Boolean} [nested]  true嵌套的所有下级人员;false直接下级人员;默认false。
         * @param {(Boolean|Function)} [asyncOrCallback] 当参数为boolean,表示是否异步执行,默认为false。当参数为function,表示回调方法。
         * @return {Promise|PersonData[]} 当async为true时,返回
         * {@link https://developer.mozilla.org/zh-CN/docs/Web/JavaScript/Reference/Global_Objects/Promise|Promise}。
         * 否则返回人员数组。
         * @o2ActionOut x_organization_assemble_express.PersonAction.listWithPersonSubDirectObject|example=Person
         * @o2syntax
         * //同步执行,返回嵌套下级人员数组。
         * var personList = this.org.listSubPerson( name, true );
         *
         * //异步执行,返回Promise对象
         * var promise = this.org.listSubPerson( name, false, true);
         * promise.then(function(personList){
         *     //personList 为返回的直接下级人员数组。
         * })
         *
         * //异步执行,在回调方法中获取
         * this.org.listSubPerson( name, true, function(personList){
         *     //personList 为返回嵌套下级人员数组。
         * })
         */
        listSubPerson: function(name, nested, async){
            getOrgActions();
            var data = {"personList": getNameFlag(name)};

            var v = null;
            var cb = function(json){
                v = json.data;
                if (async && o2.typeOf(async)=="function") return async(v);
                return v;
            };

            var promise;
            if (nested){
                promise = orgActions.listPersonSubNested(data, cb, null, !!async);
            }else{
                promise = orgActions.listPersonSubDirect(data, cb, null, !!async);
            }
            return (!!async) ? promise : v;
        },
        //查询上级人员--返回人员的对象数组
        //nested  布尔  true嵌套上级;false直接上级;默认false;
        /**
         *根据人员标识获取上级人员的对象数组:person对象数组。该上下级关系被人员的汇报对象值(superior)决定。
         * @method listSupPerson
         * @o2membercategory person
         * @methodOf module:org
         * @static
         * @param {PersonFlag|PersonFlag[]} name - 人员的distinguishedName、id、unique属性值,人员对象,或上述属性值和对象的数组。
         * @param {Boolean} [nested]  true嵌套的所有上级人员;false直接上级人员;默认false。
         * @param {(Boolean|Function)} [asyncOrCallback] 当参数为boolean,表示是否异步执行,默认为false。当参数为function,表示回调方法。
         * @return {Promise|PersonData[]} 当async为true时,返回
         * {@link https://developer.mozilla.org/zh-CN/docs/Web/JavaScript/Reference/Global_Objects/Promise|Promise}。
         * 否则返回人员数组。
         * @o2ActionOut x_organization_assemble_express.PersonAction.listWithPersonSupDirectObject|example=Person
         * @o2syntax
         * //同步执行,返回嵌套上级人员数组。
         * var personList = this.org.listSupPerson( name, true );
         *
         * //异步执行,返回Promise对象
         * var promise = this.org.listSupPerson( name, false, true);
         * promise.then(function(personList){
         *     //personList 为返回的直接上级人员数组。
         * })
         *
         * //异步执行,在回调方法中获取
         * this.org.listSupPerson( name, true, function(personList){
         *     //personList 为返回嵌套上级人员数组。
         * })
         */
        listSupPerson: function(name, nested, async){
            getOrgActions();
            var data = {"personList": getNameFlag(name)};
            var v = null;
            var cb = function(json){
                v = json.data;
                if (async && o2.typeOf(async)=="function") return async(v);
                return v;
            };

            var promise;
            if (nested){
                promise = orgActions.listPersonSupNested(data, cb, null, !!async);
            }else{
                promise = orgActions.listPersonSupDirect(data, cb, null, !!async);
            }
            return (!!async) ? promise : v;
        },
        //获取群组的所有人员--返回人员的对象数组
        /**
         * 根据群组标识获取人员对象成员:person对象数组。
         * @method listPersonWithGroup
         * @o2membercategory person
         * @methodOf module:org
         * @static
         * @param {GroupFlag|GroupFlag[]} name - 群组的distinguishedName、name、id、unique属性值,群组对象,或上述属性值和对象的数组。
         * @param {(Boolean|Function)} [asyncOrCallback] 当参数为boolean,表示是否异步执行,默认为false。当参数为function,表示回调方法。
         * @return {Promise|PersonData[]} 当async为true时,返回
         * {@link https://developer.mozilla.org/zh-CN/docs/Web/JavaScript/Reference/Global_Objects/Promise|Promise}。
         * 否则返回人员对象数组。
         * @o2ActionOut x_organization_assemble_express.PersonAction.listWithGroupObject|example=Person
         * @o2syntax
         * //同步执行,返回人员数组。
         * var personList = this.org.listPersonWithGroup( group );
         *
         * //异步执行,返回Promise对象
         * var promise = this.org.listPersonWithGroup( group, true);
         * promise.then(function(personList){
         *     //personList 为返回的人员数组。
         * })
         *
         * //异步执行,在回调方法中获取
         * this.org.listPersonWithGroup( group, function(personList){
         *     //personList 为返回的人员数组。
         * })
         */
        listPersonWithGroup: function(name, async){
            getOrgActions();
            var data = {"groupList": getNameFlag(name)};

            var v = null;
            var cb = function(json){
                v = json.data;
                if (async && o2.typeOf(async)=="function") return async(v);
                return v;
            };

            var promise = orgActions.listPersonWithGroup(data, cb, null, !!async);
            return (!!async) ? promise : v;
        },
        //获取角色的所有人员--返回人员的对象数组
        /**
         * 根据角色标识获取人员对象数组:person对象数组。
         * @method listPersonWithRole
         * @o2membercategory person
         * @methodOf module:org
         * @static
         * @param {RoleFlag|RoleFlag[]} name - 角色的distinguishedName、name、id、unique属性值,角色对象,或上述属性值和对象的数组。
         * @param {(Boolean|Function)} [asyncOrCallback] 当参数为boolean,表示是否异步执行,默认为false。当参数为function,表示回调方法。
         * @return {Promise|PersonData[]} 当async为true时,返回
         * {@link https://developer.mozilla.org/zh-CN/docs/Web/JavaScript/Reference/Global_Objects/Promise|Promise}。
         * 否则返回人员对象数组。
         * @o2ActionOut x_organization_assemble_express.PersonAction.listWithRoleObject|example=Person
         * @o2syntax
         * //同步执行,返回人员数组。
         * var personList = this.org.listPersonWithRole( role );
         *
         * //异步执行,返回Promise对象
         * var promise = this.org.listPersonWithRole( role, true);
         * promise.then(function(personList){
         *     //personList 为返回的人员数组。
         * })
         *
         * //异步执行,在回调方法中获取
         * this.org.listPersonWithRole( role, function(personList){
         *     //personList 为返回的人员数组。
         * })
         */
        listPersonWithRole: function(name, async){
            getOrgActions();
            var data = {"roleList": getNameFlag(name)};
            var v = null;
            var cb = function(json){
                v = json.data;
                if (async && o2.typeOf(async)=="function") return async(v);
                return v;
            };

            var promise
            promise = orgActions.listPersonWithRole(data, cb, null, !!async);
            return (!!async) ? promise : v;
        },
        //获取身份的所有人员--返回人员的对象数组
        /**
         * 根据身份标识获取人员对象成员:person对象数组。
         * @method listPersonWithIdentity
         * @o2membercategory person
         * @methodOf module:org
         * @static
         * @param {IdentityFlag|IdentityFlag[]} name - 身份的distinguishedName、id、unique属性值,身份对象,或上述属性值和对象的数组。
         * @param {(Boolean|Function)} [asyncOrCallback] 当参数为boolean,表示是否异步执行,默认为false。当参数为function,表示回调方法。
         * @return {Promise|PersonData[]} 当async为true时,返回
         * {@link https://developer.mozilla.org/zh-CN/docs/Web/JavaScript/Reference/Global_Objects/Promise|Promise}。
         * 否则返回人员对象数组。
         * @o2ActionOut x_organization_assemble_express.PersonAction.listWithIdentityObject|example=Person
         * @o2syntax
         * //同步执行,返回人员数组。
         * var personList = this.org.listPersonWithIdentity( identity );
         *
         * //异步执行,返回Promise对象
         * var promise = this.org.listPersonWithIdentity( identity, true);
         * promise.then(function(personList){
         *     //personList 为返回的人员数组。
         * })
         *
         * //异步执行,在回调方法中获取
         * this.org.listPersonWithIdentity( identity, function(personList){
         *     //personList 为返回的人员数组。
         * })
         */
        listPersonWithIdentity: function(name, async){
            getOrgActions();
            var data = {"identityList": getNameFlag(name)};
            var v = null;
            var cb = function(json){
                v = json.data;
                if (async && o2.typeOf(async)=="function") return async(v);
                return v;
            };

            var promise = orgActions.listPersonWithIdentity(data, cb, null, !!async);
            return (!!async) ? promise : v;
        },
        //获取身份的所有人员--返回人员的对象数组或人员对象
        getPersonWithIdentity: function(name, async){
            getOrgActions();
            var data = {"identityList": getNameFlag(name)};
            var v = null;
            var cb = function(json){
                v = json.data;
                v =  (v && v.length===1) ? v[0] : v;
                if (async && o2.typeOf(async)=="function") return async(v);
                return v;
            };

            var promise = orgActions.listPersonWithIdentity(data, cb, null, !!async);
            return (!!async) ? promise : v;
        },
        //查询组织成员的人员--返回人员的对象数组
        //nested  布尔  true嵌套的所有成员;false直接成员;默认false;
        /**
         * 根据组织标识获取人员对象成员:person对象数组。
         * @method listPersonWithUnit
         * @o2membercategory person
         * @methodOf module:org
         * @static
         * @param {UnitFlag|UnitFlag[]} name - 组织的distinguishedName、id、unique属性值,组织对象,或上述属性值和对象的数组。
         * @param {Boolean} [nested] 是否嵌套获取组织以及下级组织的人员,true表示嵌套,flase表示获取直接组织。默认为false
         * @param {(Boolean|Function)} [asyncOrCallback] 当参数为boolean,表示是否异步执行,默认为false。当参数为function,表示回调方法。
         * @return {Promise|PersonData[]} 当async为true时,返回
         * {@link https://developer.mozilla.org/zh-CN/docs/Web/JavaScript/Reference/Global_Objects/Promise|Promise}。
         * 否则返回人员对象数组。
         * @o2ActionOut x_organization_assemble_express.PersonAction.listWithUnitSubDirectObject|example=Person
         * @o2syntax
         * //同步执行,返回组织的直接人员数组。
         * var personList = this.org.listPersonWithUnit( unit );
         *
         * //同步执行,返回组织的以及嵌套下级组织所有的人员数组。
         * var personList = this.org.listPersonWithUnit( unit, true );
         *
         * //异步执行,返回Promise对象
         * var promise = this.org.listPersonWithUnit( unit, false, true);
         * promise.then(function(personList){
         *     //personList 为返回的组织的直接人员数组。
         * })
         *
         * //异步执行,在回调方法中获取
         * this.org.listPersonWithUnit( unit, false, function(personList){
         *     //personList 为返回的群组的直接人员数组。
         * })
         */
        listPersonWithUnit: function(name, nested, async){
            getOrgActions();
            var data = {"unitList": getNameFlag(name)};
            var v = null;
            var cb = function(json){
                v = json.data;
                if (async && o2.typeOf(async)=="function") return async(v);
                return v;
            };

            var promise;
            if (nested){
                promise = orgActions.listPersonWithUnitNested(data, cb, null, !!async);
            }else{
                promise = orgActions.listPersonWithUnitDirect(data, cb, null, !!async);
            }
            return (!!async) ? promise : v;
        },
        //根据属性查询人员--返回人员的对象数组
        //name  string 属性名
        //value  string 属性值
        /**
         * 根据人员属性名称和属性值获取人员对象成员:person对象数组。
         * @method listPersonWithAttribute
         * @o2membercategory person
         * @methodOf module:org
         * @static
         * @param {String} name 人员属性名称。
         * @param {String} value 人员属性值。
         * @param {(Boolean|Function)} [asyncOrCallback] 当参数为boolean,表示是否异步执行,默认为false。当参数为function,表示回调方法。
         * @return {Promise|PersonData[]} 当async为true时,返回
         * {@link https://developer.mozilla.org/zh-CN/docs/Web/JavaScript/Reference/Global_Objects/Promise|Promise}。
         * 否则返回人员对象数组。
         * @o2ActionOut x_organization_assemble_express.PersonAction.listWithPersonAttributeObject|example=Person
         * @o2syntax
         * //同步执行,返回拥有对应属性名和属性值人员数组。
         * var personList = this.org.listPersonWithAttribute( name, value );
         *
         * //异步执行,返回Promise对象
         * var promise = this.org.listPersonWithAttribute( name, value, true);
         * promise.then(function(personList){
         *     //personList 返回拥有对应属性名和属性值人员数组。
         * })
         *
         * //异步执行,在回调方法中获取
         * this.org.listPersonWithAttribute( name, value, function(personList){
         *     //personList 返回拥有对应属性名和属性值人员数组。
         * })
         */
        listPersonWithAttribute: function(name, value, async){
            getOrgActions();
            var data = {"name": name, "attribute": value};
            var v = null;
            var cb = function(json){
                v = json.data;
                if (async && o2.typeOf(async)=="function") return async(v);
                return v;
            };

            var promise = orgActions.listPersonWithAttribute(data, cb, null, !!async);
            return (!!async) ? promise : v;
        },
        //根据属性查询人员--返回人员的全称数组
        //name  string 属性名
        //value  string 属性值
        listPersonNameWithAttribute: function(name, value, async){
            getOrgActions();
            var data = {"name": name, "attribute": value};
            var v = null;
            var cb = function(json){
                v = json.data.personList;
                if (async && o2.typeOf(async)=="function") return async(v);
                return v;
            };

            var promise = orgActions.listPersonWithAttributeValue(data, cb, null, !!async);
            return (!!async) ? promise : v;
        },

        //组织职务***********
        //获取指定的组织职务的身份
        /**
         * 根据职务名称和组织名称获取身份。
         * @method getDuty
         * @o2membercategory duty
         * @methodOf module:org
         * @static
         * @param {String} dutyName 组织职务名称。
         * @param {UnitFlag} unit 组织的distinguishedName、id、unique属性值,组织对象。
         * @param {(Boolean|Function)} [asyncOrCallback] 当参数为boolean,表示是否异步执行,默认为false。当参数为function,表示回调方法。
         * @return {Promise|IdentityData[]} 当async为true时,返回
         * {@link https://developer.mozilla.org/zh-CN/docs/Web/JavaScript/Reference/Global_Objects/Promise|Promise}。
         * 否则返回身份数组。
         * @o2ActionOut x_organization_assemble_express.UnitDutyAction.getWithUnitWithName|example=Identity
         * @o2syntax
         * //同步执行,返回身份数组。
         * var identityList = this.org.getDuty( dutyName, unit );
         *
         * //异步执行,返回Promise对象
         * var promise = this.org.getDuty( dutyName, unit, true);
         * promise.then(function(identityList){
         *     //identityList 返回身份数组。
         * })
         *
         * //异步执行,在回调方法中获取
         * this.org.getDuty( dutyName, unit, function(unitList){
         *     //unitList 返回身份数组。
         * })
         */
        getDuty: function(duty, id, async){
            getOrgActions();
            var data = {"name":duty,"unit":(typeOf(id)==="object") ? (id.distinguishedName || id.id || id.unique || id.name) : id};
            var v = null;

            var cb = function(json){
                v = json.data;
                if (async && o2.typeOf(async)=="function") return async(v);
                return v;
            };

            var promise = orgActions.getDuty(data, cb, null, !!async);
            return (!!async) ? promise : v;
        },
        //获取身份的所有职务名称
        /**
         * 批量获取身份的所有职务名称。
         * @method listDutyNameWithIdentity
         * @o2membercategory duty
         * @methodOf module:org
         * @static
         * @param {IdentityFlag|IdentityFlag[]} identity - 身份的distinguishedName、id、unique属性值,身份对象,或上述属性值和对象的数组。
         * @param {(Boolean|Function)} [asyncOrCallback] 当参数为boolean,表示是否异步执行,默认为false。当参数为function,表示回调方法。
         * @return {Promise|String[]} 当async为true时,返回
         * {@link https://developer.mozilla.org/zh-CN/docs/Web/JavaScript/Reference/Global_Objects/Promise|Promise}。
         * 否则返回职务名称数组。
         * @o2syntax
         * //同步执行,返回职务名称数组。
         * var dutyNameList = this.org.listDutyNameWithIdentity( identity );
         *
         * //异步执行,返回Promise对象
         * var promise = this.org.listDutyNameWithIdentity( identity, true);
         * promise.then(function(dutyNameList){
         *     //dutyNameList 返回职务名称数组。
         * })
         *
         * //异步执行,在回调方法中获取
         * this.org.listDutyNameWithIdentity( identity, function(dutyNameList){
         *     //dutyNameList 返回职务名称数组。
         * })
         */
        listDutyNameWithIdentity: function(name, async){
            getOrgActions();
            var data = {"identityList":getNameFlag(name)};
            var v = null;
            var cb = function(json){
                v = json.data.nameList;
                if (async && o2.typeOf(async)=="function") return async(v);
                return v;
            };

            var promise = orgActions.listDutyNameWithIdentity(data, cb, null, !!async);
            return (!!async) ? promise : v;
        },
        //批量获取组织的所有职务名称
        /**
         * 批量获取组织的所有职务名称。
         * @method listDutyNameWithUnit
         * @o2membercategory duty
         * @methodOf module:org
         * @static
         * @param {UnitFlag|UnitFlag[]} unit - 组织的distinguishedName、id、unique属性值,组织对象,或上述属性值和对象的数组。
         * @param {(Boolean|Function)} [asyncOrCallback] 当参数为boolean,表示是否异步执行,默认为false。当参数为function,表示回调方法。
         * @return {Promise|String[]} 当async为true时,返回
         * {@link https://developer.mozilla.org/zh-CN/docs/Web/JavaScript/Reference/Global_Objects/Promise|Promise}。
         * 否则返回职务名称数组。
         * @o2syntax
         * //同步执行,返回职务名称数组。
         * var dutyNameList = this.org.listDutyNameWithUnit( unit );
         *
         * //异步执行,返回Promise对象
         * var promise = this.org.listDutyNameWithUnit( unit, true);
         * promise.then(function(dutyNameList){
         *     //dutyNameList 返回职务名称数组。
         * })
         *
         * //异步执行,在回调方法中获取
         * this.org.listDutyNameWithUnit( unit, function(dutyNameList){
         *     //dutyNameList 返回职务名称数组。
         * })
         */
        listDutyNameWithUnit: function(name, async){
            getOrgActions();
            var data = {"unitList":getNameFlag(name)};
            var v = null;
            var cb = function(json){
                v = json.data.nameList;
                if (async && o2.typeOf(async)=="function") return async(v);
                return v;
            };

            var promise = orgActions.listDutyNameWithUnit(data, cb, null, !!async);
            return (!!async) ? promise : v;
        },
        //获取组织的所有职务
        /**
         * 批量获取组织的所有职务。
         * @method listUnitAllDuty
         * @o2membercategory duty
         * @methodOf module:org
         * @static
         * @param {UnitFlag|UnitFlag[]} unit - 组织的distinguishedName、id、unique属性值,组织对象,或上述属性值和对象的数组。
         * @param {(Boolean|Function)} [asyncOrCallback] 当参数为boolean,表示是否异步执行,默认为false。当参数为function,表示回调方法。
         * @return {Promise|Object[]} 当async为true时,返回
         * {@link https://developer.mozilla.org/zh-CN/docs/Web/JavaScript/Reference/Global_Objects/Promise|Promise}。
         * 否则返回职务数组
         * @o2ActionOut x_organization_assemble_express.UnitDutyAction.listWithUnitObject|example=Duty
         * @o2syntax
         * //同步执行,返回职务数组。
         * var dutyList = this.org.listUnitAllDuty( unit );
         *
         * //异步执行,返回Promise对象
         * var promise = this.org.listUnitAllDuty( unit, true);
         * promise.then(function(dutyList){
         *     //dutyList 返回职务数组。
         * })
         *
         * //异步执行,在回调方法中获取
         * this.org.listUnitAllDuty( unit, function(dutyList){
         *     //dutyList 返回职务数组。
         * })
         */
        listUnitAllDuty: function(name, async){
            getOrgActions();
            var data = {"unitList":getNameFlag(name)};
            var v = null;
            var cb = function(json){
                v = json.data;
                if (async && o2.typeOf(async)=="function") return async(v);
                return v;
            };

            var promise = orgActions.listUnitAllDuty(data, cb, null, !!async);
            return (!!async) ? promise : v;
        },

        //群组***************
        //获取群组--返回群组的对象数组
        /**
         根据群组标识获取对应的群组对象或数组:group对象或数组
         * @method getGroup
         * @o2membercategory group
         * @methodOf module:org
         * @static
         * @param {GroupFlag|GroupFlag[]} name - 群组的distinguishedName、name、id、unique属性值,群组对象,或上述属性值和对象的数组。
         * @param {(Boolean|Function)} [asyncOrCallback] 当参数为boolean,表示是否异步执行,默认为false。当参数为function,表示回调方法。
         * @return {Promise|GroupData|GroupData[]} 当async为true时,返回
         * {@link https://developer.mozilla.org/zh-CN/docs/Web/JavaScript/Reference/Global_Objects/Promise|Promise}。
         * 否则返回群组,单个是Object,多个是Array。
         * @o2ActionOut x_organization_assemble_express.GroupAction.listObject|example=Group
         * @o2syntax
         * //同步执行,返回群组,单个是Object,多个是Array。
         * var groupList = this.org.getGroup( name );
         *
         * //异步执行,返回Promise对象
         * var promise = this.org.getGroup( name, true);
         * promise.then(function(groupList){
         *     //groupList 为返回的群组,单个是Object,多个是Array。
         * })
         *
         * //异步执行,在回调方法中获取群组
         * this.org.getGroup( name, function(groupList){
         *     //groupList 为返回的群组,单个是Object,多个是Array。
         * })
         */
        getGroup: function(name, async){
            getOrgActions();
            var data = {"groupList": getNameFlag(name)};

            var v = null;

            var cb = function(json){
                v = json.data;
                v = (v && v.length===1) ? v[0] : v
                if (async && o2.typeOf(async)=="function") return async(v);
                return v;
            };

            var promise = orgActions.listGroup(data, cb, null, !!async);
            return (!!async) ? promise : v;

            // var v = null;
            // orgActions.listGroup(data, function(json){v = json.data;}, null, false);
            // return (v && v.length===1) ? v[0] : v;
        },
        //查询下级群组--返回群组的对象数组
        //nested  布尔  true嵌套下级;false直接下级;默认false;
        /**
         根据群组标识获取下级群组的对象数组:group对象数组。
         * @method listSubGroup
         * @o2membercategory group
         * @methodOf module:org
         * @static
         * @param {GroupFlag|GroupFlag[]} name - 群组的distinguishedName、name、id、unique属性值,群组对象,或上述属性值和对象的数组。
         * @param {Boolean} [nested]  true嵌套的所有下级群组;false直接下级群组;默认false。
         * @param {(Boolean|Function)} [asyncOrCallback] 当参数为boolean,表示是否异步执行,默认为false。当参数为function,表示回调方法。
         * @return {Promise|GroupData[]} 当async为true时,返回
         * {@link https://developer.mozilla.org/zh-CN/docs/Web/JavaScript/Reference/Global_Objects/Promise|Promise}。
         * 否则返回群组数组。
         * @o2ActionOut x_organization_assemble_express.GroupAction.listWithGroupSubDirectObject|example=Group
         * @o2syntax
         * //同步执行,返回嵌套下级群组数组。
         * var groupList = this.org.listSubGroup( name, true );
         *
         * //异步执行,返回Promise对象
         * var promise = this.org.listSubGroup( name, false, true);
         * promise.then(function(groupList){
         *     //groupList 为返回的直接下级群组数组。
         * })
         *
         * //异步执行,在回调方法中获取群组
         * this.org.listSubGroup( name, true, function(groupList){
         *     //groupList 为返回嵌套下级群组数组。
         * })
         */
        listSubGroup: function(name, nested, async){
            getOrgActions();
            var data = {"groupList": getNameFlag(name)};

            var v = null;
            // var cb = ((async && o2.typeOf(async)=="function") ? (async.isAG ? async : async.ag()) : null) || function(json){
            //     v = json.data;
            //     return v;
            // }.ag().catch(function(json){ return json; });
            var cb = function(json){
                v = json.data;
                if (async && o2.typeOf(async)=="function") return async(v);
                return v;
            };

            var promise;
            if (nested){
                promise = orgActions.listSubGroupNested(data, cb, null, !!async);
            }else{
                promise = orgActions.listSubGroupDirect(data, cb, null, !!async);
            }
            return (!!async) ? promise : v;

            // var v = null;
            // if (nested){
            //     orgActions.listSubGroupNested(data, function(json){v = json.data;}, null, false);
            // }else{
            //     orgActions.listSubGroupDirect(data, function(json){v = json.data;}, null, false);
            // }
            // return v;
        },
        //查询上级群组--返回群组的对象数组
        //nested  布尔  true嵌套上级;false直接上级;默认false;
        /**
         根据群组标识获取上级群组的对象数组:group对象数组。
         * @method listSupGroup
         * @o2membercategory group
         * @methodOf module:org
         * @static
         * @param {GroupFlag|GroupFlag[]} name - 群组的distinguishedName、name、id、unique属性值,群组对象,或上述属性值和对象的数组。
         * @param {Boolean} [nested]  true嵌套的所有上级群组;false直接上级群组;默认false。
         * @param {(Boolean|Function)} [asyncOrCallback] 当参数为boolean,表示是否异步执行,默认为false。当参数为function,表示回调方法。
         * @return {Promise|GroupData[]} 当async为true时,返回
         * {@link https://developer.mozilla.org/zh-CN/docs/Web/JavaScript/Reference/Global_Objects/Promise|Promise}。
         * 否则返回群组数组。
         * @o2ActionOut x_organization_assemble_express.GroupAction.listWithGroupSupDirectObject|example=Group
         * @o2syntax
         * //同步执行,返回嵌套上级群组数组。
         * var groupList = this.org.listSupGroup( name, true );
         *
         * //异步执行,返回Promise对象
         * var promise = this.org.listSupGroup( name, false, true);
         * promise.then(function(groupList){
         *     //groupList 为返回的直接上级群组数组。
         * })
         *
         * //异步执行,在回调方法中获取群组
         * this.org.listSupGroup( name, true, function(groupList){
         *     //groupList 为返回嵌套上级群组数组。
         * })
         */
        listSupGroup:function(name, nested, async){
            getOrgActions();
            var data = {"groupList": getNameFlag(name)};

            var v = null;
            var cb = function(json){
                v = json.data;
                if (async && o2.typeOf(async)=="function") return async(v);
                return v;
            };

            var promise
            if (nested){
                var promise = orgActions.listSupGroupNested(data, cb, null, !!async);
            }else{
                var promise = orgActions.listSupGroupDirect(data, cb, null, !!async);
            }
            return (!!async) ? promise : v;
            // var v = null;
            // if (nested){
            //     orgActions.listSupGroupNested(data, function(json){v = json.data;}, null, false);
            // }else{
            //     orgActions.listSupGroupDirect(data, function(json){v = json.data;}, null, false);
            // }
            // return v;
        },
        //人员所在群组(嵌套)--返回群组的对象数组
        /**
         * 根据人员标识获取所有的群组对象数组。如果群组具有群组(group)成员,且群组成员中包含该人员,那么该群组也被返回。
         * @method listGroupWithPerson
         * @o2membercategory group
         * @methodOf module:org
         * @static
         * @param {PersonFlag|PersonFlag[]} name - 人员的distinguishedName、id、unique属性值,人员对象,或上述属性值和对象的数组。
         * @param {(Boolean|Function)} [asyncOrCallback] 当参数为boolean,表示是否异步执行,默认为false。当参数为function,表示回调方法。
         * @return {Promise|GroupData[]} 当async为true时,返回
         * {@link https://developer.mozilla.org/zh-CN/docs/Web/JavaScript/Reference/Global_Objects/Promise|Promise}。
         * 否则返回群组对象数组。
         * @o2ActionOut x_organization_assemble_express.GroupAction.listWithPersonObject|example=Group
         * @o2syntax
         * //同步执行,返回群组数组。
         * var groupList = this.org.listGroupWithPerson( name );
         *
         * //异步执行,返回Promise对象
         * var promise = this.org.listGroupWithPerson( name, true);
         * promise.then(function(groupList){
         *     //groupList 为返回的群组数组。
         * })
         *
         * //异步执行,在回调方法中获取群组
         * this.org.listGroupWithPerson( name, function(groupList){
         *     //groupList 为返回的群组数组。
         * })
         */
        listGroupWithPerson:function(name, async){
            getOrgActions();
            var data = {"personList": getNameFlag(name)};

            var v = null;
            var cb = function(json){
                v = json.data;
                if (async && o2.typeOf(async)=="function") return async(v);
                return v;
            };

            var promise = orgActions.listGroupWithPerson(data, cb, null, !!async);
            return (!!async) ? promise : v;
            // var v = null;
            // orgActions.listGroupWithPerson(data, function(json){v = json.data;}, null, false);
            // return v;
        },
        //身份所在群组(嵌套)--返回群组的对象数组
        /**
         * 根据身份标识获取所有的群组对象数组。如果群组具有群组(group)成员,且群组成员中包含该身份,那么该群组也被返回。
         * @method listGroupWithIdentity
         * @o2membercategory group
         * @methodOf module:org
         * @static
         * @param {IdentityFlag|IdentityFlag[]} identity - 身份的distinguishedName、id、unique属性值,身份对象,或上述属性值和对象的数组。
         * @param {(Boolean|Function)} [asyncOrCallback] 当参数为boolean,表示是否异步执行,默认为false。当参数为function,表示回调方法。
         * @return {Promise|GroupData[]} 当async为true时,返回
         * {@link https://developer.mozilla.org/zh-CN/docs/Web/JavaScript/Reference/Global_Objects/Promise|Promise}。
         * 否则返回群组对象数组。
         * @o2ActionOut x_organization_assemble_express.GroupAction.listWithPersonObject|example=Group
         * @o2syntax
         * //同步执行,返回群组数组。
         * var groupList = this.org.listGroupWithIdentity( identity );
         *
         * //异步执行,返回Promise对象
         * var promise = this.org.listGroupWithIdentity( identity, true);
         * promise.then(function(groupList){
         *     //groupList 为返回的群组数组。
         * })
         *
         * //异步执行,在回调方法中获取群组
         * this.org.listGroupWithPerson( identity, function(groupList){
         *     //groupList 为返回的群组数组。
         * })
         */
        listGroupWithIdentity:function(identity, async){
            getOrgActions();
            var data = {"identityList": getNameFlag(identity)};

            var v = null;
            var cb = function(json){
                v = json.data;
                if (async && o2.typeOf(async)=="function") return async(v);
                return v;
            };

            var promise = orgActions.listGroupWithIdentity(data, cb, null, !!async);
            return (!!async) ? promise : v;
        },

        //角色***************
        //获取角色--返回角色的对象数组
        /**
         * 根据角色标识获取对应的角色对象或数组。
         * @method getRole
         * @o2membercategory role
         * @methodOf module:org
         * @static
         * @param {RoleFlag|RoleFlag[]} name - 角色的distinguishedName、name、id、unique属性值,角色对象;或上述属性值和对象的数组。
         * @param {(Boolean|Function)} [asyncOrCallback] 当参数为boolean,表示是否异步执行,默认为false。当参数为function,表示回调方法。
         * @return {Promise|RoleData|RoleData[]} 当async为true时,返回
         * {@link https://developer.mozilla.org/zh-CN/docs/Web/JavaScript/Reference/Global_Objects/Promise|Promise}。
         * 否则返回角色,单个为Object,多个为Array。
         * @o2ActionOut x_organization_assemble_express.RoleAction.listObject|example=Role
         * @o2syntax
         * //同步执行,返回角色,单个为对象,多个为数组。
         * var roleList = this.org.getRole( name );
         *
         * //异步执行,返回Promise对象
         * var promise = this.org.getRole( name, true);
         * promise.then(function(roleList){
         *     //roleList 为返回的角色,单个为对象,多个为数组。
         * })
         *
         * //异步执行,在回调方法中获取角色,单个为对象,多个为数组
         * this.org.getRole( name, function(roleList){
         *     //roleList 为返回判断结果。
         * })
         */
        getRole: function(name, async){
            getOrgActions();
            var data = {"roleList": getNameFlag(name)};

            var v = null;
            var cb = function(json){
                v = json.data;
                v = (v && v.length===1) ? v[0] : v;
                if (async && o2.typeOf(async)=="function") return async(v);
                return v;
            };

            var promise = orgActions.listRole(data, cb, null, !!async);
            return (!!async) ? promise : v;

            // var v = null;
            // orgActions.listRole(data, function(json){v = json.data;}, null, false);
            // return (v && v.length===1) ? v[0] : v;
        },
        //人员所有角色(嵌套)--返回角色的对象数组
        /**
         * 根据人员标识获取所有的角色对象数组。如果角色具有群组(group)成员,且群组中包含该人员,那么该角色也被返回。
         * @method listRoleWithPerson
         * @o2membercategory role
         * @methodOf module:org
         * @static
         * @param {PersonFlag|PersonFlag[]} name - 人员的distinguishedName、id、unique属性值,人员对象,或上述属性值和对象的数组。
         * @param {(Boolean|Function)} [asyncOrCallback] 当参数为boolean,表示是否异步执行,默认为false。当参数为function,表示回调方法。
         * @return {Promise|RoleData[]} 当async为true时,返回
         * {@link https://developer.mozilla.org/zh-CN/docs/Web/JavaScript/Reference/Global_Objects/Promise|Promise}。
         * 否则返回角色对象数组。
         * @o2ActionOut x_organization_assemble_express.RoleAction.listWithPersonObject|example=Role
         * @o2syntax
         * //同步执行,返回角色数组。
         * var roleList = this.org.listRoleWithPerson( name );
         *
         * //异步执行,返回Promise对象
         * var promise = this.org.listRoleWithPerson( name, true);
         * promise.then(function(roleList){
         *     //roleList 为返回的角色数组。
         * })
         *
         * //异步执行,在回调方法中获取角色
         * this.org.listRoleWithPerson( name, function(roleList){
         *     //roleList 为返回的角色数组。
         * })
         */
        listRoleWithPerson:function(name, async){
            getOrgActions();
            var data = {"personList": getNameFlag(name)};

            var v = null;
            var cb = function(json){
                v = json.data;
                if (async && o2.typeOf(async)=="function") return async(v);
                return v;
            };

            var promise = orgActions.listRoleWithPerson(data, cb, null, !!async);
            return (!!async) ? promise : v;
            // var v = null;
            // orgActions.listRoleWithPerson(data, function(json){v = json.data;}, null, false);
            // return v;
        },

        //人员***************
        //人员是否拥有角色--返回true, false
        /**
         * 人员是否拥有角色。
         * @method personHasRole
         * @o2membercategory role
         * @methodOf module:org
         * @static
         * @param {PersonFlag} name - 人员的distinguishedName、id、unique属性值,人员对象。
         * @param {RoleFlag|RoleFlag[]} roleList - 角色的distinguishedName、name、id、unique属性值,角色对象;或上述属性值和对象的数组。
         * @param {(Boolean|Function)} [asyncOrCallback] 当参数为boolean,表示是否异步执行,默认为false。当参数为function,表示回调方法。
         * @return {Promise|Boolean} 当async为true时,返回
         * {@link https://developer.mozilla.org/zh-CN/docs/Web/JavaScript/Reference/Global_Objects/Promise|Promise}。
         * 否则如果人员拥有角色返回true, 否则返回false。
         * @o2syntax
         * //同步执行,返回判断结果。
         * var flag = this.org.personHasRole( name, roleList );
         *
         * //异步执行,返回Promise对象
         * var promise = this.org.personHasRole( name, roleList, true);
         * promise.then(function(flag){
         *     //flag 为返回判断结果。
         * })
         *
         * //异步执行,在回调方法中获取判断结果
         * this.org.personHasRole( name, roleList, function(flag){
         *     //flag 为返回判断结果。
         * })
         */
        personHasRole: function(name, role, async){
            getOrgActions();
            nameFlag = (typeOf(name)==="object") ? (name.distinguishedName || name.id || name.unique || name.name) : name;
            var data = {"person":nameFlag,"roleList":getNameFlag(role)};

            var v = false;
            var cb = function(json){
                v = json.data.value;
                if (async && o2.typeOf(async)=="function") return async(v);
                return v;
            };

            var promise = orgActions.personHasRole(data, cb, null, !!async);
            return (!!async) ? promise : v;

            // var v = false;
            // orgActions.personHasRole(data, function(json){v = json.data.value;}, null, false);
            // return v;
        },
        //群组是否拥有角色--返回true, false
        /**
         * 群组是否拥有角色。
         * @method groupHasRole
         * @o2membercategory role
         * @methodOf module:org
         * @static
         * @param {GroupFlag} name - 群组的distinguishedName、name、id、unique属性值,群组对象。
         * @param {RoleFlag|RoleFlag[]} roleList - 角色的distinguishedName、name、id、unique属性值,角色对象;或上述属性值和对象的数组。
         * @param {(Boolean|Function)} [asyncOrCallback] 当参数为boolean,表示是否异步执行,默认为false。当参数为function,表示回调方法。
         * @return {Promise|Boolean} 当async为true时,返回
         * {@link https://developer.mozilla.org/zh-CN/docs/Web/JavaScript/Reference/Global_Objects/Promise|Promise}。
         * 否则如果群组拥有角色返回true, 否则返回false。
         * @o2syntax
         * //同步执行,返回判断结果。
         * var groupList = this.org.groupHasRole( name, roleList );
         *
         * //异步执行,返回Promise对象
         * var promise = this.org.groupHasRole( name, roleList, true);
         * promise.then(function(flag){
         *     //flag 为返回判断结果。
         * })
         *
         * //异步执行,在回调方法中获取判断结果
         * this.org.groupHasRole( name, roleList, function(flag){
         *     //flag 为返回判断结果。
         * })
         */
        groupHasRole: function(name, role, async){
            getOrgActions();
            nameFlag = (typeOf(name)==="object") ? (name.distinguishedName || name.id || name.unique || name.name) : name;
            var data = {"group":nameFlag,"roleList":getNameFlag(role)};

            var v = false;
            var cb = function(json){
                v = json.data.value;
                if (async && o2.typeOf(async)=="function") return async(v);
                return v;
            };

            var promise = orgActions.groupHasRole(data, cb, null, !!async);
            return (!!async) ? promise : v;

            // var v = false;
            // orgActions.groupHasRole(data, function(json){v = json.data.value;}, null, false);
            // return v;
        },

        //人员属性************
        //添加人员属性值(在属性中添加values值,如果没有此属性,则创建一个)
        /**
         * 添加人员属性值(在属性中添加values值,如果没有此属性,则创建一个)
         * @method appendPersonAttribute
         * @o2membercategory personAttribute
         * @methodOf module:org
         * @static
         * @param {PersonFlag} person - 人员的distinguishedName、id、unique属性值,人员对象。
         * @param {String} attr 属性名称。
         * @param {String[]} values 属性值,必须为数组。
         * @param {Function} [success] 执行成功的回调。
         * @param {Function} [failure] 执行失败的回调。
         * @param {(Boolean)} [async] 当参数为boolean,表示是否异步执行,默认为false。
         * @o2syntax
         * //同步执行
         * this.org.appendPersonAttribute( person, attribute, valueArray);
         *
         * //异步执行
         * this.org.appendPersonAttribute( person, attribute, valueArray, function(){
         *     //执行成功的回调
         * }, null, true);
         */
        appendPersonAttribute: function(person, attr, values, success, failure, async){
            getOrgActions();
            var personFlag = (typeOf(person)==="object") ? (person.distinguishedName || person.id || person.unique || person.name) : person;
            var data = {"attributeList":values,"name":attr,"person":personFlag};

            var v = null;
            var cb = function(json){
                v = json.data;
                if (async && o2.typeOf(async)=="function") return async(v);
                return v;
            };
            var promise = orgActions.appendPersonAttribute(data, cb, null, !!async);
            return (!!async) ? promise : v;

            // var cb = function(json){
            //     if (success) return success(json);
            // }.ag().catch(function(xhr, text, error){
            //     if (failure) return failure(xhr, text, error);
            // });
            //
            // orgActions.appendPersonAttribute(data, cb, null, !!async);
        },
        //设置人员属性值(将属性值修改为values,如果没有此属性,则创建一个)
        /**
         * 设置人员属性值(将属性值修改为values,如果没有此属性,则创建一个)
         * @method setPersonAttribute
         * @o2membercategory personAttribute
         * @methodOf module:org
         * @static
         * @param {PersonFlag} person - 人员的distinguishedName、id、unique属性值,人员对象。
         * @param {String} attr 属性名称。
         * @param {String[]} values 属性值,必须为数组。
         * @param {Function} [success] 执行成功的回调。
         * @param {Function} [failure] 执行失败的回调。
         * @param {(Boolean)} [async] 当参数为boolean,表示是否异步执行,默认为false。
         * @o2syntax
         * //同步执行
         * this.org.setPersonAttribute( person, attribute, valueArray);
         *
         * //异步执行
         * this.org.setPersonAttribute( person, attribute, valueArray, function(){
         *     //执行成功的回调
         * }, null, true);
         */
        setPersonAttribute: function(person, attr, values, success, failure, async){
            getOrgActions();
            var personFlag = (typeOf(person)==="object") ? (person.distinguishedName || person.id || person.unique || person.name) : person;
            var data = {"attributeList":values,"name":attr,"person":personFlag};

            var v = null;
            var cb = function(json){
                v = json.data;
                if (async && o2.typeOf(async)=="function") return async(v);
                return v;
            };
            var promise = orgActions.setPersonAttribute(data, cb, null, !!async);
            return (!!async) ? promise : v;

            // var cb = function(json){
            //     if (success) return success(json);
            // }.ag().catch(function(xhr, text, error){
            //     if (failure) return failure(xhr, text, error);
            // });
            //
            // orgActions.setPersonAttribute(data, cb, null, !!async);
        },
        //获取人员属性值
        /**
         根据人员和属性名称获取属性值数组。
         * @method getPersonAttribute
         * @o2membercategory personAttribute
         * @methodOf module:org
         * @static
         * @param {PersonFlag} person - 人员的distinguishedName、id、unique属性值,人员对象。
         * @param {String} attr 属性名称。
         * @param {(Boolean|Function)} [asyncOrCallback] 当参数为boolean,表示是否异步执行,默认为false。当参数为function,表示回调方法。
         * @return {String[]} 当async为true时,返回
         * {@link https://developer.mozilla.org/zh-CN/docs/Web/JavaScript/Reference/Global_Objects/Promise|Promise}。
         * 否则返回属性值数组,
         * 如:<pre><code class='language-js'>[ value1, value2 ]</code></pre>
         * @o2syntax
         * //同步执行,返回该人员的属性值数组。
         * var attributeList = this.org.getPersonAttribute( person, attr );
         *
         * //异步执行,返回Promise对象
         * var promise = this.org.getPersonAttribute( person, attr, true);
         * promise.then(function(attributeList){
         *     //attributeList 为返回该人员的属性值数组。
         * })
         *
         * //异步执行,在回调方法中获取
         * this.org.getPersonAttribute( person, attr, function(attributeList){
         *     //attributeList 为返回该人员的属性值数组。
         * })
         */
        getPersonAttribute: function(person, attr, async){
            getOrgActions();
            var personFlag = (typeOf(person)==="object") ? (person.distinguishedName || person.id || person.unique || person.name) : person;
            var data = {"name":attr,"person":personFlag};
            var v = null;
            var cb = function(json){
                v = json.data.attributeList;
                if (async && o2.typeOf(async)=="function") return async(v);
                return v;
            };

            var promise = orgActions.getPersonAttribute(data, cb, null, !!async);
            return (!!async) ? promise : v;
        },
        //列出人员所有属性的名称
        /**
         列出人员所有属性的名称数组。
         * @method listPersonAttributeName
         * @o2membercategory personAttribute
         * @methodOf module:org
         * @static
         * @param {PersonFlag|PersonFlag[]} name - 人员的distinguishedName、id、unique属性值,人员对象,或上述属性值和对象的数组。
         * @param {(Boolean|Function)} [asyncOrCallback] 当参数为boolean,表示是否异步执行,默认为false。当参数为function,表示回调方法。
         * @return {String[]} 当async为true时,返回
         * {@link https://developer.mozilla.org/zh-CN/docs/Web/JavaScript/Reference/Global_Objects/Promise|Promise}。
         * 否则返回人员属性名称数组,
         * 如:<pre><code class='language-js'>[ attributeName1, attributeName2 ]</code></pre>
         * @o2syntax
         * //同步执行,返回人员所有属性的名称数组。
         * var attributeNameList = this.org.listPersonAttributeName( person );
         *
         * //异步执行,返回Promise对象
         * var promise = this.org.listPersonAttributeName( person, true);
         * promise.then(function(attributeNameList){
         *     //attributeNameList 为人员所有属性的名称数组。
         * })
         *
         * //异步执行,在回调方法中获取
         * this.org.listPersonAttributeName( person, function(attributeNameList){
         *     //attributeNameList 为人员所有属性的名称数组。
         * })
         */
        listPersonAttributeName: function(name, async){
            getOrgActions();
            var data = {"personList":getNameFlag(name)};
            var v = null;
            var cb = function(json){
                v = json.data.nameList;
                if (async && o2.typeOf(async)=="function") return async(v);
                return v;
            };

            var promise = orgActions.listPersonAttributeName(data, cb, null, !!async);
            return (!!async) ? promise : v;
        },
        //列出人员的所有属性
        /**
         列出人员的所有属性对象数组。
         * @method listPersonAllAttribute
         * @o2membercategory personAttribute
         * @methodOf module:org
         * @static
         * @param {PersonFlag|PersonFlag[]} name - 人员的distinguishedName、id、unique属性值,人员对象,或上述属性值和对象的数组。
         * @param {(Boolean|Function)} [asyncOrCallback] 当参数为boolean,表示是否异步执行,默认为false。当参数为function,表示回调方法。
         * @return {Object[]} 当async为true时,返回
         * {@link https://developer.mozilla.org/zh-CN/docs/Web/JavaScript/Reference/Global_Objects/Promise|Promise}。
         * 否则返回人员属性对象数组,如:
         * <pre><code class='language-js'>[{
         *    "name": "住址",
         *    "person": "张三@zhangsan@P",
         *    "attributeList": [
         *        "杭州市","绍兴市"
         *    ]
         * }]</code></pre>
         * @o2syntax
         * //同步执行,返回人员所有属性的对象数组。
         * var attributeObjectList = this.org.listPersonAllAttribute( person );
         *
         * //异步执行,返回Promise对象
         * var promise = this.org.listPersonAllAttribute( person, true);
         * promise.then(function(attributeObjectList){
         *     //attributeObjectList 为人员所有属性的对象数组。
         * })
         *
         * //异步执行,在回调方法中获取
         * this.org.listPersonAllAttribute( person, function(attributeObjectList){
         *     //attributeObjectList 为人员所有属性的对象数组。
         * })
         */
        listPersonAllAttribute: function(name, async){
            getOrgActions();
            var data = {"personList":getNameFlag(name)};
            var v = null;
            var cb = function(json){
                v = json.data;
                if (async && o2.typeOf(async)=="function") return async(v);
                return v;
            };

            var promise = orgActions.listPersonAllAttribute(data, cb, null, !!async);
            return (!!async) ? promise : v;
        },

        //组织属性**************
        //添加组织属性值(在属性中添加values值,如果没有此属性,则创建一个)
        /**
         * 添加组织属性值(在属性中添加values值,如果没有此属性,则创建一个)
         * @method appendUnitAttribute
         * @o2membercategory unitAttribute
         * @methodOf module:org
         * @static
         * @param {UnitFlag} unit - 组织的distinguishedName、id、unique属性值,组织对象。
         * @param {String} attribute 属性名称。
         * @param {String[]} valueArray 属性值,必须为数组。
         * @param {Function} [success] 执行成功的回调。
         * @param {Function} [failure] 执行失败的回调。
         * @param {(Boolean)} [async] 当参数为boolean,表示是否异步执行,默认为false。
         * @o2syntax
         * //同步执行
         * this.org.appendUnitAttribute( unit, attribute, valueArray);
         *
         * //异步执行
         * this.org.appendUnitAttribute( unit, attribute, valueArray, function(){
         *     //执行成功的回调
         * }, null, true);
         */
        appendUnitAttribute: function(unit, attr, values, success, failure, async){
            getOrgActions();
            var unitFlag = (typeOf(unit)==="object") ? (unit.distinguishedName || unit.id || unit.unique || unit.name) : unit;
            var data = {"attributeList":values,"name":attr,"unit":unitFlag};

            var v = null;
            var cb = function(json){
                v = json.data;
                if (async && o2.typeOf(async)=="function") return async(v);
                return v;
            };
            var promise = orgActions.appendUnitAttribute(data, cb, null, !!async);
            return (!!async) ? promise : v;

            // var cb = function(json){
            //     if (success) return success(json);
            // }.ag().catch(function(xhr, text, error){
            //     if (failure) return failure(xhr, text, error);
            // });
            //
            // orgActions.appendPersonAttribute(data, cb, null, !!async);

            // orgActions.appendUnitAttribute(data, function(json){
            //     if (json.data.value){
            //         if (success) success();
            //     }else{
            //         if (failure) failure(null, "", "append values failed");
            //     }
            // }, function(xhr, text, error){
            //     if (failure) failure(xhr, text, error);
            // }, false);
        },
        //设置组织属性值(将属性值修改为values,如果没有此属性,则创建一个)
        /**
         * 设置组织属性值(将属性值修改为values,如果没有此属性,则创建一个)
         * @method setUnitAttribute
         * @o2membercategory unitAttribute
         * @methodOf module:org
         * @static
         * @param {UnitFlag} unit - 组织的distinguishedName、id、unique属性值,组织对象。
         * @param {String} attribute 属性名称。
         * @param {String[]} valueArray 属性值,必须为数组。
         * @param {Function} [success] 执行成功的回调。
         * @param {Function} [failure] 执行失败的回调。
         * @param {(Boolean)} [async] 当参数为boolean,表示是否异步执行,默认为false。
         * @o2syntax
         * //同步执行
         * this.org.setUnitAttribute( unit, attribute, valueArray);
         *
         * //异步执行
         * this.org.setUnitAttribute( unit, attribute, valueArray, function(){
         *     //执行成功的回调
         * }, null, true);
         */
        setUnitAttribute: function(unit, attr, values, success, failure, async){
            getOrgActions();
            var unitFlag = (typeOf(unit)==="object") ? (unit.distinguishedName || unit.id || unit.unique || unit.name) : unit;
            var data = {"attributeList":values,"name":attr,"unit":unitFlag};

            var v = null;
            var cb = function(json){
                v = json.data;
                if (async && o2.typeOf(async)=="function") return async(v);
                return v;
            };
            var promise = orgActions.setUnitAttribute(data, cb, null, !!async);
            return (!!async) ? promise : v;

            // var cb = function(json){
            //     if (success) return success(json);
            // }.ag().catch(function(xhr, text, error){
            //     if (failure) return failure(xhr, text, error);
            // });
            // orgActions.setUnitAttribute(data, cb, null, !!async);

            // orgActions.setUnitAttribute(data, function(json){
            //     if (json.data.value){
            //         if (success) success();
            //     }else{
            //         if (failure) failure(null, "", "append values failed");
            //     }
            // }, function(xhr, text, error){
            //     if (failure) failure(xhr, text, error);
            // }, false);
        },
        //获取组织属性值
        /**
         根据组织标识和属性名称获取对应属性值。
         * @method getUnitAttribute
         * @o2membercategory unitAttribute
         * @methodOf module:org
         * @static
         * @param {UnitFlag} unit - 组织的distinguishedName、id、unique属性值,组织对象。
         * @param {String} attr 属性名称。
         * @param {(Boolean|Function)} [asyncOrCallback] 当参数为boolean,表示是否异步执行,默认为false。当参数为function,表示回调方法。
         * @return {String[]} 当async为true时,返回
         * {@link https://developer.mozilla.org/zh-CN/docs/Web/JavaScript/Reference/Global_Objects/Promise|Promise}。
         * 否则返回属性值数组,
         * 如:<pre><code class='language-js'>[ value1, value2 ]</code></pre>
         * @o2syntax
         * //同步执行,返回该组织的属性值数组。
         * var attributeList = this.org.getUnitAttribute( unit, attr );
         *
         * //异步执行,返回Promise对象
         * var promise = this.org.getUnitAttribute( unit, attr, true);
         * promise.then(function(attributeList){
         *     //attributeList 为返回该组织的属性值数组。
         * })
         *
         * //异步执行,在回调方法中获取
         * this.org.getUnitAttribute( unit, attr, function(attributeList){
         *     //attributeList 为返回该组织的属性值数组。
         * })
         */
        getUnitAttribute: function(unit, attr, async){
            getOrgActions();
            var unitFlag = (typeOf(unit)==="object") ? (unit.distinguishedName || unit.id || unit.unique || unit.name) : unit;
            var data = {"name":attr,"unit":unitFlag};
            var v = null;
            var cb = function(json){
                v = json.data.attributeList;
                if (async && o2.typeOf(async)=="function") return async(v);
                return v;
            };

            var promise = orgActions.getUnitAttribute(data, cb, null, !!async);
            return (!!async) ? promise : v;
        },
        //列出组织所有属性的名称
        /**
         列出组织所有属性的名称数组。
         * @method listUnitAttributeName
         * @o2membercategory unitAttribute
         * @methodOf module:org
         * @static
         * @param {UnitFlag|UnitFlag[]} name - 组织的distinguishedName、id、unique属性值,组织对象,或上述属性值和对象的数组。
         * @param {(Boolean|Function)} [asyncOrCallback] 当参数为boolean,表示是否异步执行,默认为false。当参数为function,表示回调方法。
         * @return {String[]} 当async为true时,返回
         * {@link https://developer.mozilla.org/zh-CN/docs/Web/JavaScript/Reference/Global_Objects/Promise|Promise}。
         * 否则返回组织属性名称数组,
         * 如:<pre><code class='language-js'>[ attributeName1, attributeName2 ]</code></pre>
         * @o2syntax
         * //同步执行,返回组织所有属性的名称数组。
         * var attributeNameList = this.org.listUnitAttributeName( unit );
         *
         * //异步执行,返回Promise对象
         * var promise = this.org.listUnitAttributeName( unit, true);
         * promise.then(function(attributeNameList){
         *     //attributeNameList 为组织所有属性的名称数组。
         * })
         *
         * //异步执行,在回调方法中获取
         * this.org.listUnitAttributeName( unit, function(attributeNameList){
         *     //attributeNameList 为组织所有属性的名称数组。
         * })
         */
        listUnitAttributeName: function(name, async){
            getOrgActions();
            var data = {"unitList":getNameFlag(name)};
            var v = null;
            var cb = function(json){
                v = json.data.nameList;
                if (async && o2.typeOf(async)=="function") return async(v);
                return v;
            };

            var promise = orgActions.listUnitAttributeName(data, cb, null, !!async);
            return (!!async) ? promise : v;
        },
        //列出组织的所有属性
        /**
         列出组织的所有属性对象数组。
         * @method listUnitAllAttribute
         * @o2membercategory unitAttribute
         * @methodOf module:org
         * @static
         * @param {UnitFlag|UnitFlag[]} name - 组织的distinguishedName、id、unique属性值,组织对象,或上述属性值和对象的数组。
         * @param {(Boolean|Function)} [asyncOrCallback] 当参数为boolean,表示是否异步执行,默认为false。当参数为function,表示回调方法。
         * @return {Object[]} 当async为true时,返回
         * {@link https://developer.mozilla.org/zh-CN/docs/Web/JavaScript/Reference/Global_Objects/Promise|Promise}。
         * 否则返回组织属性对象数组,如:
         * <pre><code class='language-js'>[{
         *    "name": "部门类别",
         *    "unit": "开发部@kfb@U",
         *    "attributeList": [
         *        "生产部门",
         *        "二级部门"
         *    ]
         * }]</code></pre>
         * @o2syntax
         * //同步执行,返回组织所有属性的对象数组。
         * var attributeObjectList = this.org.listUnitAllAttribute( unit );
         *
         * //异步执行,返回Promise对象
         * var promise = this.org.listUnitAllAttribute( unit, true);
         * promise.then(function(attributeObjectList){
         *     //attributeObjectList 为组织所有属性的对象数组。
         * })
         *
         * //异步执行,在回调方法中获取
         * this.org.listUnitAllAttribute( unit, function(attributeObjectList){
         *     //attributeObjectList 为组织所有属性的对象数组。
         * })
         */
        listUnitAllAttribute: function(name, async){
            getOrgActions();
            var data = {"unitList":getNameFlag(name)};
            var v = null;
            var cb = function(json){
                v = json.data;
                if (async && o2.typeOf(async)=="function") return async(v);
                return v;
            };

            var promise = orgActions.listUnitAllAttribute(data, cb, null, !!async);
            return (!!async) ? promise : v;
        }
    };

    this.Action = (function () {
        var actions = [];
        return function (root, json) {
            var action = actions[root] || (actions[root] = new MWF.xDesktop.Actions.RestActions("", root, ""));
            action.getActions = function (callback) {
                if (!this.actions) this.actions = {};
                Object.merge(this.actions, json);
                if (callback) callback();
            };
            this.invoke = function (option) {
                action.invoke(option)
            }
        }
    })();
    // this.service = {
    //     "jaxwsClient": {},
    //     "jaxrsClient": {}
    // };

    var lookupAction = null;
    var getLookupAction = function (callback) {
        if (!lookupAction) {
            MWF.require("MWF.xDesktop.Actions.RestActions", function () {
                lookupAction = new MWF.xDesktop.Actions.RestActions("", "x_processplatform_assemble_surface", "");
                lookupAction.getActions = function (actionCallback) {
                    this.actions = {
                        //"lookup": {"uri": "/jaxrs/view/flag/{view}/application/flag/{application}"},
                        //"getView": {"uri": "/jaxrs/view/{id}/design"}
                        "lookup": { "uri": "/jaxrs/queryview/flag/{view}/application/flag/{application}/execute", "method": "PUT" },
                        "getView": { "uri": "/jaxrs/queryview/flag/{view}/application/flag/{application}" }
                    };
                    if (actionCallback) actionCallback();
                }
                if (callback) callback();
            });
        } else {
            if (callback) callback();
        }
    };

    /**
     * 您可以通过view对象,获取视图数据或选择视图数据。<br/>
     * @module view
     * @o2cn 视图执行
     * @o2category web
     * @o2ordernumber 70
     * @o2syntax
     * //您可以在流程表单、内容管理表单或门户页面中,通过this来获取view对象,如下:
     * var view = this.view;
     */
    this.view = {
        /**
         * 获取指定视图的数据。
         * @method lookup
         * @static
         * @param {Object} view - 要访问的视图信息。数据格式如下:<br/>
         * <caption>以下的filter参数参考<a href='global.html#ViewFilter'>ViewFilter</a></caption>
         * <pre><code class='language-js'>
         * {
         *  "view" : "testView", //(String)必选,视图的名称、别名或ID
         *  "application" : "test数据中心应用", //(String)必选,视图所在数据应用的名称、别名或ID
         *  "filter": [ //(Array of Object)可选,对视图进行过滤的条件。json数组格式,每个数组元素描述一个过滤条件。
         *       {
         *           "logic":"and",
         *           "path":"$work.title",
         *           "comparison":"like",
         *           "value":"7月",
         *           "formatType":"textValue"
         *       }
         *  ]
         * }
         * </code></pre>
         * @param {Function} callback - 访问成功后的回调函数
         * @param {Boolean} [async] - 同步或异步调用。true:异步;false:同步。默认为true。
         * @return {Promise} 返回Promise
         * @o2syntax
         * //通过回调方法获取数据
         * this.view.lookup(view, callback, async);
         *
         * //返回Promise对象后处理
         * var promise = this.view.lookup( view );
         * promise.then(function(data){
         *     //data 为返回的数据。
         * })
         * @example
         * //获取“财务管理”应用中“报销审批数据”视图中的数据
         * //过滤条件为标题($work.title)包含包含(like))“7月”。
         * this.view.lookup({
         *   "view": "报销审批数据",
         *   "application": "财务管理",
         *   "filter": [
         *       {
         *           "logic":"and",
         *           "path":"$work.title",
         *           "comparison":"like",
         *           "value":"7月",
         *           "formatType":"textValue"
         *       }
         *   ]
         *}, function(data){
         *   var grid = data.grid; //得到过滤后的数据
         *   var groupGrid = data.groupGrid; //如果有分类,得到带分类的数据
         *   //......
         *});
         * @example
         * //同上,通过返回值获取数据
         * var promise = this.view.lookup({
         *   "view": "报销审批数据",
         *   "application": "财务管理",
         *   "filter": [
         *       {
         *           "logic":"and",
         *           "path":"$work.title",
         *           "comparison":"like",
         *           "value":"7月",
         *           "formatType":"textValue"
         *       }
         *   ]
         *});
         * promise.then(function(data){
         *   var grid = data.grid; //得到过滤后的数据
         *   var groupGrid = data.groupGrid; //如果有分类,得到带分类的数据
         *   //......
         *})
         * @example
         * //获取“财务管理”应用中“报销审批数据”视图中的数据
         * //过滤条件为标题($work.title)包含包含(like))“7月”,并且总金额大于500小于5000
         * this.view.lookup({
         *   "view": "报销审批数据",
         *   "application": "财务管理",
         *   "filter": [
         *       {
         *           "logic":"and",
         *           "path":"$work.title",
         *           "comparison":"like",
         *           "value":"7月",
         *           "formatType":"textValue"
         *       },
         *       {
         *           "logic":"and",
         *           "path":"amount",
         *           "comparison":"range",
         *           "value":500,
         *           "otherValue":5000,
         *           "formatType":"numberValue"
         *       },
         *   ]
         *}, function(data){
         *   var grid = data.grid; //得到过滤后的数据
         *   var groupGrid = data.groupGrid; //如果有分类,得到带分类的数据
         *   //......
         *});
         */
        "lookup": function (view, callback, async) {
            var filterList = { "filterList": (view.filter || null) };
            return MWF.Actions.load("x_query_assemble_surface").ViewAction.executeWithQuery(view.view, view.application, filterList, function (json) {
                var data = {
                    "grid": json.data.grid || json.data.groupGrid,
                    "groupGrid": json.data.groupGrid
                };
                if (callback) callback(data);
                return data;
            }, null, async);
        },

        "lookupV1": function (view, callback) {
            getLookupAction(function () {
                lookupAction.invoke({
                    "name": "lookup", "async": true, "parameter": { "view": view.view, "application": view.application }, "success": function (json) {
                        var data = {
                            "grid": json.data.grid,
                            "groupGrid": json.data.groupGrid
                        };
                        if (callback) callback(data);
                    }.bind(this)
                });
            }.bind(this));
        },

        /**
         * 通过视图进行数据选择。
         * @method select
         * @static
         * @param {Object} view - 要访问的视图信息。数据格式如下:<br/>
         * <caption>以下的filter参数参考<a href='global.html#ViewFilter'>ViewFilter</a></caption>
         * <pre><code class='language-js'>
         * {
         *  "view" : "testView", //(String)必选,视图的名称、别名或ID
         *  "application" : "test数据中心应用", //(String)必选,视图所在数据应用的名称、别名或ID
         *  "isTitle" : true, //(Boolean)可选,是否显示视图标题。默认true
         *  "isMulti" : true,  //(Boolean)可选,是否允许多选。默认true
         *  "width" : 700, //(Number)可选,选择框的宽度。默认700
         *  "height" : 400,  //(Number)可选,选择框的高度。默认400
         *  "caption" : "标题", //(String)可选,选择框的标题
         *  "filter": [ //(Array of Object)可选,对视图进行过滤的条件。json数组格式,每个数组元素描述一个过滤条件。
         *       {
         *           "logic":"and",
         *           "path":"$work.title",
         *           "comparison":"like",
         *           "value":"7月",
         *           "formatType":"textValue"
         *       }
         *  ]
         * }
         * </code></pre>
         * @param {Function} callback - 必选,当选择完成,点击“确定”之后的回调函数。
         * @o2syntax
         * this.view.select(view, callback);
         * @example
         * this.view.select({
         *    "application": "物业材料",  //数据中心中的应用
         *    "view": "物业材料视图",     //视图的名称
         *    "isMulti": false,           //只允许单选
         * }, function(items) {
         *    //如果选择了某个数据,将数据赋值给表单输入框
         *    if (items.length) {
         *        //物料名称,表单中输入框名为“materialName”, 视图中列的名称为“ylmc”
         *        this.data.materialName = items[0].data.ylmc;
         *        //规格,表单中输入框名为“specification”, 视图中列的名称为“gg”
         *        this.data.specification = items[0].data.gg;
         *        //单价,表单中输入框名为“price”, 视图中列的名称为“dj”
         *        this.data.price = items[0].data.dj;
         *    }
         * }.bind(this));
         */
        "select": function (view, callback, options) {
            if (view.view) {
                var viewJson = {
                    "application": view.application || _form.json.application,
                    "viewName": view.view || "",
                    "isTitle": (view.isTitle === false) ? "no" : "yes",
                    "select": (view.isMulti === false) ? "single" : "multi",
                    "filter": view.filter
                };
                if (!options) options = {};
                options.width = view.width;
                options.height = view.height;
                options.title = view.caption;

                var width = options.width || "700";
                var height = options.height || "400";

                if (layout.mobile) {
                    var size = document.body.getSize();
                    width = size.x;
                    height = size.y;
                    options.style = "viewmobile";
                }
                width = width.toInt();
                height = height.toInt();

                var size = _form.app.content.getSize();
                var x = (size.x - width) / 2;
                var y = (size.y - height) / 2;
                if (x < 0) x = 0;
                if (y < 0) y = 0;
                if (layout.mobile) {
                    x = 20;
                    y = 0;
                }

                var _self = this;
                MWF.require("MWF.xDesktop.Dialog", function () {
                    var dlg = new MWF.xDesktop.Dialog({
                        "title": options.title || "select view",
                        "style": options.style || "view",
                        "top": y,
                        "left": x - 20,
                        "fromTop": y,
                        "fromLeft": x - 20,
                        "width": width,
                        "height": height,
                        "html": "<div style='height: 100%;'></div>",
                        "maskNode": _form.app.content,
                        "container": _form.app.content,
                        "buttonList": [
                            {
                                "text": MWF.LP.process.button.ok,
                                "action": function () {
                                    //if (callback) callback(_self.view.selectedItems);
                                    if (callback) callback(_self.view.getData());
                                    this.close();
                                }
                            },
                            {
                                "text": MWF.LP.process.button.cancel,
                                "action": function () { this.close(); }
                            }
                        ]
                    });
                    dlg.show();

                    if (layout.mobile) {
                        var backAction = dlg.node.getElement(".MWF_dialod_Action_back");
                        var okAction = dlg.node.getElement(".MWF_dialod_Action_ok");
                        if (backAction) backAction.addEvent("click", function (e) {
                            dlg.close();
                        }.bind(this));
                        if (okAction) okAction.addEvent("click", function (e) {
                            //if (callback) callback(this.view.selectedItems);
                            if (callback) callback(this.view.getData());
                            dlg.close();
                        }.bind(this));
                    }

                    MWF.xDesktop.requireApp("query.Query", "Viewer", function () {
                        this.view = new MWF.xApplication.query.Query.Viewer(dlg.content.getFirst(), viewJson, { "style": "select" }, _form.app, _form.Macro);
                    }.bind(this));
                }.bind(this));
            }
        }
    };

    /**
     * 您可以通过statement对象,获取执行查询语句或者对查询结果进行选择。<br/>
     * @module statement
     * @o2cn 查询视图执行
     * @o2category web
     * @o2ordernumber 90
     * @o2syntax
     * //您可以在流程表单、内容管理表单、门户页面或视图中,通过this来获取statement对象,如下:
     * var statement = this.statement;
     */
    this.statement = {
        /**
         * 执行指定的查询语句。
         * @method execute
         * @static
         * @param {Object} statement - 要执行的查询语句的信息。数据格式如下:
         * <div>以下的filter参数参考<a href='global.html#StatementFilter'>StatementFilter</a>,
         * parameter参数参考<a href='global.html#StatementParameter'>StatementParameter</a></div>
         * <pre><code class='language-js'>
         * {
         *  "name" : "tesStatement", //(String)必选,查询配置的名称、别名或ID
         *  "mode" : "all", //(String)必选,“all”、“data”或者“count”,all表示同时执行查询语句和总数语句,data表示执行查询语句,count表示执行总数语句
         *  "page" : 1, //(number)可选,当前页码,默认为1
         *  "pageSize" : 20, //(number)可选,每页的数据条数,默认为20
         *  "filter": [ //(Array)可选,对查询进行过滤的条件。json数组格式,每个数组元素描述一个过滤条件,每个元素数据格式如下:
         *       {
         *           "path":"o.title", //查询语句格式为jpql使用o.title,为原生sql中使用xtitle
         *           "comparison":"like",
         *           "value":"关于",
         *           "formatType":"textValue"
         *       }
         *  ],
         *  parameter : {
         *       "person" : "", //参数名称为下列值时,后台默认赋值,person(当前人),identityList(当前人身份列表),unitList(当前人所在直接组织), unitAllList(当前人所在所有组织), groupList(当前人所在群组),roleList(当前人拥有的角色)。v8.0以后系统自动解析,不需要再传这类参数。
         *       "startTime" : (new Date("2020-01-01")), //如果对比的是日期,需要传入 Date 类型
         *       "applicationName" : "%test%", //如果运算符用的是 like, noLike,模糊查询
         *       "processName" : "test流程", //其他写确定的值
         *       "?1": "关于" //v8.0后查询语句支持问号加数字的传参
         *     }
         * }
         * </code></pre>
         * @param {Function} callback - 访问成功后的回调函数
         * @param {Boolean} [async] - 同步或异步调用。true:异步;false:同步。默认为true。
         * @return {Promise} 返回Promise
         * @o2syntax
         * this.statement.execute(statement, callback, async);
         *
         * //返回Promise对象后处理
         * var promise = this.statement.execute( statement );
         * promise.then(function(data){
         *     //data 为返回的数据。
         * })
         * @example
         * //获取“task”查询中的数据
         * //查询语句为 select o from Task o where (o.person = :person) and (o.startTime > :startTime) and (o.applicationName like :applicationName) and (o.processName = :processName)
         * //总数语句为 select count(o.id) from Task o where (o.person = :person) and (o.startTime > :startTime) and (o.applicationName like :applicationName) and (o.processName = :processName)
         * //过滤条件为标题o.title包含包含(like))“7月”。
         * this.statement.execute({
         *  "name": "task",
         *  "mode" : "all",
         *  "filter": [
         *      {
         *      "path":"o.title", //查询语句格式为jpql使用o.title,为原生sql中使用xtitle
         *      "comparison":"like",
         *      "value":"7月",
         *      "formatType":"textValue"
         *      }
         * ],
         * "parameter" : {
         *     "person" : "", //参数名称为下列值时,后台默认赋值,person(当前人),identityList(当前人身份列表),unitList(当前人所在直接组织), unitAllList(当前人所在所有组织), groupList(当前人所在群组),roleList(当前人拥有的角色)。v8.0以后系统自动解析,不需要再传这类参数。
         *     "startTime" : (new Date("2020-01-01")), //如果对比的是日期,需要传入 Date 类型
         *     "applicationName" : "%test%", //如果运算符用的是 like, noLike,模糊查询
         *     "processName" : "test流程", //其他写确定的值
         *     "?1": "关于" //v8.0后查询语句支持问号加数字的传参
         *   }
         * }, function(json){
         *  var count = json.count; //总数语句执行后返回的数字
         *  var list = json.data; //查询语句后返回的数组
         *   //......
         * });
         * @example
         * //同上,使用返回值接收参数
         * var promise = this.statement.execute({
         *  "name": "task",
         *  "mode" : "all",
         *  "filter": [
         *      {
         *      "path":"o.title", //查询语句格式为jpql使用o.title,为原生sql中使用xtitle
         *      "comparison":"like",
         *      "value":"7月",
         *      "formatType":"textValue"
         *      }
         * ],
         * "parameter" : {
         *     "person" : "", //参数名称为下列值时,后台默认赋值,person(当前人),identityList(当前人身份列表),unitList(当前人所在直接组织), unitAllList(当前人所在所有组织), groupList(当前人所在群组),roleList(当前人拥有的角色)。v8.0以后系统自动解析,不需要再传这类参数。
         *     "startTime" : (new Date("2020-01-01")), //如果对比的是日期,需要传入 Date 类型
         *     "applicationName" : "%test%", //如果运算符用的是 like, noLike,模糊查询
         *     "processName" : "test流程", //其他写确定的值
         *    "?1": "关于" //v8.0后查询语句支持问号加数字的传参
         *   }
         * });
         * promise.then(function(json){
         *  var count = json.count; //总数语句执行后返回的数字
         *  var list = json.data; //查询语句后返回的数组
         *   //......
         * })
         */
        execute: function (obj, callback, async) {
            if( obj.format ){
                return this._execute(obj, callback, async, obj.format);
            }else{
                if( this.needCheckFormat(obj) ){
                    var result;
                    var p = MWF.Actions.load("x_query_assemble_surface").StatementAction.getFormat(obj.name, function(json){
                        result = this._execute(obj, callback, async, json.data.format);
                        return result;
                    }.bind(this), null, async);
                    return result || p;
                }else{
                    return this._execute(obj, callback, async, "");
                }

            }
        },
            needCheckFormat: function(s){
                if( s.format )return false;
                if( typeOf(s.parameter) === "object" ){
                    for( var p in s.parameter ){
                        if( typeOf( s.parameter[p] ) === "date" )return true;
                    }
                }
                if( typeOf(s.filter) === "array" ){
                    for( var i=0; i< s.filter.length; i++){
                        var fType = s.filter[i].formatType;
                        if( ["dateTimeValue", "datetimeValue", "dateValue", "timeValue"].contains( fType ) )return true;
                    }
                }
                return false;
            },
            _execute: function(statement, callback, async, format){
                var parameter = this.parseParameter(statement.parameter, format);
                var filterList = this.parseFilter(statement.filter, parameter, format);
                var obj = {
                    "filterList": filterList,
                    "parameter" : parameter
                };
                return MWF.Actions.load("x_query_assemble_surface").StatementAction.executeV2(
                    statement.name, statement.mode || "data", statement.page || 1, statement.pageSize || 20, obj,
                    function (json) {
                        if (callback) callback(json);
                        return json;
                    }, null, async);
            },
            parseFilter: function (filter, parameter, format) {
                if (typeOf(filter) !== "array") return [];
                if( !parameter )parameter = {};
                var filterList = [];
                (filter || []).each(function (d) {
                    if( !d.logic )d.logic = "and";

                    //var parameterName = d.path.replace(/\./g, "_");
                    var pName = d.path.replace(/\./g, "_");

                    var parameterName = pName;
                    var suffix = 1;
                    while( parameter[parameterName] ){
                        parameterName = pName + "_" + suffix;
                        suffix++;
                    }
                    var value = d.value;
                    if (d.comparison === "like" || d.comparison === "notLike") {
                        if (value.substr(0, 1) !== "%") value = "%" + value;
                        if (value.substr(value.length - 1, 1) !== "%") value = value + "%";
                        parameter[parameterName] = value; //"%"+value+"%";
                    } else {
                         if( ["sql", "sqlScript"].contains(format) ) {
                            if (d.formatType === "numberValue") {
                                value = parseFloat(value);
                            }
                        }else{
                            if (d.formatType === "dateTimeValue" || d.formatType === "datetimeValue") {
                                value = "{ts '" + value + "'}"
                            } else if (d.formatType === "dateValue") {
                                value = "{d '" + value + "'}"
                            } else if (d.formatType === "timeValue") {
                                value = "{t '" + value + "'}"
                            } else if (d.formatType === "numberValue") {
                                value = parseFloat(value);
                            }
                        }
                        parameter[parameterName] = value;
                    }
                    d.value = parameterName;

                    filterList.push(d);
                }.bind(this));
                return filterList;
            },
            parseParameter : function( obj, format ){
                if( typeOf(obj) !== "object" )return {};
                var parameter = {};
                //传入的参数
                for( var p in obj ){
                    var value = obj[p];
                    if( typeOf( value ) === "date" ){
                        if( ["sql", "sqlScript"].contains(format) ){
                            value = value.format("db");
                        }else{
                            value = "{ts '"+value.format("db")+"'}"
                        }
                    }
                    parameter[ p ] = value;
                }
                return parameter;
            },

        /**
         * 如果查询的类型是"select",并且配置了查询视图,可以通过本方法进行数据选择。
         * @method select
         * @static
         * @param {Object} statement - 要访问的查询配置的信息。数据格式如下:
         * <div>以下的filter参数参考<a href='global.html#StatementFilter'>StatementFilter</a>,
         * parameter参数参考<a href='global.html#StatementParameter'>StatementParameter</a></div>
         * <pre><code class='language-js'>
         * {
         *  "name" : "tesStatement", //(String)必选,查询配置的名称、别名或ID
         *  "isTitle" : true, //(Boolean)可选,是否显示视图标题。默认true
         *  "isMulti" : true,  //(Boolean)可选,是否允许多选。默认true
         *  "width" : 700, //(Number)可选,选择框的宽度。默认700
         *  "height" : 400,  //(Number)可选,选择框的高度。默认400
         *  "caption" : "标题", //(String)可选,选择框的标题
         *  "filter": [ //(Array)可选,对查询进行过滤的条件。json数组格式,每个数组元素描述一个过滤条件,每个元素数据格式如下:
         *       {
         *           "path":"o.title", //查询语句格式为jpql使用o.title,为原生sql中使用xtitle
         *           "comparison":"like",
         *           "value":"关于",
         *           "formatType":"textValue"
         *       }
         *  ],
         *  parameter : {
         *       "person" : "", //参数名称为下列值时,后台默认赋值,person(当前人),identityList(当前人身份列表),unitList(当前人所在直接组织), unitAllList(当前人所在所有组织), groupList(当前人所在群组),roleList(当前人拥有的角色)。v8.0以后系统自动解析,不需要再传这类参数。
         *       "startTime" : (new Date("2020-01-01")), //如果对比的是日期,需要传入 Date 类型
         *       "applicationName" : "%test%", //如果运算符用的是 like, noLike,模糊查询
         *       "processName" : "test流程", //其他写确定的值
         *       "?1": "关于" //v8.0后查询语句支持问号加数字的传参
         *     }
         * }
         * </code></pre>
         * @param {Function} callback - 访问成功后的回调函数
         * @o2syntax
         * this.statement.select(statement, callback);
         * @example
         * this.statement.select({
         *     "name": "物业材料查询",     //查询的名称
         *     "isMulti": false,           //只允许单选
         * }, function(items) {
         *     //如果选择了某个数据,将数据赋值给表单输入框
         *     if (items.length) {
         *         //物料名称,表单中输入框名为“materialName”, 查询语句返回的字段名为“ylmc”
         *         this.data.materialName = items[0].ylmc;
         *         //规格,表单中输入框名为“specification”, 查询语句返回的字段名为“gg”
         *         this.data.specification = items[0].gg;
         *         //单价,表单中输入框名为“price”, 查询语句返回的字段名为“dj”
         *         this.data.price = items[0].dj;
         *     }
         * }.bind(this));
         */
        "select": function (statement, callback, options) {
            if (statement.name) {
                // var parameter = this.parseParameter(statement.parameter);
                // var filterList = this.parseFilter(statement.filter, parameter);
                var statementJson = {
                    "statementId": statement.name || "",
                    "isTitle": (statement.isTitle === false) ? "no" : "yes",
                    "select": (statement.isMulti === false) ? "single" : "multi",
                    "filter": statement.filter,
                    "parameter": statement.parameter
                };
                if (!options) options = {};
                options.width = statement.width;
                options.height = statement.height;
                options.title = statement.caption;

                var width = options.width || "700";
                var height = options.height || "400";

                if (layout.mobile) {
                    var size = document.body.getSize();
                    width = size.x;
                    height = size.y;
                    options.style = "viewmobile";
                }
                width = width.toInt();
                height = height.toInt();

                var size = _form.app.content.getSize();
                var x = (size.x - width) / 2;
                var y = (size.y - height) / 2;
                if (x < 0) x = 0;
                if (y < 0) y = 0;
                if (layout.mobile) {
                    x = 20;
                    y = 0;
                }

                var _self = this;
                MWF.require("MWF.xDesktop.Dialog", function () {
                    var dlg = new MWF.xDesktop.Dialog({
                        "title": options.title || "select statement view",
                        "style": options.style || "view",
                        "top": y,
                        "left": x - 20,
                        "fromTop": y,
                        "fromLeft": x - 20,
                        "width": width,
                        "height": height,
                        "html": "<div style='height: 100%;'></div>",
                        "maskNode": _form.app.content,
                        "container": _form.app.content,
                        "buttonList": [
                            {
                                "text": MWF.LP.process.button.ok,
                                "action": function () {
                                    //if (callback) callback(_self.view.selectedItems);
                                    if (callback) callback(_self.statement.getData());
                                    this.close();
                                }
                            },
                            {
                                "text": MWF.LP.process.button.cancel,
                                "action": function () { this.close(); }
                            }
                        ]
                    });
                    dlg.show();

                    if (layout.mobile) {
                        var backAction = dlg.node.getElement(".MWF_dialod_Action_back");
                        var okAction = dlg.node.getElement(".MWF_dialod_Action_ok");
                        if (backAction) backAction.addEvent("click", function (e) {
                            dlg.close();
                        }.bind(this));
                        if (okAction) okAction.addEvent("click", function (e) {
                            //if (callback) callback(this.view.selectedItems);
                            if (callback) callback(this.statement.getData());
                            dlg.close();
                        }.bind(this));
                    }

                    MWF.xDesktop.requireApp("query.Query", "Statement", function () {
                        this.statement = new MWF.xApplication.query.Query.Statement(dlg.content.getFirst(), statementJson, { "style": "select" }, _form.app, _form.Macro);
                    }.bind(this));
                }.bind(this));
            }
        }
    };

    /**
     * 您可以通过importer对象,执行导入模型的Excel导入数据功能。<br/>
     * @module importer
     * @o2cn 导入数据
     * @o2category web
     * @o2ordernumber 93
     * @o2syntax
     * //您可以在流程表单、内容管理表单、门户页面或视图中,通过this来获取statement对象,如下:
     * var importer = this.importer;
     */
    this.importer = {
        /**
         * 根据指定的导入模型进行上传。
         * @method upload
         * @static
         * @param {Object} options - 要执行的导入模型的选项。数据格式如下:
         * <pre><code class='language-js'>{
         *  "name" : "testImporter", //(String)必选,导入模型的名称、别名或ID
         *  "application" : "testQuery" //(String)必选,导入模型所在应用的名称、别名或ID
         * }
         * </code></pre>
         * @param {Function} [callback] - 导入成功后的回调函数
         * @param {Boolean} [async] - 同步或异步调用。true:异步;false:同步。默认为true。
         * @o2syntax
         * this.importer.upload(options, callback, async);
         * @example
         * this.importer.upload({
         *  "name": "testImporter",
         *  "application" : "testQuery",
         * }, function(json){
         *
         * });
         */
        "upload": function (options, callback, async) {
            MWF.xDesktop.requireApp("query.Query", "Importer", function () {
                var importer = new MWF.xApplication.query.Query.Importer(_form.app.content, options, {}, _form.app, _form.Macro);
                importer.addEvent("afterImport", function (data) {
                    if(callback)callback(data);
                });
                importer.load();
            }.bind(this));

            // MWF.Actions.load("x_query_assemble_surface").StatementAction.executeV2(
            //     statement.name, statement.mode || "data", statement.page || 1, statement.pageSize || 20, obj,
            //     function (json) {
            //         if (callback) callback(json);
            //     }, null, async);
        },
        /**
         * 根据指定的导入模型下载Excel模板。
         * @method downloadTemplate
         * @static
         * @param {Object} options - 要执行的导入模型的选项。数据格式如下:
         * <pre><code class='language-js'>{
         *  "name" : "testImporter", //(String)必选,导入模型的名称、别名或ID
         *  "application" : "testQuery" //(String)必选,导入模型所在应用的名称、别名或ID
         * }
         * </code></pre>
         * @param {String} fileName - 导出的Excel名称
         * @param {Function} callback - 整理好数据,在导出之前执行的方法,可接收参数如下:
         * <pre><code class='language-js'>{
         *  "data" : ["标题","拟稿人"], //导出的表头数组
         *  "colWidthArray" : [200, 150] //列宽度
         * }
         * </code></pre>
         * @o2syntax
         * this.importer.downloadTemplate(object, fileName, callback);
         * @example
         * this.importer.downloadTemplate({
         *  "name": "testImporter",
         *  "application" : "testQuery",
         * },"导入模板", function( object ){
         *     //添加一项
         *     object.data[0].push("备注");
         *     object.colWidthArray.push(300)
         * });
         */
        "downloadTemplate": function(options, fileName,callback){
            MWF.xDesktop.requireApp("query.Query", "Importer", function () {
                var importer = new MWF.xApplication.query.Query.Importer(_form.app.content, options, {}, _form.app, _form.Macro);
                importer.downloadTemplate(fileName, callback);
            }.bind(this));
        }
    };

    //include 引用脚本
    //optionsOrName : {
    //  type : "", 默认为portal, 可以为 portal  process  cms
    //  application : "", 门户/流程/CMS的名称/别名/id, 默认为当前应用
    //  name : "" // 脚本名称/别名/id
    //}
    //或者name: "" // 脚本名称/别名/id
    // if( !window.includedScripts ){
    //     var includedScripts = window.includedScripts = [];
    // }else{
    //     var includedScripts = window.includedScripts;
    // }
        var _getScriptAction = function ( type ){
            var scriptAction;
            switch (type) {
                case "portal" :
                    if (this.scriptActionPortal) {
                        scriptAction = this.scriptActionPortal;
                    } else {
                        MWF.require("MWF.xScript.Actions.PortalScriptActions", null, false);
                        scriptAction = this.scriptActionPortal = new MWF.xScript.Actions.PortalScriptActions();
                    }
                    break;
                case "process" :
                    if (this.scriptActionProcess) {
                        scriptAction = this.scriptActionProcess;
                    } else {
                        MWF.require("MWF.xScript.Actions.ScriptActions", null, false);
                        scriptAction = this.scriptActionProcess = new MWF.xScript.Actions.ScriptActions();
                    }
                    break;
                case "cms" :
                    if (this.scriptActionCMS) {
                        scriptAction = this.scriptActionCMS;
                    } else {
                        MWF.require("MWF.xScript.Actions.CMSScriptActions", null, false);
                        scriptAction = this.scriptActionCMS = new MWF.xScript.Actions.CMSScriptActions();
                    }
                    break;
                case "service" :
                    if (this.scriptActionService) {
                        scriptAction = this.scriptActionService;
                    } else {
                        MWF.require("MWF.xScript.Actions.ServiceScriptActions", null, false);
                        scriptAction = this.scriptActionService = new MWF.xScript.Actions.ServiceScriptActions();
                    }
                    break;
            }
            return scriptAction;
        }


        //缓存名称、别名、id
        var _parseScriptImportList = function (json, type){
            var includedScripts = [];
            var importedList = json.data.importedList || [];
            importedList.each(function (flag) {
                if (type === "portal") {
                    includedScripts.push(type + "-" + json.data.portal + "-" + flag);
                    if (json.data.portalName) includedScripts.push(type + "-" + json.data.portalName + "-" + flag);
                    if (json.data.portalAlias) includedScripts.push(type + "-" + json.data.portalAlias + "-" + flag);
                } else if (type === "cms") {
                    includedScripts.push(type + "-" + json.data.appId + "-" + flag);
                    if (json.data.appName) includedScripts.push(type + "-" + json.data.appName + "-" + flag);
                    if (json.data.appAlias) includedScripts.push(type + "-" + json.data.appAlias + "-" + flag);
                } else if (type === "process") {
                    includedScripts.push(type + "-" + json.data.application + "-" + flag);
                    if (json.data.appName) includedScripts.push(type + "-" + json.data.appName + "-" + flag);
                    if (json.data.appAlias) includedScripts.push(type + "-" + json.data.appAlias + "-" + flag);
                }else if (type === "service") {
                    includedScripts.push(type + "-" + flag);
                }
            });
            return includedScripts.concat(importedList);
        }

    var includedScripts = [];
    var _includeSingle = function (optionsOrName, callback, async) {
        var options = optionsOrName;
        if (typeOf(options) === "string") {
            options = { name: options };
        }
        var name = options.name;
        var type;
        if( options.type === "service" ){
            type = options.type;
        }else{
            type = (options.type && options.application) ? options.type : "portal";
        }
        var application = options.application || _form.json.application;
        var key = type + "-" + application + "-" + name;
        if( type === "service" ){
            key = type + "-" + name;
        }

        //js 加载过就不重新加载了
        if (includedScripts.indexOf(key) > -1) {
            if (callback) callback.apply(this);
            return;
        }

        var successCallback = function (json) {
                if (json.data) {
                    includedScripts.push(key);
                    includedScripts = includedScripts.concat( _parseScriptImportList(json, type) );
                    if( (options.enableAnonymous || options.anonymous ) && type === "cms" ){
                        MWF.CMSMacro.exec(json.data.text, this)
                    }else{
                        MWF.Macro.exec(json.data.text, this);
                    }
                    if (callback) callback.apply(this);
                } else {
                    if (callback) callback.apply(this);
                }
            }.bind(this);

            if (( options.enableAnonymous || options.anonymous ) && type === "cms") {
                o2.Actions.load("x_cms_assemble_control").ScriptAnonymousAction.getWithAppWithName(application, name, successCallback, null, !!async);
            } else {
                var scriptAction = _getScriptAction.call(this, type);
            if( type === "service" ){
                scriptAction.getScriptByName(name, includedScripts, successCallback, null, !!async);
            }else{
                scriptAction.getScriptByName(application, name, includedScripts, successCallback, null, !!async);
            }
        }
    };
    this.include = function( optionsOrName , callback, async){
        if (o2.typeOf(optionsOrName)==="array"){
            if (!!async){
                var count = optionsOrName.length;
                var loaded = 0;
                optionsOrName.each(function(option){
                    _includeSingle.apply(this, [option, function(){
                        loaded++;
                        if (loaded>=count) if (callback) callback.apply(this);
                    }.bind(this), true]);
                }.bind(this));

            }else{
                optionsOrName.each(function(option){
                    _includeSingle.apply(this, [option]);
                }.bind(this));
                if (callback) callback.apply(this);
            }
        }else{
            _includeSingle.apply(this, [optionsOrName , callback, async])
        }
    };

    var includedSourceMap = {};
    var _includeSource = function (optionsOrName, callback, async, fileType) {
        var options = typeOf(optionsOrName) === "string" ? {name: optionsOrName} : optionsOrName;
        var name = options.name;
        var type = options.type === "service" ? options.type : ((options.type && options.application) ? options.type : "portal");
        var application = options.application || _form.json.application;
        var key = type === "service" ? (type + "-" + name) : (type + "-" + application + "-" + name);
            var data, result;
        if( includedSourceMap[key] ){
                data = includedSourceMap[key];
                if(callback)callback( data.text );
                return !!async ? Promise.resolve( data.text ) : data.text;
        }
        var successCallback = function (json) {
            if (json.data) {
                var includeds = [key];
                includeds = includeds.concat( _parseScriptImportList(json, type) );
                includeds.each(function(k){
                    includedSourceMap[k] = json.data;
                })
                result = json.data.text;
                if( fileType === 'json' ){
                    result = JSON.parse(result);
                }
                if (callback) callback.call(this, result);
            } else {
                result = '';
                if (callback) callback.call(this, '');
            }
            return result;
        }.bind(this);
        var p;
        if (( options.enableAnonymous || options.anonymous ) && type === "cms") {
            p = o2.Actions.load("x_cms_assemble_control").ScriptAnonymousAction.getWithAppWithName(application, name, !!async ? null : successCallback, null, !!async);
        } else {
            var scriptAction = _getScriptAction.call(this, type);
            if( type === "service" ){
                p = scriptAction.getScriptByName(name, includedScripts, successCallback, !!async ? null : successCallback, null, !!async);
            }else{
                p = scriptAction.getScriptByName(application, name, includedScripts, !!async ? null : successCallback, null, !!async);
            }
        }
        return !!async ? p.then( successCallback ) : result;
    };
    this.includeHtml = function (optionsOrName, callback, async){
         return _includeSource.apply(this, [optionsOrName, callback, async!==false, 'html'])
    };
    this.includeJson = function (optionsOrName, callback, async){
          return _includeSource.apply(this, [optionsOrName, callback, async!==false, 'json'])
    };
    this.includeCss = function (optionsOrName, callback, async){
         return _includeSource.apply(this, [optionsOrName, callback, async!==false, 'css']);
    };

    this.define = function (name, fun, overwrite) {
        var over = true;
        if (overwrite === false) over = false;
        var o = {};
        o[name] = { "value": fun, "configurable": over };
        MWF.defineProperties(this, o);
    }.bind(this);

    this.exec = function(text, bind, arg=null, throwError=true){
        const b = bind || globalThis;
        const p = (arg) ? Object.values(arg) : [];
        const k = (arg) ? Object.keys(arg).join(',') : '';
        try {
            return Function('return function('+k+'){' + text + '}')().apply(b, p);
        }catch(e){
            if (throwError) throw e;
            console.error(e);
            return '';
        }
    }


    //仅前台对象-----------------------------------------
    //form

    /**
     * 当查询设计中使用了select语句,并且配置了视图,可以在查询视图中使用本章API。<br/>
     * queryStatement对象在查询视图中可用。它的很多方法与queryView类似。<b>(仅前端脚本可用)</b><br/>
     * @module queryStatement
     * @o2cn 查询视图
     * @o2category web
     * @o2range {QueryStatement}
     * @o2ordernumber 80
     * @borrows module:queryView.confirm as confirm
     * @borrows module:queryView.alert as alert
     * @borrows module:queryView.notice as notice
     * @borrows module:queryView.dialog as dialog
     * @borrows module:queryView.selectOrg as selectOrg
     * @borrows module:queryView.addEvent as addEvent
     * @borrows module:queryView.openWork as openWork
     * @borrows module:queryView.openJob as openJob
     * @borrows module:queryView.openDocument as openDocument
     * @borrows module:queryView.openPortal as openPortal
     * @borrows module:queryView.openCMS as openCMS
     * @borrows module:queryView.openProcess as openProcess
     * @borrows module:queryView.openApplication as openApplication
     * @borrows module:queryView.createDocument as createDocument
     * @borrows module:queryView.startProcess as startProcess
     * @o2syntax
     * //您可以在查询视图中,通过this来获取queryStatement对象,如下:
     * var queryStatement = this.queryStatement;
     */

    /**
     * 当查询视图被嵌入到门户页面、流程表单或内容管理表单的时候,可以在查询视图写脚本的地方通过这个方法来获取页面或表单的上下文。
     * @method getParentEnvironment
     * @memberOf module:queryStatement
     * @static
     * @return {MWF.xScript.Environment|MWF.xScript.CMSEnvironment} 页面或表单的上下文.
     * @o2syntax
     * this.queryStatement.getParentEnvironment();
     * @example
     * var env = this.queryStatement.getParentEnvironment(); //当视图被嵌入到页面的时候,可以在视图里获取页面的上下文
     * env.page.toPortal( "公文门户" ); //调用page的toPage() 跳转到其他门户
     */

    /**
     * 获取查询视图当前页的基本信息。
     * @method getPageInfor
     * @memberOf module:queryStatement
     * @static
     * @return {Object} 当前页的信息,格式如下:
     *<pre><code class='language-js'>{
     *     "pages": 3, //总页数
     *     "perPageCount": 50, //每页的条数
     *     "currentPageNumber": 1 // 当前页数
     * }
     * </code></pre>
     * @o2syntax
     * this.queryStatement.getPageInfor();
     */

    /**
     * 获取当前页的数据。
     * @method getPageData
     * @memberOf module:queryStatement
     * @static
     * @return {Object[]|Array[]} 当前页数据。
     * <div>数据格式和 jpql 语句的写法有关</div>
     * 如:  "select o from table o" 返回 json数组
     *<pre><code class='language-js'>[
     {
        "id" : "id1",
        "title" : "title1"
    },
     {
        "id" : "id2",
        "title" : "title2"
    },
     ...
     *]
     * </code></pre>
     * 如:"select id, title from table o" 返回 二维数组:
     *<pre><code class='language-js'>[
     ["id1", "title1"],
     ["id2", "title2"],
     ...
     *]
     *</code></pre>
     * @o2syntax
     * var data = this.queryStatement.getPageData();
     */

    /**
     * 跳转到指定的页面。
     * @method toPage
     * @memberOf module:queryStatement
     * @static
     * @param {Number} pageNumber - 需要跳转的页码。
     * @param {Function} [callback ] - 跳转的页面数据加载完成以后的回调方法。
     * @o2syntax
     * var data = this.queryStatement.toPage( pageNumber, callback );
     * @example
     * // 跳转到第2页并且获取该页的数据。
     * this.queryStatement.toPage( 2, function(){
     *      var data = this.queryStatement.getPageData();
     * }.bind(this) )
     */

    /**
     * 当查询视图设置了允许多选的时候,可以通过这个方法全部选中当前页面的条目。
     * @method selectAll
     * @memberOf module:queryStatement
     * @static
     * @o2syntax
     * this.queryStatement.selectAll();
     */

    /**
     * 当查询视图设置了允许多选的时候,可以通过这个方法取消选中的条目。
     * @method unSelectAll
     * @memberOf module:queryStatement
     * @static
     * @o2syntax
     * this.queryStatement.unSelectAll();
     */

    /**
     * 获取选中的条目的数据。
     * @method getSelectedData
     * @memberOf module:queryStatement
     * @static
     * @return {Object[]|Array[]} 选中的条目的数据。
     * <div>数据格式和 jpql 语句的写法有关</div>
     * 如:  "select o from table o" 返回 json数组
     *<pre><code class='language-js'>[
     {
        "id" : "id1",
        "title" : "title1"
    },
     {
        "id" : "id2",
        "title" : "title2"
    },
     ...
     *]
     * </code></pre>
     * 如:"select id, title from table o" 返回 二维数组:
     *<pre><code class='language-js'>[
     ["id1", "title1"],
     ["id2", "title2"],
     ...
     *]
     *</code></pre>
     * @o2syntax
     * var data = this.queryStatement.getSelectedData();
     */

    /**获取queryStatement对应的DOM对象。
     * @method node
     * @static
     * @methodOf module:queryStatement
     * @see module:form.node
     */


    /**
     * 重新加载查询视图。
     * @method reload
     * @methodOf module:queryStatement
     * @static
     * @o2syntax
     * this.queryStatement.reload( callback );
     */


    /**
     * queryView对象可在视图中可用。它的很多方法与form类似。<b>(仅前端脚本可用)</b><br/>
     * @module queryView
     * @o2cn 视图
     * @o2category web
     * @o2range {QueryView}
     * @o2ordernumber 60
     * @o2syntax
     * //您可以在视图中,通过this来获取queryView对象,如下:
     * var queryView = this.queryView;
     */
    this.page = this.form = this.queryView = this.queryStatement = {
        /**
         * 当视图被嵌入到门户页面、流程表单或内容管理表单的时候,可以在视图可以写脚本的地方中通过这个方法来获取页面或表单的上下文。
         * @method getParentEnvironment
         * @static
         * @return {MWF.xScript.Environment|MWF.xScript.CMSEnvironment} 页面或表单的上下文.
         * @o2syntax
         * this.queryView.getParentEnvironment();
         * @example
         * var env = this.queryView.getParentEnvironment(); //当视图被嵌入到页面的时候,可以在视图里获取页面的上下文
         * env.page.toPortal( "公文门户" ); //调用page的toPage() 跳转到其他门户
         */
        "getParentEnvironment" : function () { return _form.getParentEnvironment(); }, //视图嵌入的表单或页面的上下文

        /**
         * 获取查询的配置信息。
         * @method getStatementInfor
         * @memberOf module:queryStatement
         * @static
         * @return {Object} 查询的配置信息.
         * <pre><code class='language-js'>{
         *    "query": "26d21c71-5114-4496-8ca1-a69e56324841", //所属应用id
         *    "id": "ee334220-66d3-4f78-afce-8ccf6b995c8c", //查询id
         *    "name": "测试查询", //名称
         *    "alias": "", //别名
         *    "description": "", //描述
         *    "table": "", //自建表的id
         *    "entityClassName": "com.x.processplatform.core.entity.content.Task", //系统表表名
         *    "entityCategory": "official", //表类型 official(系统表) 或 dynamic(自建表)
         *    "format": "jpql", //语句类型,jpql 或者 script(脚本) , v8.0后还有 sql, sqlScript
         *    "type": "select", //select/update/delete
         *    "data": "SELECT o FROM Task o where o.person = :person", //查询语句
         *    "countData": "SELECT count(o.id) FROM Task o where o.person = :person", //总数语句
         *    "countScriptText" : "", //总数语句脚本
         *    "scriptText" : "", //查询语句脚本
         *    "viewJson": { ... } //视图相关信息
         * }</code></pre>
         * @o2syntax
         * this.queryStatement.getStatementInfor();
         */
        "getStatementInfor" : function () { return _form.getStatementInfor ? _form.getStatementInfor() : null; },

        /**
         * 获取查询的配置信息。
         * @method getViewInfor
         * @memberOf module:queryView
         * @static
         * @return {Object} 视图的配置信息.
         * <pre><code class='language-js'>{
         *      "application": "db9fc893-7dbc-4e0f-a617-99089d2c6323", //视图所在应用
         *      "query": "db9fc893-7dbc-4e0f-a617-99089d2c6323",  //视图所在应用,同application
         *      "name": "视图自定义测试", //视图名称
         *      "viewName": "视图自定义测试",  //视图名称,同name
         *      "isExpand": "no",  //如果有分类,默认是否展开开
         *      "id": "705ce967-2f9c-425c-8873-3bd729249e1d", //视图id
         *      "alias": "", //视图别名
         *      "description": "",  //视图描述
         *      "display": true, //视图是否显示
         *      "type": "cms", //视图嵌入的数据类型, cms 或 process
         *      "count": 2000, //最多返回2000条
         *      "pageSize": 20, //每页的条数
         *      "createTime": "2019-09-02 10:18:27",
         *      "updateTime": "2020-03-26 15:53:03"
         * }</code></pre>
         * @o2syntax
         * this.queryView.getViewInfor();
         */
        "getViewInfor" : function () { return _form.getViewInfor(); },

        /**
         * 获取视图当前页的基本信息。
         * @method getPageInfor
         * @memberOf module:queryView
         * @static
         * @return {Object} 当前页的信息,格式如下:
         *<pre><code class='language-js'>{
         *     "pages": 3, //总页数
         *     "perPageCount": 50, //每页的条数
         *     "currentPageNumber": 1 // 当前页数
         * }
         * </code></pre>
         * @o2syntax
         * this.queryView.getPageInfor();
         */
        "getPageInfor" : function () { return _form.getPageInfor(); },

        /**
         * 获取当前页的数据。
         * @method getPageData
         * @memberOf module:queryView
         * @static
         * @return {Object[]} 当前页数据。
         * <div>没有分类时候,数据格式如下:</div>
         *<pre><code class='language-js'>[
         *   {
         *       "bundle": "099ed3c9-dfbc-4094-a8b7-5bfd6c5f7070", //cms 的 documentId, process 的 jobId
         *      "data": {  //视图中配置的数据
         *        "title": "考勤管理-配置-统计周期设置", //列名称及列值
         *        "time": "2018-08-25 11:29:45"
         *      }
         *    },
         *   ...
         *]
         * </code></pre>
         * 有分类的时候,数据格式如下:
         *<pre><code class='language-js'>[
         *  {
         *   "group": "工作日志",  //分类1
         *   "list": [  //分类下的数据
         *     {
         *       "bundle": "001257be-725a-43cf-9679-3892bbab696a", //cms 的 documentId, process 的 jobId
         *       "data": { //视图中配置的数据
         *         "title": "标题",  //列名称及列值
         *         "time": "2018-07-31 15:39:13",
         *         "category": "工作日志"
         *       }
         *     },
         *     ...
         *   ]
         *  },
         *  ...
         *]
         *</code></pre>
         * @o2syntax
         * var data = this.queryView.getPageData();
         */
        "getPageData" : function () { return _form.getPageData(); },

        /**
         * 跳转到指定的页面。
         * @method toPage
         * @memberOf module:queryView
         * @static
         * @param {Number} pageNumber - 需要跳转的页码。
         * @param {Function} [callback ] - 跳转的页面数据加载完成以后的回调方法。
         * @o2syntax
         * var data = this.queryView.toPage( pageNumber, callback );
         * @example
         * // 跳转到第2页并且获取该页的数据。
         * this.queryView.toPage( 2, function(){
         *      var data = this.queryView.getPageData();
         * }.bind(this) )
         */
        "toPage" : function ( pageNumber, callback ) { return _form.toPage(pageNumber, callback); },

        /**
         * 当视图设置了允许多选的时候,可以通过这个方法全部选中当前页面的条目。
         * @method selectAll
         * @memberOf module:queryView
         * @static
         * @o2syntax
         * this.queryView.selectAll();
         */
        "selectAll" : function () { return _form.selectAll(); },

        /**
         * 当视图设置了允许多选的时候,可以通过这个方法取消选中的条目。
         * @method unSelectAll
         * @memberOf module:queryView
         * @static
         * @o2syntax
         * this.queryView.unSelectAll();
         */
        "unSelectAll" : function () { return _form.unSelectAll(); },

        /**
         * 获取选中的条目的数据。
         * @method getSelectedData
         * @memberOf module:queryView
         * @static
         * @return {Object[]} 选中的条目的数据。
         * <div>格式如下:</div>
         * <pre><code class='language-js'>
         * [
         {
            "bundle": "099ed3c9-dfbc-4094-a8b7-5bfd6c5f7070", //cms 的 documentId, process 的 jobId
            "data": {  //视图中配置的数据
              "title": "考勤管理-配置-统计周期设置", //列名称及列值
              "time": "2018-08-25 11:29:45"
            }
          },
         ...
         * ]
         </code></pre>
         * @o2syntax
         * var data = this.queryView.getSelectedData();
         */
        "getSelectedData" : function () { return _form.getSelectedData(); },

        /**
         * 设置视图的过滤条件,该方法不能修改视图中默认的过滤条件(在开发视图的时候添加的过滤条件),而是在这上面新增。
         * @method setFilter
         * @memberOf module:queryView
         * @static
         * @param {(ViewFilter[]|ViewFilter|Null)} [filter] 过滤条件。<br/>
         * 当不传参数、参数为null或为空数组的情况下,表示清空非视图默认的过滤条件。<br/>
         * 如果传入对象或者非空数组的时候,参数如下:
         * <pre><code class='language-js'>[
         *    {
         *       "logic":"and",
         *       "path":"$work.title",
         *       "comparison":"like",
         *       "value":"7月",
         *       "formatType":"textValue"
         *   }
         *]
         * </code></pre>
         * @param {Function} [callback] 过滤完成并重新加载数据后的回调方法。
         * @o2syntax
         * this.queryView.setFilter( filter );
         */
        "setFilter" : function ( filter, callback ) { return _form.setFilter(filter, callback); },

        /**
         * 增加查询语句where子句的过滤条件。
         * @method setStatementFilter
         * @memberOf module:queryStatement
         * @static
         * @param {(StatementFilter[]|Null)} [filter] 过滤条件。<br/>
         * 过滤条件。当不传参数、参数为null或为空数组的情况下,表示清空非视图默认的过滤条件。<br/>
         * 如果传入非空数组的时候,参数如下:
         * <pre><code class='language-js'>[
         *    {
         *      "path":"o.title", //查询语句格式为jpql使用o.title,为原生sql中使用xtitle
         *      "comparison":"like",
         *      "value":"关于",
         *      "formatType":"textValue"
         *  }
         *]
         * </code></pre>
         * @param {StatementParameter} [parameter] 过滤条件。对查询语句where子句的形如":person"的参数部分进行赋值,参数如下:
         * <pre><code class='language-js'>
         * //假设语句为 select count(o.id) from Read o where (o.person = :person) and (o.startTime > :startTime) and (o.applicationName like :applicationName) and (o.processName = :processName)。
         * //那么可能的参数如下:
         * {
         *    "person" : "", //出于安全考虑参数名称为下列值时,不需要填写参数值,后台默认赋值,person(当前人),identityList(当前人身份列表),unitList(当前人所在直接组织), unitAllList(当前人所在所有组织), groupList(当前人所在群组),roleList(当前人拥有的角色)。v8.0以后系统自动解析,不需要再传这类参数。
         *    "startTime" : (new Date("2020-01-01")), //如果对比的是日期,需要传入 Date 类型
         *    "applicationName" : "%test%", //如果运算符用的是 like, noLike,模糊查询
         *    "processName" : "test流程", //其他写确定的值
         *    "?1": "关于" //v8.0后查询语句支持问号加数字的传参
         * }
         * </code></pre>
         * @param {Function} [callback] 过滤完成并重新加载数据后的回调方法。
         * @o2syntax
         * this.queryStatement.setStatementFilter( filter, parameter, callback );
         */
        "setStatementFilter" : function ( filter , parameter, callback) { return _form.setFilter(filter, parameter, callback); },

        /**
         * 把当前视图切换成另外一个视图。
         * @method switchView
         * @memberOf module:queryView
         * @static
         * @param {Object} options 需要跳转的参数配置。参数说明如下:
         * <div>下列说明的filter属性参考<a href='global.html#ViewFilter'>ViewFilter</a></div>
         * <pre><code class='language-js'>{
         *     "application": application, //必选,视图的所在应用id
         *     "viewName": viewName, //必选,视图的名称
         *     "filter": [
         *         {
         *            "logic":"and",
         *             "path":"$work.title",
         *             "comparison":"like",
         *             "value":"7月",
         *             "formatType":"textValue"
         *         }
         *     ], //可选,增加视图的过滤条件(ViewFilter),如果不传,则使用原视图的配置;如果需要去掉原视图的配置,则传入空数组 []
         *     "isTitle": "yes", //可选,是否显示t视图的标题行,可选值有:yes no
         *     "select": "none", //可选,是否允许新视图选择,如果不传,则使用原视图的配置, 可选值有: 不允许选择 none, 单选 single,多选 multi
         *     "titleStyles": {
         *       "color" : "red",
         *       "font-size" : "14px"
         *     }, //可选,标题行样式,如果不传,则使用原视图的配置
         *     "itemStyles": {
         *       "color" : "#333",
         *       "font-size" : "12px"
         *     }, //可选,内容行样式,如果不传,则使用原视图的配置
         *     "isExpand": "no", //可选,默认是否展开分类,如果不传,则使用原视图的配置, 可选值有:yes no
         *   }
         * </code></pre>
         * @o2syntax
         * this.queryView.switchView( options );
         */
        "switchView" : function ( options ) { return _form.switchView(options); },

        /**
         * 把当前查询视图切换成另外一个查询视图。
         * @method switchStatement
         * @memberOf module:queryStatement
         * @static
         * @param {Object} options 需要跳转的参数配置。参数说明如下:
         * <div>下列说明的filter属性参考<a href='global.html#StatementFilter'>StatementFilter</a>,
         * parameter属性参考<a href='global.html#StatementParameter'>StatementParameter</a></div>
         * <pre><code class='language-js'>this.queryStatement.switchStatement({
         *     "statementId": statementId, //必选,查询的名称、别名、id
         *     "isTitle": "yes", //可选,是否显示视图的标题行,可选值有:yes no
         *     "select": "multi", //可选,是否允许新视图选择,如果不传,则使用原视图的配置, 可选值有: 不允许选择 none, 单选 single,多选 multi
         *     "showActionbar": false, //可选,是否显示操作条
         *     "filter": [  //可选,增加查询语句where子句的过滤条件
         *       {
         *         "path": "o.title", //查询语句格式为jpql使用o.title,为原生sql中使用xtitle
         *         "title": "标题",
         *         "type": "filter",
         *         "comparison": "like",
         *         "formatType": "textValue",
         *         "value": "测试"
         *       }
         *     ],
         *     //假设语句为 select count(o.id) from Read o where (o.person = :person) and (o.startTime > :startTime) and (o.applicationName like :applicationName) and (o.processName = :processName)
         *     "parameter" : { //可选,对查询语句where语句的形如":person"的参数部分进行赋值
         *       "person" : "", //出于安全考虑参数名称为下列值时,不需要填写参数值,后台默认赋值,person(当前人),identityList(当前人身份列表),unitList(当前人所在直接组织), unitAllList(当前人所在所有组织), groupList(当前人所在群组),roleList(当前人拥有的角色)。v8.0以后系统自动解析,不需要再传这类参数。
         *       "startTime" : (new Date("2020-01-01")), //如果对比的是日期,需要传入 Date 类型
         *       "applicationName" : "%test%", //如果运算符用的是 like, noLike,模糊查询
         *       "processName" : "test流程" //其他写确定的值
         *       "?1": "关于" //v8.0后查询语句支持问号加数字的传参
         *     }
         *   })
         * </code></pre>
         * @o2syntax
         * this.queryStatement.switchStatement( options );
         */
        "switchStatement" : function ( options ) { if(_form.switchStatement)_form.switchStatement(options) ; },

        /**
         * 重新加载视图。
         * @method reload
         * @methodOf module:queryView
         * @static
         * @o2syntax
         * this.queryView.reload( callback );
         */
        "reload" : function ( callback ) { _form.reload( callback ); },

        // "getInfor": function () { return ev.pageInfor; },
        // "infor": ev.pageInfor,
        /**获取打开当前页面的component对象。
         * @method getApp
         * @static
         * @see module:form.getApp
         */
        getApp: function () { return _form.app; },
        // "app": _form.app,

        /**获取queryView对应的DOM对象。
         * @method node
         * @static
         * @methodOf module:queryView
         * @see module:form.node
         */
        "node": function () { return _form.node; },

        // "get": function (name) { return (_form.all) ? _form.all[name] : null; },
        // "getWidgetModule": function (widget, moduleName) {
        //     if (!_form.widgetModules || !_form.widgetModules[widget]) return null;
        //     var module = _form.widgetModules[widget][moduleName];
        //     return module || null;
        // },
        // "getField": function (name) { return _forms[name]; },
        // "getAction": function () { return _form.workAction },
        "getDesktop": function () { return _form.app ? _form.app.desktop : null},
        // "getData": function () { return new MWF.xScript.JSONData(_form.getData()); },
        //"save": function(callback){_form.saveWork(callback);},
        // "close": function () { _form.closeWork(); },

        // "print": function (application, form) {
        //     _form.printWork(application, form);
        // },

        /**弹出一个确认框。
         * @method confirm
         * @static
         * @methodOf module:queryView
         * @see module:form.confirm
         */
        "confirm": function (type, title, text, width, height, ok, cancel, callback, mask, style) {
            // var p = MWF.getCenter({"x": width, "y": height});
            // e = {"event": {"clientX": p.x,"x": p.x,"clientY": p.y,"y": p.y}};
            // _form.confirm(type, e, title, text, width, height, ok, cancel, callback, mask, style);
            if ((arguments.length <= 1) || o2.typeOf(arguments[1]) === "string") {
                var p = MWF.getCenter({ "x": width, "y": height });
                e = { "event": { "clientX": p.x, "x": p.x, "clientY": p.y, "y": p.y } };
                _form.confirm(type, e, title, text, width, height, ok, cancel, callback, mask, style);
            } else {
                e = (arguments.length > 1) ? arguments[1] : null;
                title = (arguments.length > 2) ? arguments[2] : null;
                text = (arguments.length > 3) ? arguments[3] : null;
                width = (arguments.length > 4) ? arguments[4] : null;
                height = (arguments.length > 5) ? arguments[5] : null;
                ok = (arguments.length > 6) ? arguments[6] : null;
                cancel = (arguments.length > 7) ? arguments[7] : null;
                callback = (arguments.length > 8) ? arguments[8] : null;
                mask = (arguments.length > 9) ? arguments[9] : null;
                style = (arguments.length > 10) ? arguments[10] : null;
                _form.confirm(type, e, title, text, width, height, ok, cancel, callback, mask, style);
            }
        },

        /**显示一个带关闭按钮的信息框。
         * @method alert
         * @static
         * @methodOf module:queryView
         * @see module:form.alert
         */
        "alert": function(type, title, text, width, height){
            _form.alert(type, title, text, width, height);
        },

        /**显示一个信息框。
         * @method notice
         * @static
         * @methodOf module:queryView
         * @see module:form.notice
         */
        "notice": function (content, type, target, where, offset, option) {
            _form.notice(content, type, target, where, offset, option);
        },

        /**打开一个对话框
         * @method dialog
         * @static
         * @see module:form.dialog
         */
        "dialog": function ( options ) {
            return  _form.dialog( options );
        },

        /**打开人员组织选择界面
         * @method selectOrg
         * @static
         * @see module:form.selectOrg
         */
        "selectOrg": function ( container, options,  delayLoad) {
            if( !container )container = _form.app.content;
            MWF.xDesktop.requireApp("Selector", "package", null, false);
            return new MWF.O2Selector(container, options, delayLoad);
        },

        /** 给视图添加事件。
         * @method addEvent
         * @static
         * @methodOf module:queryView
         * @see module:form.addEvent
         */
        "addEvent": function (e, f) { _form.addEvent(e, f); },

        // "openWindow": function (form, app) {
        //     _form.openWindow(form, app);
        // },
        // "toPage": function (name, par, nohis) {
        //     _form.app.toPage(name, par, nohis);
        // },
        // "toPortal": function (portal, page, par) {
        //     _form.app.toPortal(portal, page, par);
        // },

        /**打开一个在流转或已完成的流程实例。
         * @method openWork
         * @static
         * @methodOf module:queryView
         * @see module:form.openWork
         */
        "openWork": function (id, completedId, title, options) {
            var op = options || {};
            op.workId = id;
            op.workCompletedId = completedId;
            op.docTitle = title;
            op.appId = "process.Work" + (op.workId || op.workCompletedId);
            return layout.desktop.openApplication(this.event, "process.Work", op);
        },

        /**根据流程的jobId打开工作。
         * @method openJob
         * @static
         * @methodOf module:queryView
         * @see module:form.openJob
         */
        "openJob": function (id, choice, options, callback) {
            var workData = null, handel;
            o2.Actions.get("x_processplatform_assemble_surface").listWorkByJob(id, function(json){
                if (json.data) workData = json.data;
            }.bind(this), null, false);

            if( !layout.inBrowser && o2.typeOf(callback) === "function" ){
                if( !options )options = {};
                var queryLoad = options.onQueryLoad;
                options.onQueryLoad = function () {
                    if( o2.typeOf(queryLoad) === "function" )queryLoad.call(this);
                    callback(this);
                }
            }

            runCallback = function ( handel ) {
                if( o2.typeOf(callback) === "function" ) {
                    if (layout.inBrowser) {
                        callback(handel);
                    } else if (options && options.appId) {
                        if (layout.desktop && layout.desktop.apps && layout.desktop.apps[options.appId]) {
                            callback(layout.desktop.apps[options.appId], true);
                        }else{
                            callback(handel, false);
                        }
                    }else{
                        callback(handel, false);
                    }
                }
            };

            if (workData){
                var len = workData.workList.length + workData.workCompletedList.length;
                if (len){
                    if (len>1 && choice){
                        var node = new Element("div", {"styles": {"padding": "20px", "width": "500px"}}).inject(_form.node);
                        workData.workList.each(function(work){
                            var workNode = new Element("div", {
                                "styles": {
                                    "background": "#ffffff",
                                    "border-radius": "10px",
                                    "clear": "both",
                                    "margin-bottom": "10px",
                                    "height": "40px",
                                    "padding": "10px 10px"
                                }
                            }).inject(node);
                            var html = "<div style='height: 40px; width: 40px; float: left; background: url(../x_component_process_Xform/$Form/default/icon/work.png) no-repeat center center'></div>" +
                                "<div style='height: 40px; width: 40px; float: right'><div class='MWFAction' style='height: 20px; width: 40px; margin-top: 10px; border: 1px solid #999999; border-radius: 5px;text-align: center; cursor: pointer'>"+o2.LP.widget.open+"</div></div>"+
                                "<div style='height: 20px; line-height: 20px; margin: 0px 40px'>"+work.title+"</div>" +
                                "<div style='margin: 0px 40px'><div style='color:#999999; float: left; margin-right: 10px'>"+work.activityName+"</div>" +
                                "<div style='color:#999999; float: left; margin-right: 10px'>"+work.activityArrivedTime+"</div>" +
                                "<div style='color:#999999; float: left; margin-right: 10px'>"+(work.manualTaskIdentityText || "")+"</div></div>";
                            workNode.set("html", html);
                            var action = workNode.getElement(".MWFAction");
                            action.store("work", work);
                            action.addEvent("click", function(e){
                                var work = e.target.retrieve("work");
                                if (work){
                                    handel =  this.openWork(work.id, null, work.title, options);
                                    runCallback( handel );
                                }
                                dlg.close();
                            }.bind(this));

                        }.bind(this));
                        workData.workCompletedList.each(function(work){
                            var workNode = new Element("div", {
                                "styles": {
                                    "background": "#ffffff",
                                    "border-radius": "10px",
                                    "clear": "both",
                                    "margin-bottom": "10px",
                                    "height": "40px",
                                    "padding": "10px 10px"
                                }
                            }).inject(node);
                            var html = "<div style='height: 40px; width: 40px; float: left; background: url(../x_component_process_Xform/$Form/default/icon/work.png) no-repeat center center'></div>" +
                                "<div style='height: 40px; width: 40px; float: right'><div class='MWFAction' style='height: 20px; width: 40px; margin-top: 10px; border: 1px solid #999999; border-radius: 5px;text-align: center; cursor: pointer'>"+o2.LP.widget.open+"</div></div>"+
                                "<div style='height: 20px; line-height: 20px; margin: 0px 40px'>"+work.title+"</div>" +
                                "<div style='margin: 0px 40px'><div style='color:#999999; float: left; margin-right: 10px'>"+o2.LP.widget.workcompleted+"</div>" +
                                "<div style='color:#999999; float: left; margin-right: 10px'>"+work.completedTime+"</div>";
                            workNode.set("html", html);
                            var action = workNode.getElement(".MWFAction");
                            action.store("work", work);
                            action.addEvent("click", function(e){
                                var work = e.target.retrieve("work");
                                if (work){
                                    handel =  this.openWork(null, work.id, work.title, options);
                                    runCallback( handel );
                                }
                                dlg.close();
                            }.bind(this));

                        }.bind(this));
                        var height = node.getSize().y+20;
                        if (height>600) height = 600;

                        var dlg = o2.DL.open({
                            "title": o2.LP.widget.choiceWork,
                            "style" : "user",
                            "isResize": false,
                            "content": node,
                            "buttonList": [
                                {
                                    "type" : "cancel",
                                    "text": o2.LP.widget.close,
                                    "action": function(){dlg.close();}
                                }
                            ]
                        });
                    }else{
                        if (workData.workList.length){
                            var work =  workData.workList[0];
                            handel = this.openWork(work.id, null, work.title, options);
                            runCallback( handel );
                            return handel;
                        }else{
                            var work =  workData.workCompletedList[0];
                            handel = this.openWork(null, work.id, work.title, options);
                            runCallback( handel );
                            return handel;
                        }
                    }
                }else{
                    runCallback(new Error("Can't open this Job", {
                        cause: workData
                    }));
                }
            }else{
                runCallback(new Error("Can't open this Job", {
                    cause: workData
                }));
            }
        },

        /**打开一个内容管理文档。
         * @method openDocument
         * @static
         * @methodOf module:queryView
         * @see module:form.openDocument
         */
        "openDocument": function (id, title, options) {
            var op = options || {};
            op.documentId = id;
            op.docTitle = title || "";
            op.appId = (op.appId) || ("cms.Document"+id);
            if( op.onPostPublish ){
                op.postPublish = op.onPostPublish;
                delete op.onPostPublish;
            }
            if( op.onAfterPublish ){
                op.afterPublish = op.onAfterPublish;
                delete op.onAfterPublish;
            }
            if( op.onAfterSave ){
                op.afterSave = op.onAfterSave;
                delete op.onAfterSave;
            }
            if( op.onBeforeClose ){
                op.beforeClose = op.onBeforeClose;
                delete op.onBeforeClose;
            }
            if( op.onPostDelete ){
                op.postDelete = op.onPostDelete;
                delete op.onPostDelete;
            }
            return layout.desktop.openApplication(this.event, "cms.Document", op);
        },

        /**打开一个门户页面。
         * @method openPortal
         * @static
         * @methodOf module:queryView
         * @see module:form.openPortal
         */
        "openPortal": function (name, page, par) {
            var action = MWF.Actions.get("x_portal_assemble_surface");
            action.getApplication(name, function (json) {
                if (json.data) {
                    if (page) {
                        action.getPageByName(page, json.data.id, function (pageJson) {
                            var pageId = (pageJson.data) ? pageJson.data.id : "";
                            layout.desktop.openApplication(null, "portal.Portal", {
                                "portalId": json.data.id,
                                "pageId": pageId,
                                "parameters": par,
                                "appId": (par && par.appId) || ("portal.Portal" + json.data.id + pageId)
                            })
                        });
                    } else {
                        layout.desktop.openApplication(null, "portal.Portal", {
                            "portalId": json.data.id,
                            "parameters": par,
                            "appId": (par && par.appId) || ("portal.Portal" + json.data.id)
                        })
                    }
                }

            });
        },

        /**打开一个内容管理栏目。
         * @method openCMS
         * @static
         * @methodOf module:queryView
         * @see module:form.openCMS
         */
        "openCMS": function (name) {
            var action = MWF.Actions.get("x_cms_assemble_control");
            action.getColumn(name, function (json) {
                if (json.data) {
                    layout.desktop.openApplication(null, "cms.Module", {
                        "columnId": json.data.id,
                        "appId": "cms.Module" + json.data.id
                    });
                }
            });
        },

        /**打开一个流程应用。
         * @method openProcess
         * @static
         * @methodOf module:queryView
         * @see module:form.openProcess
         */
        "openProcess": function (name) {
            var action = MWF.Actions.get("x_processplatform_assemble_surface");
            action.getApplication(name, function (json) {
                if (json.data) {
                    layout.desktop.openApplication(null, "process.Application", {
                        "id": json.data.id,
                        "appId": "process.Application" + json.data.id
                    });
                }
            });
        },

        /**打开一个任意一个component应用。
         * @method openApplication
         * @static
         * @methodOf module:queryView
         * @see module:form.openApplication
         */
        "openApplication": function (name, options, status) {
            return layout.desktop.openApplication(null, name, options, status);
        },

        /**创建一个内容管理文档。
         * @method createDocument
         * @static
         * @methodOf module:queryView
         * @see module:form.createDocument
         */
        "createDocument": function (columnOrOptions, category, data, identity, callback, target, latest, selectColumnEnable, ignoreTitle, restrictToColumn) {
            var column = columnOrOptions;
            var onAfterPublish, onPostPublish;
            if (typeOf(columnOrOptions) == "object") {
                column = columnOrOptions.column;
                category = columnOrOptions.category;
                data = columnOrOptions.data;
                identity = columnOrOptions.identity;
                callback = columnOrOptions.callback;
                target = columnOrOptions.target;
                latest = columnOrOptions.latest;
                selectColumnEnable = columnOrOptions.selectColumnEnable;
                ignoreTitle = columnOrOptions.ignoreTitle;
                restrictToColumn = columnOrOptions.restrictToColumn;
                onAfterPublish = columnOrOptions.onAfterPublish;
                onPostPublish = columnOrOptions.onPostPublish;
            }
            // 移动端 app相关的实现
            if (layout.mobile) {
                var options = {};
                if (column) {
                    options["column"] = column
                }
                if (category) {
                    options["category"] = category
                }
                if (data) {
                    options["data"] = data
                }
                if (identity) {
                    options["identity"] = identity
                }
                if (typeof(latest) == 'undefined' || latest === null) {
                    options["latest"] = true
                } else {
                    options["latest"] = latest
                }
                if (typeof(ignoreTitle) == 'undefined' || ignoreTitle === null) {
                    options["ignoreTitle"] = false
                } else {
                    options["ignoreTitle"] = ignoreTitle
                }
                if (window.o2android && window.o2android.postMessage) {
                        var body = {
                        type: "createO2CmsDocument",
                        data: options
                    };
                    window.o2android.postMessage(JSON.stringify(body));
                    return;
                } else if (window.o2android && window.o2android.createO2CmsDocument){
                    window.o2android.createO2CmsDocument(JSON.stringify(options));
                    return;
                } else if (window.webkit && window.webkit.messageHandlers && window.webkit.messageHandlers.createO2CmsDocument) {
                    window.webkit.messageHandlers.createO2CmsDocument.postMessage(options);
                    return;
                }
            }
            // 下面是pc端
            if (target) {
                if (layout.app && layout.app.inBrowser) {
                    layout.app.content.empty();
                    layout.app = null;
                }
            }

            MWF.xDesktop.requireApp("cms.Index", "Newer", function () {
                var starter = new MWF.xApplication.cms.Index.Newer(null, null, _form.app, null, {
                    "documentData": data,
                    "identity": identity,

                    "ignoreTitle": ignoreTitle === true,
                    "ignoreDrafted": latest === false,
                    "selectColumnEnable": !category || selectColumnEnable === true,
                    "restrictToColumn": restrictToColumn === true || (!!category && selectColumnEnable !== true),

                    "categoryFlag": category, //category id or name
                    "columnFlag": column, //column id or name,
                    "onStarted": function (documentId, data, windowHandle) {
                        if (callback) callback(documentId, data, windowHandle);
                    },
                    "onPostPublish": function () {
                        if(onPostPublish)onPostPublish();
                    },
                    "onAfterPublish": function () {
                        if(onAfterPublish)onAfterPublish();
                    }
                });
                starter.load();
            })
        },

        /** 启动一个流程实例。
         * @method startProcess
         * @static
         * @methodOf module:queryView
         * @see module:form.startProcess
         */
        "startProcess": function (app, process, data, identity, callback, target, latest, afterCreated, skipDraftCheck) {

            if (arguments.length > 2) {
                for (var i = 2; i < arguments.length; i++) {
                    if (typeOf(arguments[i]) == "boolean") {
                        target = arguments[i];
                        break;
                    }
                }
            }

            if (target) {
                if (layout.app && layout.app.inBrowser) {
                    //layout.app.content.empty();
                    layout.app.$openWithSelf = true;
                }
            }
            if (!app || !process){
                var cmpt = this.getApp();
                o2.requireApp([["process.TaskCenter", "lp."+o2.language], ["process.TaskCenter", ""]],"", function(){
                    var obj = {
                        "lp": o2.xApplication.process.TaskCenter.LP,
                        "content": cmpt.content,
                        "addEvent": function(type, fun){
                            cmpt.addEvent(type, fun);
                        },
                        "getAction": function (callback) {
                            if (!this.action) {
                                this.action = o2.Actions.get("x_processplatform_assemble_surface");
                                if (callback) callback();
                            } else {
                                if (callback) callback();
                            }
                        },
                        "desktop": layout.desktop,
                        "refreshAll": function(){},
                        "notice": cmpt.notice,
                    }
                    o2.JSON.get("../x_component_process_TaskCenter/$Main/default/css.wcss", function(data){
                        obj.css = data;
                    }, false);

                    if (!cmpt.processStarter) cmpt.processStarter = new o2.xApplication.process.TaskCenter.Starter(obj);
                    cmpt.processStarter.load({
                        "appFlag": app
                    });
                }, true, true);
                return "";
            }
            MWF.xDesktop.requireApp("process.TaskCenter", "ProcessStarter", null, false);
            var action = MWF.Actions.get("x_processplatform_assemble_surface").getProcessByName(process, app, function (json) {
                if (json.data) {
                    var starter = new MWF.xApplication.process.TaskCenter.ProcessStarter(json.data, _form.app, {
                        "workData": data,
                        "identity": identity,
                        "latest": latest,
                        "skipDraftCheck": skipDraftCheck,
                        "onStarted": function (data, title, processName) {
                            var application;
                            if (data.work){
                                var work = data.work;
                                var options = {
                                    "draft": work,
                                    "draftData":data.data||{},
                                    "appId": "process.Work"+(new o2.widget.UUID).toString(),
                                    "desktopReload": false
                                };
                                if( !layout.inBrowser && afterCreated )options.onPostLoadForm = afterCreated;
                                application = layout.desktop.openApplication(null, "process.Work", options);
                            }else{
                                var currentTask = [];
                                data.each(function(work){
                                    if (work.currentTaskIndex != -1) currentTask.push(work.taskList[work.currentTaskIndex].work);
                                }.bind(this));

                                if (currentTask.length==1){
                                    var options = {"workId": currentTask[0], "appId": currentTask[0]};
                                    if( !layout.inBrowser && afterCreated )options.onPostLoadForm = afterCreated;
                                    application =layout.desktop.openApplication(null, "process.Work", options);
                                }else{}
                            }

                            // var currentTask = [];
                            // data.each(function (work) {
                            //     if (work.currentTaskIndex != -1) currentTask.push(work.taskList[work.currentTaskIndex].work);
                            // }.bind(this));
                            //
                            // if (currentTask.length == 1) {
                            //     var options = { "workId": currentTask[0], "appId": currentTask[0] };
                            //     layout.desktop.openApplication(null, "process.Work", options);
                            // } else { }

                            if (callback) callback(data);

                            if(layout.inBrowser && afterCreated){
                                afterCreated(application)
                            }
                        }.bind(this)
                    });
                    starter.load();
                }
            });
        }
        //"app": _form.app
    };
    // this.form.currentRouteName = _form.json.currentRouteName;
    // this.form.opinion = _form.json.opinion;

    this.target = ev.target;
    this.event = ev.event;
    this.status = ev.status;

    /**
     * 在前端脚本中,可以通过this.session.user来获取当前用户信息。<br/>
     * @module session
     * @o2cn 当前用户
     * @o2category web
     * @o2ordernumber 110
     * @o2syntax
     * var user = this.session.user;
     * @example
     * //获取当前用户信息
     * var user = this.session.user
     * @return {Object} 当前用户信息,内容和格式如下:
     * <pre><code class='language-js'>{
     *    "id": "267a7bcc-f27a-49c8-8364-f1c12061085a",       //人员ID
     *    "genderType": "m",                                  //性别
     *    "icon": "...",                                      //头像
     *    "signature": "",                                    //个人签名
     *    "name": "张三",                                     //姓名
     *    "employee": "zhansan",                                 //员工号
     *    "unique": "zhansan",                                   //唯一标识
     *    "distinguishedName": "xx@zhansan@P",                   //人员全称
     *    "superior": "",                                     //上级人员id
     *    "changePasswordTime": "2017-03-13",                 //修改密码时间
     *    "lastLoginTime": "2019-01-02",                      //最后登录时间
     *    "mail": "zhansan@zoneland.net",                        //邮件地址
     *    "weixin": "",                                       //微信号
     *    "qq": "",                                           //QQ
     *    "mobile": "18057190078",                            //手机号码
     *    "officePhone": "",                                  //办公电话
     *    "createTime": "2017-03-13 12:27:04",                //人员创建时间
     *    "updateTime": "2019-01-02 13:00:04",                //人员修改时间
     *    "token": "...",                         //当前用户token
     *    "roleList": [                                       //人员角色
     *      "ProcessPlatformCreator@ProcessPlatformCreatorSystemRole@R",
     *      "ProcessPlatformManager@ProcessPlatformManagerSystemRole@R",
     *      "Manager@ManagerSystemRole@R"
     *   ],
     *    "identityList": [                                   //人员身份列表
     *      {
     *        "id": "709328c8-44a0-4f5d-a3fa-3c31208232d5",                       //身份ID
     *        "name": "xx",                                                       //身份名称
     *        "unique": "709328c8-44a0-4f5d-a3fa-3c31208232d5",                   //身份唯一标识
     *        "distinguishedName": "xx@709328c8-44a0-4f5d-a3fa-3c31208232d5@I",   //身份全称
     *        "person": "267a7bcc-f27a-49c8-8364-f1c12061085a",                   //人员ID
     *        "unit": "d5356fd4-6675-45ad-9a00-5eff20b83dfa",                     //所属组织ID
     *        "unitName": "开发部",                                               //所属组织名称
     *        "unitLevel": 2,                                                     //所属组织层级
     *        "unitLevelName": "兰德纵横/开发部",                                 //所属组织层次名
     *        "major": true                                                       //是否是主身份
     *      },
     *      {
     *        "id": "343510af-57c2-4a55-a1f2-f30d7af6d284",
     *        "description": "",
     *        "name": "xx",
     *        "unique": "343510af-57c2-4a55-a1f2-f30d7af6d284",
     *        "distinguishedName": "xx@343510af-57c2-4a55-a1f2-f30d7af6d284@I",
     *        "person": "267a7bcc-f27a-49c8-8364-f1c12061085a",
     *        "unit": "108b1b7c-cc78-49ab-9ab1-e67073bd6541",
     *        "unitName": "开发部",
     *        "unitLevel": 2,
     *        "unitLevelName": "浙江兰德纵横/开发部",
     *        "major": false
     *      }
     *    ]
     *  }
     * </code></pre>
     */
    this.session = layout.desktop.session;



    /**
     * 本文档说明如何在前台脚本中使用Actions调用平台的RESTful服务。<br/>
     * 通过访问以下地址来查询服务列表:http://server/x_program_center/jest/list.html (v7.2之前版本需要加端口20030)
     * @module Actions
     * @o2cn 服务调用
     * @o2category web
     * @o2ordernumber 130
     * @o2syntax
     * //获取Actions
     * this.Actions
     */

    /**
     * 本平台不同的服务根可能对应不同的域名或端口,您可以使用this.Actions.getHost来获取服务跟对应的host。
     * @method getHost
     * @methodOf module:Actions
     * @static
     * @param {String} root 平台RESTful服务根,具体服务列表参见:http://server/x_program_center/jest/list.html。(v7.2之前版本需要加端口20030)
     *如:<pre><code class='language-js'>
     * "x_processplatform_assemble_surface" //流程平台相关服务根
     * </code></pre>
     * @return {String} 对应服务跟对应的host。如:http://127.0.0.1 (v7.2之前版本可能带端口20020)
     * @o2syntax
     * var actions = this.Actions.getHost( root );
     */

    /**
     * 平台预置了Actions对象用于调用平台提供的服务,您可以使用this.Actions.load来获取这些方法。
     * @method load
     * @methodOf module:Actions
     * @instance
     * @param {String} root 平台RESTful服务根,具体服务列表参见:http://server/x_program_center/jest/list.html (v7.2之前版本需要加端口20030)。
     * 如:
     *<pre><code class='language-js'>
     * "x_processplatform_assemble_surface" //流程平台相关服务根
     * </code></pre>
     * @return {Object} 返回action对象,用于后续服务调用
     * @o2syntax
     * var actions = this.Actions.load( root );
     * @o2syntax
     * //获取流程平台服务对象。
     * var processAction = this.Actions.load("x_processplatform_assemble_surface");
     * @o2syntax
     * <caption>
     *     通过this.Actions.load(root)方法得到action对象,就可以访问此服务下的方法了。<br/>
     *     访问方法的规则如下:
     *  </caption>
     *  var promise = this.Actions.load( root )[actionName][methodName]( arguements );
     *
     *  promise : Promise对象,可以通过promise.then接收返回数据
     *      promise.then(
     *        function(json){
     *             //json为返回的数据
     *        }
     *      )
     *
     *  root : 平台服务根名称,如果 x_processplatform_assemble_surface
     *
     *  actionName : 服务下的Action分类名称,如 TaskAction
     *
     *  methodName : Action分类下的方法名称,如 get
     *
     *  arguements : 需调用的RESTful服务的相关参数。这些参数需要按照先后顺序传入。根据实际情况可以省略某些参数。参数序列分别是:
     *
     *      uri的参数, data/formData(Post, Put方法), file(附件), success, failure, async。
     *
     *      uri参数:如果有uri有多个参数,需要按先后顺序传入。
     *
     *      data(formData)参数:提交到后台的数据,如果是上传附件,传入formData。POST 和 PUT 方法需要传入,GET方法和DELETE方法省略。
     *
     *      file参数:POST 或者 PUT方法中有效,当需要上传附件时传入,否则可以省略。
     *
     *      success参数:服务执行成功时的回调方法,形如 function(json){
     *          json为后台服务传回的数据
     *      }。
     *
     *      failure 参数:服务执行失败时的回调方法,形如 function(xhr){
     *          xhr XmlHttpRequest对象,服务器请求失败时有值
     *          return true; //从v6.3开始,返回true, 可以避免系统自动弹出错误信息;否则弹出错误信息。
     *       }
     *      此参数可以省略,如果省略,系统会自动弹出错误信息。
     *
     *      async : 方法同步或者异步执行,默认为true。
     *
     *      urlEncode : 是否对url进行编码,默认为true。
     *
     *      cache : 是否使用缓存,默认为true。
     *  @o2syntax
     *  <caption>
     *  处理返回的数据有两种方式,二选一即可:<br/>
     *  1、该方法return的结果是Promise对象,可以通过 promise.then() 方法来处理。<br/>
     *  {@link https://developer.mozilla.org/zh-CN/docs/Web/JavaScript/Reference/Global_Objects/Promise|Promise说明}<br/>
     *  2、通过success方法作为第一个参数来处理结果
     *  </caption>
     *  promise.then(
     *    function(json){
     *         //json为返回的数据
     *    }
     *  )
     *
     *  //success:arguements中的第一个function对象
     *  function(json){
     *    //json为后台服务传回的数据
     *  }
     *  @example
     * <caption>
     *     <b>样例1:</b>
     *     根据x_processplatform_assemble_surface服务获取当前用户的待办列表:<br/>
     *     可以通过对应服务的查询页面,http://server/x_processplatform_assemble_surface/jest/index.html(v7.2之前版本需要带端口20020)<br/>
     *     可以看到以下界面:<img src="img/module/Actions/Actions.png"/>
     *     我们可以找到TaskAction的V2ListPaging服务是列式当前用户待办的服务。<br/>
     *     该服务有以下信息:<br/>
     *     1、actionName是:TaskAction<br/>
     *     2、methodName是:V2ListPaging<br/>
     *     3、有两个url参数,分别是 page(分页), size(每页数量)<br/>
     *     4、有一系列的body参数<br/>
     *     5、该服务方法类型是POST<br/>
     *     根据这些信息我们可以组织出下面的方法:
     * </caption>
     * var processAction = this.Actions.load("x_processplatform_assemble_surface"); //获取action
     * var method = processAction.TaskAction.V2ListPaging; //获取列式方法
     * //执行方法1
     * method(
     *  1,  //uri 第1个参数,如果无uri参数,可以省略
     *  20, //uri 第2个参数,如果无uri参数,可以省略,如果还有其他uri参数,可以用逗号, 分隔
     *  {   //body 参数,对POST和PUT请求,该参数必须传,可以为空对象
     *      processList : [xxx] //具体参数
     *  },
     *  function(json){ //正确调用的回调
     *       //json.data得到服务返回数据
     *  },
     *  function(xhr){ //可选,错误调用的回调
     *      //xhr为XmlHttpRequest对象,服务器请求失败时有值
     *      var responseJSON = JSON.parse( xhr.responseText ) //xhr.responseText {String}是后台返回的出错信息
     *      //responseJSON见下面的说明
     *
     *      var message = responseJSON.message; //message为错误提示文本
     *  },
     *  true //可选,true为异步调用,false为同步调用,默认为true
     * );
     *
     * //执行方法2
     * var promise = method( 1, 20, {processList : [xxx]} )
     * promise.then( function(json){
     *     //json.data得到服务返回数据
     * })
     * @example
     * <caption>出错信息responseJSON的格式</caption>
     * {
     *       "type": "error", //类型为错误
     *       "message": "标识为:343434 的 Task 对象不存在.", //提示文本
     *       "date": "2020-12-29 17:02:13", //出错时间
     *       "prompt": "com.x.base.core.project.exception.ExceptionEntityNotExist" //后台错误类
     *}
     * @example
     * <caption>
     *     <b>样例2:</b>
     *      已知流程实例的workid,在脚本中获取数据,修改后进行保存。
     * </caption>
     * //查询服务列表找到获取data数据服务为DataAction的getWithWork方法
     * //查询服务列表找到更新data数据服务为DataAction的updateWithWork方法
     *
     * var workid = "cce8bc22-225a-4f85-8132-7374d546886e";
     * var data;
     * this.Actions.load("x_processplatform_assemble_surface").DataAction.getWithWork( //平台封装好的方法
     *      workid, //uri的参数
     *      function( json ){ //服务调用成功的回调函数, json为服务传回的数据
     *          data = json.data; //为变量data赋值
     *      }.bind(this),
     *      false //同步执行
     * )
     *
     * data.subject = "新标题"; //修改数据
     * this.Actions.load("x_processplatform_assemble_surface").DataAction.updateWithWork(
     *      workid, //uri的参数
     *      data, //保存的数据
     *      function(){ //服务调用成功的回调函数
     *          o2.xDesktop.notice("success", {"y":"top", "x": "right"}, "保存成功");  //提示,{"y":"top", "x": "right"}指提示框在顶部右边
     *      }.bind(this)
     * );
     */

    /**
     * <b>已过时。</b>平台预置了Action的调用方法,您可以使用this.Actions.get来获取这些方法。RESTful配置文件在{服务器目录}/webServer/o2_core/o2/xAction/services下。
     * @method get
     * @deprecated
     * @methodOf module:Actions
     * @instance
     * @param {String} root 平台RESTful服务根,具体服务列表参见:http://server/x_program_center/jest/list.html(v7.2之前版本需要加端口20030)。
     *如:<pre><code class='language-js'>
     *  "x_processplatform_assemble_surface" //流程平台相关服务根
     * </code></pre>
     * @return {String} 对应服务根的host。如:http://127.0.0.1 (v7.2之前版本可能带端口20020)
     * @o2syntax
     * var actions = this.Actions.get( root );
     * actions[ methodName ]( arguements );
     *
     * or
     *
     * this.Actions.get( root )[methodName]( arguements );
     * @o2syntax
     * methodName :(string)方法名称。
     * arguements : 见load方法的arguements说明
     * @example
     * <caption><b>样例一:</b>已知流程实例的workid,在脚本中获取数据,修改后进行保存。</caption>
     * //现已知获取数据的方法名称是 getWorkData  uri为: /jaxrs/data/work/{workid}
     * //已知更新数据的方法名称是 saveData  uri为: /jaxrs/data/work/{workid}
     * var workid = "cce8bc22-225a-4f85-8132-7374d546886e";
     * var action = this.Actions.get("x_processplatform_assemble_surface");
     * var data;
     * action.getWorkData( //平台封装好的方法
     *      workid, //uri的参数
     *      function( json ){ //服务调用成功的回调函数, json为服务传回的数据
     *          data = json.data; //为变量data赋值
     *      }.bind(this),
     *      false //同步执行
     * );

     * data.subject = "新标题"; //修改数据
     * action.saveData(
     *      workid, //uri的参数
     *      data, //保存的数据
     *      function(){ //服务调用成功的回调函数
     *          o2.xDesktop.notice("success", {"y":"top", "x": "right"}, "保存成功");  //提示,{"y":"top", "x": "right"}指提示框在顶部右边
     *      }.bind(this)
     * );
     * @example
     * <caption><b>样例二:</b>已知流程应用的workid,分页列式出流程实例。</caption>
     * //现已知获取数据的方法名称是 listWorkNext  uri为: {"uri": "/jaxrs/work/list/{id}/next/{count}/application/{applicationId}"},
     * var id = "(0)"; //如果是第一页id是(0),否则传上一页最后一个流程实例的id
     * var count = 10;
     * var applicationId = "dde8bc22-225a-4f85-8132-7374d546886e";
     * this.Actions.get("x_processplatform_assemble_surface").listWorkNext(
     *      id, //listWorkNext服务有3个uri参数,要按先后顺序列出
     *      count,
     *      applicationId,
     *       function( json ){
     *          //json.data
     *      }.bind(this)
     * );
     */
    this.Actions = o2.Actions;

    this.query = function (option) {
        // options = {
        //      "name": "statementName",
        //      "data": "json data",
        //      "firstResult": 1,
        //      "maxResults": 100,
        //      "success": function(){},
        //      "error": function(){},
        //      "async": true or false, default is true
        // }
        if (option) {
            var json = (option.data) || {};
            if (option.firstResult) json.firstResult = option.firstResult.toInt();
            if (option.maxResults) json.maxResults = option.maxResults.toInt();
            o2.Actions.get("x_query_assemble_surface").executeStatement(option.name, json, success, error, options.async);
        }
    };

    /**
     * this.Table是一个工具类,您可以使用这个类对数据中心的数据表进行增删改查操作。
     * @module Table
     * @o2cn 数据表执行
     * @o2category web
     * @o2ordernumber 135
     * @param {String} tableName 数据表的id、名称或别名。
     * @return {Object} table对象
     * @o2syntax
     * //您可以在页面、表单、流程各个嵌入脚本中,通过this.Table()来返回Table的对象,如下:
     * var table = new this.Table( tableName )
     */

    /**
     * 列示表中的行对象,下一页。
     * @method listRowNext
     * @methodOf module:Table
     * @instance
     * @param {String} id  当前页最后一条数据的Id,如果是第一页使用"(0)"。
     * @param {String|Number} count 下一页的行数
     * @param {Function} [success] 调用成功时的回调函数。
     * @param {Function} [failure] 调用错误时的回调函数。
     * @param {Boolean} [async] 是否异步调用,默认为true。
     * @return {Promise} 返回Promise
     * @o2syntax
     * table.listRowNext( id, count, success, failure, async )
     * //或
     * var promise = table.listRowNext( id, count );
     * promise.then(function(json){
     *     //json为返回的数据
     * })
     * @example
     * var table = new this.Table("table1");
     *
     * table.listRowNext( "0", 20, function(data){
     *    //data 形如
     *    //{
     *    //    "type": "success",
     *    //    "data":[
     *    //       {
     *    //        "id": "5584e6d1-8088-4694-a948-8968ac8d4923", //数据的id
     *    //        "createTime": "2021-11-01 16:23:41", //数据创建时间
     *    //        "updateTime": "2021-11-01 16:23:41", //数据更新时间
     *    //         ... //定义的字段(列)和值
     *    //        }
     *   //     ],
     *   //       "message": "",
     *   //     "date": "2021-11-01 18:34:19",
     *   //     "spent": 13,
     *   //}
     * }, function(xhr){
     *    //xhr 为 xmlHttpRequest
     * });
     */

    /**
     * 列示表中的行对象,上一页。
     * @method listRowPrev
     * @methodOf module:Table
     * @instance
     * @param {String} id  当前页第一条数据的Id,如果是最后一页使用"(0)"。
     * @param {String|Number} count 上一页的行数
     * @param {Function} [success] 调用成功时的回调函数。
     * @param {Function} [failure] 调用错误时的回调函数。
     * @param {Boolean} [async] 是否异步调用,默认为true。
     * @return {Promise} 返回Promise
     * @o2syntax
     * table.listRowPrev( id, count, success, failure, async )
     * //或
     * var promise = table.listRowPrev( id, count );
     * promise.then(function(json){
     *     //json为返回的数据
     * })
     * @example
     * var table = new this.Table("table1");
     *
     * table.listRowPrev( "0", 20, function(data){
     *    //data 形如
     *    //{
     *    //    "type": "success",
     *    //    "data":[
     *    //       {
     *    //        "id": "5584e6d1-8088-4694-a948-8968ac8d4923", //数据的id
     *    //        "createTime": "2021-11-01 16:23:41", //数据创建时间
     *    //        "updateTime": "2021-11-01 16:23:41", //数据更新时间
     *    //         ... //定义的字段(列)和值
     *    //        }
     *   //     ],
     *   //       "message": "",
     *   //     "date": "2021-11-01 18:34:19",
     *   //     "spent": 13,
     *   //}
     * }, function(xhr){
     *    //xhr 为 xmlHttpRequest
     * });
     */

    /**
     * 根据条件获取表中的数据。
     * @method listRowSelect
     * @methodOf module:Table
     * @instance
     * @param {String} [where] 查询条件,格式为jpql语法,o.name='zhangsan',允许为空。
     * @param {String} [orderBy] 排序条件,格式为:o.updateTime desc,允许为空
     * @param {String|Number} [size] 返回结果集数量,允许为空。
     * @param {Function} [success] 调用成功时的回调函数。
     * @param {Function} [failure] 调用错误时的回调函数。
     * @param {Boolean} [async] 是否异步调用,默认为true。
     * @return {Promise} 返回Promise
     * @o2syntax
     * table.listRowSelect( where, orderBy, size, success, failure, async )
     * //或
     * var promise = table.listRowSelect( where, orderBy, size );
     * promise.then(function(json){
     *     //json为返回的数据
     * })
     * @example
     * var table = new this.Table("table1");
     *
     * //查询字段name等于zhangsan的数据,结果按updateTime倒序
     * table.listRowSelect( "o.name='zhangsan'", "o.updateTime desc", 20, function(data){
     *    //data 形如
     *    //{
     *    //    "type": "success",
     *    //    "data":[
     *    //       {
     *    //        "id": "5584e6d1-8088-4694-a948-8968ac8d4923", //数据的id
     *    //        "createTime": "2021-11-01 16:23:41", //数据创建时间
     *    //        "updateTime": "2021-11-01 16:23:41", //数据更新时间
     *    //         ... //定义的字段(列)和值
     *    //        }
     *   //     ],
     *   //       "message": "",
     *   //     "date": "2021-11-01 18:34:19",
     *   //     "spent": 13,
     *   //}
     * }, function(xhr){
     *    //xhr 为 xmlHttpRequest
     * });
     */

    /**
     * 通过where 统计数量。
     * @method rowCountWhere
     * @methodOf module:Table
     * @instance
     * @param {String} where 查询条件,格式为jpql语法,o.name='zhangsan'。
     * @param {Function} [success] 调用成功时的回调函数。
     * @param {Function} [failure] 调用错误时的回调函数。
     * @param {Boolean} [async] 是否异步调用,默认为true。
     * @return {Promise} 返回Promise
     * @o2syntax
     * table.rowCountWhere( where, success, failure, async )
     * //或
     * var promise = table.rowCountWhere( where );
     * promise.then(function(json){
     *     //json为返回的数据
     * })
     * @example
     * var table = new this.Table("table1");
     *
     * //查询字段name等于zhangsan的数据,结果按updateTime倒序
     * table.rowCountWhere( "o.name='zhangsan", function(data){
     *    //data 形如
     *    //{
     *    //   "type": "success",
     *    //  "data": {
     *    //      "value": 5 //符合条件数据的总条数
     *    //  },
     *    //  "message": "",
     *    //  "date": "2021-11-01 18:32:27"
     *    //}
     * }, function(xhr){
     *    //xhr 为 xmlHttpRequest
     * });
     */

    /**
     * 删除数据表中指定id的记录。
     * @method deleteRow
     * @methodOf module:Table
     * @instance
     * @param {String} id 需要删除记录的id。
     * @param {Function} [success] 调用成功时的回调函数。
     * @param {Function} [failure] 调用错误时的回调函数。
     * @param {Boolean} [async] 是否异步调用,默认为true。
     * @return {Promise} 返回Promise
     * @o2syntax
     * table.deleteRow( id, success, failure, async )
     * //或
     * var promise = table.deleteRow( id );
     * promise.then(function(json){
     *     //json为返回的数据
     * })
     * @example
     * var table = new this.Table("table1");
     *
     * table.deleteRow( "e1f89185-d8b0-4b66-9e34-aed3323d0d79", function(data){
     *    //data 形如
     *    //{
     *    //   "type": "success",
     *    //  "data": {
     *    //      "value": true //true表示删除成功,false表示无此数据
     *    //  },
     *    //  "message": "",
     *    //  "date": "2021-11-01 18:32:27"
     *    //}
     * }, function(xhr){
     *    //xhr 为 xmlHttpRequest
     * });
     */

    /**
     * 删除指定表中所有行的数据。
     * @method deleteAllRow
     * @methodOf module:Table
     * @instance
     * @param {Function} [success] 调用成功时的回调函数。
     * @param {Function} [failure] 调用错误时的回调函数。
     * @param {Boolean} [async] 是否异步调用,默认为true。
     * @return {Promise} 返回Promise
     * @o2syntax
     * table.deleteAllRow( success, failure, async )
     * //或
     * var promise = table.deleteAllRow();
     * promise.then(function(json){
     *     //json为返回的数据
     * })
     * @example
     * var table = new this.Table("table1");
     *
     * table.deleteAllRow( function(data){
     *    //data 形如
     *    //{
     *    //   "type": "success",
     *    //  "data": {
     *    //      "value": 1 //表示删除的条数,0表示无数据
     *    //  },
     *    //  "message": "",
     *    //  "date": "2021-11-01 18:32:27"
     *    //}
     * }, function(xhr){
     *    //xhr 为 xmlHttpRequest
     * });
     */

    /**
     * 获取数据表中指定id的记录。
     * @method getRow
     * @methodOf module:Table
     * @instance
     * @param {id} 需要获取记录的id。
     * @param {Function} [success] 调用成功时的回调函数。
     * @param {Function} [failure] 调用错误时的回调函数。
     * @param {Boolean} [async] 是否异步调用,默认为true。
     * @return {Promise} 返回Promise
     * @o2syntax
     * table.getRow( id, success, failure, async )
     * //或
     * var promise = table.getRow( id );
     * promise.then(function(json){
     *     //json为返回的数据
     * })
     * @example
     * var table = new this.Table("table1");
     *
     * table.getRow( "e1f89185-d8b0-4b66-9e34-aed3323d0d79", function(data){
     *    //data 形如
     *    //{
     *    //    "type": "success",
     *    //    "data":{
     *    //        "id": "5584e6d1-8088-4694-a948-8968ac8d4923", //数据的id
     *    //        "createTime": "2021-11-01 16:23:41", //数据创建时间
     *    //        "updateTime": "2021-11-01 16:23:41", //数据更新时间
     *    //         ... //定义的字段(列)和值
     *    //     },
     *   //     "message": "",
     *   //     "date": "2021-11-01 18:34:19",
     *   //     "spent": 13,
     *   //}
     * }, function(xhr){
     *    //xhr 为 xmlHttpRequest
     * });
     */

    /**
     * 往数据表中批量插入数据。
     * @method insertRow
     * @methodOf module:Table
     * @instance
     * @param {Object[]} data 需要插入的数据。
     * @param {Function} [success] 调用成功时的回调函数。
     * @param {Function} [failure] 调用错误时的回调函数。
     * @param {Boolean} [async] 是否异步调用,默认为true。
     * @return {Promise} 返回Promise
     * @o2syntax
     * table.insertRow( data, success, failure, async )
     * //或
     * var promise = table.insertRow( data );
     * promise.then(function(json){
     *     //json为返回的数据
     * })
     * @example
     * var table = new this.Table("table1");
     * var data = [
     *  {
     *    "subject": "标题一",
     *    ... //其他字段
     *  },
     *  ...
     * ];
     * table.insertRow( data, function(data){
     *    //data 形如
     *    //{
     *    //   "type": "success",
     *    //  "data": {
     *    //      "value": true //true表示插入成功
     *    //  },
     *    //  "message": "",
     *    //  "date": "2021-11-01 18:32:27"
     *    //}
     * }, function(xhr){
     *    //xhr 为 xmlHttpRequest
     * });
     */

    /**
     * 往数据表中插入单条数据。
     * @method addRow
     * @methodOf module:Table
     * @instance
     * @param {Object} data 需要插入的数据。
     * @param {Function} [success] 调用成功时的回调函数。
     * @param {Function} [failure] 调用错误时的回调函数。
     * @param {Boolean} [async] 是否异步调用,默认为true。
     * @return {Promise} 返回Promise
     * @o2syntax
     * table.addRow( data, success, failure, async )
     * //或
     * var promise = table.addRow( data );
     * promise.then(function(json){
     *     //json为返回的数据
     * })
     * @example
     * var table = new this.Table("table1");
     * var data = {
     *    "subject": "标题一",
     *    ... //其他字段
     *  };
     * table.addRow( data, function(data){
     *    //data 形如
     *    //{
     *    //   "type": "success",
     *    //  "data": {
     *    //      "id": 2cf3a20d-b166-490b-8d29-05544db3d79b //true表示修改成功
     *    //  },
     *    //  "message": "",
     *    //  "date": "2021-11-01 18:32:27"
     *    //}
     * }, function(xhr){
     *    //xhr 为 xmlHttpRequest
     * });
     */

    /**
     * 往数据表中修改单条数据。
     * @method updateRow
     * @methodOf module:Table
     * @instance
     * @param {String} id 需要修改的数据id。
     * @param {Object} data 需要修改的数据。
     * @param {Function} [success] 调用成功时的回调函数。
     * @param {Function} [failure] 调用错误时的回调函数。
     * @param {Boolean} [async] 是否异步调用,默认为true。
     * @return {Promise} 返回Promise
     * @o2syntax
     * table.updateRow( id, data, success, failure, async )
     * //或
     * var promise = table.updateRow( id, data );
     * promise.then(function(json){
     *     //json为返回的数据
     * })
     * @example
     * var table = new this.Table("table1");
     * var data = {
     *    "id" : "2cf3a20d-b166-490b-8d29-05544db3d79b",
     *    "subject": "标题一",
     *    ... //其他字段
     *  };
     * table.updateRow( "2cf3a20d-b166-490b-8d29-05544db3d79b", data, function(data){
     *    //data 形如
     *    //{
     *    //   "type": "success",
     *    //  "data": {
     *    //      "value": true //true表示修改成功
     *    //  },
     *    //  "message": "",
     *    //  "date": "2021-11-01 18:32:27"
     *    //}
     * }, function(xhr){
     *    //xhr 为 xmlHttpRequest
     * });
     */

    /**
     * 往数据表中部分修改单条数据。
     * @method partUpdateRow
     * @methodOf module:Table
     * @instance
     * @param {String} id 需要修改的数据id。
     * @param {Object} data 需要修改的部分数据,其他数据不变。
     * @param {Function} [success] 调用成功时的回调函数。
     * @param {Function} [failure] 调用错误时的回调函数。
     * @param {Boolean} [async] 是否异步调用,默认为true。
     * @return {Promise} 返回Promise
     * @o2syntax
     * table.partUpdateRow( id, data, success, failure, async )
     * //或
     * var promise = table.partUpdateRow( id, data );
     * promise.then(function(json){
     *     //json为返回的数据
     * })
     * @example
     * var table = new this.Table("table1");
     * var data = {
     *    "id" : "2cf3a20d-b166-490b-8d29-05544db3d79b",
     *    "subject": "标题一"
     *  };
     * table.partUpdateRow( "2cf3a20d-b166-490b-8d29-05544db3d79b", data, function(data){
     *    //data 形如
     *    //{
     *    //   "type": "success",
     *    //  "data": {
     *    //      "value": true //true表示修改成功
     *    //  },
     *    //  "message": "",
     *    //  "date": "2021-11-01 18:32:27"
     *    //}
     * }, function(xhr){
     *    //xhr 为 xmlHttpRequest
     * });
     */
    this.Table = MWF.xScript.createTable();
};

if( !MWF.xScript.createTable )MWF.xScript.createTable = function(){
    return function(name){
        this.name = name;
        this.action = o2.Actions.load("x_query_assemble_surface").TableAction;

        this.listRowNext = function(id, count, success, error, async){
            return this.action.listRowNext(this.name, id, count, success, error, async);
        };
        this.listRowPrev = function(id, count, success, error, async){
            return this.action.listRowPrev(this.name, id, count, success, error, async);
        };
        this.listRowSelect = function(where, orderBy, size, success, error, async){
            return this.action.listRowSelect(this.name, {"where": where, "orderBy": orderBy, "size": size || ""}, success, error, async);
        };
        this.listRowSelectWhere = function(where, success, error, async){
            return this.action.listRowSelectWhere(this.name, where, success, error, async);
        };
        this.rowCountWhere = function(where, success, error, async){
            return this.action.rowCountWhere(this.name, where, success, error, async);
        };
        this.deleteRow = function(id, success, error, async){
            return this.action.rowDelete(this.name, id, success, error, async);
        };
        this.deleteAllRow = function(success, error, async){
            return this.action.rowDeleteAll(this.name, success, error, async);
        };
        this.getRow = function(id, success, error, async){
            return this.action.rowGet(this.name, id, success, error, async);
        };
        this.insertRow = function(data, success, error, async){
            return this.action.rowInsert(this.name, data, success, error, async);
        };
        this.addRow = function(data, success, error, async){
            return this.action.rowInsertOne(this.name, data, success, error, async);
        };
        this.updateRow = function(id, data, success, error, async){
            return this.action.rowUpdate(this.name, id, data, success, error, async);
        };
        this.partUpdateRow = function(id, data, success, error, async){
            return this.action.rowPartUpdate(this.name, id, data, success, error, async);
        };
    }
};


if( !MWF.xScript.dictLoaded )MWF.xScript.dictLoaded = {};

if( !MWF.xScript.createDict ){
    MWF.xScript.addDictToCache = function ( options, path, json ) {
        if( !path )path = "root";
        if( path.indexOf("root") !== 0 )path = "root." + path ;

        var type = options.appType || "process";
        var enableAnonymous = ( options.enableAnonymous || options.anonymous ) || false;

        var appFlagList = [];
        if( options.application )appFlagList.push( options.application );
        if( options.appId )appFlagList.push( options.appId );
        if( options.appName )appFlagList.push( options.appName );
        if( options.appAlias )appFlagList.push( options.appAlias );

        var dictFlagList = [];
        if( options.id )dictFlagList.push( options.id );
        if( options.name )dictFlagList.push( options.name );
        if( options.alias )dictFlagList.push( options.alias );

        var cache = {};
        cache[path] = json;

        for( var i=0; i<appFlagList.length; i++ ){
            for( var j=0; j<dictFlagList.length; j++ ){
                var k = dictFlagList[j] + type + appFlagList[i] + enableAnonymous;
                if( !MWF.xScript.dictLoaded[k] ){
                    MWF.xScript.dictLoaded[k] = cache; //指向同一个对象
                    // MWF.xScript.dictLoaded[k][path] = json; //指向不同的对象
                }else if( i===0 && j===0 ){
                    MWF.xScript.setDictToCache( k, path ,json );
                    var arr = path.split(/\./g);
                    var p;
                    var cache = MWF.xScript.dictLoaded[k];
                    for( var l=0 ; l<arr.length; l++ ){
                        p = l === 0 ? arr[0] : ( p + "." + arr[l] );
                        if( cache[ p ] )break;
                    }
                    if( p ){
                        var mathP = p+".";
                        Object.keys( cache ).each( function( path, idx){
                            if( path.indexOf( mathP ) === 0 )delete cache[path];
                        })
                    }
                }
            }
        }
    };

    MWF.xScript.getMatchedDict = function(key, path){
        if( !path )path = "root";
        if( path.indexOf("root") !== 0 )path = "root." + path ;

        var arr = path.split(/\./g);
        if( MWF.xScript.dictLoaded[key] ){
            var dicts = MWF.xScript.dictLoaded[key];
            var list = Array.clone(arr);
            var p;
            var dict;
            for( var i=0 ; i<arr.length; i++ ){
                p = i === 0 ? arr[0] : ( p + "." + arr[i] );
                list.shift();
                if( dicts[ p ] ){
                    dict = dicts[ p ];
                    break;
                }
            }
            return {
                dict : dict,
                unmatchedPathList : list
            }
        }
        return {
            dict : null,
            unmatchedPathList : list
        }
    };

    MWF.xScript.insertDictToCache = function(key, path, json){
        var p = path;
        if( !p )p = "root";
        if( p.indexOf("root") !== 0 )p = "root." + p ;

        if( MWF.xScript.dictLoaded[key] ){
            var matchedDict = MWF.xScript.getMatchedDict( key, path );
            var dict = matchedDict.dict;
            var list = matchedDict.unmatchedPathList;
            if( !dict ){
                MWF.xScript.dictLoaded[key][p] = json;
            }else if( !list || list.length === 0 ){
                MWF.xScript.dictLoaded[key][p] = json;
            }else{
                for( var j=0; j<list.length-1; j++ ){
                    if( !dict[ list[j] ] ){
                        dict[ list[j] ] = {};
                    }
                    dict = dict[ list[j] ];
                }
                var lastPath = list[list.length-1];
                if( !dict[lastPath] ){
                    dict[lastPath] = json;
                }else if( typeOf( dict[lastPath] ) === "array" ){
                    dict[lastPath].push( json );
                }
            }
        }else{
            MWF.xScript.dictLoaded[key] = {};
            MWF.xScript.dictLoaded[key][p] = json;
        }
    };


    MWF.xScript.setDictToCache = function(key, path, json){
        var p = path;
        if( !p )p = "root";
        if( p.indexOf("root") !== 0 )p = "root." + p ;

        if( MWF.xScript.dictLoaded[key] ){
            var matchedDict = MWF.xScript.getMatchedDict( key, path );
            var dict = matchedDict.dict;
            var list = matchedDict.unmatchedPathList;
            if( !dict ){
                MWF.xScript.dictLoaded[key][p] = json;
            }else if( !list || list.length === 0 ){
                MWF.xScript.dictLoaded[key][p] = json;
            }else{
                for( var j=0; j<list.length-1; j++ ){
                    if( !dict[ list[j] ] ){
                        dict[ list[j] ] = {};
                    }
                    dict = dict[ list[j] ];
                }
                dict[list[list.length-1]] = json;
            }
        }else{
            MWF.xScript.dictLoaded[key] = {};
            MWF.xScript.dictLoaded[key][p] = json;
        }
    };

    MWF.xScript.getDictFromCache = function( key, path ){
        var matchedDict = MWF.xScript.getMatchedDict( key, path );
        var dict = matchedDict.dict;
        var list = matchedDict.unmatchedPathList;
        if( dict ){
            for( var j=0; j<list.length; j++ ){
                dict = dict[ list[j] ];
                if( !dict )return null;
            }
            return dict;
        }
        return null;
    };

    MWF.xScript.deleteDictToCache = function(key, path){
        var matchedDict = MWF.xScript.getMatchedDict( key, path );
        var dict = matchedDict.dict;
        var list = matchedDict.unmatchedPathList;

        if( dict){
            for( var j=0; j<list.length-1; j++ ){
                dict = dict[ list[j] ];
                if( !dict )return;
            }
            if( list.length ){
                delete dict[list[list.length-1]];
            }
        }
    };


    MWF.xScript.createDict = function(application, appType){
        //optionsOrName : {
        //  type : "", //默认为process, 可以为  process  cms
        //  application : "", //流程/CMS的名称/别名/id, 默认为当前应用
        //  name : "", // 数据字典名称/别名/id
        //  anonymous : false //允许在未登录的情况下读取CMS的数据字典, 改参数名也可以是enableAnonymous
        //}
        //或者name: "" // 数据字典名称/别名/id
        return function(optionsOrName){
            var options = optionsOrName;
            if( typeOf( options ) == "string" ){
                options = {
                    name : options,
                    type: appType,
                    application: application
                };
            }
            var name = this.name = options.name;
            var type;
            if( options.type === "service"){
                type = options.type;
            }else{
                type = ( options.type && options.application ) ?  options.type : "process";
            }
            var applicationId = options.application || application;
            var enableAnonymous = ( options.enableAnonymous || options.anonymous ) || false;

            var opt = {
                "appType" : type,
                "name" : name,
                "appId" : applicationId,
                "enableAnonymous" : enableAnonymous
            };

            var key = name+type+applicationId+enableAnonymous;
            // if (!dictLoaded[key]) dictLoaded[key] = {};
            // this.dictData = dictLoaded[key];

            //MWF.require("MWF.xScript.Actions.DictActions", null, false);
            var action;
            if (type === "cms") {
                action = MWF.Actions.get("x_cms_assemble_control");
            } else if( type === "portal" ){
                action = MWF.Actions.get("x_portal_assemble_surface");
            }else if( type === "service" ){
                key = name+type+enableAnonymous;
                action = MWF.Actions.get("x_program_center");
            } else {
                action = MWF.Actions.get("x_processplatform_assemble_surface");
            }

            var encodePath = function( path ){
                var arr = path.split(/\./g);
                var ar = arr.map(function(v){
                    return encodeURIComponent(v);
                });
                return ( type === "portal" || type === "service" ) ? ar.join(".") : ar.join("/");
            };

            this.get = function(path, success, failure, async, refresh){
                var value = null;

                if (success===true) async=true;
                if (failure===true) async=true;

                if (!refresh ){
                    var data = MWF.xScript.getDictFromCache( key, path );
                    if( data ){
                        if (success && o2.typeOf(success)=="function") success( data );
                        if( !!async ){
                            return Promise.resolve( data );
                        }else{
                            return data;
                        }
                    }
                }

                // var cb = function(json){
                //     value = json.data;
                //     MWF.xScript.addDictToCache(opt, path, value);
                //     if (success && o2.typeOf(success)=="function") value = success(json.data);
                //     return value;
                // }.ag().catch(function(xhr, text, error){ if (failure && o2.typeOf(failure)=="function") return failure(xhr, text, error); });

                var cb = function(json){
                    value = json.data;
                    MWF.xScript.addDictToCache(opt, path, value);
                    if (success && o2.typeOf(success)=="function") value = success(json.data);
                    return value;
                };

                var promise;
                if( type === "service" ){
                    if (path){
                        var p = encodePath( path );
                        promise = action.getDictData(encodeURIComponent(this.name), p, cb, null, !!async, false);
                    }else{
                        promise = action.getDictRoot(this.name, cb, null, !!async, false);
                    }
                }else{
                    if (path){
                        var p = encodePath( path );
                        promise = action[ ( (enableAnonymous && type == "cms") ? "getDictDataAnonymous" : "getDictData" ) ](encodeURIComponent(this.name), applicationId, p, cb, null, !!async, false);
                    }else{
                        promise = action[ ( (enableAnonymous && type == "cms") ? "getDictRootAnonymous" : "getDictRoot" ) ](this.name, applicationId, cb, null, !!async, false);
                    }
                }
                return (!!async) ? promise : value;

                // if (path){
                //     var p = encodePath( path );
                //     //var p = path.replace(/\./g, "/");
                //     action[ ( (enableAnonymous && type == "cms") ? "getDictDataAnonymous" : "getDictData" ) ](encodeURIComponent(this.name), applicationId, p, function(json){
                //         value = json.data;
                //         // this.dictData[path] = value;
                //         MWF.xScript.addDictToCache(opt, path, value);
                //         if (success) success(json.data);
                //     }.bind(this), function(xhr, text, error){
                //         if (failure) failure(xhr, text, error);
                //     }, !!async);
                // }else{
                //     action[ ( (enableAnonymous && type == "cms") ? "getDictRootAnonymous" : "getDictRoot" ) ](this.name, applicationId, function(json){
                //         value = json.data;
                //         // this.dictData["root"] = value;
                //         MWF.xScript.addDictToCache(opt, path, value);
                //         if (success) success(json.data);
                //     }.bind(this), function(xhr, text, error){
                //         if (failure) failure(xhr, text, error);
                //     }, !!async);
                // }

                //return value;
            };

            this.set = function(path, value, success, failure){
                var p = encodePath( path );
                //var p = path.replace(/\./g, "/");
                var successCallback = function(json){
                    MWF.xScript.setDictToCache(key, path, value);
                    if (success) return success(json.data);
                };
                var failureCallback = function(xhr, text, error){
                    if (failure) return failure(xhr, text, error);
                };
                if( type === "service" ){
                    return action.setDictData(encodeURIComponent(this.name), p, value, successCallback, failureCallback, false, false);
                }else{
                    return action.setDictData(encodeURIComponent(this.name), applicationId, p, value, successCallback, failureCallback, false, false);
                }
            };
            this.add = function(path, value, success, failure){
                var p = encodePath( path );
                //var p = path.replace(/\./g, "/");
                var successCallback = function(json){
                    MWF.xScript.insertDictToCache(key, path, value);
                    if (success) return success(json.data);
                };
                var failureCallback = function(xhr, text, error){
                    if (failure) return failure(xhr, text, error);
                };
                if( type === "service" ) {
                    return action.addDictData(encodeURIComponent(this.name), p, value, successCallback, failureCallback, false, false);
                }else{
                    return action.addDictData(encodeURIComponent(this.name), applicationId, p, value, successCallback, failureCallback, false, false);
                }
            };
            this["delete"] = function(path, success, failure){
                var p = encodePath( path );
                //var p = path.replace(/\./g, "/");
                var successCallback = function(json){
                    MWF.xScript.deleteDictToCache(key, path);
                    if (success) return success(json.data);
                };
                var failureCallback = function(xhr, text, error){
                    if (failure) return failure(xhr, text, error);
                };
                if( type === "service" ) {
                    return action.deleteDictData(encodeURIComponent(this.name), p, successCallback, failureCallback, false, false);
                }else{
                    return action.deleteDictData(encodeURIComponent(this.name), applicationId, p, successCallback, failureCallback, false, false);
                }
            };
            this.destory = this["delete"];
        }
    };
}


results matching

    No results matching ''