1
0
Fork 0
mirror of https://github.com/immich-app/immich.git synced 2025-01-01 08:31:59 +00:00

feat(web): bulk deduplicate (#10448)

* bulk deduplicate

* notification for keeping all duplicates

* fix notification

* remove unused text

* pr feedback

* wording

* formatting
This commit is contained in:
Mert 2024-06-19 12:11:59 -04:00 committed by GitHub
parent 86cbc6e125
commit b3f9641edf
No known key found for this signature in database
GPG key ID: B5690EEEBB952194
4 changed files with 123 additions and 24 deletions

View file

@ -4,9 +4,9 @@
import Portal from '$lib/components/shared-components/portal/portal.svelte'; import Portal from '$lib/components/shared-components/portal/portal.svelte';
import DuplicateAsset from '$lib/components/utilities-page/duplicates/duplicate-asset.svelte'; import DuplicateAsset from '$lib/components/utilities-page/duplicates/duplicate-asset.svelte';
import { assetViewingStore } from '$lib/stores/asset-viewing.store'; import { assetViewingStore } from '$lib/stores/asset-viewing.store';
import { suggestDuplicateByFileSize } from '$lib/utils';
import { type AssetResponseDto } from '@immich/sdk'; import { type AssetResponseDto } from '@immich/sdk';
import { mdiCheck, mdiTrashCanOutline } from '@mdi/js'; import { mdiCheck, mdiTrashCanOutline } from '@mdi/js';
import { sortBy } from 'lodash-es';
import { onDestroy, onMount } from 'svelte'; import { onDestroy, onMount } from 'svelte';
import { t } from 'svelte-i18n'; import { t } from 'svelte-i18n';
@ -20,7 +20,7 @@
$: trashCount = assets.length - selectedAssetIds.size; $: trashCount = assets.length - selectedAssetIds.size;
onMount(() => { onMount(() => {
const suggestedAsset = sortBy(assets, (asset) => asset.exifInfo?.fileSizeInByte).pop(); const suggestedAsset = suggestDuplicateByFileSize(assets);
if (!suggestedAsset) { if (!suggestedAsset) {
selectedAssetIds = new Set(assets[0].id); selectedAssetIds = new Set(assets[0].id);

View file

@ -323,6 +323,9 @@
"back": "Back", "back": "Back",
"backward": "Backward", "backward": "Backward",
"blurred_background": "Blurred background", "blurred_background": "Blurred background",
"bulk_delete_duplicates_confirmation": "Are you sure you want to bulk delete {count} duplicate assets? This will keep the largest asset of each group and permanently delete all other duplicates. You cannot undo this action!",
"bulk_keep_duplicates_confirmation": "Are you sure you want to keep {count} duplicate assets? This will resolve all duplicate groups without deleting anything.",
"bulk_trash_duplicates_confirmation": "Are you sure you want to bulk trash {count} duplicate assets? This will keep the largest asset of each group and trash all other duplicates.",
"camera": "Camera", "camera": "Camera",
"camera_brand": "Camera brand", "camera_brand": "Camera brand",
"camera_model": "Camera model", "camera_model": "Camera model",
@ -392,6 +395,7 @@
"date_before": "Date before", "date_before": "Date before",
"date_range": "Date range", "date_range": "Date range",
"day": "Day", "day": "Day",
"deduplicate_all": "Deduplicate All",
"default_locale": "Default Locale", "default_locale": "Default Locale",
"default_locale_description": "Format dates and numbers based on your browser locale", "default_locale_description": "Format dates and numbers based on your browser locale",
"delete": "Delete", "delete": "Delete",
@ -699,6 +703,7 @@
"permanent_deletion_warning_setting_description": "Show a warning when permanently deleting assets", "permanent_deletion_warning_setting_description": "Show a warning when permanently deleting assets",
"permanently_delete": "Permanently delete", "permanently_delete": "Permanently delete",
"permanently_deleted_asset": "Permanently deleted asset", "permanently_deleted_asset": "Permanently deleted asset",
"permanently_deleted_assets": "Permanently deleted {count, plural, one {# asset} other {# assets}}",
"photos": "Photos", "photos": "Photos",
"photos_count": "{count, plural, one {{count, number} Photo} other {{count, number} Photos}}", "photos_count": "{count, plural, one {{count, number} Photo} other {{count, number} Photos}}",
"photos_from_previous_years": "Photos from previous years", "photos_from_previous_years": "Photos from previous years",
@ -740,6 +745,7 @@
"reset": "Reset", "reset": "Reset",
"reset_password": "Reset password", "reset_password": "Reset password",
"reset_people_visibility": "Reset people visibility", "reset_people_visibility": "Reset people visibility",
"resolved_all_duplicates": "Resolved all duplicates",
"restore": "Restore", "restore": "Restore",
"restore_all": "Restore all", "restore_all": "Restore all",
"restore_user": "Restore user", "restore_user": "Restore user",

View file

@ -15,9 +15,11 @@ import {
linkOAuthAccount, linkOAuthAccount,
startOAuth, startOAuth,
unlinkOAuthAccount, unlinkOAuthAccount,
type AssetResponseDto,
type SharedLinkResponseDto, type SharedLinkResponseDto,
} from '@immich/sdk'; } from '@immich/sdk';
import { mdiCogRefreshOutline, mdiDatabaseRefreshOutline, mdiImageRefreshOutline } from '@mdi/js'; import { mdiCogRefreshOutline, mdiDatabaseRefreshOutline, mdiImageRefreshOutline } from '@mdi/js';
import { sortBy } from 'lodash-es';
import { t } from 'svelte-i18n'; import { t } from 'svelte-i18n';
import { derived, get } from 'svelte/store'; import { derived, get } from 'svelte/store';
@ -310,3 +312,7 @@ export const withError = async <T>(fn: () => Promise<T>): Promise<[undefined, T]
return [error, undefined]; return [error, undefined];
} }
}; };
export const suggestDuplicateByFileSize = (assets: AssetResponseDto[]): AssetResponseDto | undefined => {
return sortBy(assets, (asset) => asset.exifInfo?.fileSizeInByte).pop();
};

View file

@ -11,42 +11,129 @@
import { deleteAssets, updateAssets } from '@immich/sdk'; import { deleteAssets, updateAssets } from '@immich/sdk';
import { t } from 'svelte-i18n'; import { t } from 'svelte-i18n';
import type { PageData } from './$types'; import type { PageData } from './$types';
import { suggestDuplicateByFileSize } from '$lib/utils';
import LinkButton from '$lib/components/elements/buttons/link-button.svelte';
import { mdiCheckOutline, mdiTrashCanOutline } from '@mdi/js';
import Icon from '$lib/components/elements/icon.svelte';
export let data: PageData; export let data: PageData;
const handleResolve = async (duplicateId: string, duplicateAssetIds: string[], trashIds: string[]) => { $: hasDuplicates = data.duplicates.length > 0;
try {
if (!$featureFlags.trash && trashIds.length > 0) {
const isConfirmed = await dialogController.show({
prompt: $t('delete_duplicates_confirmation'),
confirmText: $t('permanently_delete'),
});
if (!isConfirmed) { const withConfirmation = async (callback: () => Promise<void>, prompt?: string, confirmText?: string) => {
return; if (prompt && confirmText) {
} const isConfirmed = await dialogController.show({ prompt, confirmText });
} if (!isConfirmed) {
await updateAssets({ assetBulkUpdateDto: { ids: duplicateAssetIds, duplicateId: null } });
await deleteAssets({ assetBulkDeleteDto: { ids: trashIds, force: !$featureFlags.trash } });
data.duplicates = data.duplicates.filter((duplicate) => duplicate.duplicateId !== duplicateId);
if (trashIds.length === 0) {
return; return;
} }
}
notificationController.show({ try {
message: $t('assets_moved_to_trash', { values: { count: trashIds.length } }), return await callback();
type: NotificationType.Info,
});
} catch (error) { } catch (error) {
handleError(error, $t('errors.unable_to_resolve_duplicate')); handleError(error, $t('errors.unable_to_resolve_duplicate'));
} }
}; };
const deletedNotification = (trashedCount: number) => {
if (!trashedCount) {
return;
}
notificationController.show({
message: $featureFlags.trash
? $t('assets_moved_to_trash', { values: { count: trashedCount } })
: $t('permanently_deleted_assets', { values: { count: trashedCount } }),
type: NotificationType.Info,
});
};
const handleResolve = async (duplicateId: string, duplicateAssetIds: string[], trashIds: string[]) => {
return withConfirmation(
async () => {
await deleteAssets({ assetBulkDeleteDto: { ids: trashIds, force: !$featureFlags.trash } });
await updateAssets({ assetBulkUpdateDto: { ids: duplicateAssetIds, duplicateId: null } });
data.duplicates = data.duplicates.filter((duplicate) => duplicate.duplicateId !== duplicateId);
deletedNotification(trashIds.length);
},
trashIds.length > 0 && !$featureFlags.trash ? $t('delete_duplicates_confirmation') : undefined,
trashIds.length > 0 && !$featureFlags.trash ? $t('permanently_delete') : undefined,
);
};
const handleDeduplicateAll = async () => {
const idsToKeep = data.duplicates
.map((group) => suggestDuplicateByFileSize(group.assets))
.map((asset) => asset?.id);
const idsToDelete = data.duplicates.flatMap((group, i) =>
group.assets.map((asset) => asset.id).filter((asset) => asset !== idsToKeep[i]),
);
let prompt, confirmText;
if ($featureFlags.trash) {
prompt = $t('bulk_trash_duplicates_confirmation', { values: { count: idsToDelete.length } });
confirmText = $t('confirm');
} else {
prompt = $t('bulk_delete_duplicates_confirmation', { values: { count: idsToDelete.length } });
confirmText = $t('permanently_delete');
}
return withConfirmation(
async () => {
await deleteAssets({ assetBulkDeleteDto: { ids: idsToDelete, force: !$featureFlags.trash } });
await updateAssets({
assetBulkUpdateDto: {
ids: [...idsToDelete, ...idsToKeep.filter((id): id is string => !!id)],
duplicateId: null,
},
});
data.duplicates = [];
deletedNotification(idsToDelete.length);
},
prompt,
confirmText,
);
};
const handleKeepAll = async () => {
const ids = data.duplicates.flatMap((group) => group.assets.map((asset) => asset.id));
return withConfirmation(
async () => {
await updateAssets({ assetBulkUpdateDto: { ids, duplicateId: null } });
data.duplicates = [];
notificationController.show({
message: $t('resolved_all_duplicates'),
type: NotificationType.Info,
});
},
$t('bulk_keep_duplicates_confirmation', { values: { count: ids.length } }),
$t('confirm'),
);
};
</script> </script>
<UserPageLayout title={data.meta.title + ` (${data.duplicates.length})`} scrollbar={true}> <UserPageLayout title={data.meta.title + ` (${data.duplicates.length})`} scrollbar={true}>
<div class="flex place-items-center gap-2" slot="buttons">
<LinkButton on:click={() => handleDeduplicateAll()} disabled={!hasDuplicates}>
<div class="flex place-items-center gap-2 text-sm">
<Icon path={mdiTrashCanOutline} size="18" />
{$t('deduplicate_all')}
</div>
</LinkButton>
<LinkButton on:click={() => handleKeepAll()} disabled={!hasDuplicates}>
<div class="flex place-items-center gap-2 text-sm">
<Icon path={mdiCheckOutline} size="18" />
{$t('keep_all')}
</div>
</LinkButton>
</div>
<div class="mt-4"> <div class="mt-4">
{#if data.duplicates && data.duplicates.length > 0} {#if data.duplicates && data.duplicates.length > 0}
<div class="mb-4 text-sm dark:text-white"> <div class="mb-4 text-sm dark:text-white">