
注:本文用到了前面几篇相关博客的技术实现,如果有需要的话可以点击下面的链接前去了解:
1)详解Javascript的继承实现:提供一个class.js,用来定义javascript的类和构建类的继承关系;
2)jquery技巧之让任何组件都支持类似DOM的事件管理:提供一个eventBase.js,用来给任意组件实例提供类似DOM的事件管理功能;
3)对jquery的ajax进行二次封装以及ajax缓存代理组件:AjaxCache:提供ajax.js和ajaxCache.js,简化jquery的ajax调用,以及对请求进行客户端的缓存代理。
下面先来详细了解下这个功能的要求。
1. 功能分析
以包含三个级联项的级联组件来说明这个功能:
1)每个级联项可能需要一个用作输入提示的option:

这种情况每个级联项的数据列表中都能选择一个空的option(就是输入提示的那个):

也可能不需要用作输入提示的option:

这种情况每个级联项的数据列表中只能选数据option,选不到空的option:

2)如果当前这个页面是从数据库中查询出来跟级联组件对应的字段有值,那么就把查询出来的值回显到级联组件上:

如果查询出来的对应字段没有值,那么就按第1)点需求描述的2种情况显示。
3)各个级联项在数据结构上构成单链表的关系,后一个级联项的数据列表,跟前一个级联项所选择的数据有关联的。
4)考虑到性能方面的问题,各个级联项的数据列表都采用ajax异步加载显示。
5)在级联组件初始化完成以后,自动加载第一个级联项的列表。
6)当前一个级联项发生改变时,清空后面所有直接或间接关联的级联项的数据列表,同时如果前一个级联项改变后的值不为空则自动加载跟它直接关联的下一个级联项的数据列表。清空级联项的数据列表时要注意:如果级联项需要显示输入提示的option,在清空的时候得保留该option。
7)要充分考虑性能问题,避免重复加载。
8)考虑到表单提交的问题,当级联组件任意级联项发生改变后,得把级联组件所选的值体现到一个隐藏的文本域内,方便把级联组件的值通过该文本域提交到后台。
功能大致如上。
2. 实现思路
1)数据结构
级联组件跟别的组件不太一样的是,它跟后台的数据有一些依赖,我考虑的比较好实现的数据结构是:
{"id": 1,"text": "北京市","code": 110000,"parentId": 0},{"id": 2,"text": "河北省","code": 220000,"parentId": 0},{"id": 3,"text": "河南省","code": 330000,"parentId": 0}id是数据的唯一标识,数据之间的关联关系通过parentId来构建,text,code这种都属于普通的业务字段。如果按这个数据结构,我们查询级联项数据列表的接口就会变得很简单://查第一个级联项的列表/api/cascade?parentId=0//根据第一个级联项选的值,查第二个级联项的列表/api/cascade?parentId=1//根据第二个级联项选的值,查第三个级联项的列表/api/cascade?parentId=4这个结构对于后台来说也很好处理,虽然在结构上它们是一种树形的表结构,但是查询都是单层的,所以很好实现。
<ul id="licenseLocation-view" class="cascade-view clearfix"><li><select class="form-control"><option value="">请选择省份</option></select></li><li><select class="form-control"><option value="">请选择城市</option></select></li><li><select class="form-control"><option value="">请选择区县</option></select></li></ul>或
<ul id="companyLocation-view" class="cascade-view clearfix"><li><select class="form-control"></select></li><li><select class="form-control"></select></li><li><select class="form-control"></select></li></ul>这两个结构唯一的区别就在于是否配置了用作输入提示的option。另外需要注意的是如果需要这个空的option,一定得把value属性设置成空,否则这个空的option在表单提交的时候会把option的提示信息提交到后台。

