1
0
Fork 0
mirror of https://github.com/immich-app/immich.git synced 2025-01-10 13:56:47 +01:00
immich/web/src/lib/stores/upload.ts
Villena Guillaume ca35e5557b
feat(web): Improved assets upload (#3850)
* Improved asset upload algorithm.

- Upload Queue: New process algorithm
- Upload Queue: Concurrency correctly respected when dragging / adding multiple group of files to the queue
- Upload Task: Add more information about progress (upload speed and remaining time)
- Upload Panel: Add more information to about the queue status (Remaining, Errors, Duplicated, Uploaded)
- Error recovery: asset information are kept in the queue to give the user a chance to read the error message
- Error recovery: on error allow the user to retry the upload or hide the error / all errors

* Support "live" editing of the upload concurrency

* Fixed some issues

* Reformat

* fix: merge, linting, dark mode, upload to share

---------

Co-authored-by: Jason Rasmussen <jrasm91@gmail.com>
2023-09-01 12:00:51 -04:00

104 lines
2.8 KiB
TypeScript

import { derived, writable } from 'svelte/store';
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 { 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 remainingUploads = derived(
uploadAssets,
(values) => values.filter((a) => a.state === UploadState.PENDING || a.state === UploadState.STARTED).length,
);
const addNewUploadAsset = (newAsset: UploadAsset) => {
totalUploadCounter.update((c) => c + 1);
uploadAssets.update((assets) => [
...assets,
{
...newAsset,
speed: 0,
state: UploadState.PENDING,
progress: 0,
eta: 0,
},
]);
};
const updateProgress = (id: string, loaded: number, total: number) => {
updateAssetMap(id, (v) => {
const uploadSpeed = v.startDate ? loaded / ((Date.now() - v.startDate) / 1000) : 0;
return {
...v,
progress: Math.floor((loaded / total) * 100),
speed: uploadSpeed,
eta: Math.ceil((total - loaded) / uploadSpeed),
};
});
};
const markStarted = (id: string) => {
updateAsset(id, {
state: UploadState.STARTED,
startDate: Date.now(),
});
};
const updateAssetMap = (id: string, mapper: (assets: UploadAsset) => UploadAsset) => {
uploadAssets.update((uploadingAssets) => {
return uploadingAssets.map((asset) => {
if (asset.id == id) {
return mapper(asset);
}
return asset;
});
});
};
const updateAsset = (id: string, partialObj: Partial<UploadAsset>) => {
updateAssetMap(id, (v) => ({ ...v, ...partialObj }));
};
const removeUploadAsset = (id: string) => {
uploadAssets.update((uploadingAsset) => uploadingAsset.filter((a) => a.id != id));
};
const dismissErrors = () => uploadAssets.update((value) => value.filter((e) => e.state !== UploadState.ERROR));
const resetStore = () => {
uploadAssets.set([]);
duplicateCounter.set(0);
successCounter.set(0);
totalUploadCounter.set(0);
};
return {
subscribe,
errorCounter,
duplicateCounter,
successCounter,
totalUploadCounter,
remainingUploads,
hasError,
dismissErrors,
isUploading,
resetStore,
addNewUploadAsset,
markStarted,
updateProgress,
updateAsset,
removeUploadAsset,
};
}
export const uploadAssetsStore = createUploadStore();