mirror of
https://github.com/laurent22/joplin.git
synced 2025-01-23 18:53:36 +02:00
86 lines
3.1 KiB
TypeScript
86 lines
3.1 KiB
TypeScript
import { useContext, useCallback, useMemo, useRef } from 'react';
|
|
import { StateLastDeletion } from '@joplin/lib/reducer';
|
|
import { _, _n } from '@joplin/lib/locale';
|
|
import NotyfContext from '../NotyfContext';
|
|
import { waitForElement } from '@joplin/lib/dom';
|
|
import useAsyncEffect from '@joplin/lib/hooks/useAsyncEffect';
|
|
import { htmlentities } from '@joplin/utils/html';
|
|
import restoreItems from '@joplin/lib/services/trash/restoreItems';
|
|
import { ModelType } from '@joplin/lib/BaseModel';
|
|
import { themeStyle } from '@joplin/lib/theme';
|
|
import { Dispatch } from 'redux';
|
|
import { NotyfNotification } from 'notyf';
|
|
|
|
interface Props {
|
|
lastDeletion: StateLastDeletion;
|
|
lastDeletionNotificationTime: number;
|
|
themeId: number;
|
|
dispatch: Dispatch;
|
|
}
|
|
|
|
export default (props: Props) => {
|
|
const notyfContext = useContext(NotyfContext);
|
|
const notificationRef = useRef<NotyfNotification | null>(null);
|
|
|
|
const theme = useMemo(() => {
|
|
return themeStyle(props.themeId);
|
|
}, [props.themeId]);
|
|
|
|
const notyf = useMemo(() => {
|
|
const output = notyfContext;
|
|
output.options.types = notyfContext.options.types.map(type => {
|
|
if (type.type === 'success') {
|
|
type.background = theme.backgroundColor5;
|
|
// eslint-disable-next-line @typescript-eslint/no-explicit-any -- Old code before rule was applied
|
|
(type.icon as any).color = theme.backgroundColor5;
|
|
}
|
|
return type;
|
|
});
|
|
return output;
|
|
}, [notyfContext, theme]);
|
|
|
|
// eslint-disable-next-line @typescript-eslint/no-explicit-any -- Old code before rule was applied
|
|
const onCancelClick = useCallback(async (event: any) => {
|
|
notyf.dismiss(notificationRef.current);
|
|
notificationRef.current = null;
|
|
|
|
const lastDeletion: StateLastDeletion = JSON.parse(event.currentTarget.getAttribute('data-lastDeletion'));
|
|
|
|
if (lastDeletion.folderIds.length) {
|
|
await restoreItems(ModelType.Folder, lastDeletion.folderIds);
|
|
}
|
|
|
|
if (lastDeletion.noteIds.length) {
|
|
await restoreItems(ModelType.Note, lastDeletion.noteIds);
|
|
}
|
|
}, [notyf]);
|
|
|
|
useAsyncEffect(async (event) => {
|
|
if (!props.lastDeletion || props.lastDeletion.timestamp <= props.lastDeletionNotificationTime) return;
|
|
|
|
props.dispatch({ type: 'DELETION_NOTIFICATION_DONE' });
|
|
|
|
let msg = '';
|
|
|
|
if (props.lastDeletion.folderIds.length) {
|
|
msg = _('The notebook and its content was successfully moved to the trash.');
|
|
} else if (props.lastDeletion.noteIds.length) {
|
|
msg = _n('The note was successfully moved to the trash.', 'The notes were successfully moved to the trash.', props.lastDeletion.noteIds.length);
|
|
} else {
|
|
return;
|
|
}
|
|
|
|
const linkId = `deletion-notification-cancel-${Math.floor(Math.random() * 1000000)}`;
|
|
const cancelLabel = _('Cancel');
|
|
|
|
const notification = notyf.success(`${msg} <a href="#" class="cancel" data-lastDeletion="${htmlentities(JSON.stringify(props.lastDeletion))}" id="${linkId}">${cancelLabel}</a>`);
|
|
notificationRef.current = notification;
|
|
|
|
const element: HTMLAnchorElement = await waitForElement(document, linkId);
|
|
if (event.cancelled) return;
|
|
element.addEventListener('click', onCancelClick);
|
|
}, [props.lastDeletion, notyf, props.dispatch]);
|
|
|
|
return <div style={{ display: 'none' }}/>;
|
|
};
|