Element UI框架中巧用樹選擇器

介紹
在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.$util.getNameById( _.children, item, key, value )).filter(item => item); return arr; }).reduce((acc, cur) => { return acc.concat(cur); }, []); } } 複製程式碼
到這裡為止,已經完成了樹形控制元件到選擇器的單向繫結,現在處理選擇器的值發生改變時,樹形控制元件也變化。因為此時是多選,所以要在remove-tag事件中處理,具體程式碼如下:
handleTagChange: function() { //handleDataTransform已經在之前定義過 this.selectedData = this.handleDataTransform(this.dataArr, 'label', 'key'); this.$refs.tree.setCheckedKeys(this.selectedData); }, 複製程式碼
此時,這個樹形選擇器已經完成了~:clap::clap:,但是,我們還可以進一步優化,比如:如果選擇後的內容與選擇前的內容一樣,不再發生請求的處理。 在選擇器中繫結的visible-change事件可以處理,思想是:
- item值為true,即展開下拉框時,把此時的值儲存下來,注意:warning::這時候儲存下來的值必須放在一個全域性變數中,函式內的變數會在每次進入這個函式時初始化,所以在下拉框收起再進來這個函式時,之前儲存的值已經沒有了。
- 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); } } }, 複製程式碼
以上就是全部內容了,如果哪裡寫的差點意思,請告訴我哈~記得點贊:+1:,thanks~