immich/web/src/lib/managers/modal-manager.svelte.ts

43 lines
1.2 KiB
TypeScript
Raw Normal View History

import ConfirmDialog from '$lib/components/shared-components/dialog/confirm-dialog.svelte';
import { mount, unmount, type Component, type ComponentProps } from 'svelte';
2025-05-08 00:08:19 +02:00
type OnCloseData<T> = T extends { onClose: (data?: infer R) => void } ? R : never;
type ExtendsEmptyObject<T> = keyof T extends never ? Record<string, never> : T;
class ModalManager {
2025-05-08 00:08:19 +02:00
show<T extends object>(Component: Component<T>, props: ExtendsEmptyObject<Omit<T, 'onClose'>>) {
return this.open(Component, props).onClose;
}
open<T extends object, K = OnCloseData<T>>(Component: Component<T>, props: ExtendsEmptyObject<Omit<T, 'onClose'>>) {
let modal: object = {};
let onClose: () => Promise<void>;
const deferred = new Promise<K | undefined>((resolve) => {
onClose = async (data?: K) => {
await unmount(modal);
resolve(data);
};
modal = mount(Component, {
target: document.body,
props: {
2025-05-08 00:08:19 +02:00
...(props as T),
onClose,
},
});
});
2025-05-08 00:08:19 +02:00
return {
onClose: deferred,
close: () => onClose(),
};
}
2025-05-12 17:48:05 -04:00
showDialog(options: Omit<ComponentProps<typeof ConfirmDialog>, 'onClose'>) {
2025-05-08 00:08:19 +02:00
return this.show(ConfirmDialog, options);
}
}
export const modalManager = new ModalManager();