最近公司在做一個基於vue開發的項目,項目需要開發一個展示組織架構的樹組件,在網上搜了半天,沒有找到合適的,下面小編給大家分享一個基於Vue製作組織架構樹組件,需要的朋友參考下吧
由於公司業務需求,需要開發一個展示組織架構的樹組件(公司的專案是基於Vue)。在GitHub上找了半天,這類組件不多,也沒有符合業務需求的組件,所以決定自己造輪子吧!
分析
#既然是樹,那麼每個節點都應該是相同的元件
#節點下面套節點,所以節點元件應該是遞迴元件
那麼,問題就來了。遞歸組件怎麼寫?
遞迴元件
Vue官方文件是這樣說的:
元件在它的範本內可以遞歸地調用自己。不過,只有當它有name 選項時才可以這麼做
接下來,我們來寫一個樹節點遞歸元件:
<template> <p class="org-tree-node"> <p class="org-tree-node-label">{{data.label}}</p> <p class="org-tree-node-children" v-if="data.children"> <org-tree-node v-for="node in data.children" :data="node" :key="data.id"></org-tree-node> </p> </p> </template> <script> export default { name: 'OrgTreeNode', props: { data: Object } } </script> <style> /* ... */ </style>
然後渲染這個這個元件,效果如下
至此,一個簡單的組織架構樹元件就完成了。
然而,事情還遠遠沒有結束。 。 。
需求說:節點的label要支援定制,樹要支援水平展示!
因此,我們對遞歸組件作如下修改:
<template> <p class="org-tree-node"> <p class="org-tree-node-label"> <slot>{{data.label}}</slot> </p> <p class="org-tree-node-children" v-if="data.children"> <org-tree-node v-for="node in data.children" :data="node" :key="data.id"></org-tree-node> </p> </p> </template> <script> export default { name: 'OrgTreeNode', props: { data: Object } } </script> <style> /* ... */ </style>
我們使用slot插槽來支援label可定制,但是問題又來了:我們發現只有第一層級的節點label能定制,嵌套的子節點不能有效的傳遞slot插槽。上網查了半天,還是沒有結果,於是再看官方文件。發現有個函數式組件。由於之前使用過 element-ui 的 tree 元件,受到啟發,就想到了可以像 element-ui 的 tree 元件一樣傳一個 renderContent 函數,由呼叫者自己渲染節點label,這樣就達到了節點定制的目的!
函數式元件
接下來,我們將樹節點模板元件改造成函數式元件。寫node.js:
首先我們實作一個render函數
export const render = (h, context) => { const {props} = context return renderNode(h, props.data, context) }
實作renderNode函數
export const renderNode = (h, data, context) => { const {props} = context const childNodes = [] childNodes.push(renderLabel(h, data, context)) if (props.data.children && props.data.children.length) { childNodes.push(renderChildren(h, props.data.children, context)) } return h('p', { domProps: { className: 'org-tree-node' } }, childNodes) }
實作renderLabel函數。節點label自訂關鍵在這裡:
export const renderLabel = (h, data, context) => { const {props} = context const renderContent = props.renderContent const childNodes = [] // 节点label定制,由调用者传入的renderContent实现 if (typeof renderContent === 'function') { let vnode = renderContent(h, props.data) vnode && childNodes.push(vnode) } else { childNodes.push(props.data.label) } return h('p', { domProps: { className: 'org-tree-node-label' } }, childNodes) }
實作renderChildren函數。這裡遞歸呼叫renderNode,實作了遞迴元件
export const renderChildren = (h, list, context) => { if (Array.isArray(list) && list.length) { const children = list.map(item => { return renderNode(h, item, context) }) return h('p', { domProps: { className: 'org-tree-node-children' } }, children) } return '' }
至此我們的render函數完成了,接下來使用render函數定義函數式元件。在tree元件裡面宣告:
<template> <!-- ... --> </template> <script> import render from './node.js' export default { name: 'OrgTree', components: { OrgTreeNode: { render, // 定义函数式组件 functional: true } } } </script>
至此我們的函數式元件改造完成了,至於水平顯示用樣式控制就可以了。
CSS樣式
樣式使用less預編譯。節點之間的線條採用了:before 、 :after 偽元素的border 繪製
功能擴展
- ##添加了labelClassName 屬性,以支援對節點label的樣式自訂
- 新增了labelWidth 屬性,用於限制節點label的寬度
- 新增了props 屬性,參考element-ui 的tree 元件的props屬性,以支援複雜的資料結構
- 新增了collapsable 屬性,以支援子節點的展開和折疊(展開和折疊操作需呼叫者實作)
- 剛開始採用了flex 佈局,但要相容IE9,後來改成了display: table 佈局
##問題總結可以定義一個樹的store,儲存每個節點狀態,這樣就可以在內部維護樹節點的展開可收起狀態
上面是我整理給大家的,希望今後對大家有幫助。
相關文章:
在微信小程式中如何實作圖片上傳等一系列功能在javaScript中有關空值和假值的說法在Webpack中有關自動化建置(詳細教學)在JavaScript中遇到的BUG以上是使用Vue如何製作組織架構樹組件的詳細內容。更多資訊請關注PHP中文網其他相關文章!

vue中props可以传递函数;vue中可以将字符串、数组、数字和对象作为props传递,props主要用于组件的传值,目的为了接收外面传过来的数据,语法为“export default {methods: {myFunction() {// ...}}};”。

如何覆盖组件库样式?下面本篇文章给大家介绍一下React和Vue项目中优雅地覆盖组件库样式的方法,希望对大家有所帮助!

本篇文章带大家聊聊vue指令中的修饰符,对比一下vue中的指令修饰符和dom事件中的event对象,介绍一下常用的事件修饰符,希望对大家有所帮助!

react与vue的虚拟dom没有区别;react和vue的虚拟dom都是用js对象来模拟真实DOM,用虚拟DOM的diff来最小化更新真实DOM,可以减小不必要的性能损耗,按颗粒度分为不同的类型比较同层级dom节点,进行增、删、移的操作。


熱AI工具

Undresser.AI Undress
人工智慧驅動的應用程序,用於創建逼真的裸體照片

AI Clothes Remover
用於從照片中去除衣服的線上人工智慧工具。

Undress AI Tool
免費脫衣圖片

Clothoff.io
AI脫衣器

AI Hentai Generator
免費產生 AI 無盡。

熱門文章

熱工具

Dreamweaver CS6
視覺化網頁開發工具

禪工作室 13.0.1
強大的PHP整合開發環境

SAP NetWeaver Server Adapter for Eclipse
將Eclipse與SAP NetWeaver應用伺服器整合。

mPDF
mPDF是一個PHP庫,可以從UTF-8編碼的HTML產生PDF檔案。原作者Ian Back編寫mPDF以從他的網站上「即時」輸出PDF文件,並處理不同的語言。與原始腳本如HTML2FPDF相比,它的速度較慢,並且在使用Unicode字體時產生的檔案較大,但支援CSS樣式等,並進行了大量增強。支援幾乎所有語言,包括RTL(阿拉伯語和希伯來語)和CJK(中日韓)。支援嵌套的區塊級元素(如P、DIV),

Atom編輯器mac版下載
最受歡迎的的開源編輯器