element-plus/packages/tree/src/tree-node.vue

266 lines
7.7 KiB
Vue
Raw Normal View History

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'
import ElCollapseTransition from '@element-plus/transition/collapse-transition/index.vue'
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'
import { TreeOptionProps, TreeNodeData, RootTreeType } from './tree.d'
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>