ExtJs通用筛选查询控件

在ExtJs项目开发过程中少不了筛选查询功能,而对于这部分功能又基本是大同小异,无非就是对相应条件数据进行筛选查看了,所以我们针对这部分专门去实现了一个控件,用于对所有的数据列表进行查询的通用筛选!

 

先看下示例图:


ExtJs通用筛选查询控件
 

控件核心代码如下:

 

这部分代码比较多,可能再打开浏览器的时候看不到,请大家在附件中下载即可。

 

function createFilter(store,moduleCode,callback)
{
	Ext.form.Field.prototype.msgTarget = "side";
////////////////////变量、常量区////////////////////////////////////////////////////////////////////////////////////////////////////////////////////////////////////
	/**
	 * 当前筛选条件节点
	 * @type  TreeNode
	 */
	var currentFilter={attributes:{}};
	var allOperator={EQ:"0",NE:"1",LIKE:"2",NLIKE:"3",GT:"4",LT:"5",GE:"6",LE:"7",NULL:"8",NOTNULL:"9",ASC:"10",DESC:"11"};
	
	/**
	 * 数字操作运算符集合
	 * @type Array
	 */
	var numericOp = datetimeOp = [[allOperator.EQ, "等于"], [allOperator.NE, "不等于"], [allOperator.GT, "大于"]
					, [allOperator.LT, "小于"], [allOperator.GE, "大于等于"],[allOperator.LE, "小于等于"],[allOperator.NULL, "空白"], [allOperator.NOTNULL, "非空白"]];
	/**
	 * 所有的运算符
	 * @type  Array
	 */
	var allOp = [[allOperator.EQ, "等于"], [allOperator.NE, "不等于"],[allOperator.LIKE, "包含字符"], [allOperator.NLIKE, "不包含字符"], [allOperator.GT, "大于"]
					, [allOperator.LT, "小于"], [allOperator.GE, "大于等于"],[allOperator.LE, "小于等于"],[allOperator.NULL, "空白"], [allOperator.NOTNULL, "非空白"], [allOperator.ASC, "升序"], [allOperator.DESC, "降序"]];
	/**
	 * 字符操作运算符
	 * @type Array
	 */
	var stringOp = [[allOperator.EQ, "等于"], [allOperator.NE, "不等于"],[allOperator.LIKE, "包含字符"], [allOperator.NLIKE, "不包含字符"],[allOperator.NULL, "空白"], [allOperator.NOTNULL, "非空白"]];
	/**
	 * 自定义,下拉框,boolean操作运算符
	 * @type 
	 */
	var lookupOp = booleanOp = [[allOperator.EQ, "等于"], [allOperator.NE, "不等于"], [allOperator.NULL, "空白"], [allOperator.NOTNULL, "非空白"]];
	/**
	 * 最后一行下标
	 * @type Number
	 */
	var lineIndex = 0;// 新增行下标
	/**
	 * 当前编辑行
	 * @type  int
	 */
	var currentIndex=0;// 当前编辑行号
	/**
	 * 左边树面板折叠状态,默认为打开
	 * @type Boolean
	 */
	var treeColFlag=false;
	/**
	 * 第一行下标,默认为0
	 */
	var firstIndex=0;
	/**
	 * 存放排序字段的下标, 避免重复
	 */
	var sortInfo={};
	/**
	 * 筛选结果数据源
	 */
	var filterStore = new Ext.data.Store({//模块列数据源
		id:moduleCode+"_filterStore",
		proxy : new Ext.data.HttpProxy({
					url : "filterAction!getFilterInfo.action"
				}),
		reader : new Ext.data.JsonReader({
					root : "root",
					totalProperty : "total",
					fields : ["fieldName", "fieldId", "fieldLabel", "operator",
								"setValue", "inputType", "fieldType", "inputSql",
								"method", "relation", "rightBr", "leftBr", "sort",
								"displayValue", "delFlag", "idRender","userQueryId","custom"]
				})
	});
	
////////////////////初始化方法区////////////////////////////////////////////////////////////////////////////////////////////////////////////////////////////////////
	//初始化store,如果不是公用Grid 来的,要把fieldType转成相对应的数字
	if(store.getCount()>0)
		store.each(function(rec){
			switch(rec.get("fieldType"))
			{
				case "string":rec.set("fieldType",1);break;
				case "int":rec.set("fieldType",2);break;
				case "float":rec.set("fieldType",3);break;
				case "boolean":rec.set("fieldType",4);break;
				case "date":rec.set("fieldType",5);break;
				case "gender":rec.set("fieldType",6);break;
			}
		})
	else
		store.load();
	
	/**
	 * 文本输入框
	 * @param {int} index 行索引
	 */
	var objGridTextEditor = function(index) {
		return new Ext.form.TextField({
					id : moduleCode+"_setValue" + index,
					emptyText : "请输入..."
				});
	}
	
	/**
	 * 整型输入框
	 * @param {int} index 行索引
	 */
	var objGridIntegerEditor = function(index) {
		return new Ext.form.NumberField({
					id : moduleCode+"_setValue" + index,
					allowNegative : false,
					allowDecimals : false,
					emptyText : "请输入整数..."
				});
	}
	
	/**
	 * 浮点型输入框
	 * @param {int} index 行索引
	 */
	var objGridFloatEditor = function(index) {
		return new Ext.form.NumberField({
					id : moduleCode+"_setValue" + index,
					allowNegative : false,
					emptyText : "请输入数字..."
				});
	}
	/**
	 * boolean选择器
	 * @param {int} index 行索引
	 */
	var objGridBooleanEditor = function(index) {
		return new Ext.form.ComboBox({
					store : [["1", "是"], ["0", "否"]],
					mode : "local",
					triggerAction : "all",
					id : moduleCode+"_setValue" + index,
					editable : false,
					emptyText : "请选择..."
				});
	}
	/**
	 * 性别选择器
	 * @param {int} index 行索引
	 */
	var objGridGenderEditor = function(index) {
		return new Ext.form.ComboBox({
					store : [[1, "男"], [0, "女"]],
					mode : "local",
					id : moduleCode+"_setValue" + index,
					triggerAction : "all",
					editable : false,
					emptyText : "请选择..."
				});
	}
////////////////////公用方法区////////////////////////////////////////////////////////////////////////////////////////////////////////////////////////////////////
	/**
	 * 从store中根据prop的propValue取field值
	 * @param {Ext.data.Store} store 
	 * @param {String} prop 字段
	 * @param {String} propValue 字段值
	 * @param {String} field 结果字段
	 * @return {Object}
	 */
	function findRecordValue(store, prop, propValue, field) {
		var record;
		if (store.getCount() > 0) {
			var _i=0;
			store.each(function(r) {
						if (r.data[prop] == propValue) {
							record = r.data[field];
							if(field=="refData")
								record=store.reader.jsonData[_i].refData;
							return false;
						}
						_i++;
					});
		}
		return record;
	}
	
	/**
	 * 删除当前行
	 */
	var deleteLine = function() {
		panel.remove(moduleCode+"_line"+currentIndex);
		var record = filterStore.getAt(currentIndex);
		var fieldName=record.data.fieldName;
		if(sortInfo[fieldName]+1==currentIndex+1)
			sortInfo[fieldName]=null;
		if(win.customFilter[fieldName]+1==currentIndex+1)
			win.customFilter[fieldName]=null;
		record.set("delFlag", "1");
	}
	
	/**
	 * 删除所有行
	 */
	function delAllLine()
	{
		var count=lineIndex||0;
		for(var j=0;j<=count;j++)
		{
			currentIndex=j;
			deleteLine();
		}
		if(!currentFilter.id){
			filterStore.removeAll();
			lineIndex=0;
		}
		currentIndex=null;
	}
	/**
	 * 根据左边树是否折叠取输入框宽度值
	 * @return {int}
	 */
	function getValueWidth()
	{
		return treeColFlag?197:162;
	}
	/**
	 * 根据左边树是否折叠改变值输入框宽度
	 */
	function changeValueWidth()// 改变值的宽度
	{
		for(var i=0;i<=lineIndex;i++)
		{
			var setv=Ext.getCmp(moduleCode+"_setValue"+i);
			if(!setv)
				continue;
			setv.setWidth(getValueWidth());
		}
	}
	/**
	 * 检查条件是否合法
	 * @return {Boolean}
	 */
	var checkFilter = function() {
		var n = filterStore.getCount();
		var leftPLen = 0;
		var rightPLen = 0;
		for (var i = 0; i < n; i++) {
			var record = filterStore.getAt(i);
			if (record.get("delFlag"))
				continue;
			var leftBr = record.get("leftBr");
			var rightBr = record.get("rightBr");
			leftPLen += leftBr ? leftBr.length : 0;
			rightPLen += rightBr ? rightBr.length : 0;
		}
		if (leftPLen != rightPLen) {
			Ext.Msg.alert("错误", "左括号与右括号数量不匹配,请检查。");
			return false;
		}
		return true;
	}
	/**
	 * 取筛选条件
	 * @return {Array} [修改过的,删除的(只记录有id的)]
	 */
	function getData()
	{
		var removed=[];
		var modified=[];
		filterStore.each(function(record)
		{
			if(!record.get("fieldName"))
				return;
			var operator=record.get("operator");
			var sortFlag=findRecordValue(store, "fieldName", record
							.get("fieldName"), "sort");
			if(sortFlag!=true&&(operator == allOperator.ASC || operator == allOperator.DESC))
				return;
			var dataType = findRecordValue(store, "fieldName", record
							.get("fieldName"), "fieldType");
			var _setValue=record.get("setValue");
			_setValue=_setValue+""=="0"?"0":_setValue;
			if (operator != allOperator.NULL && operator != allOperator.NOTNULL && operator != allOperator.ASC && operator != allOperator.DESC
					&& _setValue== "" && dataType != "6"
					&& dataType != "4")
					return;
			if(record.get("delFlag"))
			{
				if(!record.get("fieldId"))
					return;
				removed.push(Ext.encode(record.data));
			}else
				modified.push(Ext.encode(record.data));
		})
		return [modified,removed];
	}
	/**
	 * 点确认按钮触发事件
	 */
	function okQuery(){
		var respText=Ext.Ajax.request({
			sync:true,
			url:"query!putFilterInSession.action",
			params:{
				modified:getData()[0],
				moduleCode:moduleCode
			}
		});
		var result=Ext.util.JSON.decode(respText.conn.responseText);
		return result;
   }
   function addLineFromRecord(i,rec){
   		var _has=Ext.getCmp(moduleCode+"_fieldName"+i);
   		if(!_has){
			panel.add(getLine(i,true));
			panel.doLayout();
   		}
		Ext.getCmp(moduleCode+"_fieldName"+i).setValue(rec.get("fieldName"));
		if(!_has)
			Ext.getCmp(moduleCode+"_fieldName"+i).fireEvent("change",{value:rec.get("fieldName")});
		var operator=Ext.getCmp(moduleCode+"_operator"+i);
		if(operator==allOperator.ASC||operator==allOperator.DESC)//如果操作符为排序,存入对象
			sortInfo[rec.get('fieldName')]=i;
		var value=Ext.getCmp(moduleCode+"_setValue"+i);
		operator.setValue(rec.get("operator"));
		Ext.getCmp(moduleCode+"_relation"+i).setValue(rec.get("relation"));
		Ext.getCmp(moduleCode+"_operator"+i).fireEvent("change",{value:rec.get("operator")});
		Ext.getCmp(moduleCode+"_rightBr"+i).setValue(rec.get("rightBr"));
		Ext.getCmp(moduleCode+"_leftBr"+i).setValue(rec.get("leftBr"));
		var dataType=findRecordValue(store,"fieldName",rec.get("fieldName"),"fieldType");
		value.setValue(dataType==5?StringToDate(rec.get("setValue")):rec.get("setValue"));
		if(value.store){
			if(value.store.getCount()==0)	
				value.store.reload({callback:function(){
					value.setValue(value.getValue());
				}})
		}
		lineIndex=_has?lineIndex:i;
   }
/**
 * 条件store加载完成后事件,生成筛选框
 */
	function afterFilterStoreLoad()
	{
		var i=0;
		if(filterStore.getCount()>0)
		{
			filterStore.each(function(rec){
				addLineFromRecord(i,rec);
				i=i+1;
			})
			lineIndex=i-1;
		}else
		{
			panel.add(getLine(0));
			panel.doLayout();
		}
	}
	/**
	 * 初始化条件,取当前session中的条件
	 */
	function initFilter(){
		Ext.Ajax.request({
			url : "query!getFilterFromSession.action",
			params:{moduleCode:moduleCode},
			success:function(resp,action){
				var result=Ext.util.JSON.decode(resp.responseText);
				if(result.success){
					delAllLine();
					filterStore.loadData(result,true);
					filterStore.each(function(rec){
						if(rec.data.custom)
							win.customFilter[rec.data.fieldName]=rec.data.sort;
					})
				   filterStore.commitChanges();
					if(store.getCount()==0)
						store.load({callback:function(){afterFilterStoreLoad()}})
					else
						afterFilterStoreLoad();
				}
			}
		})
	}
	/**
	 * 树节点单击后事件
	 */
	function afterTreeClick()
	{
		delAllLine();
		filterStore.reload({params:{"head.id":currentFilter.id},callback:function(){
			afterFilterStoreLoad();
		}});
	}
/**
 * 保存筛选条件
 */
function saveFilter(data)
{
	if(!checkFilter())
			return;
		Ext.Ajax.request({
			url:"filterAction!doSaveFilter.action",
			params:{
				"head.id":currentFilter.id,
				"head.name":currentFilter.text,
				"head.moduleId":currentFilter.attributes.moduleId,
				"head.empId":currentFilter.attributes.empId,
				"head.orgId":currentFilter.attributes.orgId,
				moduleCode:moduleCode,
				modified:data[0],
				removed:data[1]
			},
			success:function(resp, opt) {
				var text = Ext.util.JSON
						.decode(resp.responseText);
				if (text.success){
						filterStore.commitChanges();
						currentFilter.id=text.id;
						filterTree.getRootNode().reload();
					}
			}
		})
}


//////////////////控件区///////////////////////////////////////////////////////////////////////////////////////////////////////////////////////////////////////

	/**
	 * 根据字段名获取编辑器和操作符集合
	 * @param {} fieldName 字段名
	 * @return {}
	 */
	var getChange=function(fieldName,index){
		var _flag=Ext.isNumber(index);
		var _store;
		var _valueStore;
		var dataType;
		var editorType= findRecordValue(store, "fieldName", fieldName, "inputType")
		var idRender= findRecordValue(store, "fieldName", fieldName, "idRender")
		var label=findRecordValue(store, "fieldName", fieldName, "fieldLabel");
		if(editorType)
		{
			if(editorType=="1"&&idRender)
				dataType="text";
			else if(editorType=="4")
				dataType="reference";
			else if(editorType!="1")
				dataType="combo";
		}
		var type=findRecordValue(store, "fieldName", fieldName, "fieldType");
		
		if(!dataType){
			if(!type)
			{
				alert(label+":"+fieldName+"列不能没有‘类型’并且也没有‘输入类型’");
				throw new Error("无效列:"+fieldName);
			}
			switch (type.toString()) {
					case "6" :
						dataType="gender";
						break; 
					case "5" :
						dataType="date";
						break; 
					case "4" :
						dataType="boolean";
						break; 
					case "2" :
						dataType="int";
						break; 
					case "3" :
						dataType="float";
						break;
					default:
						dataType = "text";
						break;
				}
			}
			switch (dataType) {
				case "text" :
					_store = stringOp;
					if(_flag)
						_valueStore = objGridTextEditor(index);
					break;
				case "date" :
					_store = datetimeOp;
					if(_flag)
						_valueStore = new Ext.form.DateField({
									emptyText:"请选择日期...",
									id : moduleCode+"_setValue" + index,
									format:findRecordValue(store, "fieldName", fieldName,"format")||"Y-m-d",
									dateFlag:true
								});
					break;
				case "gender" :
					_store = lookupOp;
					if(_flag)
						_valueStore = objGridGenderEditor(index);
					break;
				case "boolean" :
					_store = booleanOp;
					if(_flag)
						_valueStore = objGridBooleanEditor(index);
					break;
				case "int" :
					_store = numericOp;
					if(_flag)
						_valueStore = objGridIntegerEditor(index);
					break;
				case "float" :
					_store = numericOp;
					if(_flag)
						_valueStore = objGridFloatEditor(index);
					break;
				case "combo" :
					_store = lookupOp;
					if(_flag){
						_valueStore = new Ext.form.ComboBox({
								store : findRecordValue(store, "fieldName", fieldName,"store")||new Ext.data.Store({
										baseParams : {
											type : editorType,
											data : findRecordValue(store, "fieldName", fieldName,
													(editorType=="2")?"inputSql":"method")
										},
										proxy : new Ext.data.HttpProxy({
													url : "querySet!getFieldInputValues.action"
												}),
										reader : new Ext.data.JsonReader({
													root : "root"
												}, [{name:"fieldName",mapping:"value"}, {name:"fieldLabel",mapping:"text"}])
								}),
								mode : "local",
								triggerAction : "all",
								id : moduleCode+"_setValue" + index,
								editable : false,
								valueField : findRecordValue(store, "fieldName", fieldName,"disValue")||"fieldName",
								displayField : findRecordValue(store, "fieldName", fieldName,"disText")||"fieldLabel",
								emptyText : "请选择..."
							});
						if(_valueStore.store.getCount()==0)
							_valueStore.store.load();
					}
					break;
				case "reference" :
					_store = lookupOp;
					if(_flag){
						var _refData=findRecordValue(store, "fieldName", fieldName,"refData");
						_valueStore = new Ext.form.ReferenceField({
								refData : _refData,
								id : moduleCode+"_setValue" + index,
								editable : false,
								emptyText : "请选择...",
								multiple:_refData.multiple||false
							});
					}
					break;
			}
		var sort=findRecordValue(store, "fieldName", fieldName, "sort")&&(!_flag||(sortInfo[fieldName]+""!="0")&&!sortInfo[fieldName]);
		_store=_store.slice(0,_store.length);
		if(sort==true)
			_store.push(allOp[10],allOp[11]);
		return {store:_store,valueStore:_valueStore,dataType:dataType,type:type,inputType:editorType};
	}
	/**
	 * 新增一行条件
	 * @param {int} index 下标
	 * @param {Boolean} edit 如果手动新增不传或传false,自动新增为true
	 */
	var getLine = function(index,edit) {
		/**
		 * 操作符改变事件
		 */
		var operChange = function(ctl, nv, ov) {
			nv=nv||ctl.value;
			Ext.getCmp(moduleCode+"_setValue" + index).setDisabled(false);
			if (nv == allOperator.NULL || nv == allOperator.NOTNULL || nv == allOperator.ASC || nv == allOperator.DESC){
				Ext.getCmp(moduleCode+"_setValue" + index).setDisabled(true);
			}
			var _sort=nv==allOperator.ASC||nv==allOperator.DESC;
			var _reValue=Ext.getCmp(moduleCode+"_relation" + index).getValue();
			var _rec=filterStore.getAt(index);
			var hasSort=true;
			this.store.each(function(_r){
				if(_r.data.value==allOperator.ASC||_r.data.value==allOperator.DESC)
					return hasSort=false;
			})
			if(_sort){
				var _cfieldName=_rec.get("fieldName");
				if(!hasSort)
					sortInfo[_cfieldName]=index;
				Ext.getCmp(moduleCode+"_relation" + index).setValue("");
				Ext.getCmp(moduleCode+"_leftBr" + index).setValue("");
				Ext.getCmp(moduleCode+"_rightBr" + index).setValue("");
				valueChange('relation', "");
				valueChange('leftBr', "");
				valueChange('rightBr', "");
			}else{
				if(!hasSort)
					sortInfo[_cfieldName]=null;
				Ext.getCmp(moduleCode+"_relation" + index).setValue(_reValue||((index==0)?"":"and"));
				valueChange('relation', _reValue||((index==0)?"":"and"));
			}
			
			Ext.getCmp(moduleCode+"_relation" + index).setDisabled(_sort||(index==0));
			Ext.getCmp(moduleCode+"_leftBr" + index).setDisabled(_sort);
			Ext.getCmp(moduleCode+"_rightBr" + index).setDisabled(_sort);
			valueChange('operator', nv)
		}
		/**
		 * 属性名改变后事件
		 */
		var fileListChange = function(ctl, nv) {// 字段名改变后事件
			if(nv)
				edit=false;
			nv=nv||ctl.value;
			var _store;
			var _valueStore = Ext.getCmp(moduleCode+"_setValue" + index);
			if (_valueStore)
				_valueStore.destroy();
//			var dataType;

			var _changeResult=getChange(nv,index);//获取类型相关的编辑器和操作符集合
			filterStore.getAt(index).set("inputType",_changeResult.inputType);
			_store=_changeResult.store;
			_valueStore=_changeResult.valueStore;
			var oper = Ext.getCmp(moduleCode+"_operator" + index);
			oper.clearValue();
			oper.store.loadData(_store);

			_valueStore.setWidth(getValueWidth());
			_valueStore.on("focus", focus);
			_valueStore.on("change", function(ctl, v, ov) {
				Ext.getCmp(moduleCode+"_fieldName" + index).focus();
				valueChange("setValue", ctl.dateFlag?v.dateFormat(this.format):v)
			});
			var valuePanel = Ext.getCmp(moduleCode+"_valuePanel" + index);
			var dataType=_changeResult.dataType;
			var type=_changeResult.type;
			valuePanel.add(_valueStore);
			valuePanel.doLayout(true);
			if(!edit)
			{
				valueChange("setValue", dataType=="4"?"0":""); 
				valueChange("operator", allOperator.EQ); 
				valueChange("fieldName", nv);
				valueChange("fieldLabel", findRecordValue(store, "fieldName", nv,
										"fieldLabel"));
				valueChange("fieldType", type||"string");
				valueChange("idRender",findRecordValue(store, "fieldName", nv,
										"idRender"));
				oper.setValue(allOperator.EQ);
				Ext.getCmp(moduleCode+"_setValue"+index).setValue("");
			}
		}
		/**
		 * 所有构造条件控件获焦事件
		 */
		function focus() {
			currentIndex = index;
		}
		/**
		 * 所有构造条件控件值改变后事件
		 * @param {String} name 控件基础名
		 * @param {String} v 值
		 */
		function valueChange(name, v) {
			var record = filterStore.getAt(index);
			record.set(name, v);
		}
		if(!edit)
			filterStore.insert(index, new filterStore.recordType({relation:(index>0)?"and":"",sort:index}));
		return {layout:"column",border:false,id:moduleCode+"_line"+index,items:[new Ext.form.ComboBox({
							columnWidth : .1,
							store : [["and", "并且"], ["or", "或者"]],
							value : (index > 0) ? "and" : "",
							disabled : (index == 0),
							id : moduleCode+"_relation" + index,
							triggerAction : "all",
							listeners : {
								focus : focus,
								change : function(ctl, v, ov) {
									valueChange("relation", v)
								}
							}
						}), new Ext.form.ComboBox({
							id : moduleCode+"_leftBr" + index,
							columnWidth : .1,
							store : ["(", "((", "((", "(((", "(((("],
							triggerAction : "all",
							regex :/^\({1,4}$/,
							editable : true,
							listeners : {
								focus : focus,
								change : function(ctl, v, ov) {
									valueChange("leftBr", this.isValid()?v:"")
									if(!this.isValid())
										this.setValue("")
								}
							}
						}), new Ext.form.ComboBox({
							id : moduleCode+"_fieldName" + index,
							columnWidth : .2,
							store : store,
							mode : "local",
							hiddenName : "value",
							triggerAction : "all",
							valueField : "fieldName",
							displayField : "fieldLabel",
							editable : false,
							listeners : {
								change : fileListChange,
								focus : focus
							}
						}), new Ext.form.ComboBox({
							columnWidth : .18,
							id : moduleCode+"_operator" + index,
							mode : "local",
							valueField : "value",
							displayField : "text",
							triggerAction : "all",
							store : new Ext.data.SimpleStore({
										fields : ["value", "text"]
									}),
							listeners : {
								focus : focus,
								change : operChange
							}
						})

				, {
					layout : "table",
					column : 1,
					id : moduleCode+"_valuePanel" + index,
					border : false,
					columnWidth : .32,
					items : new Ext.form.TextField({
								width:getValueWidth(),
								id : moduleCode+"_setValue" + index,
								listeners : {
									focus : focus,
									change : function(ctl, v, ov) {
										valueChange("setValue", v)
									}
								}
							})
				}, new Ext.form.ComboBox({
							id : moduleCode+"_rightBr" + index,
							columnWidth : .1,
							store : [")", "))", ")))", "))))"],
							mode : "local",
							triggerAction : "all",
							valueField : "value",
							displayField : "text",
							regex :/^\){1,4}$/,
							editable : true,
							listeners : {
								focus : focus,
								change : function(ctl, v, ov) {
									valueChange("rightBr",this.isValid()?v:"")
									if(!this.isValid())
										this.setValue("")
								}
							}
						})]}
	}
	/**
	 * 工具条
	 */
	var tool = new Ext.Toolbar({
				items : ["->",{
							xtype : "button",
							text : "增行",
							tooltip:"AddLine",
							icon : KANGSOFT+"/js/images/add_16.gif",
							listeners : {
								click : function() {
									panel.add(getLine(++lineIndex));
									panel.doLayout();
									currentIndex=lineIndex;
								}
							}
						},"-", {
							xtype : "button",
							text : "删行",
							tooltip:"DeleteLine",
							icon : KANGSOFT+"/js/images/delete.gif",
							handler : function() {
								if (currentIndex||(currentIndex+""=="0")) {
									Ext.Msg.confirm("删除提示", "你要删除此行吗?",
										function(v) {
											if (v == "yes"){
												deleteLine();
												if(currentIndex==firstIndex){
													var i=0;
													var _flag=true;
													filterStore.each(function(rec){
														if(!rec.get("delFlag"))
															return _flag=false;
														i++;
													})
													if(_flag)
													{
														panel.add(getLine(0));
														panel.doLayout();
														currentIndex=null;
														firstIndex=0;
													}
													else{
														firstIndex=i;
														var record = filterStore.getAt(firstIndex);
														record.set("relation","");
														Ext.getCmp(moduleCode+"_relation"+i).setValue("");
														Ext.getCmp(moduleCode+"_relation"+i).setDisabled(true);
													}
													
												}
											}
									})
								}else
									Ext.Msg.alert("删除提示","没有选择行");
							}
						},"-", {
							text : "全部清除",
							tooltip:"RemoveAll",
							icon : KANGSOFT+"/js/images/reset.png",
							handler : function() {
								delAllLine();
								panel.add(getLine(lineIndex));
								panel.doLayout();
							}
						},"-",{text:"刷新",tooltip:"Refresh",icon : KANGSOFT+"/js/images/refresh.png",handler:function(){
				        	if(currentFilter.id)
				        		afterTreeClick();
				        	else
				        		initFilter();
				        }},"-",{text:"保存",
				        	disabled:moduleCode?false:true,
							tooltip:"Save",
							icon : KANGSOFT+"/js/images/save.gif",handler:function(){
							if(!checkFilter())
								return;
							var data=getData();
							if(!data[0][0]&&!data[1][0])
							{
								Ext.Msg.alert("保存提示","没有修改条件,不需要保存");
								return;
							}
							Ext.MessageBox.prompt("筛选保存提示", "筛选名称:", function(btn,text){
							 	if(btn=="ok")
								 	{
								 		currentFilter.text=text;
								 		saveFilter(data);
								 	}
							 },this,false,currentFilter.text||"新增条件1");
				        }},"-", {
							text : "确认",
				        	disabled:moduleCode?false:true,
							tooltip:"Ok/Query",
							icon : KANGSOFT+"/js/images/search.png",
							handler : function() {
								if (checkFilter()) {
									if(true||filterStore.getModifiedRecords().length>0){
//										var respText=Ext.Ajax.request({
//											sync:true,
//											url:"query!putFilterInSession.action",
//											params:{
//												modified:getData()[0],
//												moduleCode:moduleCode
//											}
//										});
//										var result=Ext.util.JSON.decode(respText.conn.responseText);
										var result=okQuery();
										if(result.success){
											 var fn=callback.createCallback(result.success);
											 fn();
										}
									}
									 win.hide();
									 filterStore.commitChanges();
								}

							}
						},"-", {
							text : "关闭/返回",
							tooltip:"Cancel",
							icon : KANGSOFT+"/js/images/cancel.png",
							handler : function() {
								win.hide();
							}
						}],
				height : 25
			});
	/**
	 * 构造筛选条件面板
	 */
	var panel = new Ext.Panel({
				id : moduleCode+"_main_panel",
				tbar:tool,
				anchor:"101%",
				autoScroll:true,
				border:false,
				height:355,
				items : [{layout:"column",border:false,items:[{
							height:0,
							title : "关系",
							columnWidth:.1
						}, {
							title : "括号",
							height:0,
							columnWidth:.1
						}, {
							height:0,
							title : "字段名",
							columnWidth:.2
						}, {
							height:0,
							title : "运算符",
							columnWidth:.18
						}, {
							height:0,
							title : "内容",
							columnWidth:.32
						}, {
							height:0,
							title : "括号",
							columnWidth:.1
						}]}]
			});
	panel.add(getLine(0));
	
	var treeLoader = new Ext.tree.TreeLoader({
				dataUrl : "filterAction!getFilterTree.action",
				baseParams:{moduleCode:moduleCode}
			});
	/**
	 * 筛选条件树
	 */		
	var filterTree=new Ext.tree.TreePanel({//已保存的筛选树
				tools:[{qtip:"删除条件",handler:function(){
					if(currentFilter&&currentFilter.id)
					Ext.Msg.confirm("删除提示","你确认删除此条件吗?",function(v){
						if(v=="yes")
						Ext.Ajax.request({
							url:"filterAction!doDeleteFilter.action",
							params:{
								"head.id":currentFilter.id
							},
							success:function(resp, opt) {
								var text = Ext.util.JSON.decode(resp.responseText);
								if (text.success){
										Ext.Msg.alert("删除提示","删除成功!");
										currentFilter={attributes:{}};
										filterTree.getRootNode().reload();
									}
							}
						});
					})
				}},{id:"right",qtip:"停靠",hidden:true,handler:function(){filterTree.expand();filterTree.tools.left.show();this.hide();panel.doLayout(true,false)}},
							{id:"left",qtip:"收起条件树",handler:function(){filterTree.collapse();treeColFlag=!treeColFlag;changeValueWidth();filterTree.tools.right.show();this.hide()}}],				
				autoScroll : true,
				region:"west",
				width : 130,
				lines : true,
				rootVisible : true
			});
	filterTree.on("expand",function(p){
		treeColFlag=!treeColFlag;changeValueWidth();
	})
	var treeRoot = new Ext.tree.AsyncTreeNode({
				id : "0",
				text : "root",
				draggable : false
			})
	filterTree.setRootNode(treeRoot);
	filterTree.loader = treeLoader;
	
	initFilter();
	filterTree.on("click",function(node){
		if(node.id==currentFilter.id)
			return;
		if(!node.parentNode)
		{
			if(currentFilter.id)
			{
				currentFilter={attributes:{}};
				delAllLine();
				panel.add(getLine(0));
				panel.doLayout();
			}
			return;
		}
		if(store.getCount()==0)
			store.load({});
		currentFilter=node;
		afterTreeClick();
	})

/**
 * 筛选面板
 */
function addFilter(data){
//	alert(1)
	
	function add(filter){	
		if(!Ext.isObject(filter))
			throw new Error("手动增加筛选条件不符合规定, 只能是对象或对象数组");
		if(!filter.fieldName)
			return;
		var result=getChange(filter.fieldName);
		if(!result||!result.store)
			return;
		if(filter.operator){
			var hasFlag=false;
			Ext.each(result.store,function(rec){
				if(rec[0]==filter.operator){
					hasFlag=true;
					return false;
				}
			});
			if(!hasFlag)
				throw new Error("指定的:'"+filter.fieldName+"'列的运算符不在允许范围内,请检查!");
		}
		var record={};
		store.each(function(rec){
			if(rec.data.fieldName==filter.fieldName)
			{
				record=rec.data;
				return false;
			}
		})
		record.operator=allOperator.EQ;
		Ext.applyIf(filter,{
			fieldLabel:record.fieldLabel,
			fieldType:record.fieldType,
			inputType:record.inputType,
			inputSql:record.inputSql,
			method:record.method,
			idRender:record.idRender,
			operator:record.operator
		});
		Ext.apply(filter,{
				relation:(lineIndex>0)?"and":""
			}
		)
		if(lineIndex==0&&!filterStore.getAt(0).data.fieldName){
//			filterStore.removeAll();
			delAllLine();
			lineIndex=-1;
		}
		var filterRec=new filterStore.recordType(filter);
		var _index=win.customFilter[filter.fieldName]||sortInfo[filter.fieldName];
		if(_index+""==0||_index){
			var _rec=filterStore.getAt(_index);
			for(var p in filter)
				_rec.set(p,filter[p]);
			currentIndex=_index;
		}else
		{
			filterRec.data.custom="1";
			filterRec.data.sort=lineIndex+1;
			filterStore.insert(++lineIndex, filterRec);
			win.customFilter[filter.fieldName]=lineIndex;
			currentIndex=lineIndex;
		}
		addLineFromRecord(currentIndex,filterRec);
	}
	if(Ext.isArray(data)){
		Ext.each(data,function(f){
			add(f);
		})
	}else
		add(data);
}
var getFilters=function(flag){
	if(flag){
		
	}else
		return filterStore.data;
}
var win=Ext.getCmp("win"+moduleCode);
if(!win)
	win = new Ext.Window({
			plain : true,
			resizable : false,
			draggable : true,
			loadMask : true,
			id : "win"+moduleCode,
			modal : true,
			layout:"border",
			closeAction : "hide",
			width : 650,
			height : 400,
			closable:false,
			customFilter:{},
			title : "<img src='"+KANGSOFT+"/js/images/search.png' style='vertical-align: middle;'>筛选窗体</img>",
			items : [filterTree,{layout:"anchor",region:"center",items:[panel]}],
			allOperator:allOperator,
			addFilter:addFilter,//自增加筛选行
			getFilters:getFilters,
			clearFilter:function(afterClear){//手动清除所有行
				delAllLine();
				panel.add(getLine(lineIndex));
				panel.doLayout();
				okQuery();
				if(afterClear)
					afterClear.call();
			},
			run:function(){
//				if (filterStore.getModifiedRecords().length>0&&checkFilter()) {
//					var result=okQuery();
				var result=this.save();
				if(result){
					 var fn=callback.createCallback(result.success);
					 fn();
					filterStore.commitChanges();
				}
//				}
			},
			save:function(){
				if (filterStore.getModifiedRecords().length>0&&checkFilter()) {
					var result=okQuery();
					if(!result.success){
						Ext.Msg.alert("操作提示","保存条件失败");
					}
					return result;
				}
				return false;
			}
		});
	return win;
}

