From f4ec8425775c37c25e11c89b3dfd5b11de46b42d Mon Sep 17 00:00:00 2001
From: Jason Rasmussen <jason@rasm.me>
Date: Wed, 4 Sep 2024 23:38:55 -0400
Subject: [PATCH] refactor(web): upload panel (#12326)

Co-authored-by: Alex <alex.tran1502@gmail.com>
---
 web/src/lib/components/elements/icon.svelte   |   3 +-
 .../upload-asset-preview.svelte               | 146 +++++++++---------
 .../shared-components/upload-panel.svelte     |  37 +++--
 web/src/lib/models/upload-asset.ts            |   2 +-
 web/src/lib/stores/upload.ts                  |  89 +++++++----
 web/src/lib/utils/file-uploader.ts            |  47 ++++--
 6 files changed, 184 insertions(+), 140 deletions(-)

diff --git a/web/src/lib/components/elements/icon.svelte b/web/src/lib/components/elements/icon.svelte
index bb22276286..5965928718 100644
--- a/web/src/lib/components/elements/icon.svelte
+++ b/web/src/lib/components/elements/icon.svelte
@@ -16,13 +16,14 @@
   export let ariaLabelledby: string | undefined = undefined;
   export let strokeWidth: number = 0;
   export let strokeColor: string = 'currentColor';
+  export let spin = false;
 </script>
 
 <svg
   width={size}
   height={size}
   {viewBox}
-  class="{className} {flipped ? '-scale-x-100' : ''}"
+  class="{className} {flipped ? '-scale-x-100' : ''} {spin ? 'animate-spin' : ''}"
   {role}
   stroke={strokeColor}
   stroke-width={strokeWidth}
diff --git a/web/src/lib/components/shared-components/upload-asset-preview.svelte b/web/src/lib/components/shared-components/upload-asset-preview.svelte
index d3f12555c7..a7ba3430a0 100644
--- a/web/src/lib/components/shared-components/upload-asset-preview.svelte
+++ b/web/src/lib/components/shared-components/upload-asset-preview.svelte
@@ -1,21 +1,32 @@
 <script lang="ts">
+  import Icon from '$lib/components/elements/icon.svelte';
+  import { AppRoute } from '$lib/constants';
   import type { UploadAsset } from '$lib/models/upload-asset';
   import { UploadState } from '$lib/models/upload-asset';
   import { locale } from '$lib/stores/preferences.store';
-  import { getByteUnitString } from '$lib/utils/byte-units';
-  import { fade } from 'svelte/transition';
-  import ImmichLogo from './immich-logo.svelte';
-  import { getFilenameExtension } from '$lib/utils/asset-utils';
   import { uploadAssetsStore } from '$lib/stores/upload';
-  import Icon from '$lib/components/elements/icon.svelte';
+  import { getByteUnitString } from '$lib/utils/byte-units';
   import { fileUploadHandler } from '$lib/utils/file-uploader';
-  import { mdiRefresh, mdiCancel } from '@mdi/js';
+  import {
+    mdiAlertCircle,
+    mdiCheckCircle,
+    mdiCircleOutline,
+    mdiClose,
+    mdiLoading,
+    mdiOpenInNew,
+    mdiRestart,
+  } from '@mdi/js';
   import { t } from 'svelte-i18n';
+  import { fade } from 'svelte/transition';
 
   export let uploadAsset: UploadAsset;
 
+  const handleDismiss = (uploadAsset: UploadAsset) => {
+    uploadAssetsStore.removeItem(uploadAsset.id);
+  };
+
   const handleRetry = async (uploadAsset: UploadAsset) => {
-    uploadAssetsStore.removeUploadAsset(uploadAsset.id);
+    uploadAssetsStore.removeItem(uploadAsset.id);
     await fileUploadHandler([uploadAsset.file], uploadAsset.albumId);
   };
 </script>
@@ -23,86 +34,69 @@
 <div
   in:fade={{ duration: 250 }}
   out:fade={{ duration: 100 }}
-  class="flex flex-col rounded-lg bg-immich-bg text-xs dark:bg-immich-dark-bg"
+  class="flex flex-col rounded-lg bg-immich-bg text-xs dark:bg-immich-dark-bg p-2 gap-1"
 >
-  <div class="grid grid-cols-[65px_auto_auto] max-h-[70px]">
-    <div class="relative">
-      <div in:fade={{ duration: 250 }}>
-        <ImmichLogo noText class="h-[65px] w-[65px] rounded-bl-lg rounded-tl-lg object-cover p-2" />
-      </div>
-      <div class="absolute bottom-0 left-0 h-[25px] w-full rounded-bl-md bg-immich-primary/30">
-        <p
-          class="absolute bottom-1 right-1 stroke-immich-primary object-right-bottom font-semibold uppercase text-white/95 dark:text-gray-100"
-        >
-          .{getFilenameExtension(uploadAsset.file.name)}
-        </p>
-      </div>
+  <div class="flex items-center gap-2">
+    <div class="flex items-center justify-center">
+      {#if uploadAsset.state === UploadState.PENDING}
+        <Icon path={mdiCircleOutline} size="24" class="text-immich-primary" title={$t('pending')} />
+      {:else if uploadAsset.state === UploadState.STARTED}
+        <Icon path={mdiLoading} size="24" spin class="text-immich-primary" title={$t('asset_skipped')} />
+      {:else if uploadAsset.state === UploadState.ERROR}
+        <Icon path={mdiAlertCircle} size="24" class="text-immich-error" title={$t('error')} />
+      {:else if uploadAsset.state === UploadState.DUPLICATED}
+        <Icon path={mdiAlertCircle} size="24" class="text-immich-warning" title={$t('asset_skipped')} />
+      {:else if uploadAsset.state === UploadState.DONE}
+        <Icon path={mdiCheckCircle} size="24" class="text-immich-success" title={$t('asset_uploaded')} />
+      {/if}
     </div>
-    <div class="flex flex-col justify-between p-2 pr-2">
-      <input
-        disabled
-        class="w-full rounded-md border bg-gray-100 p-1 px-2 text-[10px] dark:border-immich-dark-gray dark:bg-gray-900"
-        value={`[${getByteUnitString(uploadAsset.file.size, $locale)}] ${uploadAsset.file.name}`}
-      />
+    <!-- <span>[{getByteUnitString(uploadAsset.file.size, $locale)}]</span> -->
+    <span class="grow break-all">{uploadAsset.file.name}</span>
 
-      <div
-        class="relative mt-[5px] h-[15px] w-full rounded-md bg-gray-300 text-white dark:bg-immich-dark-gray"
-        class:dark:text-black={uploadAsset.state === UploadState.STARTED}
-      >
-        {#if uploadAsset.state === UploadState.STARTED}
-          <div class="h-[15px] rounded-md bg-immich-primary transition-all" style={`width: ${uploadAsset.progress}%`} />
-          <p class="absolute top-0 h-full w-full text-center text-[10px]">
-            {#if uploadAsset.message}
-              {uploadAsset.message}
-            {:else}
-              {uploadAsset.progress}% - {getByteUnitString(uploadAsset.speed || 0, $locale)}/s - {uploadAsset.eta}s
-            {/if}
-          </p>
-        {:else if uploadAsset.state === UploadState.PENDING}
-          <div class="h-[15px] rounded-md bg-immich-dark-gray transition-all dark:bg-immich-gray" style="width: 100%" />
-          <p class="absolute top-0 h-full w-full text-center text-[10px]">{$t('pending')}</p>
-        {:else if uploadAsset.state === UploadState.ERROR}
-          <div class="h-[15px] rounded-md bg-immich-error transition-all" style="width: 100%" />
-          <p class="absolute top-0 h-full w-full text-center text-[10px]">{$t('error')}</p>
-        {:else if uploadAsset.state === UploadState.DUPLICATED}
-          <div class="h-[15px] rounded-md bg-immich-warning transition-all" style="width: 100%" />
-          <p class="absolute top-0 h-full w-full text-center text-[10px]">
-            {$t('asset_skipped')}
-            {#if uploadAsset.message}
-              ({uploadAsset.message})
-            {/if}
-          </p>
-        {:else if uploadAsset.state === UploadState.DONE}
-          <div class="h-[15px] rounded-md bg-immich-success transition-all" style="width: 100%" />
-          <p class="absolute top-0 h-full w-full text-center text-[10px]">
-            {$t('asset_uploaded')}
-            {#if uploadAsset.message}
-              ({uploadAsset.message})
-            {/if}
-          </p>
-        {/if}
-      </div>
-    </div>
-    {#if uploadAsset.state === UploadState.ERROR}
-      <div class="flex h-full flex-col place-content-evenly place-items-center justify-items-center pr-2">
-        <button type="button" on:click={() => handleRetry(uploadAsset)} title={$t('retry_upload')} class="flex text-sm">
-          <span class="text-immich-dark-gray dark:text-immich-dark-fg"><Icon path={mdiRefresh} size="20" /></span>
-        </button>
-        <button
-          type="button"
-          on:click={() => uploadAssetsStore.removeUploadAsset(uploadAsset.id)}
-          title={$t('dismiss_error')}
-          class="flex text-sm"
+    {#if uploadAsset.state === UploadState.DUPLICATED && uploadAsset.assetId}
+      <div class="flex items-center justify-between gap-1">
+        <a
+          href="{AppRoute.PHOTOS}/{uploadAsset.assetId}"
+          target="_blank"
+          rel="noopener noreferrer"
+          class=""
+          aria-hidden="true"
+          tabindex={-1}
         >
-          <span class="text-immich-error"><Icon path={mdiCancel} size="20" /></span>
+          <Icon path={mdiOpenInNew} size="20" />
+        </a>
+        <button type="button" on:click={() => handleDismiss(uploadAsset)} class="" aria-hidden="true" tabindex={-1}>
+          <Icon path={mdiClose} size="20" />
+        </button>
+      </div>
+    {:else if uploadAsset.state === UploadState.ERROR}
+      <div class="flex items-center justify-between gap-1">
+        <button type="button" on:click={() => handleRetry(uploadAsset)} class="" aria-hidden="true" tabindex={-1}>
+          <Icon path={mdiRestart} size="20" />
+        </button>
+        <button type="button" on:click={() => handleDismiss(uploadAsset)} class="" aria-hidden="true" tabindex={-1}>
+          <Icon path={mdiClose} size="20" />
         </button>
       </div>
     {/if}
   </div>
 
+  {#if uploadAsset.state === UploadState.STARTED}
+    <div class="text-black relative mt-[5px] h-[15px] w-full rounded-md bg-gray-300 dark:bg-immich-dark-gray">
+      <div class="h-[15px] rounded-md bg-immich-primary transition-all" style={`width: ${uploadAsset.progress}%`} />
+      <p class="absolute top-0 h-full w-full text-center text-[10px]">
+        {#if uploadAsset.message}
+          {uploadAsset.message}
+        {:else}
+          {uploadAsset.progress}% - {getByteUnitString(uploadAsset.speed || 0, $locale)}/s - {uploadAsset.eta}s
+        {/if}
+      </p>
+    </div>
+  {/if}
+
   {#if uploadAsset.state === UploadState.ERROR}
     <div class="flex flex-row justify-between">
-      <p class="w-full rounded-md py-1 px-2 text-justify text-[10px] text-immich-error">
+      <p class="w-full rounded-md text-justify text-immich-error">
         {uploadAsset.error}
       </p>
     </div>
diff --git a/web/src/lib/components/shared-components/upload-panel.svelte b/web/src/lib/components/shared-components/upload-panel.svelte
index ee213d7969..d536053286 100644
--- a/web/src/lib/components/shared-components/upload-panel.svelte
+++ b/web/src/lib/components/shared-components/upload-panel.svelte
@@ -15,8 +15,7 @@
   let showOptions = false;
   let concurrency = uploadExecutionQueue.concurrency;
 
-  let { isUploading, hasError, remainingUploads, errorCounter, duplicateCounter, successCounter, totalUploadCounter } =
-    uploadAssetsStore;
+  let { stats, isDismissible, isUploading, remainingUploads } = uploadAssetsStore;
 
   const autoHide = () => {
     if (!$isUploading && showDetail) {
@@ -33,29 +32,29 @@
   }
 </script>
 
-{#if $hasError || $isUploading}
+{#if $isUploading}
   <div
     in:fade={{ duration: 250 }}
     out:fade={{ duration: 250 }}
     on:outroend={() => {
-      if ($errorCounter > 0) {
+      if ($stats.errors > 0) {
         notificationController.show({
-          message: $t('upload_errors', { values: { count: $errorCounter } }),
+          message: $t('upload_errors', { values: { count: $stats.errors } }),
           type: NotificationType.Warning,
         });
-      } else if ($successCounter > 0) {
+      } else if ($stats.success > 0) {
         notificationController.show({
           message: $t('upload_success'),
           type: NotificationType.Info,
         });
       }
-      if ($duplicateCounter > 0) {
+      if ($stats.duplicates > 0) {
         notificationController.show({
-          message: $t('upload_skipped_duplicates', { values: { count: $duplicateCounter } }),
+          message: $t('upload_skipped_duplicates', { values: { count: $stats.duplicates } }),
           type: NotificationType.Warning,
         });
       }
-      uploadAssetsStore.resetStore();
+      uploadAssetsStore.reset();
     }}
     class="fixed bottom-6 right-6 z-[10000]"
   >
@@ -70,20 +69,20 @@
               {$t('upload_progress', {
                 values: {
                   remaining: $remainingUploads,
-                  processed: $successCounter + $errorCounter,
-                  total: $totalUploadCounter,
+                  processed: $stats.total - $remainingUploads,
+                  total: $stats.total,
                 },
               })}
             </p>
             <p class="immich-form-label text-xs">
               {$t('upload_status_uploaded')}
-              <span class="text-immich-success">{$successCounter.toLocaleString($locale)}</span>
+              <span class="text-immich-success">{$stats.success.toLocaleString($locale)}</span>
               -
               {$t('upload_status_errors')}
-              <span class="text-immich-error">{$errorCounter.toLocaleString($locale)}</span>
+              <span class="text-immich-error">{$stats.errors.toLocaleString($locale)}</span>
               -
               {$t('upload_status_duplicates')}
-              <span class="text-immich-warning">{$duplicateCounter.toLocaleString($locale)}</span>
+              <span class="text-immich-warning">{$stats.duplicates.toLocaleString($locale)}</span>
             </p>
           </div>
           <div class="flex flex-col items-end">
@@ -103,7 +102,7 @@
                 on:click={() => (showDetail = false)}
               />
             </div>
-            {#if $hasError}
+            {#if $isDismissible}
               <CircleIconButton
                 title={$t('dismiss_all_errors')}
                 icon={mdiCancel}
@@ -115,7 +114,7 @@
           </div>
         </div>
         {#if showOptions}
-          <div class="immich-scrollbar mb-4 max-h-[400px] overflow-y-auto rounded-lg pr-2">
+          <div class="immich-scrollbar mb-4 max-h-[400px] overflow-y-auto rounded-lg">
             <div class="flex h-[26px] place-items-center gap-1">
               <label class="immich-form-label" for="upload-concurrency">{$t('upload_concurrency')}</label>
             </div>
@@ -133,7 +132,7 @@
             />
           </div>
         {/if}
-        <div class="immich-scrollbar flex max-h-[400px] flex-col gap-2 overflow-y-auto rounded-lg pr-2">
+        <div class="immich-scrollbar flex max-h-[400px] flex-col gap-2 overflow-y-auto rounded-lg">
           {#each $uploadAssetsStore as uploadAsset (uploadAsset.id)}
             <UploadAssetPreview {uploadAsset} />
           {/each}
@@ -149,14 +148,14 @@
         >
           {$remainingUploads.toLocaleString($locale)}
         </button>
-        {#if $hasError}
+        {#if $stats.errors > 0}
           <button
             type="button"
             in:scale={{ duration: 250, easing: quartInOut }}
             on:click={() => (showDetail = true)}
             class="absolute -right-4 -top-4 flex h-10 w-10 place-content-center place-items-center rounded-full bg-immich-error p-5 text-xs text-gray-200"
           >
-            {$errorCounter.toLocaleString($locale)}
+            {$stats.errors.toLocaleString($locale)}
           </button>
         {/if}
         <button
diff --git a/web/src/lib/models/upload-asset.ts b/web/src/lib/models/upload-asset.ts
index c213f77e3e..eb8cb44c5e 100644
--- a/web/src/lib/models/upload-asset.ts
+++ b/web/src/lib/models/upload-asset.ts
@@ -9,8 +9,8 @@ export enum UploadState {
 export type UploadAsset = {
   id: string;
   file: File;
-  albumId?: string;
   assetId?: string;
+  albumId?: string;
   progress?: number;
   state?: UploadState;
   startDate?: number;
diff --git a/web/src/lib/stores/upload.ts b/web/src/lib/stores/upload.ts
index 16f967edb6..4bf597b932 100644
--- a/web/src/lib/stores/upload.ts
+++ b/web/src/lib/stores/upload.ts
@@ -3,32 +3,36 @@ import { UploadState, type UploadAsset } from '../models/upload-asset';
 
 function createUploadStore() {
   const uploadAssets = writable<Array<UploadAsset>>([]);
-
-  const duplicateCounter = writable(0);
-  const successCounter = writable(0);
-  const totalUploadCounter = writable(0);
+  const stats = writable<{ errors: number; duplicates: number; success: number; total: number }>({
+    errors: 0,
+    duplicates: 0,
+    success: 0,
+    total: 0,
+  });
 
   const { subscribe } = uploadAssets;
 
-  const isUploading = derived(uploadAssets, ($uploadAssets) => {
-    return $uploadAssets.length > 0;
-  });
-  const errorsAssets = derived(uploadAssets, (a) => a.filter((e) => e.state === UploadState.ERROR));
-  const errorCounter = derived(errorsAssets, (values) => values.length);
-  const hasError = derived(errorCounter, (values) => values > 0);
+  const isUploading = derived(uploadAssets, (items) => items.length > 0);
+  const isDismissible = derived(uploadAssets, (items) =>
+    items.some((item) => item.state === UploadState.ERROR || item.state === UploadState.DUPLICATED),
+  );
   const remainingUploads = derived(
     uploadAssets,
     (values) => values.filter((a) => a.state === UploadState.PENDING || a.state === UploadState.STARTED).length,
   );
 
-  const addNewUploadAsset = (newAsset: UploadAsset) => {
+  const addItem = (newAsset: UploadAsset) => {
     uploadAssets.update(($assets) => {
       const duplicate = $assets.find((asset) => asset.id === newAsset.id);
       if (duplicate) {
         return $assets.map((asset) => (asset.id === newAsset.id ? newAsset : asset));
       }
 
-      totalUploadCounter.update((c) => c + 1);
+      stats.update((stats) => {
+        stats.total++;
+        return stats;
+      });
+
       $assets.push({
         ...newAsset,
         speed: 0,
@@ -36,6 +40,7 @@ function createUploadStore() {
         progress: 0,
         eta: 0,
       });
+
       return $assets;
     });
   };
@@ -53,7 +58,7 @@ function createUploadStore() {
   };
 
   const markStarted = (id: string) => {
-    updateAsset(id, {
+    updateItem(id, {
       state: UploadState.STARTED,
       startDate: Date.now(),
     });
@@ -70,39 +75,61 @@ function createUploadStore() {
     });
   };
 
-  const updateAsset = (id: string, partialObject: Partial<UploadAsset>) => {
+  const updateItem = (id: string, partialObject: Partial<UploadAsset>) => {
     updateAssetMap(id, (v) => ({ ...v, ...partialObject }));
   };
 
-  const removeUploadAsset = (id: string) => {
+  const removeItem = (id: string) => {
     uploadAssets.update((uploadingAsset) => uploadingAsset.filter((a) => a.id != id));
   };
 
-  const dismissErrors = () => uploadAssets.update((value) => value.filter((e) => e.state !== UploadState.ERROR));
+  const dismissErrors = () =>
+    uploadAssets.update((value) =>
+      value.filter((e) => e.state !== UploadState.ERROR && e.state !== UploadState.DUPLICATED),
+    );
 
-  const resetStore = () => {
+  const reset = () => {
     uploadAssets.set([]);
-    duplicateCounter.set(0);
-    successCounter.set(0);
-    totalUploadCounter.set(0);
+    stats.set({ errors: 0, duplicates: 0, success: 0, total: 0 });
+  };
+
+  const track = (value: 'success' | 'duplicate' | 'error') => {
+    stats.update((stats) => {
+      switch (value) {
+        case 'success': {
+          stats.success++;
+          break;
+        }
+
+        case 'duplicate': {
+          stats.duplicates++;
+          break;
+        }
+
+        case 'error': {
+          stats.errors++;
+          break;
+        }
+      }
+
+      return stats;
+    });
   };
 
   return {
-    subscribe,
-    errorCounter,
-    duplicateCounter,
-    successCounter,
-    totalUploadCounter,
+    stats,
     remainingUploads,
-    hasError,
-    dismissErrors,
+    isDismissible,
     isUploading,
-    resetStore,
-    addNewUploadAsset,
+    track,
+    dismissErrors,
+    reset,
     markStarted,
+    addItem,
+    updateItem,
+    removeItem,
     updateProgress,
-    updateAsset,
-    removeUploadAsset,
+    subscribe,
   };
 }
 
diff --git a/web/src/lib/utils/file-uploader.ts b/web/src/lib/utils/file-uploader.ts
index 2d244e9bea..d7275c8aa3 100644
--- a/web/src/lib/utils/file-uploader.ts
+++ b/web/src/lib/utils/file-uploader.ts
@@ -17,6 +17,25 @@ import { t } from 'svelte-i18n';
 import { get } from 'svelte/store';
 import { getServerErrorMessage, handleError } from './handle-error';
 
+export const addDummyItems = () => {
+  uploadAssetsStore.addItem({ id: 'asset-0', file: { name: 'asset0.jpg', size: 123_456 } as File });
+  uploadAssetsStore.updateItem('asset-0', { state: UploadState.PENDING });
+  uploadAssetsStore.addItem({ id: 'asset-1', file: { name: 'asset1.jpg', size: 123_456 } as File });
+  uploadAssetsStore.updateItem('asset-1', { state: UploadState.STARTED });
+  uploadAssetsStore.updateProgress('asset-1', 75, 100);
+  uploadAssetsStore.addItem({ id: 'asset-2', file: { name: 'asset2.jpg', size: 123_456 } as File });
+  uploadAssetsStore.updateItem('asset-2', { state: UploadState.ERROR, error: new Error('Internal server error') });
+  uploadAssetsStore.addItem({ id: 'asset-3', file: { name: 'asset3.jpg', size: 123_456 } as File });
+  uploadAssetsStore.updateItem('asset-3', { state: UploadState.DUPLICATED, assetId: 'asset-2' });
+  uploadAssetsStore.addItem({ id: 'asset-4', file: { name: 'asset3.jpg', size: 123_456 } as File });
+  uploadAssetsStore.updateItem('asset-4', { state: UploadState.DONE });
+  uploadAssetsStore.track('error');
+  uploadAssetsStore.track('success');
+  uploadAssetsStore.track('duplicate');
+};
+
+// addDummyItems();
+
 let _extensions: string[];
 
 export const uploadExecutionQueue = new ExecutorQueue({ concurrency: 2 });
@@ -68,7 +87,7 @@ export const fileUploadHandler = async (files: File[], albumId?: string, assetId
   for (const file of files) {
     const name = file.name.toLowerCase();
     if (extensions.some((extension) => name.endsWith(extension))) {
-      uploadAssetsStore.addNewUploadAsset({ id: getDeviceAssetId(file), file, albumId, assetId });
+      uploadAssetsStore.addItem({ id: getDeviceAssetId(file), file, albumId });
       promises.push(uploadExecutionQueue.addTask(() => fileUploader(file, albumId, assetId)));
     }
   }
@@ -106,7 +125,7 @@ async function fileUploader(assetFile: File, albumId?: string, replaceAssetId?:
     let responseData: AssetMediaResponseDto | undefined;
     const key = getKey();
     if (crypto?.subtle?.digest && !key) {
-      uploadAssetsStore.updateAsset(deviceAssetId, { message: $t('asset_hashing') });
+      uploadAssetsStore.updateItem(deviceAssetId, { message: $t('asset_hashing') });
       await tick();
       try {
         const bytes = await assetFile.arrayBuffer();
@@ -127,7 +146,7 @@ async function fileUploader(assetFile: File, albumId?: string, replaceAssetId?:
     }
 
     if (!responseData) {
-      uploadAssetsStore.updateAsset(deviceAssetId, { message: $t('asset_uploading') });
+      uploadAssetsStore.updateItem(deviceAssetId, { message: $t('asset_uploading') });
       if (replaceAssetId) {
         const response = await uploadRequest<AssetMediaResponseDto>({
           url: getBaseUrl() + getAssetOriginalPath(replaceAssetId) + (key ? `?key=${key}` : ''),
@@ -152,30 +171,34 @@ async function fileUploader(assetFile: File, albumId?: string, replaceAssetId?:
     }
 
     if (responseData.status === AssetMediaStatus.Duplicate) {
-      uploadAssetsStore.duplicateCounter.update((count) => count + 1);
+      uploadAssetsStore.track('duplicate');
     } else {
-      uploadAssetsStore.successCounter.update((c) => c + 1);
+      uploadAssetsStore.track('success');
     }
 
     if (albumId) {
-      uploadAssetsStore.updateAsset(deviceAssetId, { message: $t('asset_adding_to_album') });
+      uploadAssetsStore.updateItem(deviceAssetId, { message: $t('asset_adding_to_album') });
       await addAssetsToAlbum(albumId, [responseData.id], false);
-      uploadAssetsStore.updateAsset(deviceAssetId, { message: $t('asset_added_to_album') });
+      uploadAssetsStore.updateItem(deviceAssetId, { message: $t('asset_added_to_album') });
     }
 
-    uploadAssetsStore.updateAsset(deviceAssetId, {
+    uploadAssetsStore.updateItem(deviceAssetId, {
       state: responseData.status === AssetMediaStatus.Duplicate ? UploadState.DUPLICATED : UploadState.DONE,
+      assetId: responseData.id,
     });
 
-    setTimeout(() => {
-      uploadAssetsStore.removeUploadAsset(deviceAssetId);
-    }, 1000);
+    if (responseData.status !== AssetMediaStatus.Duplicate) {
+      setTimeout(() => {
+        uploadAssetsStore.removeItem(deviceAssetId);
+      }, 1000);
+    }
 
     return responseData.id;
   } catch (error) {
     handleError(error, $t('errors.unable_to_upload_file'));
     const reason = getServerErrorMessage(error) || error;
-    uploadAssetsStore.updateAsset(deviceAssetId, { state: UploadState.ERROR, error: reason });
+    uploadAssetsStore.track('error');
+    uploadAssetsStore.updateItem(deviceAssetId, { state: UploadState.ERROR, error: reason });
     return;
   }
 }