欧美bbbwbbbw肥妇,免费乱码人妻系列日韩,一级黄片

Element-UI控件Tree實現(xiàn)數(shù)據(jù)樹形結構的方法

 更新時間:2024年01月31日 10:56:26   作者:覺醒法師  
這篇文章主要介紹了Element-UI控件Tree實現(xiàn)數(shù)據(jù)樹形結構,本期介紹添加、修改等功能也比較簡單,可以通過element-ui的$prompt彈框控件來實現(xiàn),需要的朋友可以參考下

在前端開發(fā)中,有時會遇到所有菜單數(shù)據(jù)在同一級的情況,后端未對數(shù)據(jù)進行分級處理;但前端渲染需要是樹狀結構的數(shù)據(jù),如何實現(xiàn)數(shù)據(jù)的樹狀化?將數(shù)組中通過父節(jié)點的ID與子節(jié)點的parentId關聯(lián),通過遞歸函數(shù)來實現(xiàn)。

前端框架這里使用element-ui的tree控件來實現(xiàn),對其不了解可以去官網(wǎng)查看文檔。

地址:Element - The world's most popular Vue UI framework

一、創(chuàng)建頁面

這里就不講vue項目的搭建了,基礎不好的,可以去官網(wǎng)查看文檔。

首先在src/pages目錄中,創(chuàng)建element-trees文件夾,再創(chuàng)建index.vue,代碼如下:

<template>
  <div class="tree-container">
  </div>
</template>
<script>
  export default {
    data(){
      return {}
    },
    created() {},
    methods: {}
  }
</script>
<style lang="scss">
</style>

在類選擇器tree-container容器中,添加相關按鈕和Tree控件,以及給容器添加些樣式進行修飾一下,代碼如下:

<template>
  <div class="tree-container">
    <div class="top-box">
      <el-button type="primary" size="mini" @click="appendNode">添加菜單</el-button>
      <el-button type="info" size="mini" @click="removeSelected">刪除選中項</el-button>
    </div>
    <el-tree
    	ref="tree"
    	:data="treeData"
    	:props="props"
    	accordion
    	show-checkbox
    	node-key="id"
    	default-expand-all
    	:expand-on-click-node="false">
    	<span class="custom-tree-node" slot-scope="{ node, data }">
    		<span>{{ node.label }}</span>
    		<span>
    			<el-button type="text" size="mini" @click="append(data)">添加</el-button>
    			<el-button type="text" size="mini" @click="editor(data)">修改</el-button>
    			<el-button type="text" size="mini"	@click="remove(node, data)">刪除</el-button>
    		</span>
    	</span>
    </el-tree>
  </div>
</template>
<script>
  export default {
    data(){
      return {
        props: {
        	label: 'name',
        	children: 'children'
        },
        treeData: []
      }
    },
    created() {},
    methods: {
      //添加父節(jié)點數(shù)據(jù)
      appendNode(){},
      //添加子項數(shù)據(jù)
      append(data){},
      //編輯數(shù)據(jù)
      editor(data){},
      //移出項目
      remove(node, data){},
      //刪除選中的節(jié)點
      removeSelected(){}
    }
  }
</script>
<style lang="scss">
.tree-container{ width: 360px; font-size: 12px;
  .top-box{ text-align: right; padding-bottom: 20px; }
}
.custom-tree-node {
	flex: 1;
	display: flex;
	align-items: center;
	justify-content: space-between;
	font-size: 14px;
	padding-right: 8px;
}
</style>

此時頁面效果如下:

二、模擬數(shù)據(jù)

在element-trees文件夾中添加data.js文件,用來存儲模擬數(shù)據(jù)。代碼如下:

