Ext5實現樹形下拉框ComboxTree

来源:http://www.cnblogs.com/hhq365/archive/2016/06/14/5585245.html
-Advertisement-
Play Games

最近為了實現一個屬性下拉框被Ext框架折騰了好幾天。。 所以,首先要說的是,不管你要做什麼系統、強烈建議你不要選擇Ext。據我這幾天的搜索,應該這個框架現在用的人也很少了。 Ext框架的缺陷:框架沉重、擴展性差(與其他js框架相比)、各版本差別大(Ext3、4、5不相容)。 現在進入正題,這幾天研究 ...


最近為了實現一個屬性下拉框被Ext框架折騰了好幾天。。 

所以,首先要說的是,不管你要做什麼系統、強烈建議你不要選擇Ext。據我這幾天的搜索,應該這個框架現在用的人也很少了。

Ext框架的缺陷:框架沉重、擴展性差(與其他js框架相比)、各版本差別大(Ext3、4、5不相容)。

現在進入正題,這幾天研究ext實現樹形下拉框發現網上常見的兩種做法:

1、擴展Ext.form.field.ComboBox

  使用這種方式的好處是ComboBox與我們要實現的東西其行為更為相似,在取值、賦值方面比較方便、可以對觸發下拉事件上做定製。

  最後實現的東西發現瀏覽器不能相容、只有Win10的Edge可以完美呈現,無奈放棄。。

2、擴展Ext.form.field.Picker

  上一條路沒有走通最終選擇了這種方式。這種方式的好處:對樹形節點的展開、關閉不會影響到用戶的選取操作。

 

這段代碼也是根據網路上的源碼做了修改,之所以發出來因為網上實在是找不到基於Ext5的實現。所以我判斷現在使用Ext框架的人應該極少了,網上有的都是幾年前使用Ext4、3的人發表的。所有基於Ext4的實現在Ext5上都不能完美支持。。

 

定義組件:

