// Copyright 2023 The Matrix.org Foundation C.I.C. // // Licensed under the Apache License, Version 2.0 (the "License"); // you may not use this file except in compliance with the License. // You may obtain a copy of the License at // // http://www.apache.org/licenses/LICENSE-2.0 // // Unless required by applicable law or agreed to in writing, software // distributed under the License is distributed on an "AS IS" BASIS, // WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied. // See the License for the specific language governing permissions and // limitations under the License. import { Alert, H3 } from "@vector-im/compound-web"; import { atom, useAtom, useAtomValue, useSetAtom } from "jotai"; import { atomFamily } from "jotai/utils"; import { atomWithQuery } from "jotai-urql"; import { useTransition } from "react"; import { routeAtom } from "../../Router"; import { graphql } from "../../gql"; import { PageInfo } from "../../gql/graphql"; import { atomForCurrentPagination, atomWithPagination, FIRST_PAGE, Pagination, } from "../../pagination"; import BlockList from "../BlockList"; import PaginationControls from "../PaginationControls"; import UserEmail from "../UserEmail"; import AddEmailForm from "./AddEmailForm"; const QUERY = graphql(/* GraphQL */ ` query UserEmailListQuery( $userId: ID! $first: Int $after: String $last: Int $before: String ) { user(id: $userId) { id emails(first: $first, after: $after, last: $last, before: $before) { edges { cursor node { id ...UserEmail_email } } totalCount pageInfo { hasNextPage hasPreviousPage startCursor endCursor } } } } `); const PRIMARY_EMAIL_QUERY = graphql(/* GraphQL */ ` query UserPrimaryEmail($userId: ID!) { user(id: $userId) { id primaryEmail { id } } } `); export const primaryEmailResultFamily = atomFamily((userId: string) => { const primaryEmailResult = atomWithQuery({ query: PRIMARY_EMAIL_QUERY, getVariables: () => ({ userId }), }); return primaryEmailResult; }); const primaryEmailIdFamily = atomFamily((userId: string) => { const primaryEmailIdAtom = atom( async (get) => { const result = await get(primaryEmailResultFamily(userId)); return result.data?.user?.primaryEmail?.id ?? null; }, (get, set) => { set(primaryEmailResultFamily(userId)); }, ); return primaryEmailIdAtom; }); export const currentPaginationAtom = atomForCurrentPagination(); export const emailPageResultFamily = atomFamily((userId: string) => { const emailPageResult = atomWithQuery({ query: QUERY, getVariables: (get) => ({ userId, ...get(currentPaginationAtom) }), }); return emailPageResult; }); const pageInfoFamily = atomFamily((userId: string) => { const pageInfoAtom = atom(async (get): Promise => { const result = await get(emailPageResultFamily(userId)); return result.data?.user?.emails?.pageInfo ?? null; }); return pageInfoAtom; }); const paginationFamily = atomFamily((userId: string) => { const paginationAtom = atomWithPagination( currentPaginationAtom, pageInfoFamily(userId), ); return paginationAtom; }); const UserEmailList: React.FC<{ userId: string; }> = ({ userId }) => { const [pending, startTransition] = useTransition(); const [result, refreshList] = useAtom(emailPageResultFamily(userId)); const setPagination = useSetAtom(currentPaginationAtom); const setRoute = useSetAtom(routeAtom); const [prevPage, nextPage] = useAtomValue(paginationFamily(userId)); const [primaryEmailId, refreshPrimaryEmailId] = useAtom( primaryEmailIdFamily(userId), ); const paginate = (pagination: Pagination): void => { startTransition(() => { setPagination(pagination); }); }; // When removing an email, we want to refresh the list and go back to the first page const onRemove = (): void => { startTransition(() => { setPagination(FIRST_PAGE); refreshList(); }); }; // When adding an email, we want to go to the email verification form const onAdd = (id: string): void => { setRoute({ type: "verify-email", id }); }; const showNoPrimaryEmailAlert = !!result?.data && !primaryEmailId; return (

Emails

paginate(prevPage) : null} onNext={nextPage ? (): void => paginate(nextPage) : null} disabled={pending} /> {showNoPrimaryEmailAlert && ( )} {result.data?.user?.emails?.edges?.map((edge) => ( ))}
); }; export default UserEmailList;