mirror of
https://github.com/mastodon/mastodon.git
synced 2025-10-07 01:22:44 +00:00
94 lines
2.1 KiB
TypeScript
94 lines
2.1 KiB
TypeScript
import { useState, useEffect } from 'react';
|
|
|
|
import { useIntl } from 'react-intl';
|
|
import type { IntlShape } from 'react-intl';
|
|
|
|
import { dismissAlert } from 'mastodon/actions/alerts';
|
|
import type {
|
|
Alert as AlertType,
|
|
TranslatableString,
|
|
TranslatableValues,
|
|
} from 'mastodon/models/alert';
|
|
import { useAppSelector, useAppDispatch } from 'mastodon/store';
|
|
|
|
import { Alert } from './alert';
|
|
|
|
const formatIfNeeded = (
|
|
intl: IntlShape,
|
|
message: TranslatableString,
|
|
values?: TranslatableValues,
|
|
) => {
|
|
if (typeof message === 'object') {
|
|
return intl.formatMessage(message, values);
|
|
}
|
|
|
|
return message;
|
|
};
|
|
|
|
const TimedAlert: React.FC<{
|
|
alert: AlertType;
|
|
dismissAfter: number;
|
|
}> = ({
|
|
alert: { key, title, message, values, action, onClick },
|
|
dismissAfter,
|
|
}) => {
|
|
const dispatch = useAppDispatch();
|
|
const intl = useIntl();
|
|
const [active, setActive] = useState(false);
|
|
|
|
useEffect(() => {
|
|
const setActiveTimeout = setTimeout(() => {
|
|
setActive(true);
|
|
}, 1);
|
|
|
|
return () => {
|
|
clearTimeout(setActiveTimeout);
|
|
};
|
|
}, []);
|
|
|
|
useEffect(() => {
|
|
const dismissTimeout = setTimeout(() => {
|
|
setActive(false);
|
|
|
|
// Allow CSS transition to finish before removing from the DOM
|
|
setTimeout(() => {
|
|
dispatch(dismissAlert({ key }));
|
|
}, 500);
|
|
}, dismissAfter);
|
|
|
|
return () => {
|
|
clearTimeout(dismissTimeout);
|
|
};
|
|
}, [dispatch, setActive, key, dismissAfter]);
|
|
|
|
return (
|
|
<Alert
|
|
isActive={active}
|
|
title={title ? formatIfNeeded(intl, title, values) : undefined}
|
|
message={formatIfNeeded(intl, message, values)}
|
|
action={action ? formatIfNeeded(intl, action, values) : undefined}
|
|
onActionClick={onClick}
|
|
/>
|
|
);
|
|
};
|
|
|
|
export const AlertsController: React.FC = () => {
|
|
const alerts = useAppSelector((state) => state.alerts);
|
|
|
|
if (alerts.length === 0) {
|
|
return null;
|
|
}
|
|
|
|
return (
|
|
<div className='notification-list'>
|
|
{alerts.map((alert, idx) => (
|
|
<TimedAlert
|
|
key={alert.key}
|
|
alert={alert}
|
|
dismissAfter={5000 + idx * 1000}
|
|
/>
|
|
))}
|
|
</div>
|
|
);
|
|
};
|