<template>
  <div class="tree-container">
    <div class="top-box">
      <el-button type="primary" size="mini" @click="appendNode">添加菜單</el-button>
      <el-button type="info" size="mini" @click="removeSelected">刪除選中項</el-button>
    </div>
    <el-tree
    	ref="tree"
    	:data="treeData"
    	:props="props"
    	accordion
    	show-checkbox
    	node-key="id"
    	default-expand-all
    	:expand-on-click-node="false">
    	<span class="custom-tree-node" slot-scope="{ node, data }">
    		<span>{{ node.label }}</span>
    		<span>
    			<el-button type="text" size="mini" @click="append(data)">添加</el-button>
    			<el-button type="text" size="mini" @click="editor(data)">修改</el-button>
    			<el-button type="text" size="mini"	@click="remove(node, data)">刪除</el-button>
    		</span>
    	</span>
    </el-tree>
  </div>
</template>
<script>
  export default {
    data(){
      return {
        props: {
        	label: 'name',
        	children: 'children'
        },
        treeData: []
      }
    },
    created() {},
    methods: {
      //添加父節(jié)點數(shù)據(jù)
      appendNode(){},
      //添加子項數(shù)據(jù)
      append(data){},
      //編輯數(shù)據(jù)
      editor(data){},
      //移出項目
      remove(node, data){},
      //刪除選中的節(jié)點
      removeSelected(){}
    }
  }
</script>
<style lang="scss">
.tree-container{ width: 360px; font-size: 12px;
  .top-box{ text-align: right; padding-bottom: 20px; }
}
.custom-tree-node {
	flex: 1;
	display: flex;
	align-items: center;
	justify-content: space-between;
	font-size: 14px;
	padding-right: 8px;
}
</style>

文件創(chuàng)建好后,我們將其引入到index.vue文件中,代碼如下:

<script>
  import DataList from './data'
  export default {
    data(){
      return {
        props: {
        	label: 'name',
        	children: 'children'
        },
        treeData: []
      }
    },
    created() {
      this.loadNode();
    },
    methods: {
	  //加載節(jié)點數(shù)據(jù)
      loadNode(){
        this.treeData = DataList.map(item => item);
      },
      //略...
    }
  }
</script>

如上代碼,通過loadNode()函數(shù)執(zhí)行后,模擬數(shù)據(jù)則已與Tree控件進行了雙方綁定。但是頁面中顯示的數(shù)據(jù),全都顯示在同級位置。數(shù)據(jù)的歸類分級,我們在后面再講解。頁面效果如下:

三、遞歸函數(shù)

此時離成功緊一步之遙了,我們先來分析下data.js中的數(shù)據(jù)結構。如下圖:

如何將數(shù)據(jù)歸類分級呢,可以通過循環(huán)判斷,每級的id與數(shù)據(jù)中的pid進行關聯(lián);比如第一條數(shù)據(jù)id為1,將數(shù)組進行遍歷查詢判斷pid有為1的數(shù)據(jù),則表示其有子項數(shù)據(jù),將其子項數(shù)據(jù)添加到children數(shù)組中。pid為0的為一級數(shù)據(jù),所以當定義遞歸函數(shù)時,傳入的pid默認為0。

這里層次有可能是1層,2層,3層...... n層,要一層一層循環(huán)判斷,就太麻煩了;所以這里我們要使用遞歸方法,來實現(xiàn)n層級的數(shù)據(jù)處理。

在src下創(chuàng)建utils/utils.js文件,在utils.js中定義遞歸函數(shù),代碼如下:

/**
  * 遞歸函數(shù)
  * @param arr 數(shù)組
  * @param pid 父ID,不傳默認為0
  */
export const DGFilterTrees = (arr, pid = 0) => {
	let newArr = [];
	//循環(huán)數(shù)組
	arr.forEach((item, i) => {
		//判斷pid與遍歷元素中pid相同的數(shù)據(jù),相等的為同級數(shù)據(jù),追加到該級數(shù)組中
		if(item.pid == pid){
			newArr.push(item);
			/**
             * 判斷該item元素是否有子項
             * 當數(shù)組中有元素的pid與item.id相等,some函數(shù)會返回true,表示該元素有子項數(shù)據(jù)
             */
			if(arr.some(ele => ele.pid == item.id)){
				//此時,通過重新調用本函數(shù)進行遞歸處理
				item['children'] = DGFilterTrees(arr, item.id);
			}
		}
	});
	return newArr;
}

