mirror of
https://github.com/immich-app/immich.git
synced 2026-02-04 19:12:11 -08:00
* fix: clear ocr and asset cache when edits are applied * use event manager * fix: undefined check
151 lines
4.1 KiB
TypeScript
151 lines
4.1 KiB
TypeScript
import TransformTool from '$lib/components/asset-viewer/editor/transform-tool/transform-tool.svelte';
|
|
import { transformManager } from '$lib/managers/edit/transform-manager.svelte';
|
|
import { eventManager } from '$lib/managers/event-manager.svelte';
|
|
import { waitForWebsocketEvent } from '$lib/stores/websocket';
|
|
import { editAsset, removeAssetEdits, type AssetEditsDto, type AssetResponseDto } from '@immich/sdk';
|
|
import { ConfirmModal, modalManager, toastManager } from '@immich/ui';
|
|
import { mdiCropRotate } from '@mdi/js';
|
|
import type { Component } from 'svelte';
|
|
|
|
export type EditAction = AssetEditsDto['edits'][number];
|
|
export type EditActions = EditAction[];
|
|
|
|
export interface EditToolManager {
|
|
onActivate: (asset: AssetResponseDto, edits: EditActions) => Promise<void>;
|
|
onDeactivate: () => void;
|
|
resetAllChanges: () => Promise<void>;
|
|
hasChanges: boolean;
|
|
edits: EditAction[];
|
|
}
|
|
|
|
export enum EditToolType {
|
|
Transform = 'transform',
|
|
}
|
|
|
|
export interface EditTool {
|
|
type: EditToolType;
|
|
icon: string;
|
|
component: Component;
|
|
manager: EditToolManager;
|
|
}
|
|
|
|
export class EditManager {
|
|
tools: EditTool[] = [
|
|
{
|
|
type: EditToolType.Transform,
|
|
icon: mdiCropRotate,
|
|
component: TransformTool,
|
|
manager: transformManager,
|
|
},
|
|
];
|
|
|
|
currentAsset = $state<AssetResponseDto | null>(null);
|
|
selectedTool = $state<EditTool | null>(null);
|
|
hasChanges = $derived(this.tools.some((t) => t.manager.hasChanges));
|
|
|
|
// used to disable multiple confirm dialogs and mouse events while one is open
|
|
isShowingConfirmDialog = $state(false);
|
|
isApplyingEdits = $state(false);
|
|
hasAppliedEdits = $state(false);
|
|
|
|
async closeConfirm(): Promise<boolean> {
|
|
// Prevent multiple dialogs (usually happens with rapid escape key presses)
|
|
if (this.isShowingConfirmDialog) {
|
|
return false;
|
|
}
|
|
if (!this.hasChanges || this.hasAppliedEdits) {
|
|
return true;
|
|
}
|
|
|
|
this.isShowingConfirmDialog = true;
|
|
|
|
const confirmed = await modalManager.show(ConfirmModal, {
|
|
title: 'Discard Edits?',
|
|
prompt: 'You have unsaved edits. Are you sure you want to discard them?',
|
|
confirmText: 'Discard Edits',
|
|
});
|
|
|
|
this.isShowingConfirmDialog = false;
|
|
|
|
return confirmed;
|
|
}
|
|
|
|
reset() {
|
|
for (const tool of this.tools) {
|
|
tool.manager.onDeactivate?.();
|
|
}
|
|
this.selectedTool = this.tools[0];
|
|
}
|
|
|
|
async activateTool(toolType: EditToolType, asset: AssetResponseDto, edits: AssetEditsDto) {
|
|
this.hasAppliedEdits = false;
|
|
if (this.selectedTool?.type === toolType) {
|
|
return;
|
|
}
|
|
|
|
this.currentAsset = asset;
|
|
|
|
this.selectedTool?.manager.onDeactivate?.();
|
|
const newTool = this.tools.find((t) => t.type === toolType);
|
|
if (newTool) {
|
|
this.selectedTool = newTool;
|
|
await newTool.manager.onActivate?.(asset, edits.edits);
|
|
}
|
|
}
|
|
|
|
cleanup() {
|
|
for (const tool of this.tools) {
|
|
tool.manager.onDeactivate?.();
|
|
}
|
|
this.currentAsset = null;
|
|
this.selectedTool = null;
|
|
}
|
|
|
|
async resetAllChanges() {
|
|
for (const tool of this.tools) {
|
|
await tool.manager.resetAllChanges();
|
|
}
|
|
}
|
|
|
|
async applyEdits(): Promise<boolean> {
|
|
this.isApplyingEdits = true;
|
|
|
|
const edits = this.tools.flatMap((tool) => tool.manager.edits);
|
|
if (!this.currentAsset) {
|
|
return false;
|
|
}
|
|
|
|
const assetId = this.currentAsset.id;
|
|
|
|
try {
|
|
// Setup the websocket listener before sending the edit request
|
|
const editCompleted = waitForWebsocketEvent('AssetEditReadyV1', (event) => event.asset.id === assetId, 10_000);
|
|
|
|
await (edits.length === 0
|
|
? removeAssetEdits({ id: assetId })
|
|
: editAsset({
|
|
id: assetId,
|
|
assetEditActionListDto: {
|
|
edits,
|
|
},
|
|
}));
|
|
|
|
await editCompleted;
|
|
|
|
eventManager.emit('AssetEditsApplied', assetId);
|
|
|
|
toastManager.success('Edits applied successfully');
|
|
this.hasAppliedEdits = true;
|
|
|
|
return true;
|
|
} catch {
|
|
toastManager.danger('Failed to apply edits');
|
|
return false;
|
|
} finally {
|
|
this.isApplyingEdits = false;
|
|
}
|
|
}
|
|
}
|
|
|
|
export const editManager = new EditManager();
|