2020-08-21 16:16:08 +08:00
|
|
|
<template>
|
|
|
|
<div
|
|
|
|
v-show="node.visible"
|
|
|
|
ref="node$"
|
|
|
|
class="el-tree-node"
|
|
|
|
:class="{
|
|
|
|
'is-expanded': expanded,
|
|
|
|
'is-current': node.isCurrent,
|
|
|
|
'is-hidden': !node.visible,
|
|
|
|
'is-focusable': !node.disabled,
|
|
|
|
'is-checked': !node.disabled && node.checked,
|
|
|
|
}"
|
|
|
|
role="treeitem"
|
|
|
|
tabindex="-1"
|
|
|
|
:aria-expanded="expanded"
|
|
|
|
:aria-disabled="node.disabled"
|
|
|
|
:aria-checked="node.checked"
|
|
|
|
:draggable="tree.props.draggable"
|
|
|
|
@click.stop="handleClick"
|
|
|
|
@contextmenu="handleContextMenu"
|
|
|
|
@dragstart.stop="handleDragStart"
|
|
|
|
@dragover.stop="handleDragOver"
|
|
|
|
@dragend.stop="handleDragEnd"
|
|
|
|
@drop.stop="handleDrop"
|
|
|
|
>
|
|
|
|
<div
|
|
|
|
class="el-tree-node__content"
|
|
|
|
:style="{ 'padding-left': (node.level - 1) * tree.props.indent + 'px' }"
|
|
|
|
>
|
|
|
|
<span
|
|
|
|
:class="[
|
|
|
|
{
|
|
|
|
'is-leaf': node.isLeaf,
|
|
|
|
expanded: !node.isLeaf && expanded,
|
|
|
|
},
|
|
|
|
'el-tree-node__expand-icon',
|
|
|
|
tree.props.iconClass ? tree.props.iconClass : 'el-icon-caret-right',
|
|
|
|
]"
|
|
|
|
@click.stop="handleExpandIconClick"
|
|
|
|
>
|
|
|
|
</span>
|
|
|
|
<el-checkbox
|
|
|
|
v-if="showCheckbox"
|
|
|
|
:model-value="node.checked"
|
|
|
|
:indeterminate="node.indeterminate"
|
|
|
|
:disabled="!!node.disabled"
|
|
|
|
@click.stop
|
|
|
|
@change="handleCheckChange"
|
|
|
|
/>
|
|
|
|
<span
|
|
|
|
v-if="node.loading"
|
|
|
|
class="el-tree-node__loading-icon el-icon-loading"
|
|
|
|
>
|
|
|
|
</span>
|
|
|
|
<node-content :node="node" :render-content="renderContent" />
|
|
|
|
</div>
|
|
|
|
<el-collapse-transition>
|
|
|
|
<div
|
|
|
|
v-if="!renderAfterExpand || childNodeRendered"
|
|
|
|
v-show="expanded"
|
|
|
|
class="el-tree-node__children"
|
|
|
|
role="group"
|
|
|
|
:aria-expanded="expanded"
|
|
|
|
>
|
|
|
|
<el-tree-node
|
|
|
|
v-for="child in node.childNodes"
|
|
|
|
:key="getNodeKey(child)"
|
|
|
|
:render-content="renderContent"
|
|
|
|
:render-after-expand="renderAfterExpand"
|
|
|
|
:show-checkbox="showCheckbox"
|
|
|
|
:node="child"
|
|
|
|
@node-expand="handleChildNodeExpand"
|
|
|
|
/>
|
|
|
|
</div>
|
|
|
|
</el-collapse-transition>
|
|
|
|
</div>
|
|
|
|
</template>
|
|
|
|
<script lang='ts'>
|
|
|
|
import { defineComponent, getCurrentInstance, ref, watch, nextTick, inject, provide, PropType, ComponentInternalInstance } from 'vue'
|
2020-11-23 13:55:48 +08:00
|
|
|
import ElCollapseTransition from '@element-plus/collapse-transition'
|
2020-11-20 20:24:16 +08:00
|
|
|
import ElCheckbox from '@element-plus/checkbox'
|
2020-08-21 16:16:08 +08:00
|
|
|
import NodeContent from './tree-node-content.vue'
|
|
|
|
import { getNodeKey as getNodeKeyUtil } from './model/util'
|
|
|
|
import { useNodeExpandEventBroadcast } from './model/useNodeExpandEventBroadcast'
|
|
|
|
import { useDragNodeEmitter } from './model/useDragNode'
|
|
|
|
import Node from './model/node'
|
2020-11-26 23:26:03 +08:00
|
|
|
import { TreeOptionProps, TreeNodeData, RootTreeType } from './tree.type'
|
2020-08-21 16:16:08 +08:00
|
|
|
|
|
|
|
export default defineComponent({
|
|
|
|
name: 'ElTreeNode',
|
|
|
|
components: {
|
|
|
|
ElCollapseTransition,
|
|
|
|
ElCheckbox,
|
|
|
|
NodeContent,
|
|
|
|
},
|
|
|
|
props: {
|
|
|
|
node: {
|
|
|
|
type: Node,
|
|
|
|
default: () => ({}),
|
|
|
|
},
|
|
|
|
props: {
|
|
|
|
type: Object as PropType<TreeOptionProps>,
|
|
|
|
default: () => ({}),
|
|
|
|
},
|
|
|
|
renderContent: Function,
|
|
|
|
renderAfterExpand: Boolean,
|
|
|
|
showCheckbox: {
|
|
|
|
type: Boolean,
|
|
|
|
default: false,
|
|
|
|
},
|
|
|
|
},
|
|
|
|
emits: ['node-expand'],
|
|
|
|
setup(props, ctx) {
|
|
|
|
const { broadcastExpanded } = useNodeExpandEventBroadcast(props)
|
|
|
|
const tree = inject<RootTreeType>('RootTree')
|
|
|
|
const expanded = ref(false)
|
|
|
|
const childNodeRendered = ref(false)
|
|
|
|
const oldChecked = ref<boolean>(null)
|
|
|
|
const oldIndeterminate = ref<boolean>(null)
|
|
|
|
const node$ = ref<Nullable<HTMLElement>>(null)
|
|
|
|
const { emitter } = useDragNodeEmitter()
|
|
|
|
const instance = getCurrentInstance()
|
|
|
|
|
|
|
|
provide('NodeInstance', instance)
|
|
|
|
if(!tree) {
|
|
|
|
console.warn('Can not find node\'s tree.')
|
|
|
|
}
|
|
|
|
|
|
|
|
if(props.node.expanded) {
|
|
|
|
expanded.value = true
|
|
|
|
childNodeRendered.value = true
|
|
|
|
}
|
|
|
|
|
|
|
|
const childrenKey = tree.props['children'] || 'children'
|
|
|
|
watch(() => props.node.data[childrenKey], () => {
|
|
|
|
props.node.updateChildren()
|
|
|
|
})
|
|
|
|
|
|
|
|
watch(() => props.node.indeterminate, val => {
|
|
|
|
handleSelectChange(props.node.checked, val)
|
|
|
|
})
|
|
|
|
|
|
|
|
watch(() => props.node.checked, val => {
|
|
|
|
handleSelectChange(val, props.node.indeterminate)
|
|
|
|
})
|
|
|
|
|
|
|
|
watch(() => props.node.expanded, val => {
|
|
|
|
nextTick(() => expanded.value = val)
|
|
|
|
if(val) {
|
|
|
|
childNodeRendered.value = true
|
|
|
|
}
|
|
|
|
})
|
|
|
|
|
|
|
|
const getNodeKey = (node: Node): any => {
|
|
|
|
return getNodeKeyUtil(tree.props.nodeKey, node.data)
|
|
|
|
}
|
|
|
|
|
|
|
|
const handleSelectChange = (checked: boolean, indeterminate: boolean) => {
|
|
|
|
if (oldChecked.value !== checked && oldIndeterminate.value !== indeterminate) {
|
|
|
|
tree.ctx.emit('check-change', props.node.data, checked, indeterminate)
|
|
|
|
}
|
|
|
|
oldChecked.value = checked
|
|
|
|
oldIndeterminate.value = indeterminate
|
|
|
|
}
|
|
|
|
|
|
|
|
const handleClick = () => {
|
|
|
|
const store = tree.store.value
|
|
|
|
store.setCurrentNode(props.node)
|
|
|
|
tree.ctx.emit('current-change', store.currentNode ? store.currentNode.data : null, store.currentNode)
|
|
|
|
tree.currentNode.value = props.node
|
|
|
|
|
|
|
|
if(tree.props.expandOnClickNode) {
|
|
|
|
handleExpandIconClick()
|
|
|
|
}
|
|
|
|
|
|
|
|
if(tree.props.checkOnClickNode && !props.node.disabled) {
|
|
|
|
handleCheckChange(null, {
|
|
|
|
target: { checked: !props.node.checked },
|
|
|
|
})
|
|
|
|
}
|
|
|
|
tree.ctx.emit('node-click', props.node.data, props.node, instance)
|
|
|
|
}
|
|
|
|
|
|
|
|
const handleContextMenu = (event: Event) => {
|
|
|
|
if (tree.instance.vnode.props['onNode-contextmenu']) {
|
|
|
|
event.stopPropagation()
|
|
|
|
event.preventDefault()
|
|
|
|
}
|
|
|
|
tree.ctx.emit('node-contextmenu', event, props.node.data, props.node, instance)
|
|
|
|
}
|
|
|
|
|
|
|
|
const handleExpandIconClick = () => {
|
|
|
|
if (props.node.isLeaf) return
|
|
|
|
if (expanded.value) {
|
|
|
|
tree.ctx.emit('node-collapse', props.node.data, props.node, instance)
|
|
|
|
props.node.collapse()
|
|
|
|
} else {
|
|
|
|
props.node.expand()
|
|
|
|
ctx.emit('node-expand', props.node.data, props.node, instance)
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
const handleCheckChange = (value, ev) => {
|
|
|
|
props.node.setChecked(ev.target.checked, !tree.props.checkStrictly)
|
|
|
|
nextTick(() => {
|
|
|
|
const store = tree.store.value
|
|
|
|
tree.ctx.emit('check', props.node.data, {
|
|
|
|
checkedNodes: store.getCheckedNodes(),
|
|
|
|
checkedKeys: store.getCheckedKeys(),
|
|
|
|
halfCheckedNodes: store.getHalfCheckedNodes(),
|
|
|
|
halfCheckedKeys: store.getHalfCheckedKeys(),
|
|
|
|
})
|
|
|
|
})
|
|
|
|
}
|
|
|
|
|
|
|
|
const handleChildNodeExpand = (nodeData: TreeNodeData, node: Node, instance: ComponentInternalInstance) => {
|
|
|
|
broadcastExpanded(node)
|
|
|
|
tree.ctx.emit('node-expand', nodeData, node, instance)
|
|
|
|
}
|
|
|
|
|
|
|
|
const handleDragStart = (event: DragEvent) => {
|
|
|
|
if (!tree.props.draggable) return
|
|
|
|
emitter.emit('tree-node-drag-start', { event, treeNode: props })
|
|
|
|
}
|
|
|
|
|
|
|
|
const handleDragOver = (event: DragEvent) => {
|
|
|
|
if (!tree.props.draggable) return
|
|
|
|
emitter.emit('tree-node-drag-over', { event, treeNode: { $el: node$.value, node: props.node } })
|
|
|
|
event.preventDefault()
|
|
|
|
}
|
|
|
|
|
|
|
|
const handleDrop = (event: DragEvent) => {
|
|
|
|
event.preventDefault()
|
|
|
|
}
|
|
|
|
|
|
|
|
const handleDragEnd = (event: DragEvent) => {
|
|
|
|
if (!tree.props.draggable) return
|
|
|
|
emitter.emit('tree-node-drag-end', event)
|
|
|
|
}
|
|
|
|
|
|
|
|
return {
|
|
|
|
node$,
|
|
|
|
tree,
|
|
|
|
expanded,
|
|
|
|
childNodeRendered,
|
|
|
|
oldChecked,
|
|
|
|
oldIndeterminate,
|
|
|
|
emitter,
|
|
|
|
parent,
|
|
|
|
getNodeKey,
|
|
|
|
handleSelectChange,
|
|
|
|
handleClick,
|
|
|
|
handleContextMenu,
|
|
|
|
handleExpandIconClick,
|
|
|
|
handleCheckChange,
|
|
|
|
handleChildNodeExpand,
|
|
|
|
handleDragStart,
|
|
|
|
handleDragOver,
|
|
|
|
handleDrop,
|
|
|
|
handleDragEnd,
|
|
|
|
}
|
|
|
|
|
|
|
|
},
|
|
|
|
})
|
|
|
|
</script>
|