Skip to content

Commit 94dcda5

Browse files
authored
feat: Dialog support aria-* in closable (#403)
* feat: Dialog support aria-* in closable * feat: Dialog support aria-* in closable * feat: Dialog support aria-* in closable * feat: Dialog support aria-* in closable * feat: optimize code * feat: optimize code
1 parent 7a3bb93 commit 94dcda5

File tree

3 files changed

+58
-4
lines changed

3 files changed

+58
-4
lines changed

src/Dialog/Content/Panel.tsx

+17-3
Original file line numberDiff line numberDiff line change
@@ -1,9 +1,10 @@
11
import classNames from 'classnames';
22
import { useComposeRef } from 'rc-util/lib/ref';
3-
import React, { useRef } from 'react';
3+
import React, { useMemo, useRef } from 'react';
44
import { RefContext } from '../../context';
55
import type { IDialogPropTypes } from '../../IDialogPropTypes';
66
import MemoChildren from './MemoChildren';
7+
import pickAttrs from 'rc-util/lib/pickAttrs';
78

89
const sentinelStyle = { width: 0, height: 0, overflow: 'hidden', outline: 'none' };
910
const entityStyle = { outline: 'none' };
@@ -96,11 +97,24 @@ const Panel = React.forwardRef<ContentRef, PanelProps>((props, ref) => {
9697
);
9798
}
9899

100+
101+
const closableObj = useMemo(() => {
102+
if (typeof closable === 'object' && closable !== null) {
103+
return closable;
104+
}
105+
if (closable) {
106+
return { closeIcon: closeIcon ?? <span className={`${prefixCls}-close-x`} /> };
107+
}
108+
return {};
109+
}, [closable, closeIcon]);
110+
111+
const ariaProps = pickAttrs(closableObj, true);
112+
99113
let closer: React.ReactNode;
100114
if (closable) {
101115
closer = (
102-
<button type="button" onClick={onClose} aria-label="Close" className={`${prefixCls}-close`}>
103-
{closeIcon || <span className={`${prefixCls}-close-x`} />}
116+
<button type="button" onClick={onClose} aria-label="Close" {...ariaProps} className={`${prefixCls}-close`}>
117+
{closableObj.closeIcon}
104118
</button>
105119
);
106120
}

src/IDialogPropTypes.tsx

+1-1
Original file line numberDiff line numberDiff line change
@@ -28,7 +28,7 @@ export type IDialogPropTypes = {
2828
afterClose?: () => any;
2929
afterOpenChange?: (open: boolean) => void;
3030
onClose?: (e: SyntheticEvent) => any;
31-
closable?: boolean;
31+
closable?: boolean | ({ closeIcon?: React.ReactNode } & React.AriaAttributes);
3232
maskClosable?: boolean;
3333
visible?: boolean;
3434
destroyOnClose?: boolean;

tests/index.spec.tsx

+40
Original file line numberDiff line numberDiff line change
@@ -640,4 +640,44 @@ describe('dialog', () => {
640640
);
641641
spy.mockRestore();
642642
});
643+
644+
it('support aria-* in closable', () => {
645+
const onClose = jest.fn();
646+
const wrapper = mount(
647+
<Dialog
648+
closable={{
649+
closeIcon:"test",
650+
'aria-label': 'test aria-label',
651+
}}
652+
visible
653+
onClose={onClose}
654+
/>
655+
);
656+
jest.runAllTimers();
657+
wrapper.update();
658+
659+
const btn = wrapper.find('.rc-dialog-close');
660+
expect(btn.text()).toBe('test');
661+
expect(btn.getDOMNode().getAttribute('aria-label')).toBe('test aria-label');
662+
btn.simulate('click');
663+
664+
jest.runAllTimers();
665+
wrapper.update();
666+
expect(onClose).toHaveBeenCalledTimes(1);
667+
});
668+
it('should not display closeIcon when closable is false', () => {
669+
const onClose = jest.fn();
670+
const wrapper = mount(
671+
<Dialog
672+
closable={false}
673+
visible
674+
onClose={onClose}
675+
/>
676+
);
677+
jest.runAllTimers();
678+
wrapper.update();
679+
680+
const btn = wrapper.find('.rc-dialog-close');
681+
expect(btn.find('.rc-dialog-close-x')).not.toBeNull();
682+
});
643683
});

0 commit comments

Comments
 (0)