* 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
120 lines
3.6 KiB
TypeScript
120 lines
3.6 KiB
TypeScript
import produce from 'immer';
|
|
import { atom, useSetAtom } from 'jotai';
|
|
import {
|
|
ClientEvent,
|
|
MatrixClient,
|
|
MatrixEvent,
|
|
Room,
|
|
RoomEvent,
|
|
RoomStateEvent,
|
|
} from 'matrix-js-sdk';
|
|
import { useEffect } from 'react';
|
|
import { Membership, RoomToParents, StateEvent } from '../../types/matrix/room';
|
|
import {
|
|
getRoomToParents,
|
|
getSpaceChildren,
|
|
isSpace,
|
|
isValidChild,
|
|
mapParentWithChildren,
|
|
} from '../utils/room';
|
|
|
|
export type RoomToParentsAction =
|
|
| {
|
|
type: 'INITIALIZE';
|
|
roomToParents: RoomToParents;
|
|
}
|
|
| {
|
|
type: 'PUT';
|
|
parent: string;
|
|
children: string[];
|
|
}
|
|
| {
|
|
type: 'DELETE';
|
|
roomId: string;
|
|
};
|
|
|
|
const baseRoomToParents = atom<RoomToParents>(new Map());
|
|
export const roomToParentsAtom = atom<RoomToParents, [RoomToParentsAction], undefined>(
|
|
(get) => get(baseRoomToParents),
|
|
(get, set, action) => {
|
|
if (action.type === 'INITIALIZE') {
|
|
set(baseRoomToParents, action.roomToParents);
|
|
return;
|
|
}
|
|
if (action.type === 'PUT') {
|
|
set(
|
|
baseRoomToParents,
|
|
produce(get(baseRoomToParents), (draftRoomToParents) => {
|
|
mapParentWithChildren(draftRoomToParents, action.parent, action.children);
|
|
})
|
|
);
|
|
return;
|
|
}
|
|
if (action.type === 'DELETE') {
|
|
set(
|
|
baseRoomToParents,
|
|
produce(get(baseRoomToParents), (draftRoomToParents) => {
|
|
const noParentRooms: string[] = [];
|
|
draftRoomToParents.delete(action.roomId);
|
|
draftRoomToParents.forEach((parents, child) => {
|
|
parents.delete(action.roomId);
|
|
if (parents.size === 0) noParentRooms.push(child);
|
|
});
|
|
noParentRooms.forEach((room) => draftRoomToParents.delete(room));
|
|
})
|
|
);
|
|
}
|
|
}
|
|
);
|
|
|
|
export const useBindRoomToParentsAtom = (
|
|
mx: MatrixClient,
|
|
roomToParents: typeof roomToParentsAtom
|
|
) => {
|
|
const setRoomToParents = useSetAtom(roomToParents);
|
|
|
|
useEffect(() => {
|
|
setRoomToParents({ type: 'INITIALIZE', roomToParents: getRoomToParents(mx) });
|
|
|
|
const handleAddRoom = (room: Room) => {
|
|
if (isSpace(room) && room.getMyMembership() !== Membership.Invite) {
|
|
setRoomToParents({ type: 'PUT', parent: room.roomId, children: getSpaceChildren(room) });
|
|
}
|
|
};
|
|
|
|
const handleMembershipChange = (room: Room, membership: string) => {
|
|
if (isSpace(room) && membership === Membership.Join) {
|
|
setRoomToParents({ type: 'PUT', parent: room.roomId, children: getSpaceChildren(room) });
|
|
}
|
|
};
|
|
|
|
const handleStateChange = (mEvent: MatrixEvent) => {
|
|
if (mEvent.getType() === StateEvent.SpaceChild) {
|
|
const childId = mEvent.getStateKey();
|
|
const roomId = mEvent.getRoomId();
|
|
if (childId && roomId) {
|
|
if (isValidChild(mEvent)) {
|
|
setRoomToParents({ type: 'PUT', parent: roomId, children: [childId] });
|
|
} else {
|
|
setRoomToParents({ type: 'DELETE', roomId: childId });
|
|
}
|
|
}
|
|
}
|
|
};
|
|
|
|
const handleDeleteRoom = (roomId: string) => {
|
|
setRoomToParents({ type: 'DELETE', roomId });
|
|
};
|
|
|
|
mx.on(ClientEvent.Room, handleAddRoom);
|
|
mx.on(RoomEvent.MyMembership, handleMembershipChange);
|
|
mx.on(RoomStateEvent.Events, handleStateChange);
|
|
mx.on(ClientEvent.DeleteRoom, handleDeleteRoom);
|
|
return () => {
|
|
mx.removeListener(ClientEvent.Room, handleAddRoom);
|
|
mx.removeListener(RoomEvent.MyMembership, handleMembershipChange);
|
|
mx.removeListener(RoomStateEvent.Events, handleStateChange);
|
|
mx.removeListener(ClientEvent.DeleteRoom, handleDeleteRoom);
|
|
};
|
|
}, [mx, setRoomToParents]);
|
|
};
|