| 123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208209210211 | import React, { CSSProperties, ReactNode } from 'react';import PropTypes from 'prop-types';import cls from 'classnames';import { cssClasses } from '@douyinfe/semi-foundation/sideSheet/constants';import Button from '../iconButton';import { noop } from 'lodash';import { IconClose } from '@douyinfe/semi-icons';import { SideSheetProps } from "@douyinfe/semi-foundation/sideSheet/sideSheetFoundation";import getDataAttr from '@douyinfe/semi-foundation/utils/getDataAttr';let uuid = 0;const prefixCls = cssClasses.PREFIX;export interface SideSheetContentProps {    onClose?: (e: React.MouseEvent) => void;    closeIcon?: ReactNode;    mask?: boolean;    maskStyle?: CSSProperties;    maskClosable?: boolean;    maskClassName?: string;    title?: React.ReactNode;    closable?: boolean;    headerStyle?: CSSProperties;    width?: CSSProperties['width'];    height: CSSProperties['height'];    style: CSSProperties;    size: SideSheetProps['size'];    bodyStyle?: CSSProperties;    className: string;    dialogClassName?: string;    children?: React.ReactNode;    footer?: React.ReactNode;    'aria-label'?: string;    maskExtraProps?: {[key: string]: any};    wrapperExtraProps?: {[key: string]: any}}export default class SideSheetContent extends React.PureComponent<SideSheetContentProps> {    static propTypes = {        onClose: PropTypes.func,        closeIcon: PropTypes.node,    };    static defaultProps = {        onClose: noop,    };    private sideSheetId: string;    private timeoutId: number;    componentDidMount() {        this.sideSheetId = `sidesheet-${uuid++}`;    }    componentWillUnmount() {        clearTimeout(this.timeoutId);    }    onMaskClick = (e: React.MouseEvent) => {        if (e.target === e.currentTarget) {            this.close(e);        }    };    close = (e: React.MouseEvent) => {        const { onClose } = this.props;        onClose && onClose(e);    };    getMaskElement() {        const {            mask,            maskStyle,            maskClosable,        } = this.props;        if (mask) {            return (                <div                    aria-hidden={true}                    key="mask"                    className={cls(`${prefixCls}-mask`, this.props.maskClassName ?? "")}                    style={maskStyle}                    onClick={maskClosable ? this.onMaskClick : null}                    {...this.props.maskExtraProps}                />            );        }        return null;    }    renderHeader() {        const {            title,            closable,            headerStyle,            closeIcon,        } = this.props;        let header, closer;        if (title) {            header = (                <div className={`${prefixCls}-title`} x-semi-prop="title">                    {this.props.title}                </div>            );        }        if (closable) {            const iconType = closeIcon || <IconClose/>;            closer = (                <Button                    className={`${prefixCls}-close`}                    key="close-btn"                    onClick={this.close}                    type="tertiary"                    icon={iconType}                    theme="borderless"                    size="small"                />            );        }        return (            <div className={`${prefixCls}-header`} role={'heading'} aria-level={1} style={{ ...headerStyle }}>                {header}                {closer}            </div>        );    }    getDialogElement() {        const { ...props } = this.props;        const style: CSSProperties = {};        if (props.width) {            style.width = props.width;            // When the mask is false, the width is set on the wrapper. At this time, sidesheet-inner does not need to set the width again, otherwise, the percentage will be accumulated repeatedly when the width is a percentage            if (!props.mask) {                style.width = '100%';            }        }        if (props.height) {            style.height = props.height;        }        const header = this.renderHeader();        const dialogElement = (            <div                key="dialog-element"                role="dialog"                tabIndex={-1}                className={cls(`${prefixCls}-inner`, `${prefixCls}-inner-wrap`, this.props.dialogClassName??"", `${prefixCls}-size-${props.size}`)}                // onMouseDown={this.onDialogMouseDown}                style={{ ...props.style, ...style }}                {...this.props.wrapperExtraProps}                // id={this.dialogId}            >                <div className={`${prefixCls}-content`}>                    {header}                    <div className={`${prefixCls}-body`} style={props.bodyStyle} x-semi-prop="children">                        {props.children}                    </div>                    {props.footer ? (                        <div className={`${prefixCls}-footer`} x-semi-prop="footer">                            {props.footer}                        </div>                    ) : null}                </div>            </div>        );        return dialogElement;    }    render() {        const {            mask,            className,            width,            onClose,            maskStyle,            maskClosable,            maskClassName,            title,            closable,            headerStyle,            height,            style,            size,            bodyStyle,            dialogClassName,            children,            footer,            maskExtraProps,            wrapperExtraProps,            ...rest        } = this.props;        const wrapperCls = cls(className, {            [`${prefixCls}-fixed`]: !mask,            [`${prefixCls}-size-${this.props.size}`]: !mask        });        const wrapperStyle: CSSProperties = {};        if (!mask && width) {            wrapperStyle.width = width;        }        const dataAttr = getDataAttr(rest);        return (            <div className={wrapperCls} style={wrapperStyle} {...dataAttr}>                {this.getMaskElement()}                {this.getDialogElement()}            </div>        );    }}
 |