| 123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208209210211212213214215216217218219220221222223224225226227228229230231232233234235236237238239240241242243244245246247248249250251252253254255256257258259260261262263264265266267268269270271272273274275276277278279280281282283284285286287288289290291292293294295296297298299300301302303304305306307308309310311312313314315316317318319320321322323324325326327328329330331332333334335336337338339340341342343344345346347348349350351352353354355356357358359360361362363364365366367368369370371372373374375376377378379380381382383384385386387388389390391392393394395396397398399400401402403404405406407408409410411412413414415416417418419420421422423424425426427428429430431432433434435436437438439440441442443444445446447448449450451452453454455456457458459460461462463464465466467468469470471472473474475476477478479480481482483484485486487488489490491492493494495496497498499500501502503504505506507508509510511512513514515516517518519520521522523524525526527528529530531532533534535536537538539540541542543544545546547548549550551552553554555556557558559560561562563564565566567568569570571572573574575576577578579580581582583584585586587588589590591592593594595596597598599600601602603604605606607608609610611612613614615616617618619620621622623624625626627628629630631632633634635636637638639640641642643644645646647648649650651652653654655656657658659660661662663664665666667668669670671672673674675676677678679680681682683684685686687688689690691692693694695696697698699700701702703704705706707708709710711712713714715716717718719720721722723724725726727728729730731732733734735736737738739740741742743744745746747748749750751752753754755756757758759760761762763764765766767768769770771772773774775776777778779780781782783784785786787788789790791792793794795796797798799800801802803804805806807808809810811812813814815816817818819820821822823824825826827828829830831832833834835836837838839840841842843844845846847848849850851852853854855856857858859860861862863864865866867868869870871872873874875876877878879880881882883884885886887888889890891892893894895896897898899900901902903904905906907908909910911912913914915916917918919920921922923924925926927928929930931932933934935936937938939940941942943944945946947948949950951952953954955956957958959960961962963964965966967968969970971972973974975976977978979980981982983984985986987988989990991992993994995996997998999100010011002100310041005100610071008100910101011101210131014101510161017101810191020102110221023102410251026102710281029103010311032103310341035103610371038103910401041104210431044104510461047104810491050105110521053105410551056105710581059106010611062106310641065106610671068106910701071107210731074107510761077107810791080108110821083108410851086108710881089109010911092109310941095109610971098109911001101110211031104110511061107110811091110111111121113111411151116 |
- import _objectSpread from "@babel/runtime/helpers/esm/objectSpread2";
- import _extends from "@babel/runtime/helpers/esm/extends";
- import { createVNode as _createVNode } from "vue";
- import { useProvideKeysState, TreeContext } from './contextTypes';
- import { getDragChildrenKeys, parseCheckedKeys, conductExpandParent, calcSelectedKeys, calcDropPosition, arrAdd, arrDel, posToArr } from './util';
- import { flattenTreeData, convertTreeToData, convertDataToEntities, convertNodePropsToEventData, getTreeNodeProps, fillFieldNames } from './utils/treeUtil';
- import NodeList, { MOTION_KEY, MotionEntity } from './NodeList';
- import { conductCheck } from './utils/conductUtil';
- import DropIndicator from './DropIndicator';
- import { computed, defineComponent, onUnmounted, reactive, shallowRef, watch, watchEffect, nextTick, toRaw } from 'vue';
- import initDefaultProps from '../_util/props-util/initDefaultProps';
- import { treeProps } from './props';
- import { warning } from '../vc-util/warning';
- import KeyCode from '../_util/KeyCode';
- import classNames from '../_util/classNames';
- import pickAttrs from '../_util/pickAttrs';
- import useMaxLevel from './useMaxLevel';
- const MAX_RETRY_TIMES = 10;
- export default defineComponent({
- compatConfig: {
- MODE: 3
- },
- name: 'Tree',
- inheritAttrs: false,
- props: initDefaultProps(treeProps(), {
- prefixCls: 'vc-tree',
- showLine: false,
- showIcon: true,
- selectable: true,
- multiple: false,
- checkable: false,
- disabled: false,
- checkStrictly: false,
- draggable: false,
- expandAction: false,
- defaultExpandParent: true,
- autoExpandParent: false,
- defaultExpandAll: false,
- defaultExpandedKeys: [],
- defaultCheckedKeys: [],
- defaultSelectedKeys: [],
- dropIndicatorRender: DropIndicator,
- allowDrop: () => true
- }),
- setup(props, _ref) {
- let {
- attrs,
- slots,
- expose
- } = _ref;
- const destroyed = shallowRef(false);
- let delayedDragEnterLogic = {};
- const indent = shallowRef();
- const selectedKeys = shallowRef([]);
- const checkedKeys = shallowRef([]);
- const halfCheckedKeys = shallowRef([]);
- const loadedKeys = shallowRef([]);
- const loadingKeys = shallowRef([]);
- const expandedKeys = shallowRef([]);
- const loadingRetryTimes = {};
- const dragState = reactive({
- draggingNodeKey: null,
- dragChildrenKeys: [],
- // dropTargetKey is the key of abstract-drop-node
- // the abstract-drop-node is the real drop node when drag and drop
- // not the DOM drag over node
- dropTargetKey: null,
- dropPosition: null,
- dropContainerKey: null,
- dropLevelOffset: null,
- dropTargetPos: null,
- dropAllowed: true,
- // the abstract-drag-over-node
- // if mouse is on the bottom of top dom node or no the top of the bottom dom node
- // abstract-drag-over-node is the top node
- dragOverNodeKey: null
- });
- const treeData = shallowRef([]);
- watch([() => props.treeData, () => props.children], () => {
- treeData.value = props.treeData !== undefined ? props.treeData.slice() : convertTreeToData(toRaw(props.children));
- }, {
- immediate: true,
- deep: true
- });
- const keyEntities = shallowRef({});
- const focused = shallowRef(false);
- const activeKey = shallowRef(null);
- const listChanging = shallowRef(false);
- const fieldNames = computed(() => fillFieldNames(props.fieldNames));
- const listRef = shallowRef();
- let dragStartMousePosition = null;
- let dragNode = null;
- let currentMouseOverDroppableNodeKey = null;
- const treeNodeRequiredProps = computed(() => {
- return {
- expandedKeysSet: expandedKeysSet.value,
- selectedKeysSet: selectedKeysSet.value,
- loadedKeysSet: loadedKeysSet.value,
- loadingKeysSet: loadingKeysSet.value,
- checkedKeysSet: checkedKeysSet.value,
- halfCheckedKeysSet: halfCheckedKeysSet.value,
- dragOverNodeKey: dragState.dragOverNodeKey,
- dropPosition: dragState.dropPosition,
- keyEntities: keyEntities.value
- };
- });
- const expandedKeysSet = computed(() => {
- return new Set(expandedKeys.value);
- });
- const selectedKeysSet = computed(() => {
- return new Set(selectedKeys.value);
- });
- const loadedKeysSet = computed(() => {
- return new Set(loadedKeys.value);
- });
- const loadingKeysSet = computed(() => {
- return new Set(loadingKeys.value);
- });
- const checkedKeysSet = computed(() => {
- return new Set(checkedKeys.value);
- });
- const halfCheckedKeysSet = computed(() => {
- return new Set(halfCheckedKeys.value);
- });
- watchEffect(() => {
- if (treeData.value) {
- const entitiesMap = convertDataToEntities(treeData.value, {
- fieldNames: fieldNames.value
- });
- keyEntities.value = _extends({
- [MOTION_KEY]: MotionEntity
- }, entitiesMap.keyEntities);
- }
- });
- let init = false; // 处理 defaultXxxx api, 仅仅首次有效
- watch([() => props.expandedKeys, () => props.autoExpandParent, keyEntities],
- // eslint-disable-next-line @typescript-eslint/no-unused-vars
- (_ref2, _ref3) => {
- let [_newKeys, newAutoExpandParent] = _ref2;
- let [_oldKeys, oldAutoExpandParent] = _ref3;
- let keys = expandedKeys.value;
- // ================ expandedKeys =================
- if (props.expandedKeys !== undefined || init && newAutoExpandParent !== oldAutoExpandParent) {
- keys = props.autoExpandParent || !init && props.defaultExpandParent ? conductExpandParent(props.expandedKeys, keyEntities.value) : props.expandedKeys;
- } else if (!init && props.defaultExpandAll) {
- const cloneKeyEntities = _extends({}, keyEntities.value);
- delete cloneKeyEntities[MOTION_KEY];
- keys = Object.keys(cloneKeyEntities).map(key => cloneKeyEntities[key].key);
- } else if (!init && props.defaultExpandedKeys) {
- keys = props.autoExpandParent || props.defaultExpandParent ? conductExpandParent(props.defaultExpandedKeys, keyEntities.value) : props.defaultExpandedKeys;
- }
- if (keys) {
- expandedKeys.value = keys;
- }
- init = true;
- }, {
- immediate: true
- });
- // ================ flattenNodes =================
- const flattenNodes = shallowRef([]);
- watchEffect(() => {
- flattenNodes.value = flattenTreeData(treeData.value, expandedKeys.value, fieldNames.value);
- });
- // ================ selectedKeys =================
- watchEffect(() => {
- if (props.selectable) {
- if (props.selectedKeys !== undefined) {
- selectedKeys.value = calcSelectedKeys(props.selectedKeys, props);
- } else if (!init && props.defaultSelectedKeys) {
- selectedKeys.value = calcSelectedKeys(props.defaultSelectedKeys, props);
- }
- }
- });
- const {
- maxLevel,
- levelEntities
- } = useMaxLevel(keyEntities);
- // ================= checkedKeys =================
- watchEffect(() => {
- if (props.checkable) {
- let checkedKeyEntity;
- if (props.checkedKeys !== undefined) {
- checkedKeyEntity = parseCheckedKeys(props.checkedKeys) || {};
- } else if (!init && props.defaultCheckedKeys) {
- checkedKeyEntity = parseCheckedKeys(props.defaultCheckedKeys) || {};
- } else if (treeData.value) {
- // If `treeData` changed, we also need check it
- checkedKeyEntity = parseCheckedKeys(props.checkedKeys) || {
- checkedKeys: checkedKeys.value,
- halfCheckedKeys: halfCheckedKeys.value
- };
- }
- if (checkedKeyEntity) {
- let {
- checkedKeys: newCheckedKeys = [],
- halfCheckedKeys: newHalfCheckedKeys = []
- } = checkedKeyEntity;
- if (!props.checkStrictly) {
- const conductKeys = conductCheck(newCheckedKeys, true, keyEntities.value, maxLevel.value, levelEntities.value);
- ({
- checkedKeys: newCheckedKeys,
- halfCheckedKeys: newHalfCheckedKeys
- } = conductKeys);
- }
- checkedKeys.value = newCheckedKeys;
- halfCheckedKeys.value = newHalfCheckedKeys;
- }
- }
- });
- // ================= loadedKeys ==================
- watchEffect(() => {
- if (props.loadedKeys) {
- loadedKeys.value = props.loadedKeys;
- }
- });
- const resetDragState = () => {
- _extends(dragState, {
- dragOverNodeKey: null,
- dropPosition: null,
- dropLevelOffset: null,
- dropTargetKey: null,
- dropContainerKey: null,
- dropTargetPos: null,
- dropAllowed: false
- });
- };
- const scrollTo = scroll => {
- listRef.value.scrollTo(scroll);
- };
- watch(() => props.activeKey, () => {
- if (props.activeKey !== undefined) {
- activeKey.value = props.activeKey;
- }
- }, {
- immediate: true
- });
- watch(activeKey, val => {
- nextTick(() => {
- if (val !== null) {
- scrollTo({
- key: val
- });
- }
- });
- }, {
- immediate: true,
- flush: 'post'
- });
- // =========================== Expanded ===========================
- /** Set uncontrolled `expandedKeys`. This will also auto update `flattenNodes`. */
- const setExpandedKeys = keys => {
- if (props.expandedKeys === undefined) {
- expandedKeys.value = keys;
- }
- };
- const cleanDragState = () => {
- if (dragState.draggingNodeKey !== null) {
- _extends(dragState, {
- draggingNodeKey: null,
- dropPosition: null,
- dropContainerKey: null,
- dropTargetKey: null,
- dropLevelOffset: null,
- dropAllowed: true,
- dragOverNodeKey: null
- });
- }
- dragStartMousePosition = null;
- currentMouseOverDroppableNodeKey = null;
- };
- // if onNodeDragEnd is called, onWindowDragEnd won't be called since stopPropagation() is called
- const onNodeDragEnd = (event, node) => {
- const {
- onDragend
- } = props;
- dragState.dragOverNodeKey = null;
- cleanDragState();
- onDragend === null || onDragend === void 0 ? void 0 : onDragend({
- event,
- node: node.eventData
- });
- dragNode = null;
- };
- // since stopPropagation() is called in treeNode
- // if onWindowDrag is called, whice means state is keeped, drag state should be cleared
- const onWindowDragEnd = event => {
- onNodeDragEnd(event, null, true);
- window.removeEventListener('dragend', onWindowDragEnd);
- };
- const onNodeDragStart = (event, node) => {
- const {
- onDragstart
- } = props;
- const {
- eventKey,
- eventData
- } = node;
- dragNode = node;
- dragStartMousePosition = {
- x: event.clientX,
- y: event.clientY
- };
- const newExpandedKeys = arrDel(expandedKeys.value, eventKey);
- dragState.draggingNodeKey = eventKey;
- dragState.dragChildrenKeys = getDragChildrenKeys(eventKey, keyEntities.value);
- indent.value = listRef.value.getIndentWidth();
- setExpandedKeys(newExpandedKeys);
- window.addEventListener('dragend', onWindowDragEnd);
- if (onDragstart) {
- onDragstart({
- event,
- node: eventData
- });
- }
- };
- /**
- * [Legacy] Select handler is smaller than node,
- * so that this will trigger when drag enter node or select handler.
- * This is a little tricky if customize css without padding.
- * Better for use mouse move event to refresh drag state.
- * But let's just keep it to avoid event trigger logic change.
- */
- const onNodeDragEnter = (event, node) => {
- const {
- onDragenter,
- onExpand,
- allowDrop,
- direction
- } = props;
- const {
- pos,
- eventKey
- } = node;
- // record the key of node which is latest entered, used in dragleave event.
- if (currentMouseOverDroppableNodeKey !== eventKey) {
- currentMouseOverDroppableNodeKey = eventKey;
- }
- if (!dragNode) {
- resetDragState();
- return;
- }
- const {
- dropPosition,
- dropLevelOffset,
- dropTargetKey,
- dropContainerKey,
- dropTargetPos,
- dropAllowed,
- dragOverNodeKey
- } = calcDropPosition(event, dragNode, node, indent.value, dragStartMousePosition, allowDrop, flattenNodes.value, keyEntities.value, expandedKeysSet.value, direction);
- if (
- // don't allow drop inside its children
- dragState.dragChildrenKeys.indexOf(dropTargetKey) !== -1 ||
- // don't allow drop when drop is not allowed caculated by calcDropPosition
- !dropAllowed) {
- resetDragState();
- return;
- }
- // Side effect for delay drag
- if (!delayedDragEnterLogic) {
- delayedDragEnterLogic = {};
- }
- Object.keys(delayedDragEnterLogic).forEach(key => {
- clearTimeout(delayedDragEnterLogic[key]);
- });
- if (dragNode.eventKey !== node.eventKey) {
- // hoist expand logic here
- // since if logic is on the bottom
- // it will be blocked by abstract dragover node check
- // => if you dragenter from top, you mouse will still be consider as in the top node
- delayedDragEnterLogic[pos] = window.setTimeout(() => {
- if (dragState.draggingNodeKey === null) return;
- let newExpandedKeys = expandedKeys.value.slice();
- const entity = keyEntities.value[node.eventKey];
- if (entity && (entity.children || []).length) {
- newExpandedKeys = arrAdd(expandedKeys.value, node.eventKey);
- }
- setExpandedKeys(newExpandedKeys);
- if (onExpand) {
- onExpand(newExpandedKeys, {
- node: node.eventData,
- expanded: true,
- nativeEvent: event
- });
- }
- }, 800);
- }
- // Skip if drag node is self
- if (dragNode.eventKey === dropTargetKey && dropLevelOffset === 0) {
- resetDragState();
- return;
- }
- // Update drag over node and drag state
- _extends(dragState, {
- dragOverNodeKey,
- dropPosition,
- dropLevelOffset,
- dropTargetKey,
- dropContainerKey,
- dropTargetPos,
- dropAllowed
- });
- if (onDragenter) {
- onDragenter({
- event,
- node: node.eventData,
- expandedKeys: expandedKeys.value
- });
- }
- };
- const onNodeDragOver = (event, node) => {
- const {
- onDragover,
- allowDrop,
- direction
- } = props;
- if (!dragNode) {
- return;
- }
- const {
- dropPosition,
- dropLevelOffset,
- dropTargetKey,
- dropContainerKey,
- dropAllowed,
- dropTargetPos,
- dragOverNodeKey
- } = calcDropPosition(event, dragNode, node, indent.value, dragStartMousePosition, allowDrop, flattenNodes.value, keyEntities.value, expandedKeysSet.value, direction);
- if (dragState.dragChildrenKeys.indexOf(dropTargetKey) !== -1 || !dropAllowed) {
- // don't allow drop inside its children
- // don't allow drop when drop is not allowed caculated by calcDropPosition
- return;
- }
- // Update drag position
- if (dragNode.eventKey === dropTargetKey && dropLevelOffset === 0) {
- if (!(dragState.dropPosition === null && dragState.dropLevelOffset === null && dragState.dropTargetKey === null && dragState.dropContainerKey === null && dragState.dropTargetPos === null && dragState.dropAllowed === false && dragState.dragOverNodeKey === null)) {
- resetDragState();
- }
- } else if (!(dropPosition === dragState.dropPosition && dropLevelOffset === dragState.dropLevelOffset && dropTargetKey === dragState.dropTargetKey && dropContainerKey === dragState.dropContainerKey && dropTargetPos === dragState.dropTargetPos && dropAllowed === dragState.dropAllowed && dragOverNodeKey === dragState.dragOverNodeKey)) {
- _extends(dragState, {
- dropPosition,
- dropLevelOffset,
- dropTargetKey,
- dropContainerKey,
- dropTargetPos,
- dropAllowed,
- dragOverNodeKey
- });
- }
- if (onDragover) {
- onDragover({
- event,
- node: node.eventData
- });
- }
- };
- const onNodeDragLeave = (event, node) => {
- // if it is outside the droppable area
- // currentMouseOverDroppableNodeKey will be updated in dragenter event when into another droppable receiver.
- if (currentMouseOverDroppableNodeKey === node.eventKey && !event.currentTarget.contains(event.relatedTarget)) {
- resetDragState();
- currentMouseOverDroppableNodeKey = null;
- }
- const {
- onDragleave
- } = props;
- if (onDragleave) {
- onDragleave({
- event,
- node: node.eventData
- });
- }
- };
- const onNodeDrop = function (event, _node) {
- let outsideTree = arguments.length > 2 && arguments[2] !== undefined ? arguments[2] : false;
- var _a;
- const {
- dragChildrenKeys,
- dropPosition,
- dropTargetKey,
- dropTargetPos,
- dropAllowed
- } = dragState;
- if (!dropAllowed) return;
- const {
- onDrop
- } = props;
- dragState.dragOverNodeKey = null;
- cleanDragState();
- if (dropTargetKey === null) return;
- const abstractDropNodeProps = _extends(_extends({}, getTreeNodeProps(dropTargetKey, toRaw(treeNodeRequiredProps.value))), {
- active: ((_a = activeItem.value) === null || _a === void 0 ? void 0 : _a.key) === dropTargetKey,
- data: keyEntities.value[dropTargetKey].node
- });
- const dropToChild = dragChildrenKeys.indexOf(dropTargetKey) !== -1;
- warning(!dropToChild, "Can not drop to dragNode's children node. Maybe this is a bug of ant-design-vue. Please report an issue.");
- const posArr = posToArr(dropTargetPos);
- const dropResult = {
- event,
- node: convertNodePropsToEventData(abstractDropNodeProps),
- dragNode: dragNode ? dragNode.eventData : null,
- dragNodesKeys: [dragNode.eventKey].concat(dragChildrenKeys),
- dropToGap: dropPosition !== 0,
- dropPosition: dropPosition + Number(posArr[posArr.length - 1])
- };
- if (!outsideTree) {
- onDrop === null || onDrop === void 0 ? void 0 : onDrop(dropResult);
- }
- dragNode = null;
- };
- const triggerExpandActionExpand = (e, treeNode) => {
- const {
- expanded,
- key
- } = treeNode;
- const node = flattenNodes.value.filter(nodeItem => nodeItem.key === key)[0];
- const eventNode = convertNodePropsToEventData(_extends(_extends({}, getTreeNodeProps(key, treeNodeRequiredProps.value)), {
- data: node.data
- }));
- setExpandedKeys(expanded ? arrDel(expandedKeys.value, key) : arrAdd(expandedKeys.value, key));
- onNodeExpand(e, eventNode);
- };
- const onNodeClick = (e, treeNode) => {
- const {
- onClick,
- expandAction
- } = props;
- if (expandAction === 'click') {
- triggerExpandActionExpand(e, treeNode);
- }
- if (onClick) {
- onClick(e, treeNode);
- }
- };
- const onNodeDoubleClick = (e, treeNode) => {
- const {
- onDblclick,
- expandAction
- } = props;
- if (expandAction === 'doubleclick' || expandAction === 'dblclick') {
- triggerExpandActionExpand(e, treeNode);
- }
- if (onDblclick) {
- onDblclick(e, treeNode);
- }
- };
- const onNodeSelect = (e, treeNode) => {
- let newSelectedKeys = selectedKeys.value;
- const {
- onSelect,
- multiple
- } = props;
- const {
- selected
- } = treeNode;
- const key = treeNode[fieldNames.value.key];
- const targetSelected = !selected;
- // Update selected keys
- if (!targetSelected) {
- newSelectedKeys = arrDel(newSelectedKeys, key);
- } else if (!multiple) {
- newSelectedKeys = [key];
- } else {
- newSelectedKeys = arrAdd(newSelectedKeys, key);
- }
- // [Legacy] Not found related usage in doc or upper libs
- const keyEntitiesValue = keyEntities.value;
- const selectedNodes = newSelectedKeys.map(selectedKey => {
- const entity = keyEntitiesValue[selectedKey];
- if (!entity) return null;
- return entity.node;
- }).filter(node => node);
- if (props.selectedKeys === undefined) {
- selectedKeys.value = newSelectedKeys;
- }
- if (onSelect) {
- onSelect(newSelectedKeys, {
- event: 'select',
- selected: targetSelected,
- node: treeNode,
- selectedNodes,
- nativeEvent: e
- });
- }
- };
- const onNodeCheck = (e, treeNode, checked) => {
- const {
- checkStrictly,
- onCheck
- } = props;
- const key = treeNode[fieldNames.value.key];
- // Prepare trigger arguments
- let checkedObj;
- const eventObj = {
- event: 'check',
- node: treeNode,
- checked,
- nativeEvent: e
- };
- const keyEntitiesValue = keyEntities.value;
- if (checkStrictly) {
- const newCheckedKeys = checked ? arrAdd(checkedKeys.value, key) : arrDel(checkedKeys.value, key);
- const newHalfCheckedKeys = arrDel(halfCheckedKeys.value, key);
- checkedObj = {
- checked: newCheckedKeys,
- halfChecked: newHalfCheckedKeys
- };
- eventObj.checkedNodes = newCheckedKeys.map(checkedKey => keyEntitiesValue[checkedKey]).filter(entity => entity).map(entity => entity.node);
- if (props.checkedKeys === undefined) {
- checkedKeys.value = newCheckedKeys;
- }
- } else {
- // Always fill first
- let {
- checkedKeys: newCheckedKeys,
- halfCheckedKeys: newHalfCheckedKeys
- } = conductCheck([...checkedKeys.value, key], true, keyEntitiesValue, maxLevel.value, levelEntities.value);
- // If remove, we do it again to correction
- if (!checked) {
- const keySet = new Set(newCheckedKeys);
- keySet.delete(key);
- ({
- checkedKeys: newCheckedKeys,
- halfCheckedKeys: newHalfCheckedKeys
- } = conductCheck(Array.from(keySet), {
- checked: false,
- halfCheckedKeys: newHalfCheckedKeys
- }, keyEntitiesValue, maxLevel.value, levelEntities.value));
- }
- checkedObj = newCheckedKeys;
- // [Legacy] This is used for vc-tree-select`
- eventObj.checkedNodes = [];
- eventObj.checkedNodesPositions = [];
- eventObj.halfCheckedKeys = newHalfCheckedKeys;
- newCheckedKeys.forEach(checkedKey => {
- const entity = keyEntitiesValue[checkedKey];
- if (!entity) return;
- const {
- node,
- pos
- } = entity;
- eventObj.checkedNodes.push(node);
- eventObj.checkedNodesPositions.push({
- node,
- pos
- });
- });
- if (props.checkedKeys === undefined) {
- checkedKeys.value = newCheckedKeys;
- halfCheckedKeys.value = newHalfCheckedKeys;
- }
- }
- if (onCheck) {
- onCheck(checkedObj, eventObj);
- }
- };
- const onNodeLoad = treeNode => {
- const key = treeNode[fieldNames.value.key];
- const loadPromise = new Promise((resolve, reject) => {
- // We need to get the latest state of loading/loaded keys
- const {
- loadData,
- onLoad
- } = props;
- if (!loadData || loadedKeysSet.value.has(key) || loadingKeysSet.value.has(key)) {
- return null;
- }
- // Process load data
- const promise = loadData(treeNode);
- promise.then(() => {
- const newLoadedKeys = arrAdd(loadedKeys.value, key);
- const newLoadingKeys = arrDel(loadingKeys.value, key);
- // onLoad should trigger before internal setState to avoid `loadData` trigger twice.
- // https://github.com/ant-design/ant-design/issues/12464
- if (onLoad) {
- onLoad(newLoadedKeys, {
- event: 'load',
- node: treeNode
- });
- }
- if (props.loadedKeys === undefined) {
- loadedKeys.value = newLoadedKeys;
- }
- loadingKeys.value = newLoadingKeys;
- resolve();
- }).catch(e => {
- const newLoadingKeys = arrDel(loadingKeys.value, key);
- loadingKeys.value = newLoadingKeys;
- // If exceed max retry times, we give up retry
- loadingRetryTimes[key] = (loadingRetryTimes[key] || 0) + 1;
- if (loadingRetryTimes[key] >= MAX_RETRY_TIMES) {
- warning(false, 'Retry for `loadData` many times but still failed. No more retry.');
- const newLoadedKeys = arrAdd(loadedKeys.value, key);
- if (props.loadedKeys === undefined) {
- loadedKeys.value = newLoadedKeys;
- }
- resolve();
- }
- reject(e);
- });
- loadingKeys.value = arrAdd(loadingKeys.value, key);
- });
- // Not care warning if we ignore this
- loadPromise.catch(() => {});
- return loadPromise;
- };
- const onNodeMouseEnter = (event, node) => {
- const {
- onMouseenter
- } = props;
- if (onMouseenter) {
- onMouseenter({
- event,
- node
- });
- }
- };
- const onNodeMouseLeave = (event, node) => {
- const {
- onMouseleave
- } = props;
- if (onMouseleave) {
- onMouseleave({
- event,
- node
- });
- }
- };
- const onNodeContextMenu = (event, node) => {
- const {
- onRightClick
- } = props;
- if (onRightClick) {
- event.preventDefault();
- onRightClick({
- event,
- node
- });
- }
- };
- const onFocus = e => {
- const {
- onFocus
- } = props;
- focused.value = true;
- if (onFocus) {
- onFocus(e);
- }
- };
- const onBlur = e => {
- const {
- onBlur
- } = props;
- focused.value = false;
- onActiveChange(null);
- if (onBlur) {
- onBlur(e);
- }
- };
- const onNodeExpand = (e, treeNode) => {
- let newExpandedKeys = expandedKeys.value;
- const {
- onExpand,
- loadData
- } = props;
- const {
- expanded
- } = treeNode;
- const key = treeNode[fieldNames.value.key];
- // Do nothing when motion is in progress
- if (listChanging.value) {
- return;
- }
- // Update selected keys
- const index = newExpandedKeys.indexOf(key);
- const targetExpanded = !expanded;
- warning(expanded && index !== -1 || !expanded && index === -1, 'Expand state not sync with index check');
- if (targetExpanded) {
- newExpandedKeys = arrAdd(newExpandedKeys, key);
- } else {
- newExpandedKeys = arrDel(newExpandedKeys, key);
- }
- setExpandedKeys(newExpandedKeys);
- if (onExpand) {
- onExpand(newExpandedKeys, {
- node: treeNode,
- expanded: targetExpanded,
- nativeEvent: e
- });
- }
- // Async Load data
- if (targetExpanded && loadData) {
- const loadPromise = onNodeLoad(treeNode);
- if (loadPromise) {
- loadPromise.then(() => {
- // [Legacy] Refresh logic
- // const newFlattenTreeData = flattenTreeData(
- // treeData.value,
- // newExpandedKeys,
- // fieldNames.value,
- // );
- // flattenNodes.value = newFlattenTreeData;
- }).catch(e => {
- const expandedKeysToRestore = arrDel(expandedKeys.value, key);
- setExpandedKeys(expandedKeysToRestore);
- Promise.reject(e);
- });
- }
- }
- };
- const onListChangeStart = () => {
- listChanging.value = true;
- };
- const onListChangeEnd = () => {
- setTimeout(() => {
- listChanging.value = false;
- });
- };
- // =========================== Keyboard ===========================
- const onActiveChange = newActiveKey => {
- const {
- onActiveChange
- } = props;
- if (activeKey.value === newActiveKey) {
- return;
- }
- if (props.activeKey !== undefined) {
- activeKey.value = newActiveKey;
- }
- if (newActiveKey !== null) {
- scrollTo({
- key: newActiveKey
- });
- }
- if (onActiveChange) {
- onActiveChange(newActiveKey);
- }
- };
- const activeItem = computed(() => {
- if (activeKey.value === null) {
- return null;
- }
- return flattenNodes.value.find(_ref4 => {
- let {
- key
- } = _ref4;
- return key === activeKey.value;
- }) || null;
- });
- const offsetActiveKey = offset => {
- let index = flattenNodes.value.findIndex(_ref5 => {
- let {
- key
- } = _ref5;
- return key === activeKey.value;
- });
- // Align with index
- if (index === -1 && offset < 0) {
- index = flattenNodes.value.length;
- }
- index = (index + offset + flattenNodes.value.length) % flattenNodes.value.length;
- const item = flattenNodes.value[index];
- if (item) {
- const {
- key
- } = item;
- onActiveChange(key);
- } else {
- onActiveChange(null);
- }
- };
- const activeItemEventNode = computed(() => {
- return convertNodePropsToEventData(_extends(_extends({}, getTreeNodeProps(activeKey.value, treeNodeRequiredProps.value)), {
- data: activeItem.value.data,
- active: true
- }));
- });
- const onKeydown = event => {
- const {
- onKeydown,
- checkable,
- selectable
- } = props;
- // >>>>>>>>>> Direction
- switch (event.which) {
- case KeyCode.UP:
- {
- offsetActiveKey(-1);
- event.preventDefault();
- break;
- }
- case KeyCode.DOWN:
- {
- offsetActiveKey(1);
- event.preventDefault();
- break;
- }
- }
- // >>>>>>>>>> Expand & Selection
- const item = activeItem.value;
- if (item && item.data) {
- const expandable = item.data.isLeaf === false || !!(item.data.children || []).length;
- const eventNode = activeItemEventNode.value;
- switch (event.which) {
- // >>> Expand
- case KeyCode.LEFT:
- {
- // Collapse if possible
- if (expandable && expandedKeysSet.value.has(activeKey.value)) {
- onNodeExpand({}, eventNode);
- } else if (item.parent) {
- onActiveChange(item.parent.key);
- }
- event.preventDefault();
- break;
- }
- case KeyCode.RIGHT:
- {
- // Expand if possible
- if (expandable && !expandedKeysSet.value.has(activeKey.value)) {
- onNodeExpand({}, eventNode);
- } else if (item.children && item.children.length) {
- onActiveChange(item.children[0].key);
- }
- event.preventDefault();
- break;
- }
- // Selection
- case KeyCode.ENTER:
- case KeyCode.SPACE:
- {
- if (checkable && !eventNode.disabled && eventNode.checkable !== false && !eventNode.disableCheckbox) {
- onNodeCheck({}, eventNode, !checkedKeysSet.value.has(activeKey.value));
- } else if (!checkable && selectable && !eventNode.disabled && eventNode.selectable !== false) {
- onNodeSelect({}, eventNode);
- }
- break;
- }
- }
- }
- if (onKeydown) {
- onKeydown(event);
- }
- };
- expose({
- onNodeExpand,
- scrollTo,
- onKeydown,
- selectedKeys: computed(() => selectedKeys.value),
- checkedKeys: computed(() => checkedKeys.value),
- halfCheckedKeys: computed(() => halfCheckedKeys.value),
- loadedKeys: computed(() => loadedKeys.value),
- loadingKeys: computed(() => loadingKeys.value),
- expandedKeys: computed(() => expandedKeys.value)
- });
- onUnmounted(() => {
- window.removeEventListener('dragend', onWindowDragEnd);
- destroyed.value = true;
- });
- useProvideKeysState({
- expandedKeys,
- selectedKeys,
- loadedKeys,
- loadingKeys,
- checkedKeys,
- halfCheckedKeys,
- expandedKeysSet,
- selectedKeysSet,
- loadedKeysSet,
- loadingKeysSet,
- checkedKeysSet,
- halfCheckedKeysSet,
- flattenNodes
- });
- return () => {
- const {
- // focused,
- // flattenNodes,
- // keyEntities,
- draggingNodeKey,
- // activeKey,
- dropLevelOffset,
- dropContainerKey,
- dropTargetKey,
- dropPosition,
- dragOverNodeKey
- // indent,
- } = dragState;
- const {
- prefixCls,
- showLine,
- focusable,
- tabindex = 0,
- selectable,
- showIcon,
- icon = slots.icon,
- switcherIcon,
- draggable,
- checkable,
- checkStrictly,
- disabled,
- motion,
- loadData,
- filterTreeNode,
- height,
- itemHeight,
- virtual,
- dropIndicatorRender,
- onContextmenu,
- onScroll,
- direction,
- rootClassName,
- rootStyle
- } = props;
- const {
- class: className,
- style
- } = attrs;
- const domProps = pickAttrs(_extends(_extends({}, props), attrs), {
- aria: true,
- data: true
- });
- // It's better move to hooks but we just simply keep here
- let draggableConfig;
- if (draggable) {
- if (typeof draggable === 'object') {
- draggableConfig = draggable;
- } else if (typeof draggable === 'function') {
- draggableConfig = {
- nodeDraggable: draggable
- };
- } else {
- draggableConfig = {};
- }
- } else {
- draggableConfig = false;
- }
- return _createVNode(TreeContext, {
- "value": {
- prefixCls,
- selectable,
- showIcon,
- icon,
- switcherIcon,
- draggable: draggableConfig,
- draggingNodeKey,
- checkable,
- customCheckable: slots.checkable,
- checkStrictly,
- disabled,
- keyEntities: keyEntities.value,
- dropLevelOffset,
- dropContainerKey,
- dropTargetKey,
- dropPosition,
- dragOverNodeKey,
- dragging: draggingNodeKey !== null,
- indent: indent.value,
- direction,
- dropIndicatorRender,
- loadData,
- filterTreeNode,
- onNodeClick,
- onNodeDoubleClick,
- onNodeExpand,
- onNodeSelect,
- onNodeCheck,
- onNodeLoad,
- onNodeMouseEnter,
- onNodeMouseLeave,
- onNodeContextMenu,
- onNodeDragStart,
- onNodeDragEnter,
- onNodeDragOver,
- onNodeDragLeave,
- onNodeDragEnd,
- onNodeDrop,
- slots
- }
- }, {
- default: () => [_createVNode("div", {
- "role": "tree",
- "class": classNames(prefixCls, className, rootClassName, {
- [`${prefixCls}-show-line`]: showLine,
- [`${prefixCls}-focused`]: focused.value,
- [`${prefixCls}-active-focused`]: activeKey.value !== null
- }),
- "style": rootStyle
- }, [_createVNode(NodeList, _objectSpread({
- "ref": listRef,
- "prefixCls": prefixCls,
- "style": style,
- "disabled": disabled,
- "selectable": selectable,
- "checkable": !!checkable,
- "motion": motion,
- "height": height,
- "itemHeight": itemHeight,
- "virtual": virtual,
- "focusable": focusable,
- "focused": focused.value,
- "tabindex": tabindex,
- "activeItem": activeItem.value,
- "onFocus": onFocus,
- "onBlur": onBlur,
- "onKeydown": onKeydown,
- "onActiveChange": onActiveChange,
- "onListChangeStart": onListChangeStart,
- "onListChangeEnd": onListChangeEnd,
- "onContextmenu": onContextmenu,
- "onScroll": onScroll
- }, domProps), null)])]
- });
- };
- }
- });
|