anvil/striker-ui/components/MessageGroup.tsx

124 lines
2.9 KiB
TypeScript
Raw Normal View History

2022-08-12 04:11:26 +00:00
import {
forwardRef,
ReactNode,
2022-08-12 04:11:26 +00:00
useCallback,
useImperativeHandle,
useMemo,
useState,
} from 'react';
import MessageBox, { Message, MessageBoxProps } from './MessageBox';
type Messages = {
[messageKey: string]: Message | undefined;
};
2022-08-12 04:11:26 +00:00
type MessageGroupOptionalProps = {
count?: number;
2022-08-12 04:11:26 +00:00
defaultMessageType?: MessageBoxProps['type'];
};
type MessageGroupProps = MessageGroupOptionalProps;
2022-08-12 04:11:26 +00:00
type MessageGroupForwardedRefContent = {
exists?: (key: string) => boolean;
setMessage?: (key: string, message?: Message) => void;
setMessageRe?: (re: RegExp, message?: Message) => void;
2022-08-12 04:11:26 +00:00
};
const MESSAGE_GROUP_DEFAULT_PROPS: Required<MessageGroupOptionalProps> = {
count: 0,
2022-08-12 04:11:26 +00:00
defaultMessageType: 'info',
};
const MessageGroup = forwardRef<
MessageGroupForwardedRefContent,
MessageGroupProps
>(
(
{
count = MESSAGE_GROUP_DEFAULT_PROPS.count,
defaultMessageType = MESSAGE_GROUP_DEFAULT_PROPS.defaultMessageType,
},
2022-08-12 04:11:26 +00:00
ref,
) => {
const [messages, setMessages] = useState<Messages>({});
2022-08-12 04:11:26 +00:00
const exists = useCallback(
(key: string) => messages[key] !== undefined,
[messages],
);
const setMessage = useCallback((key: string, message?: Message) => {
2022-08-12 04:11:26 +00:00
setMessages((previous) => {
const result = { ...previous };
2022-08-12 04:11:26 +00:00
result[key] = message;
return result;
2022-08-12 04:11:26 +00:00
});
}, []);
const setMessageRe = useCallback((re: RegExp, message?: Message) => {
setMessages((previous) => {
const result = { ...previous };
Object.keys(previous).forEach((key: string) => {
if (re.test(key)) {
result[key] = message;
}
});
return result;
});
}, []);
2022-08-12 04:11:26 +00:00
const messageElements = useMemo(() => {
const pairs = Object.entries(messages);
const isValidCount = count > 0;
const limit = isValidCount ? count : pairs.length;
const result: ReactNode[] = [];
pairs.every(([messageKey, message]) => {
if (message) {
const { children: messageChildren, type = defaultMessageType } =
message;
result.push(
<MessageBox key={`message-${messageKey}`} type={type}>
{messageChildren}
</MessageBox>,
);
}
return result.length < limit;
});
2022-08-12 04:11:26 +00:00
if (isValidCount && result.length === 0) {
result.push(
<MessageBox
key="message-placeholder"
sx={{ visibility: 'hidden' }}
text="Placeholder"
/>,
);
}
return result;
}, [count, defaultMessageType, messages]);
2022-08-12 04:11:26 +00:00
useImperativeHandle(ref, () => ({ exists, setMessage, setMessageRe }), [
exists,
setMessage,
setMessageRe,
]);
2022-08-12 04:11:26 +00:00
return <>{messageElements}</>;
},
);
MessageGroup.defaultProps = MESSAGE_GROUP_DEFAULT_PROPS;
MessageGroup.displayName = 'MessageGroup';
export type { MessageGroupForwardedRefContent };
export default MessageGroup;