Ext.define('Ext.ux.ComboBoxTree', {
    extend: 'Ext.form.field.Picker',
    requires: ['Ext.tree.Panel'],
    alias: ['widget.comboboxtree'],
    multiSelect: false,
    multiCascade: true,
    rootVisible: false,
    displayField: 'text',
    emptyText:'',
    submitValue: '',
    url:'',
    pathValue: '',
    defaultValue: null,
    pathArray: [],
    selectNodeModel: 'all',
    setValue: function (value) {
        if (value) {//註意:此處的判斷會使id為0的值選中失效
            if (typeof value == 'number') {
                this.defaultValue = value;
            }
            this.callParent(arguments);
        }
    },
    initComponent: function () {
        var self = this;
        self.selectNodeModel = Ext.isEmpty(self.selectNodeModel) ? 'all' : self.selectNodeModel;

        Ext.apply(self, {
            fieldLabel: self.fieldLabel,
            labelWidth: self.labelWidth
        });
        self.store = Ext.create('Ext.data.TreeStore', {
            root: { expanded: true },
            proxy: { type: 'ajax', url: self.url },
            autoLoad: true
        });
        self.store.addListener('load', function (st, rds, opts) {
            if (self.defaultValue) {
                var defaultRecord = self.store.getNodeById(self.defaultValue);
                self.setDefaultValue(defaultRecord.get('id'), defaultRecord.get('text'));
            } else {
                self.setDefaultValue('', self.emptyText);
            }
        });
        self.callParent();
    },
    createPicker: function () {
        var self = this;

        self.picker = Ext.create('Ext.tree.Panel', {
            //height: self.treeHeight == null ? 200 : self.treeHeight,
            autoScroll: true,
            floating: true,
            focusOnToFront: false,
            shadow: true,
            ownerCt: this.ownerCt,
            useArrows: false,
            store: this.store,
            rootVisible: this.rootVisible,
            displayField: this.displayField,
            viewConfig: {
                onCheckboxChange: function (e, t) {
                    if (self.multiSelect) {
                        var item = e.getTarget(this.getItemSelector(), this.getTargetEl()),
                            record;
                        if (item) {
                            record = this.getRecord(item);
                            var check = !record.get('checked');
                            record.set('checked', check);
                            if (self.multiCascade) {
                                if (check) {
                                    record.bubble(function (parentNode) {
                                        if ('Root' != parentNode.get('text')) {
                                            parentNode.set('checked', true);
                                        }
                                    });
                                    record.cascadeBy(function (node) {
                                        node.set('checked', true);
                                        node.expand(true);
                                    });
                                } else {
                                    record.cascadeBy(function (node) {
                                        node.set('checked', false);
                                    });
                                    record.bubble(function (parentNode) {
                                        if ('Root' != parentNode.get('text')) {
                                            var flag = true;
                                            for (var i = 0; i < parentNode.childNodes.length; i++) {
                                                var child = parentNode.childNodes[i];
                                                if (child.get('checked')) {
                                                    flag = false;
                                                    continue;
                                                }
                                            }
                                            if (flag) {
                                                parentNode.set('checked', false);
                                            }
                                        }
                                    });
                                }
                            }
                        }
                        var records = self.picker.getView().getChecked(),
                            names = [],
                            values = [];
                        Ext.Array.each(records, function (rec) {
                            names.push(rec.get('text'));
                            values.push(rec.get('id'));
                        });
                        self.submitValue = values.join(',');
                        self.setValue(names.join(','));
                    }
                }
            }
        });

        self.picker.on({
            itemclick: function (view, recore, item, index, e, object) {
                var selModel = self.selectNodeModel;
                var isLeaf = recore.data.leaf;
                var isRoot = recore.data.root;
                var view = self.picker.getView();
                if (!self.multiSelect) {
                    if ((isRoot) && selModel != 'all') {
                        return;
                    } else if (selModel == 'exceptRoot' && isRoot) {
                        return;
                    } else if (selModel == 'folder' && isLeaf) {
                        return;
                    } else if (selModel == 'leaf' && !isLeaf) {
                        var expand = recore.get('expanded');
                        if (expand) {
                            view.collapse(recore);
                        } else {
                            view.expand(recore);
                        }
                        return;
                    }
                    self.submitValue = recore.get('id');
                    self.setValue(recore.get('text'));
                    self.eleJson = Ext.encode(recore.raw);
                    self.collapse();
                }
            }

        });
        return self.picker;
    },
    listeners: {
        expand: function (field, eOpts) {
            var picker = this.getPicker();
            if (!this.multiSelect) {
                if (this.pathValue != '') {
                    picker.expandPath(this.pathValue, 'id', '/', function (bSucess, oLastNode) {
                        picker.getSelectionModel().select(oLastNode);
                    });
                }
            } else {
                if (this.pathArray.length > 0) {
                    for (var m = 0; m < this.pathArray.length; m++) {
                        picker.expandPath(this.pathArray[m], 'id', '/', function (bSucess, oLastNode) {
                            oLastNode.set('checked', true);
                        });
                    }
                }
            }
        }
    },
    clearValue: function () {

        this.setDefaultValue('', '');

    },
    getEleJson: function () {
        if (this.eleJson == undefined) {
            this.eleJson = [];
        }
        return this.eleJson;
    },
    getSubmitValue: function () {
        if (this.submitValue == undefined) {
            this.submitValue = '';
        }
        return this.submitValue;
    },
    getDisplayValue: function () {
        if (this.value == undefined) {
            this.value = '';
        }
        return this.value;
    },
    getValue: function () {
        return this.getSubmitValue();
    },
    setPathValue: function (pathValue) {
        this.pathValue = pathValue;
    },

    setPathArray: function (pathArray) {
        this.pathArray = pathArray;
    },
    setDefaultValue: function (submitValue, displayValue) {
        this.submitValue = submitValue;
        this.setValue(displayValue);
        this.eleJson = undefined;
        this.pathArray = [];
    },
    alignPicker: function () {
        var me = this,
            picker,
            isAbove,
            aboveSfx = '-above';
        if (this.isExpanded) {
            picker = me.getPicker();
            if (me.matchFieldWidth) {
                picker.setWidth(me.bodyEl.getWidth());
            }

            if (picker.isFloating()) {
                picker.alignTo(me.inputEl, "", me.pickerOffset); // ""->tl
                isAbove = picker.el.getY() < me.inputEl.getY();
                me.bodyEl[isAbove ? 'addCls' : 'removeCls'](me.openCls + aboveSfx);
                picker.el[isAbove ? 'addCls' : 'removeCls'](picker.baseCls + aboveSfx);
            }
        }
    }
});