注:遞歸算法是一種直接或者間接調用自身函數(shù)或者方法的算法。

遞歸算法的特點:

  • 遞歸就是方法里調用自身。
  • 在使用遞增歸策略時,必須有一個明確的遞歸結束條件,稱為遞歸出口。
  • 遞歸算法解題通常顯得很簡潔,但遞歸算法解題的運行效率較低。所以一般不提倡用遞歸算法設計程序。
  • 在遞歸調用的過程當中系統(tǒng)為每一層的返回點、局部量等開辟了棧來存儲。遞歸次數(shù)過多容易造成棧溢出等,所以一般不提倡用遞歸算法設計程序。

所以在使用遞歸算法時,一定要注意判斷條件能正常退出,否則無法靠自身的控制終止的循環(huán),會導致棧溢出,程序卡死等問題。

將DGFilterTrees()遞歸函數(shù)引入到index.vue中,將模擬數(shù)據(jù)重組,代碼如下:

<script>
  import DataList from './data'
  import DGFilterTrees from '@/utils/utils.js'
  export default {
    data(){
      return {
        props: {
        	label: 'name',
        	children: 'children'
        },
        treeData: []
      }
    },
    created() {
      this.loadNode();
    },
    methods: {
	  //加載節(jié)點數(shù)據(jù)
      loadNode(){
        this.treeData = DGFilterTrees(DataList);
      },
      //略...
    }
  }
</script>

此時,所有數(shù)據(jù)都進行了歸類分級顯示,頁面的效果圖如下:

四、選中后顯示按鈕

如上圖,我們會發(fā)現(xiàn)每項后面的添加、修改、刪除全顯示出來,會顯示擁擠并不方便操作。這里我們將節(jié)點輸出后會發(fā)現(xiàn),可以使用isCurrent來判斷,只顯示選中當前項的按鈕。

在span標簽上添加v-if,判斷node.isCurrent是否為true,代碼如下:

<el-tree
	ref="tree"
	:data="treeData"
	:props="props"
	accordion
	show-checkbox
	node-key="id"
	default-expand-all
	:expand-on-click-node="false">
	<span class="custom-tree-node" slot-scope="{ node, data }">
		<span>{{ node.label }}</span>
		<span v-if="node.isCurrent">
			<el-button type="text" size="mini" @click="append(data)">添加</el-button>
			<el-button type="text" size="mini" @click="editor(data)">修改</el-button>
			<el-button type="text" size="mini"	@click="remove(node, data)">刪除</el-button>
		</span>
	</span>
</el-tree>

如下圖,點擊選擇”后端基礎SQL“后,會顯示后面的操作按鈕。

五、刪除元素

這里我們是使用本地定義模擬數(shù)據(jù)進行操作的,所以沒有真實項目中是通過接口進行數(shù)據(jù)的增刪改查部分。代碼如下:

//移出項目
remove(node, data){
	if(Array.isArray(data['children'])&&data.children.length>0){
		this.$message.error('當前數(shù)據(jù)有子項,無法刪除~');
		return;
	}
	//查詢數(shù)據(jù)位置索引
	let index = DataList.findIndex(item => item.id==data.id);
	//刪除指定位置數(shù)據(jù)
	DataList.splice(index, 1);
	//重新加載數(shù)據(jù)
	this.loadNode();
}

效果如下圖:

六、刪除多選項

這里需要使用Tree控件的getCheckedNodes()函數(shù),來獲取被選中項;然后通過循環(huán),刪除每條被選中元素,代碼如下:

//刪除選中的節(jié)點
removeSelected(){
	//獲取選中的數(shù)據(jù)
	let checks = this.$refs.tree.getCheckedNodes();
	//循環(huán)刪除選中
	checks.forEach(item => {
		DataList.splice(DataList.findIndex(sub => sub.id == item.id), 1);
	});
	//重新加載數(shù)據(jù)
	this.loadNode();
}

效果如下圖:

本期就先介紹到這,添加、修改等功能也比較簡單,可以通過element-ui的$prompt彈框控件來實現(xiàn)。

