| 123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208209210211212213214215216217218219220221222223224225226227228229230231232233234235236237238239240241242243244245246247248249250251252253254255256257258259260261262263264265266267268269270271272273274275276277278279280281282283284285286287288289290291292293294295296297298299300301302303304305306307308309310311312313314315316317318319320321322323324325326327328329330331332333334335336337338339340341342343344345346347348349350351352353354355356357358359360361362363364365366367368369370371372373374375376377378379380381382383384385 | import BaseComponent, { BaseProps } from '../_base/baseComponent';import React from 'react';import PropTypes from 'prop-types';import cls from 'classnames';import '@douyinfe/semi-foundation/navigation/navigation.scss';import isNullOrUndefined from '@douyinfe/semi-foundation/utils/isNullOrUndefined';import SubNavFoundation, { SubNavAdapter } from '@douyinfe/semi-foundation/navigation/subNavFoundation';import { strings, numbers, cssClasses } from '@douyinfe/semi-foundation/navigation/constants';import { IconChevronDown, IconChevronUp, IconChevronRight } from '@douyinfe/semi-icons';import NavItem from './Item';import Dropdown, { DropdownProps } from '../dropdown';import NavContext, { NavContextType } from './nav-context';import { times, get, isNumber, isString } from 'lodash';import Collapsible from "../collapsible";import CSSAnimation from "../_cssAnimation";export interface SubNavProps extends BaseProps {    disabled?: boolean;    dropdownStyle?: React.CSSProperties;    icon?: React.ReactNode;    indent?: boolean | number;    isCollapsed?: boolean;    isOpen?: boolean;    itemKey?: string | number;    level?: number;    maxHeight?: number;    onMouseEnter?: React.MouseEventHandler<HTMLLIElement>;    onMouseLeave?: React.MouseEventHandler<HTMLLIElement>;    text?: React.ReactNode;    expandIcon?: React.ReactNode;    dropdownProps?: DropdownProps;    subDropdownProps?: DropdownProps}export interface SubNavState {    isHovered: boolean}export default class SubNav extends BaseComponent<SubNavProps, SubNavState> {    static contextType = NavContext;    static propTypes = {        /**         * Unique identification         */        itemKey: PropTypes.oneOfType([PropTypes.string, PropTypes.number]),        /**         * Copywriting         */        text: PropTypes.oneOfType([PropTypes.string, PropTypes.node]),        /**         * Whether child navigation is expanded         */        isOpen: PropTypes.bool,        /**         * Whether it is in the state of being stowed to the sidebar         */        isCollapsed: PropTypes.bool,        /**         * Whether to keep the left Icon placeholder         */        indent: PropTypes.oneOfType([PropTypes.bool, PropTypes.number]),        /**         * Nested child elements         */        children: PropTypes.node,        style: PropTypes.object,        /**         * Icon name on the left         */        icon: PropTypes.node,        /**         * Maximum height (for animation)         */        maxHeight: PropTypes.number,        onMouseEnter: PropTypes.func,        onMouseLeave: PropTypes.func,        // Is it disabled        disabled: PropTypes.bool,        level: PropTypes.number    };    static defaultProps = {        level: 0,        indent: false,        isCollapsed: false,        isOpen: false,        maxHeight: numbers.DEFAULT_SUBNAV_MAX_HEIGHT,        disabled: false,    };    titleRef: React.RefObject<HTMLDivElement>;    itemRef: React.RefObject<HTMLLIElement>;    foundation: SubNavFoundation;    context: NavContextType;    constructor(props: SubNavProps) {        super(props);        this.state = {            isHovered: false,        };        this.adapter.setCache('firstMounted', true);        this.titleRef = React.createRef();        this.itemRef = React.createRef();        this.foundation = new SubNavFoundation(this.adapter);    }    setItemRef = (ref: HTMLLIElement | React.RefObject<HTMLLIElement>) => {        if (ref && (ref as React.RefObject<HTMLLIElement>).current) {            this.itemRef = ref as React.RefObject<HTMLLIElement>;        } else {            this.itemRef = { current: ref as HTMLLIElement };        }    };    setTitleRef = (ref: HTMLDivElement | React.RefObject<HTMLDivElement>) => {        if (ref && (ref as React.RefObject<HTMLDivElement>).current) {            this.titleRef = ref as React.RefObject<HTMLDivElement>;        } else {            this.titleRef = { current: ref as HTMLDivElement };        }    };    _invokeContextFunc(funcName: string, ...args: any[]) {        if (funcName && this.context && typeof this.context[funcName] === 'function') {            return this.context[funcName](...args);        }        return null;    }    get adapter(): SubNavAdapter<SubNavProps, SubNavState> {        return {            ...super.adapter,            updateIsHovered: isHovered => this.setState({ isHovered }),            getOpenKeys: () => this.context && this.context.openKeys,            getOpenKeysIsControlled: () => this.context && this.context.openKeysIsControlled,            getCanUpdateOpenKeys: () => this.context && this.context.canUpdateOpenKeys,            updateOpen: isOpen =>                this._invokeContextFunc(isOpen ? 'addOpenKeys' : 'removeOpenKeys', this.props.itemKey),            notifyGlobalOpenChange: (...args) => this._invokeContextFunc('onOpenChange', ...args),            notifyGlobalOnSelect: (...args) => this._invokeContextFunc('onSelect', ...args),            notifyGlobalOnClick: (...args) => this._invokeContextFunc('onClick', ...args),            getIsSelected: itemKey => Boolean(!isNullOrUndefined(itemKey) && get(this.context, 'selectedKeys', []).includes(String(itemKey))),            getIsOpen: () => {                const { itemKey } = this.props;                return Boolean(this.context && this.context.openKeys && this.context.openKeys.includes(this.props.itemKey));            }        };    }    handleClick = (e: React.MouseEvent) => {        this.foundation.handleClick(e && e.nativeEvent, this.titleRef && this.titleRef.current);    };    handleKeyPress = (e: React.KeyboardEvent) => {        this.foundation.handleKeyPress(e && e.nativeEvent, this.titleRef && this.titleRef.current);    }    handleDropdownVisible = (visible: boolean) => this.foundation.handleDropdownVisibleChange(visible);    renderIcon(icon: React.ReactNode, pos: string, withTransition?: boolean, isToggleIcon = false, key: number | string = 0) {        const { prefixCls } = this.context;        let iconSize = 'large';        if (pos === strings.ICON_POS_RIGHT) {            iconSize = 'default';        }        const className = cls(`${prefixCls}-item-icon`, {            [`${prefixCls}-item-icon-toggle-${this.context.toggleIconPosition}`]: isToggleIcon,            [`${prefixCls}-item-icon-info`]: !isToggleIcon        });        const isOpen = this.adapter.getIsOpen();        const iconElem = React.isValidElement(icon) ? (withTransition ? (            <CSSAnimation animationState={isOpen ? "enter" : "leave"} startClassName={`${cssClasses.PREFIX}-icon-rotate-${isOpen ? "180" : "0"}`}>                {({ animationClassName }) => {                    // @ts-ignore                    return React.cloneElement(icon, { size: iconSize, className: animationClassName });                }}            </CSSAnimation>            // @ts-ignore        ) : React.cloneElement(icon, { size: iconSize })) : null;        return <i key={key} className={className}>{iconElem}</i>;    }    renderTitleDiv() {        const { text, icon, itemKey, indent, disabled, level, expandIcon } = this.props;        const { mode, isInSubNav, isCollapsed, prefixCls, subNavMotion, limitIndent } = this.context;        const isOpen = this.adapter.getIsOpen();        const titleCls = cls(`${prefixCls}-sub-title`, {            [`${prefixCls}-sub-title-selected`]: this.adapter.getIsSelected(itemKey),            [`${prefixCls}-sub-title-disabled`]: disabled,        });        let withTransition = false;        let toggleIconType: React.ReactNode = '';        if (isCollapsed) {            if (isInSubNav) {                toggleIconType = <IconChevronRight />;            } else {                toggleIconType = null;            }        } else if (mode === strings.MODE_HORIZONTAL) {            if (isInSubNav) {                toggleIconType = <IconChevronRight aria-hidden={true} />;            } else {                toggleIconType = expandIcon ? expandIcon : <IconChevronDown aria-hidden={true} />;                // Horizontal mode does not require animation fix#1198                // withTransition = true;            }        } else {            if (subNavMotion) {                withTransition = true;            }            toggleIconType = expandIcon ? expandIcon : <IconChevronDown aria-hidden={true} />;        }        let placeholderIcons = null;        if (mode === strings.MODE_VERTICAL && !limitIndent && !isCollapsed) {            /* Different icons' amount means different indents.*/            const iconAmount = (icon && !indent) ? level : level - 1;            placeholderIcons = times(iconAmount, index => this.renderIcon(null, strings.ICON_POS_RIGHT, false, false, index));        }        const isIconChevronRightShow = (!isCollapsed && isInSubNav && mode === strings.MODE_HORIZONTAL) || (isCollapsed && isInSubNav);        const titleDiv = (            <div                role="menuitem"                // to avoid nested horizontal navigation be focused                tabIndex={isIconChevronRightShow ? -1 : 0}                ref={this.setTitleRef as any}                className={titleCls}                onClick={this.handleClick}                onKeyPress={this.handleKeyPress}                aria-expanded={isOpen ? 'true' : 'false'}            >                <div className={`${prefixCls}-item-inner`}>                    {placeholderIcons}                    {this.context.toggleIconPosition === strings.TOGGLE_ICON_LEFT && this.renderIcon(toggleIconType, strings.ICON_POS_RIGHT, withTransition, true, 'key-toggle-position-left')}                    {icon || indent || (isInSubNav && mode !== strings.MODE_HORIZONTAL)                        ? this.renderIcon(icon, strings.ICON_POS_LEFT, false, false, 'key-inSubNav-position-left')                        : null}                    <span className={`${prefixCls}-item-text`}>{text}</span>                    {this.context.toggleIconPosition === strings.TOGGLE_ICON_RIGHT && this.renderIcon(toggleIconType, strings.ICON_POS_RIGHT, withTransition, true, 'key-toggle-position-right')}                </div>            </div>        );        return titleDiv;    }    renderSubUl() {        const { children, maxHeight } = this.props;        const { isCollapsed, mode, subNavMotion, prefixCls } = this.context;        const isOpen = this.adapter.getIsOpen();        const isHorizontal = mode === strings.MODE_HORIZONTAL;        const subNavCls = cls(`${prefixCls}-sub`, {            [`${prefixCls}-sub-open`]: isOpen,            [`${prefixCls}-sub-popover`]: isCollapsed || isHorizontal,        });        const ulWithMotion = <Collapsible motion={subNavMotion} isOpen={isOpen} keepDOM={false} fade={true}>            {                !isCollapsed ? <ul                    className={subNavCls}                >                    {children}                </ul> : null            }        </Collapsible>;        const finalDom = isHorizontal ? null : subNavMotion ? (            ulWithMotion        ) : isOpen && !isCollapsed ? (            <ul className={subNavCls}>{children}</ul>        ) : null;        return finalDom;    }    wrapDropdown(elem: React.ReactNode = '') {        let _elem: React.ReactNode = elem;        const { children, dropdownStyle, disabled, subDropdownProps, dropdownProps: userDropdownProps } = this.props;        const { mode, isInSubNav, isCollapsed, subNavCloseDelay, subNavOpenDelay, prefixCls, getPopupContainer } = this.context;        const isOpen = this.adapter.getIsOpen();        const openKeysIsControlled = this.adapter.getOpenKeysIsControlled();        const subNavCls = cls({            [`${prefixCls}-popover`]: isCollapsed,        });        const dropdownProps: DropdownProps = {            trigger: 'hover',            style: dropdownStyle,        };        if (openKeysIsControlled) {            dropdownProps.trigger = 'custom';            dropdownProps.visible = isOpen;        }        if (getPopupContainer) {            dropdownProps.getPopupContainer = getPopupContainer;        }        if (isCollapsed || mode === strings.MODE_HORIZONTAL) {            // Do not show dropdown when disabled            _elem = !disabled ? (                <Dropdown                    className={subNavCls}                    render={(                        <Dropdown.Menu>                            {/* <li className={`${prefixCls}-popover-crumb`} /> */}                            {children}                        </Dropdown.Menu>                    )}                    position={mode === strings.MODE_HORIZONTAL && !isInSubNav ? 'bottomLeft' : 'rightTop'}                    mouseEnterDelay={subNavOpenDelay}                    mouseLeaveDelay={subNavCloseDelay}                    onVisibleChange={this.handleDropdownVisible}                    {...(userDropdownProps ? userDropdownProps : subDropdownProps)}                    {...dropdownProps}                >                    {_elem}                </Dropdown>            ) : _elem;        }        return _elem;    }    render() {        const { itemKey, style, onMouseEnter, onMouseLeave, disabled, text } = this.props;        const { mode, isCollapsed, prefixCls } = this.context;        let titleDiv: React.ReactNode = this.renderTitleDiv();        const subUl = this.renderSubUl();        // When mode=horizontal, it is displayed in Dropdown        if (isCollapsed || mode === strings.MODE_HORIZONTAL) {            titleDiv = this.wrapDropdown(titleDiv);        }        return (            // Children is not a recommended usage and may cause some bug-like performance, but some users have already used it, so here we only delete the ts definition instead of deleting the actual code            // children 并不是我们推荐的用法,可能会导致一些像 bug的表现,但是有些用户已经用了,所以此处仅作删除 ts 定义而非删除实际代码的操作            // refer https://github.com/DouyinFE/semi-design/issues/2710            // @ts-ignore                <NavItem                style={style}                isSubNav={true}                itemKey={itemKey}                forwardRef={this.setItemRef}                isCollapsed={isCollapsed}                className={`${prefixCls}-sub-wrap`}                onMouseEnter={onMouseEnter}                onMouseLeave={onMouseLeave}                disabled={disabled}                text={text}            >                   <NavContext.Provider value={{ ...this.context, isInSubNav: true }}>                    {titleDiv}                    {subUl}                </NavContext.Provider>            </NavItem>        );    }}
 |