游乐游手机版
首页/前端开发/文章详情

EXTJS记事本中复合字段与行编辑器的结合实践

时间:2026-06-15 07:01
在RowEditor中利用CompositeField嵌入三个ComboBox实现物料级联筛选,需手动处理控件初始化与Change事件。通过扩展afterstart事件进行控件赋值,在validateedit中校验重复物料,并在afteredit中手动将值写回记录。关键是通过editor items items[0]访问CompositeField子组件。
EXTJS记事本 当CompositeField遇上RowEditor 在进行BOM(物料清单)编辑时,经常需要在RowEditor中实现级联下拉菜单。这个需求看似简单,实际开发中却有不少陷阱。 背景是这样的:客户的物料种类繁多,多达一千多种。如果仅使用单个ComboBox,用户查找物料几乎只能靠猜测。因此,我设计了通过“物料分类”和“物料品牌”两个ComboBox进行级联筛选的方案。然而,问题也随之而来:在RowEditor的一个字段内嵌入多个控件,意味着需要手动管理每个控件的初始化及Change事件。我搜索了网上现有方案,发现没有合适的现成解决方案。经过整整三天的调试,终于成功解决了这一难题。 下面展示RowEditor的定义及关键事件处理代码。

复制代码 代码如下:

var editor=new Ext.ux.grid.RowEditor({ sa veText: '确定', cancelText:"放弃", commitChangesText: '请确定或放弃修改', errorText: '错误' }); // 当取消时,根据关键字段值是否为空来删除空记录 editor.on("canceledit",function(editor,pressed) { if(pressed && editor.record.get("materialid")==0) { store.remove(editor.record); } },this); /* * afterstart 这个事件是自己加的。 * 在beforeedit事件里想初始化自定义控件?不行,因为那时候RowEditor还没渲染。 * 所以自己扩展了一个afterstart事件——它在RowEditor显示后立即触发,这里做初始化再合适不过。 * 还有一处关键:通过RowEditor控件遍历来访问自定义的CompositeField。 * 注意写法:editor.items.items[0] * 这并不是写重了,而是RowEditor的items竟然不是一个数组,而是一个对象。这部分也耗费了不少时间,最后是靠FireBug打印出editor对象才发现的。 * editor.items.items[0] 就是CompositeField组件,通过它的items集合可以标准访问子组件,剩下的初始化就好办了。 * 最后一个ComboBox的数据依赖前两个ComboBox级联选取,所以需要在这里动态加载数据。 * 重点:必须在callback中执行,因为JsonStore的load是异步的,数据加载成功后,再用setValue初始化值。 */ editor.on("afterstart",function(editor,rowIndex) { var record=store.getAt(rowIndex); editor.items.items[0].items.items[0].setValue(record.get("setid")); editor.items.items[0].items.items[1].setValue(record.get("category")); var t_store=editor.items.items[0].items.items[2].getStore(); t_store.load({ params:{category:record.get("category"),setid:record.get("setid")}, callback:function(r,options,success){ if (success) editor.items.items[0].items.items[2].setValue(record.get("materialid")); } }); },this); /* * validateedit 事件在点击确认时执行,用来验证RowEditor中的控件值。 * 这里加了一个自定义校验:禁止用户添加重复物料。 * 实现逻辑是遍历JsonStore中的每条记录,将物料值与用户选择的值比对,如果重复则给出提示。 */ editor.on("validateedit",function(editor,obj,record,rowIndex){ var materialid=editor.items.items[0].items.items[2].getValue(); var exist=false; Ext.each(store.getRange(),function(o,i){ if(o!=record && o.get("materialid")==materialid) { exist=true; return(false); } }); if(exist) { Ext.MessageBox.alert("系统提示","请勿重复添加"); store.remove(record); } return(!exist); },this); /* * afteredit 在通过校验后执行。 * 这一步至关重要:因为使用了CompositeField,RowEditor无法自动将值正确赋给record的对应属性。 * 需要手动把用户选择的值写回字段。materialid是物料编号,model是型号。 * 为什么还要赋值model?因为列的显示值就是model,不赋值,界面上就空着。 */ editor.on("afteredit",function(editor,obj,record,rowIndex){ record.set("materialid",editor.items.items[0].items.items[2].getValue()); record.set("model",editor.items.items[0].items.items[2].getRawValue()); },this);

