infogo-tree
Version:
A vue tree component using virtual list.
1,032 lines (893 loc) • 36 kB
text/typescript
import TreeNode, { ITreeNodeOptions } from './tree-node'
import { TreeNodeKeyType, ignoreEnum, IgnoreType } from '../const'
//#region Interfaces
interface ITreeStoreOptions {
[key: string]: any,
keyField: string,
ignoreMode?: IgnoreType,
filteredNodeCheckable?: boolean,
cascade?: boolean,
cascadeLoose?: boolean,
defaultExpandAll?: boolean,
load?: Function,
expandOnFilter?: boolean,
}
interface IMapData {
[key: string]: TreeNode,
[key: number]: TreeNode,
}
interface IListenersMap {
[eventName: string]: Function[],
}
export interface IEventNames {
'set-data': () => void,
'visible-data-change': () => void,
'render-data-change': () => void,
'expand': NodeGeneralListenerType,
'select': NodeGeneralListenerType,
'unselect': NodeGeneralListenerType,
'selected-change': (node: TreeNode | null, key: TreeNodeKeyType | null) => void,
'check': NodeGeneralListenerType,
'uncheck': NodeGeneralListenerType,
'checked-change': (nodes: TreeNode[], keys: TreeNodeKeyType[]) => void,
}
//#endregion Interfaces
//#region Types
type NodeGeneralListenerType = (node: TreeNode) => void
export type ListenerType<T extends keyof IEventNames> = IEventNames[T]
export type FilterFunctionType = (keyword: string, node: TreeNode) => boolean
//#endregion Types
export default class TreeStore {
//#region Properties
/** 树数据 */
data: TreeNode[] = []
childStr: string = ''
/** 扁平化的树数据 */
flatData: TreeNode[] = []
disabledFn: any
/** 树数据 map */
mapData: IMapData = Object.create(null)
/** 未加载但已选中的节点 key ,每次 flattenData 的时候进行检查,将加载的节点从此数组中剔除 */
private unloadCheckedKeys: TreeNodeKeyType[] = []
/** 未加载但选中的单选节点 key */
private unloadSelectedKey: TreeNodeKeyType | null = null
/** 当前单选选中节点 key */
private currentSelectedKey: TreeNodeKeyType | null = null
/** 事件 listeners */
private listenersMap: IListenersMap = {}
//#endregion Properties
constructor(private readonly options: ITreeStoreOptions) {
}
setData(data: ITreeNodeOptions[], selectableUnloadKey: TreeNodeKeyType | null = null, checkableUnloadKeys: TreeNodeKeyType[] | null = null, childStr: string, disabledFn: any, replenishSelect: Boolean): void {
this.childStr = childStr
this.disabledFn = disabledFn
this.data = data.map((nodeData: ITreeNodeOptions): TreeNode => new TreeNode(nodeData, null, this.options.keyField, !!this.options.load, childStr, disabledFn))
// 清空 mapData
for (let key in this.mapData) delete this.mapData[key]
// 扁平化之前清空单选选中,如果 value 有值,则是 selectableUnloadKey 有值,会重新设置 currentSelectedKey ;多选选中没有存储在 store 中,因此不必事先清空。
this.currentSelectedKey = null
// 扁平化节点数据
this.flatData = this.flattenData(this.data)
// 更新未载入多选选中节点
this.setUnloadCheckedKeys(checkableUnloadKeys || [], replenishSelect)
if (selectableUnloadKey || selectableUnloadKey === 0) {
// 更新未载入单选选中节点
this.currentSelectedKey = null
this.setUnloadSelectedKey(selectableUnloadKey)
}
this.emit('visible-data-change')
this.emit('set-data')
}
//#region Set api
/**
* 设置单个节点多选选中
* @param key 设置的节点 key
* @param value 是否选中
* @param triggerEvent 是否触发事件
* @param triggerDataChange 是否触发 `data-change` 事件以通知外部刷新视图
* @param filtering 是否正在过滤,如果是,则考虑 `filteredNodeCheckable` Prop,用于判断是否是用户点击节点触发的勾选
* @param isCheckAll 是否是全选
*/
setChecked(key: TreeNodeKeyType, value: boolean, triggerEvent: boolean = true, triggerDataChange: boolean = true, filtering: boolean = false, isCheckAll: boolean = false): void {
const node = this.mapData[key]
if (!node) return this.setUnloadChecked(key, value, triggerEvent, triggerDataChange)
if (node.disabled && isCheckAll) return
// 首先确定没有变化的情况
if (node.checked && value) return // 已经勾选的再次勾选,直接返回
if (!node.checked && !node.indeterminate && !value) return // 未勾选且不是半选状态再次设置未勾选,直接返回
if (this.options.cascade) {
// 向下勾选,包括自身
this.checkNodeDownward(node, value, filtering, isCheckAll)
// 向上勾选父节点直到根节点
this.checkNodeUpward(node)
} else {
node.checked = value
}
if (triggerEvent) {
if (node.checked) {
this.emit('check', node)
} else {
this.emit('uncheck', node)
}
}
this.triggerCheckedChange(triggerEvent, triggerDataChange)
}
/**
* 设置单个未加载节点选中,不公开此 API
*/
private setUnloadChecked(key: TreeNodeKeyType, value: boolean, triggerEvent: boolean = true, triggerDataChange: boolean = true): void {
const index = this.findIndex(key, this.unloadCheckedKeys)
if (value) {
if (index === -1) {
this.unloadCheckedKeys.push(key)
}
} else {
if (index !== -1) {
this.unloadCheckedKeys.splice(index, 1)
}
}
this.triggerCheckedChange(triggerEvent, triggerDataChange)
}
/**
* 批量选中/取消多选选中节点
* @param keys 选中的节点 key 数组
* @param value 是否选中
* @param triggerEvent 是否触发事件
*/
setCheckedKeys(keys: TreeNodeKeyType[], value: boolean, triggerEvent: boolean = true, triggerDataChange: boolean = true, isCheckAll: boolean = false): void {
keys.forEach((key) => {
this.setChecked(key, value, false, false, false, isCheckAll)
})
this.triggerCheckedChange(triggerEvent, triggerDataChange)
}
/**
* 多选勾选全部节点
* @param triggerEvent 是否触发事件
* @param triggerDataChange 是否触发视图刷新
*/
checkAll(triggerEvent: boolean = true, triggerDataChange: boolean = true): void {
if (this.options.cascade) {
// 级联时,只要勾选第一层节点即可,如果第一层被禁用了,则往下一层勾选
const checkCascade = (nodes: TreeNode[]): void => {
nodes.forEach((node) => {
if (node.disabled) {
checkCascade(node.children)
} else {
this.setChecked(node[this.options.keyField], true, false, false)
}
})
}
checkCascade(this.data)
} else {
const length = this.flatData.length
for (let i = 0; i < length; i++) {
const node = this.flatData[i]
this.setChecked(node[this.options.keyField], true, false, false)
}
}
this.triggerCheckedChange(triggerEvent, triggerDataChange)
}
/**
* 清除所有多选已选
* @param triggerEvent 是否触发事件
* @param triggerDataChange 是否触发视图刷新
*/
clearChecked(triggerEvent: boolean = true, triggerDataChange: boolean = true): void {
const currentCheckedNodes = this.getCheckedNodes()
currentCheckedNodes.forEach((checkedNode) => {
this.setChecked(checkedNode[this.options.keyField], false, false, false)
})
// 清空未加载多选选中节点
this.unloadCheckedKeys = []
this.triggerCheckedChange(triggerEvent, triggerDataChange)
}
/**
* 触发 checked-change 的快捷方法
* @param triggerEvent 是否触发事件
* @param triggerDataChange 是否触发视图刷新
*/
private triggerCheckedChange(triggerEvent: boolean = true, triggerDataChange: boolean = true) {
if (triggerEvent) {
this.emit('checked-change', this.getCheckedNodes(), this.getCheckedKeys())
}
if (triggerDataChange) {
this.emit('render-data-change')
}
}
/**
* 设置单选选中
* @param key 选中节点 key
* @param value 是否选中
* @param triggerEvent 是否触发事件
*/
setSelected(key: TreeNodeKeyType, value: boolean, triggerEvent: boolean = true, triggerDataChange: boolean = true): void {
const node = this.mapData[key]
if (!node) return this.setUnloadSelected(key, value, triggerEvent, triggerDataChange)
if (node.disabled) return
if (node.selected === value) return // 当前节点已经是将要设置的状态,直接返回
if (key === this.currentSelectedKey) { // 设置的节点即当前已选中节点
if (!value) { // 取消当前选中节点
node.selected = value
this.currentSelectedKey = null
}
} else { // 设置的节点不是当前已选中节点,要么当前没有选中节点,要么当前有选中节点
if (value) {
if (this.currentSelectedKey === null) { // 当前没有选中节点
node.selected = value
this.currentSelectedKey = node[this.options.keyField]
} else { // 取消当前已选中,设置新的选中节点
if (this.mapData[this.currentSelectedKey]) {
this.mapData[this.currentSelectedKey].selected = false
}
node.selected = value
this.currentSelectedKey = node[this.options.keyField]
}
}
}
if (triggerEvent) {
if (node.selected) {
this.emit('select', node)
} else {
this.emit('unselect', node)
}
this.emit('selected-change', this.getSelectedNode(), this.getSelectedKey())
}
if (triggerDataChange) {
this.emit('render-data-change')
}
}
/**
* 设置未加载单选选中节点,不公开此 API
*/
private setUnloadSelected(key: TreeNodeKeyType, value: boolean, triggerEvent: boolean = true, triggerDataChange: boolean = true): void {
if (value) {
if (this.currentSelectedKey) {
this.setSelected(this.currentSelectedKey, false, false, false)
}
this.unloadSelectedKey = key
} else {
if (this.unloadSelectedKey === key) {
this.unloadSelectedKey = null
}
}
if (triggerEvent) {
this.emit('selected-change', this.getSelectedNode(), this.getSelectedKey())
}
if (triggerDataChange) {
this.emit('render-data-change')
}
}
/**
* 清除当前单选选中
* @param triggerEvent 是否触发事件
* @param triggerDataChange 是否触发视图刷新
*/
clearSelected(triggerEvent: boolean = true, triggerDataChange: boolean = true): void {
if (this.currentSelectedKey && this.mapData[this.currentSelectedKey]) {
this.setSelected(this.currentSelectedKey, false, triggerEvent, triggerDataChange)
} else if (this.unloadSelectedKey !== null) {
this.unloadSelectedKey = null
if (triggerEvent) {
this.emit('selected-change', this.getSelectedNode(), this.getSelectedKey())
}
if (triggerDataChange) {
this.emit('render-data-change')
}
}
}
/**
* 设置节点展开
* @param key 节点 key
* @param value 是否展开
* @param expandParent 展开节点时是否同时展开父节点
* @param triggerEvent 是否触发事件
* @param triggerDataChange 是否触发 `data-change` 事件以通知外部刷新视图
*/
setExpand(key: TreeNodeKeyType, value: boolean, expandParent: boolean = false, triggerEvent: boolean = true, triggerDataChange: boolean = true): void {
const node = this.mapData[key]
if (!node || (!expandParent && node.isLeaf)) return
if (node.expand === value) return // 当前节点已经是将要设置的状态,直接返回
if (!node.isLeaf) {
if (typeof this.options.load === 'function') {
// 如果节点未加载过且将要设置为加载,则调用 load 方法
if (!node._loaded && !node._loading && value) {
node._loading = true
if (triggerDataChange) {
this.emit('visible-data-change')
}
new Promise((resolve, reject) => {
const load = (this.options.load as Function)
load(node, resolve, reject)
}).then((children) => {
if (Array.isArray(children) && children.length > 0) {
const parentIndex: number = this.findIndex(node)
if (parentIndex === -1) return
node._loaded = true
node.expand = value
node.setChildren(children, this.childStr, this.disabledFn)
// 如果单选选中的值为空,则允许后续数据覆盖单选 value
const currentCheckedKeys = this.getCheckedKeys()
const flattenChildren = this.flattenData(node.children, this.getSelectedKey === null)
this.flatData.splice(parentIndex + 1, 0, ...flattenChildren)
// 如果有未加载的选中节点,判断其是否已加载
this.setUnloadCheckedKeys(currentCheckedKeys)
if (this.unloadSelectedKey !== null) {
this.setUnloadSelectedKey(this.unloadSelectedKey)
}
this.emit('set-data')
} else { // 当懒加载的数据为空去掉展开哦节点的箭头
if (node) {
node.isLeaf = true
}
}
}).catch((e: Error | string) => {
let err = e
if (!(e instanceof Error)) {
err = new Error(e)
}
// tslint:disable-next-line: no-console
console.error(err)
}).then(() => {
node._loading = false
if (triggerEvent) {
this.emit('expand', node)
}
if (triggerDataChange) {
this.emit('visible-data-change')
}
})
return
} else if (node._loading) return // 正在加载的节点,直接返回
}
node.expand = value
// Set children visibility
const queue = [...node.children]
while (queue.length) {
if (queue[0].expand && queue[0].children.length) {
queue.push(...queue[0].children)
}
if (queue[0]._filterVisible === false) {
queue[0].visible = false
} else {
queue[0].visible = queue[0]._parent === null || (queue[0]._parent.expand && queue[0]._parent.visible)
}
queue.shift()
}
if (triggerEvent) {
this.emit('expand', node)
}
if (triggerDataChange) {
this.emit('visible-data-change')
}
}
if (expandParent && node._parent && value) {
this.setExpand(node._parent[this.options.keyField], value, expandParent, false, triggerDataChange)
}
}
/**
* 批量设置节点展开/折叠
* @param keys 展开的节点 key 数组
* @param value 是否展开
*/
setExpandKeys(keys: TreeNodeKeyType[], value: boolean, triggerDataChange: boolean = true): void {
keys.forEach((key) => {
this.setExpand(key, value, true, false, false)
})
if (triggerDataChange) {
this.emit('visible-data-change')
}
}
setExpandAll(value: boolean, triggerDataChange: boolean = true): void {
this.flatData.forEach((node) => {
if (!this.options.load || node._loaded) {
this.setExpand(node[this.options.keyField], value, false, false, false)
}
})
if (triggerDataChange) {
this.emit('visible-data-change')
}
}
//#endregion Set api
//#region Get api
/**
* 获取多选选中节点
* @param ignoreMode 忽略模式,可选择忽略父节点或子节点,默认值是 CTree 的 ignoreMode Prop
*/
getCheckedNodes(ignoreMode = this.options.ignoreMode): TreeNode[] {
if (ignoreMode === ignoreEnum.children) {
const result: TreeNode[] = []
const traversal = (nodes: TreeNode[]) => {
nodes.forEach((node) => {
if (node.checked) {
result.push(node)
} else if (!node.isLeaf && (node.indeterminate || (this.options.cascadeLoose && !node.indeterminate))) {
traversal(node.children)
}
})
}
traversal(this.data)
return result
} else {
return this.flatData.filter((node) => {
if (ignoreMode === ignoreEnum.parents) return node.checked && node.isLeaf
return node.checked
})
}
}
/**
* 获取多选选中的节点 key ,包括未加载的 key
* @param ignoreMode 忽略模式,同 `getCheckedNodes`
*/
getCheckedKeys(ignoreMode = this.options.ignoreMode): TreeNodeKeyType[] {
return this.getCheckedNodes(ignoreMode).map((checkedNodes) => checkedNodes[this.options.keyField]).concat(this.unloadCheckedKeys)
}
/**
* 获取多选半选状态节点
*/
getIndeterminateNodes(): TreeNode[] {
return this.flatData.filter((node) => node.indeterminate)
}
/**
* 获取当前单选选中节点
*/
getSelectedNode(): TreeNode | null {
if (this.currentSelectedKey === null) return null
return this.mapData[this.currentSelectedKey] || null
}
/**
* 获取当前单选选中节点 key ,有可能是未加载的选中项
*/
getSelectedKey(): TreeNodeKeyType | null {
if (this.currentSelectedKey === 0 || this.currentSelectedKey) return this.currentSelectedKey
return this.unloadSelectedKey || null
}
/**
* 获取未加载但多选选中的节点
*/
getUnloadCheckedKeys(): TreeNodeKeyType[] {
return this.unloadCheckedKeys
}
/**
* 获取展开节点
*/
getExpandNodes(): TreeNode[] {
return this.flatData.filter((node) => !node.isLeaf && node.expand)
}
/**
* 获取展开节点 keys
*/
getExpandKeys(): TreeNodeKeyType[] {
return this.getExpandNodes().map((node) => node[this.options.keyField])
}
/**
* 根据节点 key 获取节点
* @param key 节点 key
*/
getNode(key: TreeNodeKeyType): TreeNode | null {
return this.mapData[key] || null
}
//#endregion Get api
//#region Node transfer api
// 这边的 referenceKey 都是被执行的节点,比如 insertBefore ,在这里 referenceKey 就是指要插入到 referenceKey 节点前面
insertBefore(insertedNode: TreeNodeKeyType | ITreeNodeOptions, referenceKey: TreeNodeKeyType): TreeNode | null {
const node = this.getInsertedNode(insertedNode, referenceKey)
if (!node) return null
this.remove(node[this.options.keyField], false)
const referenceNode = this.mapData[referenceKey]
const parentNode = referenceNode._parent
const childIndex = this.findIndex(referenceKey, parentNode && parentNode.children)
const flatIndex = this.findIndex(referenceKey)
const dataIndex = (parentNode && -1) || this.findIndex(referenceKey, this.data)
this.insertIntoStore(node, parentNode, childIndex, flatIndex, dataIndex)
this.emit('visible-data-change')
return node
}
insertAfter(insertedNode: TreeNodeKeyType | ITreeNodeOptions, referenceKey: TreeNodeKeyType): TreeNode | null {
const node = this.getInsertedNode(insertedNode, referenceKey)
if (!node) return null
this.remove(node[this.options.keyField], false)
const referenceNode = this.mapData[referenceKey]
const parentNode = referenceNode._parent
const childIndex = this.findIndex(referenceKey, parentNode && parentNode.children) + 1
const length = this.flatData.length
const referenceIndex = this.findIndex(referenceKey)
// 找到待插入节点应该插入 flatData 的索引
let flatIndex = referenceIndex + 1
for (let i = referenceIndex + 1; i <= length; i++) {
if (i === length) {
flatIndex = i
break
}
if (this.flatData[i]._level <= referenceNode._level) {
flatIndex = i
break
}
}
const dataIndex = (parentNode && -1) || (this.findIndex(referenceKey, this.data) + 1)
this.insertIntoStore(node, parentNode, childIndex, flatIndex, dataIndex)
this.emit('visible-data-change')
return node
}
append(insertedNode: TreeNodeKeyType | ITreeNodeOptions, parentKey: TreeNodeKeyType): TreeNode | null {
const parentNode = this.mapData[parentKey]
if (!parentNode.isLeaf) {
const childrenLength = parentNode.children.length
return this.insertAfter(insertedNode, parentNode.children[childrenLength - 1][this.options.keyField])
}
const node = this.getInsertedNode(insertedNode, parentKey, true)
if (!node) return null
this.remove(node[this.options.keyField], false)
const flatIndex = this.findIndex(parentKey) + 1
this.insertIntoStore(node, parentNode, 0, flatIndex)
this.emit('visible-data-change')
return node
}
prepend(insertedNode: TreeNodeKeyType | ITreeNodeOptions, parentKey: TreeNodeKeyType): TreeNode | null {
const parentNode = this.mapData[parentKey]
if (!parentNode.isLeaf) {
return this.insertBefore(insertedNode, parentNode.children[0][this.options.keyField])
}
const node = this.getInsertedNode(insertedNode, parentKey, true)
if (!node) return null
this.remove(node[this.options.keyField], false)
const flatIndex = this.findIndex(parentKey) + 1
this.insertIntoStore(node, parentNode, 0, flatIndex)
this.emit('visible-data-change')
return node
}
/**
* 删除节点
* @param removedKey 要删除的节点 key
*/
remove(removedKey: TreeNodeKeyType, triggerDataChange: boolean = true): TreeNode | null {
const node = this.mapData[removedKey]
if (!node) return null
// 从 flatData 中移除
const index = this.findIndex(node)
if (index === -1) return null
let deleteCount = 1
const length = this.flatData.length
for (let i = index + 1; i < length; i++) {
if (this.flatData[i]._level > node._level) {
deleteCount++
} else break
}
this.flatData.splice(index, deleteCount)
// 从 mapData 中移除
const deleteInMap = (key: TreeNodeKeyType): void => {
const node = this.mapData[key]
delete this.mapData[key]
node.children.forEach((child) => deleteInMap(child[this.options.keyField]))
}
deleteInMap(removedKey)
// 从 data 中移除
if (!node._parent) {
const index = this.findIndex(node, this.data)
if (index > -1) {
this.data.splice(index, 1)
}
}
// 从父节点 children 中移除
if (node._parent) {
const childIndex = this.findIndex(node, node._parent.children)
if (childIndex !== -1) {
node._parent.children.splice(childIndex, 1)
}
// 处理父节点 isLeaf
node._parent.isLeaf = !node._parent.children.length
// 处理父节点 expand
if (node._parent.isLeaf) {
node._parent.expand = false
node._parent.indeterminate = false
}
// 更新被移除处父节点状态
this.updateMovingNodeStatus(node)
}
if (triggerDataChange) {
this.emit('visible-data-change')
}
return node
}
private getInsertedNode(insertedNode: TreeNodeKeyType | ITreeNodeOptions, referenceKey: TreeNodeKeyType, isParent: boolean = false): TreeNode | null {
const referenceNode = this.mapData[referenceKey]
if (!referenceNode) return null
const parentNode = isParent ? referenceNode : referenceNode._parent
if (insertedNode instanceof TreeNode) {
// 与参照节点是同一个节点
if (insertedNode[this.options.keyField] === referenceKey) return null
return insertedNode
} else if (typeof insertedNode === 'object') {
if (insertedNode[this.options.keyField] === referenceKey) return null
const mapNode = this.mapData[insertedNode[this.options.keyField]]
if (mapNode) return mapNode
return new TreeNode(insertedNode, parentNode, this.options.keyField, !!this.options.load, this.childStr, this.disabledFn)
} else {
if (!this.mapData[insertedNode] || insertedNode === referenceKey) return null
return this.mapData[insertedNode]
}
}
/**
* 将一个节点插入 store 记录中
* @param node 要插入的节点
* @param parentNode 要插入节点的父节点
* @param childIndex 如果有父节点,则需提供要插入的节点在同级节点中的顺序
* @param flatIndex 在 flatData 中的索引
* @param dataIndex 如果没有父节点,需要提供节点在 data 中的索引
*/
private insertIntoStore(node: TreeNode, parentNode: TreeNode | null, childIndex: number, flatIndex: number, dataIndex?: number): void {
if (flatIndex === -1) return
// 插入父节点 children 中
if (childIndex !== -1 && parentNode) {
parentNode.children.splice(childIndex, 0, node)
}
// 更新 _parent
node._parent = parentNode
// 更新父节点 isLeaf, expand
if (parentNode) {
parentNode.isLeaf = false
this.setExpand(parentNode[this.options.keyField], true, true, false, false)
} else if (typeof dataIndex === 'number' && dataIndex > -1) {
// 没有父节点,则需要插入到 this.data 中以保证数据正确
this.data.splice(dataIndex, 0, node)
}
// 插入 flatData 与 mapData
const nodes: TreeNode[] = this.flattenData([node])
// 处理自身及子节点 _level
node._level = parentNode ? parentNode._level + 1 : 0
nodes.forEach((childNode) => childNode._level = childNode._parent ? childNode._parent._level + 1 : 0)
this.flatData.splice(flatIndex, 0, ...nodes)
// 更新被移除处父节点状态
this.updateMovingNodeStatus(node)
}
private updateMovingNodeStatus(movingNode: TreeNode): void {
// 处理多选
this.checkNodeUpward(movingNode)
this.triggerCheckedChange()
// 处理单选
if (movingNode.selected) {
this.setSelected(movingNode[this.options.keyField], true)
}
}
//#endregion Node transfer api
/**
* 过滤本地节点数据
* @param keyword 过滤关键词
* @param filterMethod 过滤方法
*/
filter(keyword: string, filterMethod: FilterFunctionType): void {
// 使用树形结构数据进行遍历
const filterVisibleNodes: TreeNode[] = []
this.flatData.forEach((node) => {
node._filterVisible = node._parent && node._parent._filterVisible || filterMethod(keyword, node)
node.visible = node._filterVisible
if (node._filterVisible) {
filterVisibleNodes.push(node)
}
})
// 对于临时列表中的节点,都是可见的,因此将它们的父节点都设为可见并展开
filterVisibleNodes.forEach((node) => {
const stack = []
let parent = node._parent
while (parent) {
stack.unshift(parent)
parent = parent._parent
}
stack.forEach((parent) => {
parent._filterVisible = true
// parent.visible = parent._filterVisible
// _filterVisible 且 无父级或者父级展开且可见,则设为可见
parent.visible = (parent._parent === null || (parent._parent.expand && parent._parent.visible)) && parent._filterVisible
this.options.expandOnFilter && this.setExpand(parent[this.options.keyField], true, true, false, false)
})
node.visible = node._parent === null || (node._parent.expand && node._parent.visible)
})
this.emit('visible-data-change')
}
/**
* 过滤未加载多选节点,对比最终勾选节点是否有变化并触发 checked-change 事件
* @param keys 全量选中节点 key 数组,包括加载与未加载选中节点
*/
private setUnloadCheckedKeys(keys: TreeNodeKeyType[], replenishSelect: Boolean = true): void {
this.unloadCheckedKeys = keys
const checkedKeysCache = keys.concat()
const length = this.unloadCheckedKeys.length
for (let i = length - 1; i >= 0; i--) {
const key = this.unloadCheckedKeys[i]
if (this.mapData[key]) {
this.setChecked(key, true, false, false)
this.unloadCheckedKeys.splice(i, 1)
}
}
const newCheckedKeys = this.getCheckedKeys()
if (newCheckedKeys.length === checkedKeysCache.length && newCheckedKeys.every((val) => (checkedKeysCache as TreeNodeKeyType[]).some((cache) => cache === val))) return
// 因为多选模式下,value 传过来的数据跟数据上标识 checked: true 的结果可能不一致,因此需要触发一个事件告诉外部最终勾选了哪些节点
if (replenishSelect) {
this.emit('checked-change', this.getCheckedNodes(), newCheckedKeys)
}
}
/**
* 过滤未加载单选选中节点,对比是否有变化并触发 selected-change 事件
* @param key 节点 key
*/
private setUnloadSelectedKey(key: TreeNodeKeyType): void {
const selectedKeyCache = this.getSelectedKey()
if (this.mapData[key]) { // 如果节点已加载
this.setSelected(key, true, false)
this.unloadSelectedKey = null
} else { // 节点未加载
this.currentSelectedKey && this.setSelected(this.currentSelectedKey, false, false)
this.unloadSelectedKey = key
}
const newSelectedKey = this.getSelectedKey()
if (newSelectedKey === selectedKeyCache) return
this.emit('selected-change', this.getSelectedNode(), newSelectedKey)
}
/**
* 保存扁平化的节点数据 `flatData` 与节点数据 Map `mapData`
* @param nodes 树状节点数据
* @param overrideSelected 是否根据数据设置 `selected`
*/
private flattenData(nodes: TreeNode[], overrideSelected: boolean = true, result: TreeNode[] = []): TreeNode[] {
const length = nodes.length
for (let i = 0; i < length; i++) {
const node = nodes[i]
const key: TreeNodeKeyType = node[this.options.keyField]
result.push(node)
if (this.mapData[key]) {
throw new Error('[CTree] Duplicate tree node key.')
}
this.mapData[key] = node
// 如果数据上就是选中的,则更新节点状态
if (node.checked && this.options.cascade) {
// 向下勾选,包括自身
this.checkNodeDownward(node, true)
// 向上勾选父节点直到根节点
this.checkNodeUpward(node)
}
if (node.selected && overrideSelected) {
this.clearSelected(false, false)
this.currentSelectedKey = node[this.options.keyField]
this.emit('selected-change', node, this.currentSelectedKey)
}
if ((this.options.defaultExpandAll || node.expand) && !this.options.load && !node.isLeaf) {
node.expand = false
this.setExpand(node[this.options.keyField], true, false, false, false)
}
if (node.children.length) {
this.flattenData(node.children, overrideSelected, result)
}
}
return result
}
//#region Check nodes
/**
* 向下勾选/取消勾选节点,包括自身
* @param node 需要向下勾选的节点
* @param value 勾选或取消勾选
* @param filtering 是否正在过滤,如果是,则考虑 `filteredNodeCheckable` Prop
*/
private checkNodeDownward(node: TreeNode, value: boolean, filtering: boolean = false, isCheckAll: boolean = false): void {
node.children.forEach((child) => {
this.checkNodeDownward(child, value, filtering, isCheckAll)
})
if (node.isLeaf || (this.options.load && !node.children.length)) {
if (!node.disabled || (node.disabled && !isCheckAll)) {
// 正在过滤,若被过滤节点不可勾选,且节点过滤后不可见,则直接返回
if (filtering && !this.options.filteredNodeCheckable && !node._filterVisible) return
node.checked = value
node.indeterminate = false
}
} else {
this.checkParentNode(node, 'down')
}
}
/**
* 向上勾选/取消勾选父节点,不包括自身
* @param node 需要勾选的节点
*/
private checkNodeUpward(node: TreeNode) {
let parent = node._parent
while (parent) {
this.checkParentNode(parent, 'up')
parent = parent._parent
}
}
/**
* 根据子节点的勾选状态更新当前父节点的勾选状态
* @param node 需要勾选的节点
*/
private checkParentNode(node: TreeNode, type: string): void {
const length = node.children.length
if (!length) return
let hasChecked = false
let hasUnchecked = false
let isInterrupted = false
for (let i = 0; i < length; i++) {
const child = node.children[i]
// && !child.disabled
if (!child.disabled) {
if (child.checked) {
hasChecked = true
} else {
hasUnchecked = true
}
}
if ((hasChecked && hasUnchecked) || child.indeterminate) {
if (!this.options.cascadeLoose) {
node.indeterminate = true
}
isInterrupted = true
node.checked = false
break
}
}
if (!isInterrupted) {
if (type === 'up' && this.options.cascadeLoose) {
node.checked = false
} else {
node.checked = hasChecked
node.indeterminate = false
}
}
}
//#endregion Check nodes
//#region Utils
/**
* 搜索节点在指定数组中的位置
*/
private findIndex(keyOrNode: TreeNode | TreeNodeKeyType, searchList: TreeNode[] | TreeNodeKeyType[] | null = this.flatData): number {
if (searchList !== null) {
let key: TreeNodeKeyType = keyOrNode instanceof TreeNode ? keyOrNode[this.options.keyField] : keyOrNode
const length = searchList.length
for (let i = 0; i < length; i++) {
if (searchList[0] instanceof TreeNode) {
if (key === (searchList as TreeNode[])[i][this.options.keyField]) {
return i
}
} else {
if (key === searchList[i]) {
return i
}
}
}
}
return -1
}
//#endregion Utils
//#region Mini EventTarget
on<T extends keyof IEventNames>(eventName: T, listener: ListenerType<T> | Array<ListenerType<T>>): void {
if (!this.listenersMap[eventName]) {
this.listenersMap[eventName] = []
}
let listeners: Array<ListenerType<T>> = []
if (!Array.isArray(listener)) {
listeners = [listener]
} else {
listeners = listener
}
listeners.forEach((listener) => {
if (this.listenersMap[eventName].indexOf(listener) === -1) {
this.listenersMap[eventName].push(listener)
}
})
}
off<T extends keyof IEventNames>(eventName: T, listener?: ListenerType<T>): void {
if (!this.listenersMap[eventName]) return
if (!listener) {
this.listenersMap[eventName] = []
} else {
const index = this.listenersMap[eventName].indexOf(listener)
if (index > -1) {
this.listenersMap[eventName].splice(index, 1)
}
}
}
emit<T extends keyof IEventNames>(eventName: T, ...args: Parameters<IEventNames[T]>): void {
if (!this.listenersMap[eventName]) return
const length: number = this.listenersMap[eventName].length
for (let i: number = 0; i < length; i++) {
this.listenersMap[eventName][i](...args)
}
}
//#endregion Mini EventTarget
}