113
common/resources/client/ui/toast/toast-store.ts
Executable file
113
common/resources/client/ui/toast/toast-store.ts
Executable file
@@ -0,0 +1,113 @@
|
||||
import {create} from 'zustand';
|
||||
import {immer} from 'zustand/middleware/immer';
|
||||
import {MessageDescriptor} from '../../i18n/message-descriptor';
|
||||
import {nanoid} from 'nanoid';
|
||||
import {ToastTimer} from './toast-timer';
|
||||
|
||||
type ToastType = 'danger' | 'default' | 'positive' | 'loading' | null;
|
||||
type ToastPosition = 'bottom-center' | 'bottom-right';
|
||||
|
||||
interface ToastAction {
|
||||
label: string | MessageDescriptor;
|
||||
action: string;
|
||||
}
|
||||
|
||||
export interface ToastOptions {
|
||||
type?: ToastType;
|
||||
action?: ToastAction;
|
||||
id?: string | number;
|
||||
duration?: number;
|
||||
position?: 'bottom-center' | 'bottom-right';
|
||||
disableExitAnimation?: boolean;
|
||||
disableEnterAnimation?: boolean;
|
||||
}
|
||||
|
||||
interface Toast {
|
||||
timer?: ToastTimer | null;
|
||||
message: string | MessageDescriptor;
|
||||
type: ToastType;
|
||||
id: string | number;
|
||||
duration: number;
|
||||
action?: ToastAction;
|
||||
position: ToastPosition;
|
||||
disableExitAnimation?: boolean;
|
||||
disableEnterAnimation?: boolean;
|
||||
}
|
||||
|
||||
interface ToastStore {
|
||||
toasts: Toast[];
|
||||
add: (message: Toast['message'], opts?: ToastOptions) => void;
|
||||
remove: (toastId: string | number) => void;
|
||||
}
|
||||
|
||||
const maximumVisible = 1;
|
||||
|
||||
function getDefaultDuration(type: ToastType) {
|
||||
switch (type) {
|
||||
case 'danger':
|
||||
return 8000;
|
||||
case 'loading':
|
||||
return 0;
|
||||
default:
|
||||
return 3000;
|
||||
}
|
||||
}
|
||||
|
||||
export const useToastStore = create<ToastStore>()(
|
||||
immer((set, get) => ({
|
||||
toasts: [],
|
||||
add: (message, opts) => {
|
||||
const amountToRemove = get().toasts.length + 1 - maximumVisible;
|
||||
if (amountToRemove > 0) {
|
||||
set(state => {
|
||||
state.toasts.splice(0, amountToRemove);
|
||||
});
|
||||
}
|
||||
|
||||
const toastId = opts?.id || nanoid(6);
|
||||
const toastType = opts?.type || 'positive';
|
||||
const duration = opts?.duration ?? getDefaultDuration(toastType);
|
||||
const toast: Toast = {
|
||||
timer:
|
||||
duration > 0
|
||||
? new ToastTimer(() => get().remove(toastId), duration)
|
||||
: null,
|
||||
message,
|
||||
...opts,
|
||||
id: toastId,
|
||||
type: toastType,
|
||||
position: opts?.position || 'bottom-center',
|
||||
duration,
|
||||
disableExitAnimation: opts?.disableExitAnimation,
|
||||
disableEnterAnimation: opts?.disableEnterAnimation,
|
||||
};
|
||||
|
||||
const toastIndex = get().toasts.findIndex(t => t.id === toast.id);
|
||||
if (toastIndex > -1) {
|
||||
set(state => {
|
||||
state.toasts[toastIndex] = toast;
|
||||
});
|
||||
} else {
|
||||
set(state => {
|
||||
state.toasts.push(toast);
|
||||
});
|
||||
}
|
||||
},
|
||||
remove: toastId => {
|
||||
const newToasts = get().toasts.filter(toast => {
|
||||
if (toastId === toast.id) {
|
||||
toast.timer?.clear();
|
||||
return false;
|
||||
}
|
||||
return true;
|
||||
});
|
||||
set(state => {
|
||||
state.toasts = newToasts;
|
||||
});
|
||||
},
|
||||
}))
|
||||
);
|
||||
|
||||
export function toastState() {
|
||||
return useToastStore.getState();
|
||||
}
|
||||
Reference in New Issue
Block a user