我们需要做的就是控制好消息的发布跟传递。
4)表单提交
为了能够方便地将级联组件的值提交到后台,可以把整个级联组件当成一个整体,对外提供一个onChanged事件,外部可通过这个事件获取所有级联项的值。由于存在多个级联项,所以在发布onChanged这个事件时,只能在任意级联项发生改变的时候,都去触发这个事件。
5)ajax缓存
在这个组件里面得考虑两个层级的ajax缓存,第一个是组件这一层级的,比如我把第一个级联项切换到了北京,这个时候第二个级联项就把北京的数据加载出来了,然后我把第一个级联项从北京切换到河北再切换到北京,这个时候第二个级联项要显示的还是北京的关联数据列表,如果我们在第一次加载这个列表的时候就把它的数据缓存下来了,那么这次就不用发起ajax请求了;第二个是ajax请求这一层级的,假如页面上有多个级联组件,我先把第一个级联组件的第一个级联项切换到北京,浏览器发起一个ajax请求加载数据,当我再把第二个级联组件的第一个级联项切换到北京的时候,浏览器还会再发一个请求去加载数据,如果我把第一个组件第一次ajax请求的返回的数据,先缓存起来,当第二个组件,用同样的参数请求同样的接口时,直接拿之前缓存觉得结果返回,这样也能减少一次ajax请求。第二个层级的ajax缓存依赖上文《对jquery的ajax进行二次封装以及ajax缓存代理组件:AjaxCache》,对于组件来说,它内部只实现了第一个层级的缓存,但是它不用考虑第二个层级的缓存,因为第二个层级的缓存实现对它来说是透明的,它不知道它用到的ajax组件有缓存的功能。
3. 实现细节
最终的实现包含了三个组件,CascadeView、CascadeItem、CascadePublicDefaults,前面两个是组件的核心,最后一个只是用来定义一些option,它的作用在CascadeItem的注释里面有详细的描述。另外在下面的代码中有非常详细的注释解释了一些关键代码的作用,结合着前面的需求来看代码,应该还是比较容易理解的。我以前倾向于用文字来解释一些实现细节,后来我慢慢觉得这种方式有点费力不讨好,第一是细节层面的语言不好组织,有的时候言不达意,明明想把一件事情解释清楚,结果反而弄得更加迷糊,至少我自己看自己写的东西就会这样的感触;第二是本身开发人员都具有阅读源码的能力,而且大部分积极的开发人员都愿意通过琢磨别人的代码来理解实现思路;所以我改用注释的方式来说明实现细节:)
CascadePublicDefaults:
define(function () {return {url: "",//数据查询接口textField: "text", //返回的数据中要在<option>元素内显示的字段名称valueField: "text", //返回的数据中要设置在<option>元素的value上的字段名称paramField: "id", //当调用数据查询接口时,要传递给后台的数据对应的字段名称paramName: "parentId", //当调用数据查询接口时,跟在url后面传递数据的参数名defaultParam: "", //当查询第一个级联项时,传递给后台的值,一般是0,"",或者-1等,表示要查询第上层的数据keepFirstOption: true, //是否保留第一个option(用作输入提示,如:请选择省份),如果为true,在重新加载级联项时,不会清除默认的第一个optionresolveAjax: function (res) {return res;}//因为级联项在加载数据的时候会发异步请求,这个回调用来解析异步请求返回的响应}});CascadeView:define(function (require, exports, module) {var $ = require("jquery");var Class = require("mod/class");var EventBase = require("mod/eventBase");var PublicDefaults = require("mod/cascadePublicDefaults");var CascadeItem = require("mod/cascadeItem");/*** PublicDefaults的作用见CascadeItem组件内的注释*/var DEFAULTS = $.extend({}, PublicDefaults, {$elements: undefined, //级联项jq对象的数组,元素在数据中的顺序代表级联的先后顺序valueSeparator: ",", //获取所有级联项的值时使用的分隔符,如果是英文逗号,返回的值形如 北京市,区,朝阳区values: "", //用valueSeparator分隔的字符串,表示初始时各个select的值onChanged: $.noop //当任意级联项的值发生改变的时候会触发这个事件});var CascadeView = Class({instanceMembers: {init: function (options) {//通过this.base调用父类EventBase的init方法this.base();var opts = this.options = this.getOptions(options),items = this.items = [],that = this,$elements = opts.$elements,values = opts.values.split(opts.valueSeparator);this.on("changed.cascadeView", $.proxy(opts.onChanged, this));$elements && $elements.each(function (i) {var $el = $(this);//实例化CascadeItem组件,并把每个实例的prevItem属性指向前一个实例//第一个prevItem属性设置为undefinedvar cascadeItem = new CascadeItem($el, $.extend(that.getItemOptions(), {prevItem: i == 0 ? undefined : items[i - 1],value: $.trim(values[i])}));items.push(cascadeItem);//每个级联项实例发生改变都会触发CascadeView组件的changed事件//外部可在这个回调内处理业务逻辑//比如将所有级联项的值设置到一个隐藏域里面,用于表单提交cascadeItem.on("changed.cascadeItem", function () {that.trigger("changed.cascadeView", that.getValue());});});//初始化完成自动加载第一个级联项items.length && items[0].load();},getOptions: function (options) {return $.extend({}, this.getDefaults(), options);},getDefaults: function () {return DEFAULTS;},getItemOptions: function () {var opts = {}, _options = this.options;for (var i in PublicDefaults) {if (PublicDefaults.hasOwnProperty(i) && i in _options) {opts[i] = _options[i];}}return opts;},//获取所有级联项的值,是一个用valueSeparator分隔的字符串//为空的级联项的值不会返回getValue: function () {var value = [];this.items.forEach(function (item) {var val = $.trim(item.getValue());val != "" && value.push(val);});return value.join(this.options.valueSeparator);}},extend: EventBase});return CascadeView;});CascadeItem:define(function (require, exports, module) {var $ = require("jquery");var Class = require("mod/class");var EventBase = require("mod/eventBase");var PublicDefaults = require("mod/cascadePublicDefaults");var AjaxCache = require("mod/ajaxCache");//这是一个可缓存的Ajax组件var Ajax = new AjaxCache();/*** 有一部分option定义在PublicDefaults里面,因为CascadeItem组件不会被外部直接使用* 外部用的是CascadeView组件,所以有一部分的option必须变成公共的,在CascadeView组件也定义一次* 外部通过CascadeView组件传递所有的option* CascadeView内部实例化CascadeItem的时候,再把PublicDefaults内的option传递给CascadeItem*/var DEFAULTS = $.extend({}, PublicDefaults, {prevItem: undefined, // 指向前一个级联项value: "" //初始时显示的value});var CascadeItem = Class({instanceMembers: {init: function ($el, options) {//通过this.base调用父类EventBase的init方法this.base($el);this.$el = $el;this.options = this.getOptions(options);this.prevItem = this.options.prevItem; //前一个级联项this.hasContent = false;//这个变量用来控制是否需要重新加载数据this.cache = {};//用来缓存数据var that = this;//代理select元素的change事件$el.on("change", function () {that.trigger("changed.cascadeItem");});//当前一个级联项的值发生改变的时候,根据需要做清空和重新加载数据的处理this.prevItem && this.prevItem.on("changed.cascadeItem", function () {//只要前一个的值发生改变并且自身有内容的时候,就得清空内容that.hasContent && that.clear();//如果不是第一个级联项,同时前一个级联项没有选中有效的option时,就不处理if (that.prevItem && $.trim(that.prevItem.getValue()) == "") return;that.load();});var value = $.trim(this.options.value);value !== "" && this.one("render.cascadeItem", function () {//设置初始值that.$el.val(value.split(","));//通知后面的级联项做清空和重新加载数据的处理that.trigger("changed.cascadeItem");});},getOptions: function (options) {return $.extend({}, this.getDefaults(), options);},getDefaults: function () {return DEFAULTS;},clear: function () {var $el = this.$el;$el.val("");if (this.options.keepFirstOption) {//保留第一个option$el.children().filter(":gt(0)").remove();} else {//清空全部$el.html("");}//通知后面的级联项做清空和重新加载数据的处理this.trigger("changed.cascadeItem");this.hasContent = false;//表示内容为空},load: function () {var opts = this.options,paramValue,that = this,dataKey;//dataKey是在cache缓存时用的键名//由于第一个级联项的数据是顶层数据,所以在缓存的时候用的是固定且唯一的键:root//其它级联项的数据缓存时用的键名跟前一个选择的option有关if (!this.prevItem) {paramValue = opts.defaultParam;dataKey = "root";} else {paramValue = this.prevItem.getParamValue();dataKey = paramValue;}//先看数据缓存中有没有加载过的数据,有就直接显示出来,避免Ajaxif (dataKey in this.cache) {this.render(this.cache[dataKey]);} else {var params = {};params[opts.paramName] = paramValue;Ajax.get(opts.url, params).done(function (res) {//resolveAjax这个回调用来在外部解析ajax返回的数据//它需要返回一个data数组var data = opts.resolveAjax(res);if (data) {that.cache[dataKey] = data;that.render(data);}});}},render: function (data) {var html = [],opts = this.options;data.forEach(function (item) {html.push(["<option value="",item[opts.valueField],"" data-param-value="",//将paramField对应的值存放在option的data-param-value属性上item[opts.paramField],"">",item[opts.textField],"</option>"].join(""));});//采用append的方式动态添加,避免影响第一个option//最后还要把value设置为空this.$el.append(html.join("")).val("");this.hasContent = true;//表示有内容this.trigger("render.cascadeItem");},getValue: function () {return this.$el.val();},getParamValue: function () {return this.$el.find("option:selected").data("paramValue");}},extend: EventBase});return CascadeItem;});4. demo说明
其中框起来的就是演示的相关部分。html/regist.html是演示效果的页面,js/app/regist.js是演示效果的入口js:
define(function (require, exports, module) {var $ = require("jquery");var CascadeView = require("mod/cascadeView");function publicSetCascadeView(fieldName, opts) {this.cascadeView = new CascadeView({$elements: $("#" + fieldName + "-view").find("select"),url: "../api/cascade.json",onChanged: this.onChanged,values: opts.values,keepFirstOption: this.keepFirstOption,resolveAjax: function (res) {if (res.code == 200) {return res.data;}}});}var LOCATION_VIEWS = {licenseLocation: {$input: $("input[name="licenseLocation"]"),keepFirstOption: true,setCascadeView: publicSetCascadeView,onChanged: function(e, value){LOCATION_VIEWS.licenseLocation.$input.val(value);}},companyLocation: {$input: $("input[name="companyLocation"]"),keepFirstOption: false,setCascadeView: publicSetCascadeView,onChanged: function(e, value){LOCATION_VIEWS.companyLocation.$input.val(value);}}};LOCATION_VIEWS.licenseLocation.setCascadeView("licenseLocation", {values: LOCATION_VIEWS.licenseLocation.$input.val()});LOCATION_VIEWS.companyLocation.setCascadeView("companyLocation", {values: LOCATION_VIEWS.companyLocation.$input.val()});});注意以上代码中LOCATION_VIEWS这个变量的作用,因为页面上有多个级联组件,这个变量其实是通过策略模式,把各个组件的相关的东西都用一种类似的方式管理起来而已。如果不这么做的话,很容易产生重复代码;这种形式也比较有利于在入口文件这种处理业务逻辑的地方,进行一些业务逻辑的分离与封装。