Element UI框架中巧用樹選擇器

程式媛兔子發表於2018-12-12

先看效果圖

樹選擇器

基本功能

在Element UI框架中有選擇器和樹形控制元件,但是沒有樹形選擇器,也就是圖上的這種方式的選擇器,所以只能自定義選擇器的slot。這裡介紹的是多選情況,如果是單選則去掉核取方塊,修改一部分的處理即可。

html部分的程式碼:

<el-select
    v-model="dataArr"
    :multiple="multiple"
    filterable
    :placeholder="placeholder"
    :disabled="disabled"
    :collapse-tags="multiple"
    @remove-tag="handleTagChange"
    @visible-change="handleOptionHidden"
    class="hi-input">
    <el-option value="0"
        class="hidden">
    </el-option>
    <!--el-tree繫結的陣列中children裡的key值不能是0-->
    <el-tree
        ref="tree"
        :data="options"
        node-key="key"
        show-checkbox
        :default-checked-keys="selectedData"
        @check="handleCheckChange"
        :props="defaultProps">
    </el-tree>
</el-select>
複製程式碼

樹形控制元件到選擇器的繫結

在el-tree中繫結的值是已選擇的key值組成的陣列,check繫結的事件函式是為了:

  • 得到現在樹選擇器上選中的值
  • 過濾undefined、null的值(是為了容錯處理) 具體程式碼如下:
handleCheckChange: function() {
    this.selectedData = this.$refs.tree.getCheckedKeys().filter(_ => _);
}
複製程式碼

因為選擇器是有label值和key值區分的,所以,每當在el-tree中選中值key值變化時,選擇器上繫結的值label值也應該隨之變化,所以在watch中監聽key值,目的是在el-tree繫結的data中找到當前key值對應的label值 具體程式碼如下:

watch: {
    selectedData: function(newValue) {
        this.$nextTick(() => { this.dataArr = this.handleDataTransform(newValue, 'key', 'label'); });
    },
},
methods: {
    getNameById(array, value, id, name, multi) {
        let arr = array || [];
        let flag;
        let result = arr.filter(item => {
            return item[id] + '' === value + '';
        });
        if (multi) {
            flag = result.map(item => {
                return item[name];
            });
        } else {
            let obj = result[0];
            flag = name ? obj && obj[name] : obj;
        }
        return flag;
    },
    handleDataTransform: function(source, key, value) {
        return this.options.map(_ => {
            let arr = source.map(item => this.getNameById(
                _.children,
                item,
                key,
                value
            )).filter(item => item);
            return arr;
        }).reduce((acc, cur) => {
            return acc.concat(cur);
        }, []);
    }
}
複製程式碼

到這裡為止,已經完成了樹形控制元件到選擇器的單向繫結

選擇器到樹形控制元件的繫結

remove-tag事件

現在處理選擇器的值發生改變時,樹形控制元件也變化。因為此時是多選,所以要在remove-tag事件中處理,具體程式碼如下:

handleTagChange: function() {
    //  handleDataTransform已經在之前定義過
    this.selectedData = this.handleDataTransform(this.dataArr, 'label', 'key');
    this.$refs.tree.setCheckedKeys(this.selectedData);
},
複製程式碼

change事件

通過測試發現,當在鍵盤上點選delete時,也會刪除選擇器內選中的元素,我嘗試過繫結@keyup.delete事件,但是繫結不成功,如果使用了.native修飾符可以監聽到事件了,但是阻止了原生刪除事件,所以也不可取。經過測試發現繫結change事件就可以處理鍵盤的delete事件,繫結的函式和handleTagChange一樣。

優化

此時,這個樹形選擇器已經完成了~??,但是,我們還可以進一步優化,比如

選擇項不變,不重複發請求

如果選擇後的內容與選擇前的內容一樣,不再發生請求的處理。 在選擇器中繫結的visible-change事件可以處理,思想是:

  1. item值為true,即展開下拉框時,把此時的值儲存下來,注意⚠️:這時候儲存下來的值必須放在一個全域性變數中,函式內的變數會在每次進入這個函式時初始化,所以在下拉框收起再進來這個函式時,之前儲存的值已經沒有了。
  2. item值為false,即收起下拉框時,判斷之前儲存下的變數值和當前變數值是否相等,如果不相等才觸發資料的更新。 具體程式碼如下:
handleOptionHidden: function(item) {
    // 處理選中內容沒變的情況
    if (item) {
        this.selectedItem = [...this.selectedData];
    } else {
        //  this.$util.isEqual()是判斷兩個陣列是否相等函式,網上很多,請自行谷歌
        if (!this.$util.isEqual(this.selectedItem, this.selectedData)) {
            this.handleUpdate(this.selectedData);
        }
    }
}
複製程式碼

選擇器的搜尋功能

當資料量大的時候,選擇器一般都會有搜尋的需求,在選擇器上封裝過的搜尋功能無法滿足需求,因為樹形控制元件本身有搜尋的函式,所以在選擇器上自定義filter-mothod事件,呼叫樹形控制元件的搜尋事件即可。

handleSelectFilter: function(val) {
    this.$refs.tree.filter(val);
},
複製程式碼

在樹形控制元件上繫結filter-node-method自定義函式,支援忽略大小寫搜尋

handleTreeFilter: function(value, data) {
    if (!value) return true;
    return data.label.toUpperCase().indexOf(value.toUpperCase()) !== -1;
},
複製程式碼

以上就是全部內容了,如果哪裡寫的差點意思,請告訴我哈~記得點贊?,thanks~

相關文章