minty/src/app/state/upload.ts
Ajay Bura 20db27fa7e
feat: URL navigation in auth (#1603)
* bump to react 18 and install react-router-dom

* Upgrade to react 18 root

* update vite

* add cs api's

* convert state/auth to ts

* add client config context

* add auto discovery context

* add spec version context

* add auth flow context

* add background dot pattern css

* add promise utils

* init url based routing

* update auth route server path as effect

* add auth server hook

* always use server from discovery info in context

* login - WIP

* upgrade jotai to v2

* add atom with localStorage util

* add multi account sessions atom

* add default IGNORE res to auto discovery

* add error type in async callback hook

* handle password login error

* fix async callback hook

* allow password login

* Show custom server not allowed error in mxId login

* add sso login component

* add token login

* fix hardcoded m.login.password in login func

* update server input on url change

* Improve sso login labels

* update folds

* fix async callback batching state update in safari

* wrap async callback set state in queueMicrotask

* wip

* wip - register

* arrange auth file structure

* add error codes

* extract filed error component form password login

* add register util function

* handle register flow - WIP

* update unsupported auth flow method reasons

* improve password input styles

* Improve UIA flow next stage calculation
complete stages can have any order so we will look for first stage which is not in completed

* process register UIA flow stages

* Extract register UIA stages component

* improve register error messages

* add focus trap & step count in UIA stages

* add reset password path and path utils

* add path with origin hook

* fix sso redirect url

* rename register token query param to token

* restyle auth screen header

* add reset password component - WIP

* add reset password form

* add netlify rewrites

* fix netlify file indentation

* test netlify redirect

* fix vite to include netlify toml

* add more netlify redirects

* add splat to public and assets path

* fix vite base name

* add option to use hash router in config and remove appVersion

* add splash screen component

* add client config loading and error screen

* fix server picker bug

* fix reset password email input type

* make auth page small screen responsive

* fix typo in reset password screen
2024-01-21 18:20:56 +05:30

146 lines
3.7 KiB
TypeScript

import { atom, useAtom } from 'jotai';
import { atomFamily } from 'jotai/utils';
import { MatrixClient, UploadResponse, UploadProgress, MatrixError } from 'matrix-js-sdk';
import { useCallback } from 'react';
import { useThrottle } from '../hooks/useThrottle';
import { uploadContent, TUploadContent } from '../utils/matrix';
export enum UploadStatus {
Idle = 'idle',
Loading = 'loading',
Success = 'success',
Error = 'error',
}
export type UploadIdle = {
file: TUploadContent;
status: UploadStatus.Idle;
};
export type UploadLoading = {
file: TUploadContent;
status: UploadStatus.Loading;
promise: Promise<UploadResponse>;
progress: UploadProgress;
};
export type UploadSuccess = {
file: TUploadContent;
status: UploadStatus.Success;
mxc: string;
};
export type UploadError = {
file: TUploadContent;
status: UploadStatus.Error;
error: MatrixError;
};
export type Upload = UploadIdle | UploadLoading | UploadSuccess | UploadError;
export type UploadAtomAction =
| {
promise: Promise<UploadResponse>;
}
| {
progress: UploadProgress;
}
| {
mxc: string;
}
| {
error: MatrixError;
};
export const createUploadAtom = (file: TUploadContent) => {
const baseUploadAtom = atom<Upload>({
file,
status: UploadStatus.Idle,
});
return atom<Upload, [UploadAtomAction], undefined>(
(get) => get(baseUploadAtom),
(get, set, update) => {
const uploadState = get(baseUploadAtom);
if ('promise' in update) {
set(baseUploadAtom, {
status: UploadStatus.Loading,
file,
promise: update.promise,
progress: { loaded: 0, total: file.size },
});
return;
}
if ('progress' in update && uploadState.status === UploadStatus.Loading) {
set(baseUploadAtom, {
...uploadState,
progress: update.progress,
});
return;
}
if ('mxc' in update) {
set(baseUploadAtom, {
status: UploadStatus.Success,
file,
mxc: update.mxc,
});
return;
}
if ('error' in update) {
set(baseUploadAtom, {
status: UploadStatus.Error,
file,
error: update.error,
});
}
}
);
};
export type TUploadAtom = ReturnType<typeof createUploadAtom>;
export const useBindUploadAtom = (
mx: MatrixClient,
file: TUploadContent,
uploadAtom: TUploadAtom,
hideFilename?: boolean
) => {
const [upload, setUpload] = useAtom(uploadAtom);
const handleProgress = useThrottle(
useCallback((progress: UploadProgress) => setUpload({ progress }), [setUpload]),
{ immediate: true, wait: 200 }
);
const startUpload = useCallback(
() =>
uploadContent(mx, file, {
hideFilename,
onPromise: (promise: Promise<UploadResponse>) => setUpload({ promise }),
onProgress: handleProgress,
onSuccess: (mxc) => setUpload({ mxc }),
onError: (error) => setUpload({ error }),
}),
[mx, file, hideFilename, setUpload, handleProgress]
);
const cancelUpload = useCallback(async () => {
if (upload.status === UploadStatus.Loading) {
await mx.cancelUpload(upload.promise);
}
}, [mx, upload]);
return {
upload,
startUpload,
cancelUpload,
};
};
export const createUploadAtomFamily = () =>
atomFamily<TUploadContent, TUploadAtom>(createUploadAtom);
export type TUploadAtomFamily = ReturnType<typeof createUploadAtomFamily>;
export const createUploadFamilyObserverAtom = (
uploadFamily: TUploadAtomFamily,
uploads: TUploadContent[]
) => atom<Upload[]>((get) => uploads.map((upload) => get(uploadFamily(upload))));
export type TUploadFamilyObserverAtom = ReturnType<typeof createUploadFamilyObserverAtom>;