以上代碼有待優化,由於只用了單選,所以多選應該還有點問題。

調用代碼:

Ext.create('Ext.ux.ComboBoxTree', {
                cId: 'cbOrganizationId',
                name: 'OrganizationId',
                fieldLabel: '所屬組織',
                editable: false,
                url: urls.SaleInfo.GetOrganizationTree,
                //emptyText: '請選擇所屬組織',
                allowBlank: false
            });

 

 

 

Ext.form.field.ComboBoxView source...


您的分享是我們最大的動力!

-Advertisement-
Play Games
更多相關文章
  • 1、塊級元素 一般用來搭建網站架構、佈局、承載內容……它包括以下這些標簽: address、blockquote、center、dir、div、dl、dt、dd、fieldset、form、h1~h6、hr、isindex、menu、noframes、noscript、ol、p、pre、table、 ...
  • 目前主流瀏覽器的相容性做的都比較好了,本文主要針對IE6,7的不相容問題進行解決。 1.有浮動存在時,計算一定要精確,不要讓內容的寬高超出我們所設置的寬高,IE6下,內容會撐開設置好的高度。 解決方法:給對應的父級加overflow:hidden;但是會有部分被隱藏掉,最好是精確計算寬高再設定 eg ...
  • 前言: 1.HTML5的發展非常迅速,可以說已經是前端開發人員的標配,在電商類型的APP中更是運用廣泛,這個系列的文章是本人自己整理,儘量將開發中不常用到的剔除,將經常使用的拿出來,使需要的朋友能夠真正快速入門,如果有哪些不清楚的地方或者錯誤,歡迎聯繫我 2.更新時間沒有規律,一般會在3天左右更新一 ...
  • 一丶 流 什麼是流? 比如 react 中的單項數據流,Node.js 中的流,或者本文中的 DOM 事件流,都是流的具體體現。專業地講,流是程式輸入或輸出的一個連續的位元組序列;通俗地講,流是有方向的數據。 二丶 事件流 什麼是事件流? 假想一下,現在有一組同心圓,你把手指在最裡面的圓心上,與此同時 ...
  • 1:視頻播放器2:地理定位 我們的支持html5 的瀏覽器給我們提供一個介面(api),可以用來獲取你當前的位置. 主要是通過geolocation(地理位置),對象 ,去訪問硬體,來獲取到經緯度.. 我們獲取到的是一個經緯度。我們調用地圖。我們調用百度地圖.(街景地圖) 3: 拖拽 html5 里 ...
  • 1.prototype和__proto__ 所有對象的__proto__都指向其構造器的prototype,即constructor的原型 2.變數作用域的問題 通常認為在當前作用域中找不到變數值時會到其父作用域中去尋找,這種說法是不准確的,應該是會到創建這個函數的作用域中去找 3.settimeo ...
  • JavaScript一直沒有模塊體系,但是伴隨著ES6的到來,module隨之而來。ES6module提倡一個js文件就是一個模塊的概念,主要包括兩個命令:export和import,用於模塊向外提供介面(export)和引入其他模塊介面(import)。該隨筆分為四部分:1、ES6module概述... ...
  • 下載地址:https://www.sublimetext.com/3 安裝Package Control Zend Coding 插件 參考文章 原文鏈接:http://www.cnsecer.com/460.html 官方文檔:http://docs.emmet.io/customization/ ...
