mirror of
https://github.com/ant-design/ant-design.git
synced 2025-01-23 01:45:05 +08:00
523b74e3b6
* feat: add successColor for Progress (#24655) * feat: add successColor for Progress * feat: update * fix: update test * remove snap * feat: add test case * refactor success * feat: adjust styyle * feat: add DevWarning * feat: Support rowSelection.dirty (#24718) * feat: Support rowSelection.dirty * rename to reserveKeys * preserveKeys will keep record also * to preserveSelectedRowKeys * feat: add ghost prop for collapse (#24734) * feat: add ghost prop for collapse * doc: version of collapse's ghost prop * refactor: make ghost collapse's less code to a nested style * chore: remove redundant codes in ghost collapse's less & doc * doc: add a background wrapper for ghost collapse demo * doc: dark-theme wrapper bg-color for ghost collapse demo * test: update snapshot of ghost collapse * doc: use softer bg-color on ghost collapse demo * doc: remove disabled panel in ghost collapse demo * feat: form instance support getFieldInstance (#24711) * support getFieldInstance * update doc * fix lint * move func * move into hooks * update ref logic * fix lint * rm only * fix docs * feat: dropdown support arrow (#23869) * feat: dropdown support arrow prop close #22758 * test: update snapshot * fix: fix dropdown cls names * test: update snapshot * test: update snapshot * doc: update demo * test: update demo snapshot * demo * fix: snapshot * chore: change the style of ghost collapse & demo modified (#24762) * refactor: reduce content padding in ghost collapse * doc: remove the wrapper outside ghost collapse Designer want the demo differs from other demos * refactor: remove redundant .less code in collapse * feat: cascader dropdown-render prop (#24812) * feat: cascader dropdown-render prop * fix: update Cascader dropdownRender type annotation * fix: set rc-cascader semver from ^ to ~ * docs: fix coding style in cascader/custom-dropdown * feat: 🆕 support Drawer closeIcon (#24842) * feat: 🆕 support Drawer closeIcon close #19283 close #19153 * add test case * update docs * feat: 🆕 Cascader expandIcon (#24865) * feat: cascader expandIcon * fix: snap * refactor: reduce CSS size (#24846) * refactor: reduce button css size * refactor: remove redundant button .less code * feat: add Table onChange an action param (#24697) * Working on tests * created TableAction type * changed TableActions to tuple * removed chinese documentation line * refactor TableActions * fix documentation * Moved action into extra param * minor doc change * feat: add closeIcon customize tag close (#24885) * feat: add closeIcon customize tag close * docs fix * update snap * fix: css name * update snapshot * snapshot * feat: add radio `optionType` api to set radio option type (#24809) * feat: radio component * docs: update md * fix: snap * test components * fix: use optionType * fix name * add warning * fix * feat: expand rate character (#24903) * feat: expand rate character * fix: demo * fix: snap * Update components/rate/index.zh-CN.md Co-authored-by: 偏右 <afc163@gmail.com> * fix Co-authored-by: 偏右 <afc163@gmail.com> * Refactor demo code box actions (#24887) * refactor: refine the styling of actions part of demo code-box * fix: lint style * refactor: move Result children to end (#24945) * feat: remove content max-width on dot-step (#24907) * feat: add Skeleton-Image (#24805) * feat: add Skeleton-Image * feat: add docs * fix: adjust skeleton * feat: adjust Image Component * feat: rebase * feat: adjust style * fix: lint * feat: remove size * feat: delete md * feat: fix style * ✨ feat: Mentions support autoSize (#24961) close #17746 * chore: replace textarea with rc-textarea (#24966) * feat: update pagination@2.3.0 support onChange called when pageSize change (#24964) * feat: update pagination@2.5.0 and add test case to relative component * fix: lint * delete * feat: add test case for pagination * adjust test case * feat: Implement centered prop in Tabs (#24958) * Implement centered in Tabs along with its tests and docs * Fix build error * Add Chinese translations and remove test case Co-authored-by: Ashkan Pourghasem <ashkan.pourghasem@gmail.com> * feat: Add modal style parameter (#24773) * add some paramters in default.less * Update components/style/themes/default.less Co-authored-by: Amumu <yoyo837@hotmail.com> * change parameter in compact.less Co-authored-by: Crystal Gao <jinggao@ebay.com> Co-authored-by: Amumu <yoyo837@hotmail.com> * feat: export Tabs addIcon (#25006) * feat: export Tabs addIcon * update snapshot * feat: showNow on timepicker and datetimepicker (#25032) * feat: update rc-picker@1.7.1 and fix icons of month and quarter picker in DatePicker Component (#25035) * feat: update rc-picker@1.7.1 * delete * add * feat: expand rate support props (#24993) * docs: 📝 Add Form.Item hidden in doc (#25108) close #25101 * fix: ⌨️ Improve Pagination accessibility issue (#25119) * ⌨️ Improve Pagination a11y by fixing a W3C error https://github.com/react-component/pagination/issues/280 * update snapshot * 🆙 rc-pagination to 2.4.1 * feat: support triggerSubMenuAction for <Menu /> (#25127) * feat(menu): add triggerSubMenuAction for Menu * feat(menu): test cases * chore: Adjust picker logic (#25135) * chore: update rc-picker 1.10.0 (#25174) * feat: table row check strictly (#24931) * feat: add checkStrictly on Table.rowSelection * fix: LGTM warnings * test: table rowSelection.checkStrictly * test: add cov [wip] * refactor: tree.rowSelection.checkStrictly [wip] * test: table.rowSelection.checkStrictly basic case * feat: support rowKey on checkStrictly table * feat: Table checkStrictly support getCheckboxProps * docs: Table checkStrictly * chore: typo * chore: remove useless comment * chore: update snapshot * chore: update snapshot * fix: fire selectAll on selection dropdown menu & changeRows incorrect in selectAll callback * docs: typo * chore * chore * fix: expand buttons of leaf rows in tree data are not hidden * feat: Table warning about rowKey index parameter * perf: only generate keyEntities when not checkStrictly * refactor: remove useless parseCheckedKeys * refactor: get derived selected & half selected keys from selectedRowKeys * chore: remove env condition stmt * chore: revert index usage & code formatting * chore: rerun ci * docs: table tree-data checkstrictly * test: update snapshots * refactor: use useMergedState hook * chore: rerun ci * chore: rerun ci 2 * chore: revert selection select all behavior * refactor: refactor code based on feature * chore: revert table code format * chore: revert table code format * fix: useMemo deps * fix: useMemo deps * fix: useMemo deps * feat: support preserve (#25186) * docs: add responsibly order for Col (#25139) * feat: add type * feat: add responsibly order cols * feat: add docs * feat: add test case * fix test Co-authored-by: 二货机器人 <smith3816@gmail.com> Co-authored-by: 偏右 <afc163@gmail.com> Co-authored-by: zoomdong <1344492820@qq.com> Co-authored-by: 07akioni <07akioni2@gmail.com> Co-authored-by: wendellhu <wendellhu95@gmail.com> Co-authored-by: xrkffgg <xrkffgg@gmail.com> Co-authored-by: Neto Braghetto <netow93@gmail.com> Co-authored-by: Kermit Xuan <kermitlx@outlook.com> Co-authored-by: Ashkan Pourghasem <64011067+ashkan-pm@users.noreply.github.com> Co-authored-by: Ashkan Pourghasem <ashkan.pourghasem@gmail.com> Co-authored-by: hicrystal <295247343@qq.com> Co-authored-by: Crystal Gao <jinggao@ebay.com> Co-authored-by: Amumu <yoyo837@hotmail.com> Co-authored-by: Li Ming <armyiljfe@gmail.com>
341 lines
9.1 KiB
TypeScript
341 lines
9.1 KiB
TypeScript
import * as React from 'react';
|
||
import RcDrawer from 'rc-drawer';
|
||
import getScrollBarSize from 'rc-util/lib/getScrollBarSize';
|
||
import CloseOutlined from '@ant-design/icons/CloseOutlined';
|
||
import classNames from 'classnames';
|
||
import omit from 'omit.js';
|
||
|
||
import { ConfigConsumerProps } from '../config-provider';
|
||
import { withConfigConsumer, ConfigConsumer } from '../config-provider/context';
|
||
import { tuple } from '../_util/type';
|
||
|
||
const DrawerContext = React.createContext<Drawer | null>(null);
|
||
|
||
type EventType =
|
||
| React.KeyboardEvent<HTMLDivElement>
|
||
| React.MouseEvent<HTMLDivElement | HTMLButtonElement>;
|
||
|
||
type getContainerFunc = () => HTMLElement;
|
||
|
||
const PlacementTypes = tuple('top', 'right', 'bottom', 'left');
|
||
type placementType = typeof PlacementTypes[number];
|
||
export interface DrawerProps {
|
||
closable?: boolean;
|
||
closeIcon?: React.ReactNode;
|
||
destroyOnClose?: boolean;
|
||
forceRender?: boolean;
|
||
getContainer?: string | HTMLElement | getContainerFunc | false;
|
||
maskClosable?: boolean;
|
||
mask?: boolean;
|
||
maskStyle?: React.CSSProperties;
|
||
style?: React.CSSProperties;
|
||
/** wrapper dom node style of header and body */
|
||
drawerStyle?: React.CSSProperties;
|
||
headerStyle?: React.CSSProperties;
|
||
bodyStyle?: React.CSSProperties;
|
||
title?: React.ReactNode;
|
||
visible?: boolean;
|
||
width?: number | string;
|
||
height?: number | string;
|
||
zIndex?: number;
|
||
prefixCls?: string;
|
||
push?: boolean;
|
||
placement?: placementType;
|
||
onClose?: (e: EventType) => void;
|
||
afterVisibleChange?: (visible: boolean) => void;
|
||
className?: string;
|
||
handler?: React.ReactNode;
|
||
keyboard?: boolean;
|
||
footer?: React.ReactNode;
|
||
footerStyle?: React.CSSProperties;
|
||
}
|
||
|
||
export interface IDrawerState {
|
||
push?: boolean;
|
||
}
|
||
|
||
class Drawer extends React.Component<DrawerProps & ConfigConsumerProps, IDrawerState> {
|
||
static defaultProps = {
|
||
width: 256,
|
||
height: 256,
|
||
closable: true,
|
||
placement: 'right' as placementType,
|
||
maskClosable: true,
|
||
mask: true,
|
||
level: null,
|
||
keyboard: true,
|
||
};
|
||
|
||
readonly state = {
|
||
push: false,
|
||
};
|
||
|
||
parentDrawer: Drawer | null;
|
||
|
||
destroyClose: boolean;
|
||
|
||
public componentDidMount() {
|
||
// fix: delete drawer in child and re-render, no push started.
|
||
// <Drawer>{show && <Drawer />}</Drawer>
|
||
const { visible } = this.props;
|
||
if (visible && this.parentDrawer) {
|
||
this.parentDrawer.push();
|
||
}
|
||
}
|
||
|
||
public componentDidUpdate(preProps: DrawerProps) {
|
||
const { visible } = this.props;
|
||
if (preProps.visible !== visible && this.parentDrawer) {
|
||
if (visible) {
|
||
this.parentDrawer.push();
|
||
} else {
|
||
this.parentDrawer.pull();
|
||
}
|
||
}
|
||
}
|
||
|
||
public componentWillUnmount() {
|
||
// unmount drawer in child, clear push.
|
||
if (this.parentDrawer) {
|
||
this.parentDrawer.pull();
|
||
this.parentDrawer = null;
|
||
}
|
||
}
|
||
|
||
push = () => {
|
||
this.setState({
|
||
push: true,
|
||
});
|
||
};
|
||
|
||
pull = () => {
|
||
this.setState({
|
||
push: false,
|
||
});
|
||
};
|
||
|
||
onDestroyTransitionEnd = () => {
|
||
const isDestroyOnClose = this.getDestroyOnClose();
|
||
if (!isDestroyOnClose) {
|
||
return;
|
||
}
|
||
if (!this.props.visible) {
|
||
this.destroyClose = true;
|
||
this.forceUpdate();
|
||
}
|
||
};
|
||
|
||
getDestroyOnClose = () => this.props.destroyOnClose && !this.props.visible;
|
||
|
||
// get drawer push width or height
|
||
getPushTransform = (placement?: placementType) => {
|
||
if (placement === 'left' || placement === 'right') {
|
||
return `translateX(${placement === 'left' ? 180 : -180}px)`;
|
||
}
|
||
if (placement === 'top' || placement === 'bottom') {
|
||
return `translateY(${placement === 'top' ? 180 : -180}px)`;
|
||
}
|
||
};
|
||
|
||
getOffsetStyle() {
|
||
const { placement, width, height, visible, mask } = this.props;
|
||
// https://github.com/ant-design/ant-design/issues/24287
|
||
if (!visible && !mask) {
|
||
return {};
|
||
}
|
||
const offsetStyle: any = {};
|
||
if (placement === 'left' || placement === 'right') {
|
||
offsetStyle.width = width;
|
||
} else {
|
||
offsetStyle.height = height;
|
||
}
|
||
return offsetStyle;
|
||
}
|
||
|
||
getRcDrawerStyle = () => {
|
||
const { zIndex, placement, mask, style } = this.props;
|
||
const { push } = this.state;
|
||
// 当无 mask 时,将 width 应用到外层容器上
|
||
// 解决 https://github.com/ant-design/ant-design/issues/12401 的问题
|
||
const offsetStyle = mask ? {} : this.getOffsetStyle();
|
||
return {
|
||
zIndex,
|
||
transform: push ? this.getPushTransform(placement) : undefined,
|
||
...offsetStyle,
|
||
...style,
|
||
};
|
||
};
|
||
|
||
renderHeader() {
|
||
const { title, prefixCls, closable, headerStyle } = this.props;
|
||
if (!title && !closable) {
|
||
return null;
|
||
}
|
||
|
||
const headerClassName = title ? `${prefixCls}-header` : `${prefixCls}-header-no-title`;
|
||
return (
|
||
<div className={headerClassName} style={headerStyle}>
|
||
{title && <div className={`${prefixCls}-title`}>{title}</div>}
|
||
{closable && this.renderCloseIcon()}
|
||
</div>
|
||
);
|
||
}
|
||
|
||
renderFooter() {
|
||
const { footer, footerStyle, prefixCls } = this.props;
|
||
if (!footer) {
|
||
return null;
|
||
}
|
||
|
||
const footerClassName = `${prefixCls}-footer`;
|
||
return (
|
||
<div className={footerClassName} style={footerStyle}>
|
||
{footer}
|
||
</div>
|
||
);
|
||
}
|
||
|
||
renderCloseIcon() {
|
||
const { closable, closeIcon = <CloseOutlined />, prefixCls, onClose } = this.props;
|
||
return (
|
||
closable && (
|
||
// eslint-disable-next-line react/button-has-type
|
||
<button
|
||
onClick={onClose}
|
||
aria-label="Close"
|
||
className={`${prefixCls}-close`}
|
||
style={
|
||
{
|
||
'--scroll-bar': `${getScrollBarSize()}px`,
|
||
} as any
|
||
}
|
||
>
|
||
{closeIcon}
|
||
</button>
|
||
)
|
||
);
|
||
}
|
||
|
||
// render drawer body dom
|
||
renderBody = () => {
|
||
const { bodyStyle, drawerStyle, prefixCls, visible } = this.props;
|
||
if (this.destroyClose && !visible) {
|
||
return null;
|
||
}
|
||
this.destroyClose = false;
|
||
|
||
const containerStyle: React.CSSProperties = {};
|
||
|
||
const isDestroyOnClose = this.getDestroyOnClose();
|
||
|
||
if (isDestroyOnClose) {
|
||
// Increase the opacity transition, delete children after closing.
|
||
containerStyle.opacity = 0;
|
||
containerStyle.transition = 'opacity .3s';
|
||
}
|
||
|
||
return (
|
||
<div
|
||
className={`${prefixCls}-wrapper-body`}
|
||
style={{
|
||
...containerStyle,
|
||
...drawerStyle,
|
||
}}
|
||
onTransitionEnd={this.onDestroyTransitionEnd}
|
||
>
|
||
{this.renderHeader()}
|
||
<div className={`${prefixCls}-body`} style={bodyStyle}>
|
||
{this.props.children}
|
||
</div>
|
||
{this.renderFooter()}
|
||
</div>
|
||
);
|
||
};
|
||
|
||
// render Provider for Multi-level drawer
|
||
renderProvider = (value: Drawer) => {
|
||
this.parentDrawer = value;
|
||
|
||
return (
|
||
<ConfigConsumer>
|
||
{({ getPopupContainer, getPrefixCls }) => {
|
||
const {
|
||
prefixCls: customizePrefixCls,
|
||
placement,
|
||
className,
|
||
mask,
|
||
direction,
|
||
visible,
|
||
...rest
|
||
} = this.props;
|
||
|
||
const prefixCls = getPrefixCls('select', customizePrefixCls);
|
||
const drawerClassName = classNames(className, {
|
||
'no-mask': !mask,
|
||
[`${prefixCls}-rtl`]: direction === 'rtl',
|
||
});
|
||
const offsetStyle = mask ? this.getOffsetStyle() : {};
|
||
|
||
return (
|
||
<DrawerContext.Provider value={this}>
|
||
<RcDrawer
|
||
handler={false}
|
||
{...omit(rest, [
|
||
'zIndex',
|
||
'style',
|
||
'closable',
|
||
'closeIcon',
|
||
'destroyOnClose',
|
||
'drawerStyle',
|
||
'headerStyle',
|
||
'bodyStyle',
|
||
'footerStyle',
|
||
'footer',
|
||
'locale',
|
||
'title',
|
||
'push',
|
||
'visible',
|
||
'getPopupContainer',
|
||
'rootPrefixCls',
|
||
'getPrefixCls',
|
||
'renderEmpty',
|
||
'csp',
|
||
'pageHeader',
|
||
'autoInsertSpaceInButton',
|
||
'width',
|
||
'height',
|
||
'dropdownMatchSelectWidth',
|
||
'getTargetContainer',
|
||
])}
|
||
getContainer={
|
||
// 有可能为 false,所以不能直接判断
|
||
rest.getContainer === undefined && getPopupContainer
|
||
? () => getPopupContainer(document.body)
|
||
: rest.getContainer
|
||
}
|
||
{...offsetStyle}
|
||
prefixCls={prefixCls}
|
||
open={visible}
|
||
showMask={mask}
|
||
placement={placement}
|
||
style={this.getRcDrawerStyle()}
|
||
className={drawerClassName}
|
||
>
|
||
{this.renderBody()}
|
||
</RcDrawer>
|
||
</DrawerContext.Provider>
|
||
);
|
||
}}
|
||
</ConfigConsumer>
|
||
);
|
||
};
|
||
|
||
render() {
|
||
return <DrawerContext.Consumer>{this.renderProvider}</DrawerContext.Consumer>;
|
||
}
|
||
}
|
||
|
||
export default withConfigConsumer<DrawerProps>({
|
||
prefixCls: 'drawer',
|
||
})(Drawer);
|