import * as React from 'react'; import classNames from 'classnames'; import omit from 'omit.js'; import debounce from 'lodash/debounce'; import { conductExpandParent, convertTreeToEntities } from 'rc-tree/lib/util'; import { ConfigConsumer, ConfigConsumerProps } from '../config-provider'; import Tree, { TreeProps, AntdTreeNodeAttribute, AntTreeNodeExpandedEvent, AntTreeNodeSelectedEvent, AntTreeNode, } from './Tree'; import { calcRangeKeys, getFullKeyList } from './util'; import Icon from '../icon'; export type ExpandAction = false | 'click' | 'doubleClick'; export interface DirectoryTreeProps extends TreeProps { expandAction?: ExpandAction; } export interface DirectoryTreeState { expandedKeys?: string[]; selectedKeys?: string[]; } function getIcon(props: AntdTreeNodeAttribute): React.ReactNode { const { isLeaf, expanded } = props; if (isLeaf) { return ; } return ; } export default class DirectoryTree extends React.Component { static defaultProps = { showIcon: true, expandAction: 'click', }; state: DirectoryTreeState; tree: Tree; onDebounceExpand: (event: React.MouseEvent, node: AntTreeNode) => void; // Shift click usage lastSelectedKey?: string; cachedSelectedKeys?: string[]; constructor(props: DirectoryTreeProps) { super(props); const { defaultExpandAll, defaultExpandParent, expandedKeys, defaultExpandedKeys, children, } = props; const { keyEntities } = convertTreeToEntities(children); // Selected keys this.state = { selectedKeys: props.selectedKeys || props.defaultSelectedKeys || [], }; // Expanded keys if (defaultExpandAll) { this.state.expandedKeys = getFullKeyList(props.children); } else if (defaultExpandParent) { this.state.expandedKeys = conductExpandParent( expandedKeys || defaultExpandedKeys, keyEntities, ); } else { this.state.expandedKeys = expandedKeys || defaultExpandedKeys; } this.onDebounceExpand = debounce(this.expandFolderNode, 200, { leading: true, }); } componentWillReceiveProps(nextProps: DirectoryTreeProps) { if ('expandedKeys' in nextProps) { this.setState({ expandedKeys: nextProps.expandedKeys }); } if ('selectedKeys' in nextProps) { this.setState({ selectedKeys: nextProps.selectedKeys }); } } onExpand = (expandedKeys: string[], info: AntTreeNodeExpandedEvent) => { const { onExpand } = this.props; this.setUncontrolledState({ expandedKeys }); // Call origin function if (onExpand) { return onExpand(expandedKeys, info); } return undefined; }; onClick = (event: React.MouseEvent, node: AntTreeNode) => { const { onClick, expandAction } = this.props; // Expand the tree if (expandAction === 'click') { this.onDebounceExpand(event, node); } if (onClick) { onClick(event, node); } }; onDoubleClick = (event: React.MouseEvent, node: AntTreeNode) => { const { onDoubleClick, expandAction } = this.props; // Expand the tree if (expandAction === 'doubleClick') { this.onDebounceExpand(event, node); } if (onDoubleClick) { onDoubleClick(event, node); } }; onSelect = (keys: string[], event: AntTreeNodeSelectedEvent) => { const { onSelect, multiple, children } = this.props; const { expandedKeys = [] } = this.state; const { node, nativeEvent } = event; const { eventKey = '' } = node.props; const newState: DirectoryTreeState = {}; // Windows / Mac single pick const ctrlPick: boolean = nativeEvent.ctrlKey || nativeEvent.metaKey; const shiftPick: boolean = nativeEvent.shiftKey; // Generate new selected keys let newSelectedKeys: string[]; if (multiple && ctrlPick) { // Control click newSelectedKeys = keys; this.lastSelectedKey = eventKey; this.cachedSelectedKeys = newSelectedKeys; } else if (multiple && shiftPick) { // Shift click newSelectedKeys = Array.from( new Set([ ...(this.cachedSelectedKeys || []), ...calcRangeKeys(children, expandedKeys, eventKey, this.lastSelectedKey), ]), ); } else { // Single click newSelectedKeys = [eventKey]; this.lastSelectedKey = eventKey; this.cachedSelectedKeys = newSelectedKeys; } newState.selectedKeys = newSelectedKeys; if (onSelect) { onSelect(newSelectedKeys, event); } this.setUncontrolledState(newState); }; setTreeRef = (node: Tree) => { this.tree = node; }; expandFolderNode = (event: React.MouseEvent, node: AntTreeNode) => { const { isLeaf } = node.props; if (isLeaf || event.shiftKey || event.metaKey || event.ctrlKey) { return; } // Get internal rc-tree const internalTree = this.tree.tree; // Call internal rc-tree expand function // https://github.com/ant-design/ant-design/issues/12567 internalTree.onNodeExpand(event, node); }; setUncontrolledState = (state: DirectoryTreeState) => { const newState = omit(state, Object.keys(this.props)); if (Object.keys(newState).length) { this.setState(newState); } }; renderDirectoryTree = ({ getPrefixCls }: ConfigConsumerProps) => { const { prefixCls: customizePrefixCls, className, ...props } = this.props; const { expandedKeys, selectedKeys } = this.state; const prefixCls = getPrefixCls('tree', customizePrefixCls); const connectClassName = classNames(`${prefixCls}-directory`, className); return ( ); }; render() { return {this.renderDirectoryTree}; } }