一周排行
    -Advertisement-
    Play Games
  • 移動開發(一):使用.NET MAUI開發第一個安卓APP 對於工作多年的C#程式員來說,近來想嘗試開發一款安卓APP,考慮了很久最終選擇使用.NET MAUI這個微軟官方的框架來嘗試體驗開發安卓APP,畢竟是使用Visual Studio開發工具,使用起來也比較的順手,結合微軟官方的教程進行了安卓 ...
  • 前言 QuestPDF 是一個開源 .NET 庫,用於生成 PDF 文檔。使用了C# Fluent API方式可簡化開發、減少錯誤並提高工作效率。利用它可以輕鬆生成 PDF 報告、發票、導出文件等。 項目介紹 QuestPDF 是一個革命性的開源 .NET 庫,它徹底改變了我們生成 PDF 文檔的方 ...
  • 項目地址 項目後端地址: https://github.com/ZyPLJ/ZYTteeHole 項目前端頁面地址: ZyPLJ/TreeHoleVue (github.com) https://github.com/ZyPLJ/TreeHoleVue 目前項目測試訪問地址: http://tree ...
  • 話不多說,直接開乾 一.下載 1.官方鏈接下載: https://www.microsoft.com/zh-cn/sql-server/sql-server-downloads 2.在下載目錄中找到下麵這個小的安裝包 SQL2022-SSEI-Dev.exe,運行開始下載SQL server; 二. ...
  • 前言 隨著物聯網(IoT)技術的迅猛發展,MQTT(消息隊列遙測傳輸)協議憑藉其輕量級和高效性,已成為眾多物聯網應用的首選通信標準。 MQTTnet 作為一個高性能的 .NET 開源庫,為 .NET 平臺上的 MQTT 客戶端與伺服器開發提供了強大的支持。 本文將全面介紹 MQTTnet 的核心功能 ...
  • Serilog支持多種接收器用於日誌存儲,增強器用於添加屬性,LogContext管理動態屬性,支持多種輸出格式包括純文本、JSON及ExpressionTemplate。還提供了自定義格式化選項,適用於不同需求。 ...
  • 目錄簡介獲取 HTML 文檔解析 HTML 文檔測試參考文章 簡介 動態內容網站使用 JavaScript 腳本動態檢索和渲染數據,爬取信息時需要模擬瀏覽器行為,否則獲取到的源碼基本是空的。 本文使用的爬取步驟如下: 使用 Selenium 獲取渲染後的 HTML 文檔 使用 HtmlAgility ...
  • 1.前言 什麼是熱更新 游戲或者軟體更新時,無需重新下載客戶端進行安裝,而是在應用程式啟動的情況下,在內部進行資源或者代碼更新 Unity目前常用熱更新解決方案 HybridCLR,Xlua,ILRuntime等 Unity目前常用資源管理解決方案 AssetBundles,Addressable, ...
  • 本文章主要是在C# ASP.NET Core Web API框架實現向手機發送驗證碼簡訊功能。這裡我選擇是一個互億無線簡訊驗證碼平臺,其實像阿裡雲,騰訊雲上面也可以。 首先我們先去 互億無線 https://www.ihuyi.com/api/sms.html 去註冊一個賬號 註冊完成賬號後,它會送 ...
  • 通過以下方式可以高效,並保證數據同步的可靠性 1.API設計 使用RESTful設計,確保API端點明確,並使用適當的HTTP方法(如POST用於創建,PUT用於更新)。 設計清晰的請求和響應模型,以確保客戶端能夠理解預期格式。 2.數據驗證 在伺服器端進行嚴格的數據驗證,確保接收到的數據符合預期格 ...