VUE餓了么樹形控件添加增刪改功能的示例代碼
本文介紹了VUE餓了么樹形控件添加增刪改功能的示例代碼,分享給大家,具體如下:
element-ui樹形控件:地址
在原文檔中有個案例是有新增和刪除功能,但是后來發(fā)現(xiàn)其修改的數(shù)據(jù)并不能直接影響到樹形數(shù)據(jù),所以采用了 render-content 的API重新寫了個組件。
寫個開發(fā)的步驟,所以文章比較長emmm
大致效果如圖:

1.省市API
在網(wǎng)上復(fù)制了個省市的list,有兩個屬性是新增的
- isEdit :控制編輯狀態(tài)
- maxexpandId :為現(xiàn)下id的最大值
export default{
maxexpandId: 95,
treelist: [{
id: 1,
name: "北京市",
ProSort: 1,
remark: "直轄市",
pid: '',
isEdit: false,
children: [{
id: 35,
name: "朝陽區(qū)",
pid: 1,
remark: '',
isEdit: false,
children: []
}]
}{...}]
}
2.el-tree Component基本
咱們一步步來,先寫個餓了么的組件
<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 :為了點擊時節(jié)點高亮
* expand-on-click-node : 只能箭頭控制樹形的展開收縮
* render-content : 節(jié)點渲染方式
* default-expanded-keys :默認展開節(jié)點
-->
同時引入API和節(jié)點渲染的組件
import TreeRender from '@/components/tree_render' import api from '@/resource/api'
然后搭建好基礎(chǔ)
data(){
return{
maxexpandId: api.maxexpandId,//新增節(jié)點開始id
non_maxexpandId: api.maxexpandId,//新增節(jié)點開始id(不更改)
isLoadingTree: false,//是否加載節(jié)點樹
setTree: api.treelist,//節(jié)點樹數(shù)據(jù)
defaultProps: {
children: 'children',
label: 'name'
},
defaultExpandKeys: [],//默認展開節(jié)點列表
}
},
添加個渲染的method
methods: {
renderContent(h,{node,data,store}){
let that = this;//指向vue
return h(TreeRender,{
props: {
DATA: data,//節(jié)點數(shù)據(jù)
NODE: node,//節(jié)點內(nèi)容
STORE: store,//完整樹形內(nèi)容
},
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){//增加節(jié)點
console.log(s,d,n)
},
handleEdit(s,d,n){//編輯節(jié)點
console.log(s,d,n)
},
handleDelete(s,d,n){//刪除節(jié)點
console.log(s,d,n)
}
}
3.tree_render Component基本
渲染組件:
<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:地址)對應(yīng)的方法:
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)
}
}
}
4.改
我們用isEdit來切換input和span的顯示狀態(tài),首先加個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>
編輯的時候按鈕同時消失,那么什么時候編輯完成呢?
- 編輯完按enter鍵=》監(jiān)聽input的enter輸入
- 點擊其他節(jié)點=》input失焦-blur=》編輯時自動聚焦-focus
- 點擊當(dāng)前節(jié)點范圍
當(dāng)以上三點發(fā)生一項,節(jié)點對應(yīng)的data都要isEdit = false;
1、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;
}
}
2、focus or blur
<!-- tree_render component --> <el-input @blur="nodeEditPass(STORE,DATA,NODE)"></el-input>
后來發(fā)現(xiàn)第一次編輯時能讓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)
}
3、當(dāng)前節(jié)點點擊
采用el-tree已有的API——node-click
<!-- el-tree component --> <el-tree @node-click="handleNodeClick"></el-tree>
添加methods:
//el-tree component
methods: {
handleNodeClick(d,n,s){//點擊節(jié)點
d.isEdit = false;//放棄編輯狀態(tài)
}
}
問題來了,如果在編輯狀態(tài)下點擊此節(jié)點也同樣會影響input,這就無法進入編輯,所以要阻止input事件冒泡:
<!-- tree_render component --> <el-input @click.stop.native="nodeEditFocus"></el-input>
添加methods:
//tree_render component
methods: {
nodeEditFocus(){}
}
4、v-show代替v-if
這里有個新的問題,當(dāng)用戶經(jīng)常編輯修改,v-if模板的開銷更高,所以改用v-show。而后者不支持template模板,所以要適當(dāng)調(diào)整一下位置:
<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>
5.增
新增節(jié)點 =》添加一條數(shù)據(jù)
- 新增的同時展開父節(jié)點
- 是否考慮無限新增
//el-tree component
handleAdd(s,d,n){//增加節(jié)點
console.log(s,d,n)
if(n.level >=6){
this.$message.error("最多只支持五級!")
return false;
}
//添加數(shù)據(jù)
d.children.push({
id: ++this.maxexpandId,
name: '新增節(jié)點',
pid: d.id,
isEdit: false,
children: []
});
//展開節(jié)點
if(!n.expanded){
n.expanded = true;
}
}
新增節(jié)點字體加粗 =》給節(jié)點添加一個class =》 如何判斷是否新增?
我們有一個參數(shù)maxexpandId
給tree_render添加一個prop:
//el-tree component
renderContent(h,{node,data,store}){//加載節(jié)點
let that = this;
return h(TreeRender,{
props: {
...
maxexpandId: that.non_maxexpandId
},
on: {...}
});
}
根據(jù)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;
}
6.刪
跟新增同義:刪除節(jié)點 =》刪除一條數(shù)據(jù)
- 新增節(jié)點直接刪除
- 已有節(jié)點需提示再刪除
- 已有子級節(jié)點不能刪除
handleDelete(s,d,n){//刪除節(jié)點
console.log(s,d,n)
let that = this;
//有子級不刪除
if(d.children && d.children.length !== 0){
this.$message.error("此節(jié)點有子級,不可刪除!")
return false;
}else{
//刪除操作
let delNode = () => {
let list = n.parent.data.children || n.parent.data,
//節(jié)點同級數(shù)據(jù),頂級節(jié)點時無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("是否刪除此節(jié)點?","提示",{
confirmButtonText: "確認",
cancelButtonText: "取消",
type: "warning"
}).then(() => {
delNode()//此處可通過ajax做刪除操作
}).catch(() => {
return false;
})
}
//新增節(jié)點直接刪除,否則要通過請求數(shù)據(jù)刪除
d.id > this.non_maxexpandId ? delNode() : isDel()
}
}
7.拓展
還有一些特別的需求,例如:
1、點擊高亮的時候顯示icon
.expand-tree .is-current>.el-tree-node__content .tree-btn,
.expand-tree .el-tree-node__content:hover .tree-btn{
display: inline-block;
}
2、添加頂級節(jié)點
添加按鈕:
<!-- el-tree component --> <el-button @click="handleAddTop">添加頂級節(jié)點</el-button>
添加methods:
//el-tree component
methods: {
handleAddTop(){
this.setTree.push({
id: ++this.maxexpandId,
name: '新增節(jié)點',
pid: '',
isEdit: false,
children: []
})
}
}
3、默認展開樹形第一級
//el-tree component
mounted(){
this.initExpand()
},
methods: {
initExpand(){
//isLoadingTree用意也是在此
this.setTree.map((a) => {
this.defaultExpandKeys.push(a.id)
});
this.isLoadingTree = true;
},
}
8.github
還有些具體的樣式都放在github了
以上就是本文的全部內(nèi)容,希望對大家的學(xué)習(xí)有所幫助,也希望大家多多支持腳本之家。
- 不依任何賴第三方,單純用vue實現(xiàn)Tree 樹形控件的案例
- Vue+Element UI 樹形控件整合下拉功能菜單(tree + dropdown +input)
- 詳解vue-element Tree樹形控件填坑路
- vue elementUI tree樹形控件獲取父節(jié)點ID的實例
- vue.js element-ui tree樹形控件改iview的方法
- 解決ant design vue中樹形控件defaultExpandAll設(shè)置無效的問題
- vue遞歸組件實戰(zhàn)之簡單樹形控件實例代碼
- vue用遞歸組件寫樹形控件的實例代碼
- vue+iview使用樹形控件的具體使用
- vue樹形控件tree的使用方法
相關(guān)文章
vue3 ElementUI 日期禁選當(dāng)日前當(dāng)日后三天后的實現(xiàn)代碼
這篇文章主要介紹了vue3 ElementUI 日期禁選當(dāng)日前當(dāng)日后三天后的實現(xiàn)代碼,本文通過示例代碼給大家介紹的非常詳細,感興趣的朋友跟隨小編一起看看吧2024-05-05
詳解Vue CLI3 多頁應(yīng)用實踐和源碼設(shè)計
這篇文章主要介紹了詳解Vue CLI3 多頁應(yīng)用實踐和源碼設(shè)計,小編覺得挺不錯的,現(xiàn)在分享給大家,也給大家做個參考。一起跟隨小編過來看看吧2018-08-08

