-
Notifications
You must be signed in to change notification settings - Fork 113
/
Copy pathNotice.tsx
110 lines (97 loc) · 2.82 KB
/
Notice.tsx
1
2
3
4
5
6
7
8
9
10
11
12
13
14
15
16
17
18
19
20
21
22
23
24
25
26
27
28
29
30
31
32
33
34
35
36
37
38
39
40
41
42
43
44
45
46
47
48
49
50
51
52
53
54
55
56
57
58
59
60
61
62
63
64
65
66
67
68
69
70
71
72
73
74
75
76
77
78
79
80
81
82
83
84
85
86
87
88
89
90
91
92
93
94
95
96
97
98
99
100
101
102
103
104
105
106
107
108
109
110
import classNames from 'classnames';
import KeyCode from 'rc-util/lib/KeyCode';
import * as React from 'react';
import usePageActiveStatus from './hooks/usePageActiveStatus';
import type { NoticeConfig } from './interface';
export interface NoticeProps extends Omit<NoticeConfig, 'onClose'> {
prefixCls: string;
className?: string;
style?: React.CSSProperties;
eventKey: React.Key;
onClick?: React.MouseEventHandler<HTMLDivElement>;
onNoticeClose?: (key: React.Key) => void;
hovering?: boolean;
}
const Notify = React.forwardRef<HTMLDivElement, NoticeProps & { times?: number }>((props, ref) => {
const {
prefixCls,
style,
className,
duration = 4.5,
pauseOnFocusLoss,
eventKey,
content,
closable,
closeIcon = 'x',
props: divProps,
onClick,
onNoticeClose,
times,
hovering: forcedHovering,
} = props;
const [hovering, setHovering] = React.useState(false);
const mergedHovering = forcedHovering || hovering;
const activeStauts = usePageActiveStatus();
const mergedStauts = pauseOnFocusLoss ? activeStauts : true;
// ======================== Close =========================
const onInternalClose = () => {
onNoticeClose(eventKey);
};
const onCloseKeyDown: React.KeyboardEventHandler<HTMLAnchorElement> = (e) => {
if (e.key === 'Enter' || e.code === 'Enter' || e.keyCode === KeyCode.ENTER) {
onInternalClose();
}
};
// ======================== Effect ========================
React.useEffect(() => {
if (mergedStauts && !mergedHovering && duration > 0) {
const timeout = setTimeout(() => {
onInternalClose();
}, duration * 1000);
return () => {
clearTimeout(timeout);
};
}
// eslint-disable-next-line react-hooks/exhaustive-deps
}, [duration, mergedHovering, times, mergedStauts]);
// ======================== Render ========================
const noticePrefixCls = `${prefixCls}-notice`;
return (
<div
{...divProps}
ref={ref}
className={classNames(noticePrefixCls, className, {
[`${noticePrefixCls}-closable`]: closable,
})}
style={style}
onMouseEnter={(e) => {
setHovering(true);
divProps?.onMouseEnter?.(e);
}}
onMouseLeave={(e) => {
setHovering(false);
divProps?.onMouseLeave?.(e);
}}
onClick={onClick}
>
{/* Content */}
<div className={`${noticePrefixCls}-content`}>{content}</div>
{/* Close Icon */}
{closable && (
<a
tabIndex={0}
className={`${noticePrefixCls}-close`}
onKeyDown={onCloseKeyDown}
onClick={(e) => {
e.preventDefault();
e.stopPropagation();
onInternalClose();
}}
>
{closeIcon}
</a>
)}
</div>
);
});
export default Notify;