mirror of
https://github.com/immich-app/immich.git
synced 2025-01-09 21:36:46 +01:00
01d6707b59
* feat: add an option to change the date formats * pr feedback * fix: change title * fix: show list supported by the browser * fix: tests * fix: dates * fix: check only if locale is set * fix: better fallback value * fix: fallback * fix: fallback * feat: add default locale option * refactor: shared components * refactor: shared components * prepare for svelte 5 * don't use relative paths * refactor: fallback value Co-authored-by: Michel Heusschen <59014050+michelheusschen@users.noreply.github.com> * fix: parsing store * fix: lint * refactor: locales --------- Co-authored-by: Michel Heusschen <59014050+michelheusschen@users.noreply.github.com> Co-authored-by: Jason Rasmussen <jrasm91@gmail.com>
196 lines
5.8 KiB
TypeScript
196 lines
5.8 KiB
TypeScript
import { goto } from '$app/navigation';
|
|
import { page } from '$app/stores';
|
|
import { NotificationType, notificationController } from '$lib/components/shared-components/notification/notification';
|
|
import { locales } from '$lib/constants';
|
|
import { handleError } from '$lib/utils/handle-error';
|
|
import {
|
|
AssetJobName,
|
|
JobName,
|
|
ThumbnailFormat,
|
|
defaults,
|
|
finishOAuth,
|
|
linkOAuthAccount,
|
|
startOAuth,
|
|
unlinkOAuthAccount,
|
|
type UserResponseDto,
|
|
} from '@immich/sdk';
|
|
import { get } from 'svelte/store';
|
|
|
|
interface UpdateParamAction {
|
|
param: string;
|
|
value: string;
|
|
add: boolean;
|
|
}
|
|
|
|
const getParamValues = (param: string) =>
|
|
new Set((get(page).url.searchParams.get(param) || '').split(' ').filter((x) => x !== ''));
|
|
|
|
export const hasParamValue = (param: string, value: string) => getParamValues(param).has(value);
|
|
|
|
export const updateParamList = async ({ param, value, add }: UpdateParamAction) => {
|
|
const values = getParamValues(param);
|
|
|
|
if (add) {
|
|
values.add(value);
|
|
} else {
|
|
values.delete(value);
|
|
}
|
|
|
|
const searchParams = new URLSearchParams(get(page).url.searchParams);
|
|
searchParams.set(param, [...values.values()].join(' '));
|
|
|
|
if (values.size === 0) {
|
|
searchParams.delete(param);
|
|
}
|
|
|
|
await goto(`?${searchParams.toString()}`, { replaceState: true, noScroll: true, keepFocus: true });
|
|
};
|
|
|
|
export const getJobName = (jobName: JobName) => {
|
|
const names: Record<JobName, string> = {
|
|
[JobName.ThumbnailGeneration]: 'Generate Thumbnails',
|
|
[JobName.MetadataExtraction]: 'Extract Metadata',
|
|
[JobName.Sidecar]: 'Sidecar Metadata',
|
|
[JobName.SmartSearch]: 'Smart Search',
|
|
[JobName.FaceDetection]: 'Face Detection',
|
|
[JobName.FacialRecognition]: 'Facial Recognition',
|
|
[JobName.VideoConversion]: 'Transcode Videos',
|
|
[JobName.StorageTemplateMigration]: 'Storage Template Migration',
|
|
[JobName.Migration]: 'Migration',
|
|
[JobName.BackgroundTask]: 'Background Tasks',
|
|
[JobName.Search]: 'Search',
|
|
[JobName.Library]: 'Library',
|
|
};
|
|
|
|
return names[jobName];
|
|
};
|
|
|
|
let _key: string | undefined;
|
|
|
|
export const setKey = (key: string) => {
|
|
_key = key;
|
|
};
|
|
|
|
export const getKey = (): string | undefined => {
|
|
return _key;
|
|
};
|
|
|
|
export const isSharedLink = () => {
|
|
return !!_key;
|
|
};
|
|
|
|
const createUrl = (path: string, parameters?: Record<string, unknown>) => {
|
|
const searchParameters = new URLSearchParams();
|
|
for (const key in parameters) {
|
|
const value = parameters[key];
|
|
if (value !== undefined && value !== null) {
|
|
searchParameters.set(key, value.toString());
|
|
}
|
|
}
|
|
|
|
const url = new URL(path, 'https://example.com');
|
|
url.search = searchParameters.toString();
|
|
|
|
return defaults.baseUrl + url.pathname + url.search + url.hash;
|
|
};
|
|
|
|
export const getAssetFileUrl = (...[assetId, isWeb, isThumb]: [string, boolean, boolean]) => {
|
|
const path = `/asset/file/${assetId}`;
|
|
return createUrl(path, { isThumb, isWeb, key: getKey() });
|
|
};
|
|
|
|
export const getAssetThumbnailUrl = (...[assetId, format]: [string, ThumbnailFormat | undefined]) => {
|
|
const path = `/asset/thumbnail/${assetId}`;
|
|
return createUrl(path, { format, key: getKey() });
|
|
};
|
|
|
|
export const getProfileImageUrl = (...[userId]: [string]) => {
|
|
const path = `/user/profile-image/${userId}`;
|
|
return createUrl(path);
|
|
};
|
|
|
|
export const getPeopleThumbnailUrl = (personId: string) => {
|
|
const path = `/person/${personId}/thumbnail`;
|
|
return createUrl(path);
|
|
};
|
|
|
|
export const getAssetJobName = (job: AssetJobName) => {
|
|
const names: Record<AssetJobName, string> = {
|
|
[AssetJobName.RefreshMetadata]: 'Refresh metadata',
|
|
[AssetJobName.RegenerateThumbnail]: 'Refresh thumbnails',
|
|
[AssetJobName.TranscodeVideo]: 'Refresh encoded videos',
|
|
};
|
|
|
|
return names[job];
|
|
};
|
|
|
|
export const getAssetJobMessage = (job: AssetJobName) => {
|
|
const messages: Record<AssetJobName, string> = {
|
|
[AssetJobName.RefreshMetadata]: 'Refreshing metadata',
|
|
[AssetJobName.RegenerateThumbnail]: `Regenerating thumbnails`,
|
|
[AssetJobName.TranscodeVideo]: `Refreshing encoded video`,
|
|
};
|
|
|
|
return messages[job];
|
|
};
|
|
|
|
export const copyToClipboard = async (secret: string) => {
|
|
try {
|
|
await navigator.clipboard.writeText(secret);
|
|
notificationController.show({ message: 'Copied to clipboard!', type: NotificationType.Info });
|
|
} catch (error) {
|
|
handleError(error, 'Cannot copy to clipboard, make sure you are accessing the page through https');
|
|
}
|
|
};
|
|
|
|
export const makeSharedLinkUrl = (externalDomain: string, key: string) => {
|
|
let url = externalDomain || window.location.origin;
|
|
if (!url.endsWith('/')) {
|
|
url += '/';
|
|
}
|
|
return `${url}share/${key}`;
|
|
};
|
|
|
|
export const oauth = {
|
|
isCallback: (location: Location) => {
|
|
const search = location.search;
|
|
return search.includes('code=') || search.includes('error=');
|
|
},
|
|
isAutoLaunchDisabled: (location: Location) => {
|
|
const values = ['autoLaunch=0', 'password=1', 'password=true'];
|
|
for (const value of values) {
|
|
if (location.search.includes(value)) {
|
|
return true;
|
|
}
|
|
}
|
|
return false;
|
|
},
|
|
authorize: async (location: Location) => {
|
|
try {
|
|
const redirectUri = location.href.split('?')[0];
|
|
const { url } = await startOAuth({ oAuthConfigDto: { redirectUri } });
|
|
window.location.href = url;
|
|
return true;
|
|
} catch (error) {
|
|
handleError(error, 'Unable to login with OAuth');
|
|
return false;
|
|
}
|
|
},
|
|
login: (location: Location) => {
|
|
return finishOAuth({ oAuthCallbackDto: { url: location.href } });
|
|
},
|
|
link: (location: Location): Promise<UserResponseDto> => {
|
|
return linkOAuthAccount({ oAuthCallbackDto: { url: location.href } });
|
|
},
|
|
unlink: () => {
|
|
return unlinkOAuthAccount();
|
|
},
|
|
};
|
|
|
|
export const findLocale = (code: string | undefined) => {
|
|
const language = locales.find((lang) => lang.code === code);
|
|
return {
|
|
code: language?.code,
|
|
name: language?.name,
|
|
};
|
|
};
|