element-ui樹形控件:地址vue
在原文檔中有個案例是有新增和刪除功能,可是後來發現其修改的數據並不能直接影響到樹形數據,因此採用了 render-content
的API從新寫了個組件。
寫個開發的步驟,因此文章比較長emmmnode
elementUI ^2.2.0提供了一個slot的自定義節點方法,相關代碼已在github
更新,原理同樣。git
大體效果如圖:
github
在網上覆制了個省市的list,有兩個屬性是新增的ajax
isEdit
:控制編輯狀態 maxexpandId
:爲現下id的最大值export default{ maxexpandId: 95, treelist: [{ id: 1, name: "北京市", ProSort: 1, remark: "直轄市", pid: '', isEdit: false, children: [{ id: 35, name: "朝陽區", pid: 1, remark: '', isEdit: false, children: [] }] }{...}] }
我們一步步來,先寫個餓了麼的組件element-ui
<template> <el-tree ref="expandMenuList" class="expand-tree" v-if="isLoadingTree" :data="setTree" node-key="id" highlight-current :props="defaultProps" :expand-on-click-node="false" :render-content="renderContent" :default-expanded-keys="defaultExpandKeys"></el-tree> </template> <!-- * highlight-current :爲了點擊時節點高亮 * expand-on-click-node : 只能箭頭控制樹形的展開收縮 * render-content : 節點渲染方式 * default-expanded-keys :默認展開節點 -->
同時引入API和節點渲染的組件api
import TreeRender from '@/components/tree_render' import api from '@/resource/api'
而後搭建好基礎字體
data(){ return{ maxexpandId: api.maxexpandId,//新增節點開始id non_maxexpandId: api.maxexpandId,//新增節點開始id(不更改) isLoadingTree: false,//是否加載節點樹 setTree: api.treelist,//節點樹數據 defaultProps: { children: 'children', label: 'name' }, defaultExpandKeys: [],//默認展開節點列表 } },
添加個渲染的methodui
methods: { renderContent(h,{node,data,store}){ let that = this;//指向vue return h(TreeRender,{ props: { DATA: data,//節點數據 NODE: node,//節點內容 STORE: store,//完整樹形內容 }, on: {//綁定方法 nodeAdd: ((s,d,n) => that.handleAdd(s,d,n)), nodeEdit: ((s,d,n) => that.handleEdit(s,d,n)), nodeDel: ((s,d,n) => that.handleDelete(s,d,n)) } }); }, handleAdd(s,d,n){//增長節點 console.log(s,d,n) }, handleEdit(s,d,n){//編輯節點 console.log(s,d,n) }, handleDelete(s,d,n){//刪除節點 console.log(s,d,n) } }
渲染組件:this
<template> <span class="tree-expand"> <span class="tree-label"> <span>{{DATA.name}}</span> </span> <span class="tree-btn"> <i class="el-icon-plus" @click.stop="nodeAdd(STORE,DATA,NODE)"></i> <i class="el-icon-edit" @click.stop="nodeEdit(STORE,DATA,NODE)"></i> <i class="el-icon-delete" @click.stop="nodeDel(STORE,DATA,NODE)"></i> </span> </span> </template>
添加好幾個按鈕(element-ui自帶icon:地址)對應的方法:
export default{ props: ['NODE', 'DATA', 'STORE'], methods: { nodeAdd(s,d,n){//新增 this.$emit('nodeAdd',s,d,n) }, nodeEdit(s,d,n){//編輯 this.$emit('nodeEdit',s,d,n) }, nodeDel(s,d,n){//刪除 this.$emit('nodeDel',s,d,n) } } }
咱們用isEdit
來切換input
和span
的顯示狀態,首先加個input:
<!-- tree_render component --> <template> <span class="tree-expand"> <span class="tree-label" v-if="DATA.isEdit"> <el-input class="edit" size="mini" :ref="'treeInput'+DATA.id" v-model="DATA.name"></el-input> </span> <template v-else> <span class="tree-label"> <span>{{DATA.name}}</span> </span> <span class="tree-btn" v-show="!DATA.isEdit"> <i class="el-icon-plus" @click.stop="nodeAdd(STORE,DATA,NODE)"></i> <i class="el-icon-edit" @click.stop="nodeEdit(STORE,DATA,NODE)"></i> <i class="el-icon-delete" @click.stop="nodeDel(STORE,DATA,NODE)"></i> </span> </template> </span> </template>
編輯的時候按鈕同時消失,那麼何時編輯完成呢?
blur
=》編輯時自動聚焦-focus
當以上三點發生一項,節點對應的data都要isEdit = false;
enter鍵
<!-- tree_render component --> <el-input @keyup.enter.native="nodeEditPass(STORE,DATA,NODE)"></el-input>
添加方法:
//tree_render component methods: { nodeEditPass(s,d,n){ d.isEdit = false; } }
focus or blur
<!-- tree_render component --> <el-input @blur="nodeEditPass(STORE,DATA,NODE)"></el-input>
後來發現第一次編輯時能讓input
聚焦,點擊第二個input
就不起做用了,加了autofocus
屬性也一樣如此。因此咱們要在點擊編輯icon
的時候,用原生的input autofocus
。
修改方法:
//tree_render component nodeEdit(s,d,n){//編輯 d.isEdit = true; this.$nextTick(() => { this.$refs['treeInput'+d.id].$refs.input.focus() }) this.$emit('nodeEdit',s,d,n) }
當前節點點擊
採用el-tree
已有的API——node-click
<!-- el-tree component --> <el-tree @node-click="handleNodeClick"></el-tree>
添加methods:
//el-tree component methods: { handleNodeClick(d,n,s){//點擊節點 d.isEdit = false;//放棄編輯狀態 } }
問題來了,若是在編輯狀態下點擊此節點也一樣會影響input,這就沒法進入編輯,因此要阻止input事件冒泡:
<!-- tree_render component --> <el-input @click.stop.native="nodeEditFocus"></el-input>
添加methods:
//tree_render component methods: { nodeEditFocus(){} }
v-show
代替v-if
這裏有個新的問題,當用戶常常編輯修改,v-if
模板的開銷更高,因此改用v-show。然後者不支持template模板,因此要適當調整一下位置:
<template> <span class="tree-expand"> <span class="tree-label" v-show="DATA.isEdit"> <el-input class="edit" size="mini" autofocus v-model="DATA.name" :ref="'treeInput'+DATA.id" @click.stop.native="nodeEditFocus" @blur="nodeEditPass(STORE,DATA,NODE)" @keyup.enter.native="nodeEditPass(STORE,DATA,NODE)"></el-input> </span> <span v-show="!DATA.isEdit"> <span>{{DATA.name}}</span> </span> <span class="tree-btn" v-show="!DATA.isEdit"> <i class="el-icon-plus" @click.stop="nodeAdd(STORE,DATA,NODE)"></i> <i class="el-icon-edit" @click.stop="nodeEdit(STORE,DATA,NODE)"></i> <i class="el-icon-delete" @click.stop="nodeDel(STORE,DATA,NODE)"></i> </span> </span> </template>
新增節點 =》添加一條數據
//el-tree component handleAdd(s,d,n){//增長節點 console.log(s,d,n) if(n.level >=6){ this.$message.error("最多隻支持五級!") return false; } //添加數據 d.children.push({ id: ++this.maxexpandId, name: '新增節點', pid: d.id, isEdit: false, children: [] }); //展開節點 if(!n.expanded){ n.expanded = true; } }
新增節點字體加粗 =》給節點添加一個class =》 如何判斷是否新增?
咱們有一個參數maxexpandId
給tree_render
添加一個prop
:
//el-tree component renderContent(h,{node,data,store}){//加載節點 let that = this; return h(TreeRender,{ props: { ... maxexpandId: that.non_maxexpandId }, on: {...} }); }
根據id判斷:
//tree_render component props: ['NODE', 'DATA', 'STORE', 'maxexpandId']
<!-- tree_render component --> <span v-show="!DATA.isEdit" :class="[DATA.id > maxexpandId ? 'tree-new tree-label' : 'tree-label']" :ref="'treeLabel'+DATA.id"> <span>{{DATA.name}}</span> </span>
.tree-expand .tree-label.tree-new{ font-weight:600; }
跟新增同義:刪除節點 =》刪除一條數據
handleDelete(s,d,n){//刪除節點 console.log(s,d,n) let that = this; //有子級不刪除 if(d.children && d.children.length !== 0){ this.$message.error("此節點有子級,不可刪除!") return false; }else{ //刪除操做 let delNode = () => { let list = n.parent.data.children || n.parent.data, //節點同級數據,頂級節點時無children _index = 99999;//要刪除的index list.map((c,i) => { if(d.id == c.id){ _index = i; } }) let k = list.splice(_index,1); //console.log(_index,k) this.$message.success("刪除成功!") } let isDel = () => { that.$confirm("是否刪除此節點?","提示",{ confirmButtonText: "確認", cancelButtonText: "取消", type: "warning" }).then(() => { delNode()//此處可經過ajax作刪除操做 }).catch(() => { return false; }) } //新增節點直接刪除,不然要經過請求數據刪除 d.id > this.non_maxexpandId ? delNode() : isDel() } }
還有一些特別的需求,例如:
點擊高亮的時候顯示icon
.expand-tree .is-current>.el-tree-node__content .tree-btn, .expand-tree .el-tree-node__content:hover .tree-btn{ display: inline-block; }
添加頂級節點
添加按鈕:
<!-- el-tree component --> <el-button @click="handleAddTop">添加頂級節點</el-button>
添加methods:
//el-tree component methods: { handleAddTop(){ this.setTree.push({ id: ++this.maxexpandId, name: '新增節點', pid: '', isEdit: false, children: [] }) } }
默認展開樹形第一級
//el-tree component mounted(){ this.initExpand() }, methods: { initExpand(){ //isLoadingTree用意也是在此 this.setTree.map((a) => { this.defaultExpandKeys.push(a.id) }); this.isLoadingTree = true; }, }
還有些具體的樣式都放在github了
若有錯漏,歡迎指正╰( ◕ ▽ ◕ )╯