定义好编辑器并完成事件绑定后,接下来将RowEditor作为插件集成到GridPanel中。

复制代码 代码如下:

{ xtype:"grid", title:"产品BOM", layout:"fit", store:store, enableDragDrop: false, border: false, frame:false, autoScroll:true ,plugins:[editor], sm:sm, height:340, clicksToEdit:2, autoWidth: true, viewConfig:{forceFit:true,autoFill:true,markDirty:false} }

下面是最关键的列定义部分,展示了CompositeField的具体配置方式。

复制代码 代码如下:

columns: [{ header: "物料名称/型号", dataIndex: "model", width: 200, menuDisabled: true, editor: { // 定义编辑器 xtype:"compositefield", name:"compositefield", items:[ { xtype: "combo", mode:"local", name:"sets", width:80, fieldLabel: "适用产品品牌", emptyText:"请选择", valueField: "id", lazyInit:false, value:this.data?this.data.title:"", hiddenName:"setid", hiddenValue:this.data?this.data.setid:"", displayField: "title", typeAhead: false, forceSelection: true, editable:true, listeners:{ "change":function(combo,newvalue,oldvalue) { // 处理品牌的change事件——在选取品牌后,重新载入后面的ComboBox。 // 这里的editor就是前文定义的RowEditor实例。 var category=editor.items.items[0].items.items[1]; var material=editor.items.items[0].items.items[2]; var c=category.getValue(); var store=material.getStore(); store.load({ params:{setid:newvalue,category:c}, callback:function(r,options,success){ if (success) material.setValue(""); } }); } }, triggerAction: "all", store: new Ext.data.JsonStore({ url: "<%=script_path%>data.asp", root: "data",autoDestroy:true, remoteSort: true, listeners:{"load":function(store,records,option){ var s=Ext.data.Record.create([{name:"id",type:"int"},{name:"title",type:"string"}]); store.add(new s({id:0,title:"通用"})) }}, baseParams: {op: "setList"}, totalProperty: "total", autoLoad: true, fields: ["title","id"] }) }, { xtype: "combo", mode:"local",width:60, name:"category", fieldLabel: "类别", emptyText:"请选择", valueField: "category", lazyInit:false, value:this.data?this.data.category:"", displayField: "category", typeAhead: false,forceSelection: true, triggerAction: "all", listeners:{ "change":function(combo,newvalue,oldvalue) { // 处理类别的change事件——同样,选取后重新载入物料ComboBox。 var sets=editor.items.items[0].items.items[0]; var material=editor.items.items[0].items.items[2]; var setid=sets.getValue(); var store=material.getStore(); store.load({ params:{category:newvalue,setid:setid}, callback:function(r,options,success){ if (success) material.setValue(""); } }); } }, store: new Ext.data.JsonStore({ url: "<%=script_path%>data.asp", root: "data",autoDestroy:true, remoteSort: true, baseParams: {op: "materialCategoryList"}, totalProperty: "total", autoLoad: true, fields: ["category"] }) }, { xtype: "combo", forceSelection: true, editable:true, mode:"local", name:"material", fieldLabel: "物料", emptyText:"请选择物料", valueField: "id", allowBlank:false, displayField: "model", width:250, lazyInit:false, typeAhead: false, triggerAction: "all", listeners:{ "change":function(combo,newvalue,oldvalue) { // 这里一定要注意!!!如果没有下面这两句,你会发现选择后显示的值不会变化,点了确认也无法更新。 // 原因是RowEditor通过检测record的isdirty属性来判断是否触发validateedit和afteredit。 // 它实际上检测的是每列对应控件的值是否变化。由于物料型号这一列对应的是CompositeField, // 所以必须强制让CompositeField的值变化,RowEditor才会正常调用validateedit和afteredit。 // 此外,CompositeField的值还会被用来显示在列里。 var comp=editor.items.items[0]; comp.setRawValue(combo.getRawValue()); } }, store: new Ext.data.JsonStore({ url: "<%=script_path%>data.asp", root: "data",autoDestroy:true, remoteSort: true, baseParams: {op: "materialList"}, totalProperty: "total", autoLoad: false, fields: ["model","id"] })} ] } }, { header: "数量", dataIndex: "qty", width: 50, menuDisabled: true, editor: { xtype: 'numberfield', minValue:1, allowDecimals:false } } ,{ header: "颜色", dataIndex: "color", width: 60, menuDisabled: true } ,{ header: "尺寸", dataIndex: "size", width: 60, menuDisabled: true } ] ]

以上经验分享给大家,希望对遇到类似问题的朋友有所帮助。
来源:https://www.jb51.net/article/27844.htm
上一篇ExtJS GridPanel横向滚动条问题解决方案 下一篇ExtJS CheckboxGroup初始值设置代码
本站内容用于信息整理与展示,如有侵权或内容问题请及时联系处理。

相关推荐

补充同频道和同主题内容,方便继续浏览更多相关内容。

同类最新

继续查看同栏目最近更新的文章。

更多
Vue应用中异步更新性能问题的优化策略详解
前端开发 · 2026-07-03

Vue应用中异步更新性能问题的优化策略详解

先来看一个令许多开发者感到困惑的场景:明明修改了数据,DOM 却“毫无反应”,无法获取最新的高度,也无法计算正确的坐标。这并非 Vue 的缺陷,反而是它精心设计的性能优化策略。核心在于——你需要学会与它“异步更新”的特性协作,而非硬碰硬。 所谓的“异步更新性能问题”,本质上是一种认知偏差。Vue 的

如何避免原型对象挂载大体积动态数组内存污染
前端开发 · 2026-07-03

如何避免原型对象挂载大体积动态数组内存污染

原型链上的大数组:一个隐蔽的内存冲击波 先给个核心判断:直接在原型对象上挂载一个大体积动态数组,这既不是传统意义上的内存“污染”,也不是安全漏洞那种“污染”,而是一种相当隐蔽但后果严重的内存管理失当。它会导致所有实例共享同一份数据,而且正因为生命周期跟整个原型链绑定得太紧,垃圾回收器(GC)根本看不

利用堆栈信息精准定位显式绑定错误对象致未定义异常
前端开发 · 2026-07-03

利用堆栈信息精准定位显式绑定错误对象致未定义异常

深入追踪:显式绑定传错对象引发的未定义异常 说实话,这类问题在JavaScript开发中相当常见——显式绑定传错了对象,然后方法执行时静默失败、访问undefined、或者抛出TypeError。但真正的难点不在于“报了什么错”,而在于“到底是哪个对象被绑错了”。要解决它,需要跳出堆栈的表层报错信息

ES模块中默认导出和具名导出的执行上下文
前端开发 · 2026-07-03

ES模块中默认导出和具名导出的执行上下文

export default 与具名导出在 ES Module 中的行为机制截然不同,核心差异不在于“值如何传递”,而在于绑定如何建立以及导入时如何使用。先给出总结性结论,再逐一详细拆解。 export default 是一种语法糖,而非真正的变量声明 这种设计容易引起误解。实际上,export d

详解HTML中iframe标签loading=lazy属性实现嵌入内容懒加载方法
前端开发 · 2026-07-03

详解HTML中iframe标签loading=lazy属性实现嵌入内容懒加载方法

先聊聊 loading= "lazy " 这个属性——它本意是让 iframe 实现延迟加载,但实际落地时常常“失效”。这并非程序漏洞,而是浏览器内置的防御机制:只有所有条件同时触发,它才会真正推迟资源请求。比如 src 必须是跨域地址(类似 https: widget example com emb