到此這篇關于Element-UI控件Tree實現(xiàn)數(shù)據(jù)樹形結構的文章就介紹到這了,更多相關Element-UI數(shù)據(jù)樹形結構內容請搜索腳本之家以前的文章或繼續(xù)瀏覽下面的相關文章希望大家以后多多支持腳本之家!

相關文章

  • 深入理解vue-loader如何使用

    深入理解vue-loader如何使用

    本篇文章主要介紹了vue-loader,vue-loader是webpack下loader插件 可以把.vue文件輸出成組件,有興趣的可以了解一下
    2017-06-06
  • vue使用中的內存泄漏【推薦】

    vue使用中的內存泄漏【推薦】

    內存泄露是指new了一塊內存,但無法被釋放或者被垃圾回收。這篇文章主要介紹了vue使用中的內存泄漏,需要的朋友可以參考下
    2018-07-07
  • 在vue中v-for循環(huán)遍歷圖片不顯示錯誤的解決方案

    在vue中v-for循環(huán)遍歷圖片不顯示錯誤的解決方案

    這篇文章主要介紹了在vue中v-for循環(huán)遍歷圖片不顯示錯誤的解決方案,具有很好的參考價值,希望對大家有所幫助。如有錯誤或未考慮完全的地方,望不吝賜教
    2022-01-01
  • vue如何截取字符串

    vue如何截取字符串

    這篇文章主要介紹了vue如何截取字符串,文中通過示例代碼介紹的非常詳細,對大家的學習或者工作具有一定的參考學習價值,需要的朋友們下面隨著小編來一起學習學習吧
    2019-05-05
  • 總結4個方面優(yōu)化Vue項目

    總結4個方面優(yōu)化Vue項目

    在本篇文章里我們給大家整理了一篇關于優(yōu)化VUE項目的四個總要點,對此有需要的朋友們學習下天。
    2019-02-02
  • vue數(shù)據(jù)監(jiān)聽解析Object.defineProperty與Proxy區(qū)別

    vue數(shù)據(jù)監(jiān)聽解析Object.defineProperty與Proxy區(qū)別

    這篇文章主要為大家介紹了vue數(shù)據(jù)監(jiān)聽解析Object.defineProperty Proxy源碼示例,有需要的朋友可以借鑒參考下,希望能夠有所幫助,祝大家多多進步,早日升職加薪
    2023-03-03
  • vue中axios的get請求和post請求的傳參方式、攔截器示例代碼

    vue中axios的get請求和post請求的傳參方式、攔截器示例代碼

    Post是向服務器提交數(shù)據(jù)的一種請求,get是向服務器發(fā)索取數(shù)據(jù)的一種請求,post在真正接受數(shù)據(jù)之前會先將請求頭發(fā)送給服務器進行確認,然后才真正發(fā)送數(shù)據(jù),本文給大家介紹vue中axios的get請求和post請求的傳參方式、攔截器示例代碼,感興趣的朋友一起看看吧
    2023-10-10
  • Vue發(fā)布項目實例講解

    Vue發(fā)布項目實例講解

    在本篇文章里小編給各位分享的是關于Vue發(fā)布項目的實例內容以及知識點講解,需要的朋友們參考下。
    2019-07-07
  • 關于Vue Router中路由守衛(wèi)的應用及在全局導航守衛(wèi)中檢查元字段的方法

    關于Vue Router中路由守衛(wèi)的應用及在全局導航守衛(wèi)中檢查元字段的方法

    這篇文章主要介紹了關于Vue Router中路由守衛(wèi)的應用及在全局導航守衛(wèi)中檢查元字段的方法,實現(xiàn)方法有兩種,本文通過實例代碼對每種方法介紹的很詳細,需要的朋友參考下
    2018-12-12
  • vue2實現(xiàn)封裝動態(tài)表單組件

    vue2實現(xiàn)封裝動態(tài)表單組件

    這篇文章主要介紹了vue2實現(xiàn)封裝動態(tài)表單組件,文章圍繞主題展開詳細的內容介紹,具有一定的參考價值,需要的小伙伴可以參考一下
    2022-08-08

最新評論