FilterProxy=function(config){
	Ext.apply(this,config);
	FilterProxy.constructor.call(this);
}
Ext.extend(FilterProxy,Object,{
	addFilter:function(data){
		var obj=this;
		function add(filter){	
			var _fname=obj[filter.fieldName]||filter.fieldName;
			filter.fieldName=_fname;
		}
		if(Ext.isArray(data)){
			Ext.each(data,function(f){
				add(f);
			})
		}else
			add(data);
		obj.win.addFilter(data);
	},
	run:function(){this.win.run()},
	save:function(){this.win.save()}
})

 使用部分代码如下:

 

/** ******************************************筛选*************************************************** */
					var reloadStore =function(flag)
					{
						if(flag)
							ds.load({params:{start:0,limit:15}})
					}
					var fieldsDef = new Ext.data.JsonStore({
						fields :  ['fieldName','fieldLabel',
									'fieldType','inputType',"idRender","store",
									'disValue', 'disText'],
						data : [{
									fieldName : 's.itemClassId',
									fieldLabel : '大类',
									fieldType : 'string',
									idRender:'itemClass_hql'
								}, {
									fieldName : 's.ornaClassId',
									fieldLabel : '小类',
									fieldType : 'string',
									idRender:'ornaClass_hql'
								}, {
									fieldName : 's.sizeName',
									fieldLabel : '尺寸/规格',
									fieldType : 'string'
								}, {
									fieldName : 's.errorStart',
									fieldLabel : '误差起始范围',
									fieldType : 'string'
								}, {
									fieldName : 's.errorEnd',
									fieldLabel : '误差截止范围',
									fieldType : 'string'
								}]
					});
					//创建筛选grid
					var filter =createFilter(fieldsDef,'040623',reloadStore);

 注:其中的后台代码部分,根据项目的具体情况由使用者自己发挥开发。建议将相应的筛选条件部分放入session,session的ID可以设置为模块代码,便于与其它模块的筛选功能部分的条件区分。此筛选功能还有待进一步完善,如果有使用到的朋友在开发过程中有什么更好的建议,希望给予支持。(QQ:329365156)

你可能感兴趣的:(json,Ajax,qq,浏览器,ext)