diff --git a/packages/interface/src/features/applications/components/ApplicationForm.tsx b/packages/interface/src/features/applications/components/ApplicationForm.tsx index 5d1924e7..0c7c7c4e 100644 --- a/packages/interface/src/features/applications/components/ApplicationForm.tsx +++ b/packages/interface/src/features/applications/components/ApplicationForm.tsx @@ -8,6 +8,7 @@ import { ImageUpload } from "~/components/ImageUpload"; import { FieldArray, Form, FormControl, FormSection, Select, Textarea } from "~/components/ui/Form"; import { Input } from "~/components/ui/Input"; import { useIsCorrectNetwork } from "~/hooks/useIsCorrectNetwork"; +import { MY_APPS_KEY } from "~/utils/types"; import { useCreateApplication } from "../hooks/useCreateApplication"; import { ApplicationSchema, contributionTypes, fundingSourceTypes, type Application } from "../types"; @@ -22,7 +23,7 @@ interface IApplicationFormProps { } export const ApplicationForm = ({ pollId }: IApplicationFormProps): JSX.Element => { - const clearDraft = useLocalStorage("application-draft")[2]; + const [myApps, setMyApps] = useLocalStorage(MY_APPS_KEY); const { isCorrectNetwork, correctNetwork } = useIsCorrectNetwork(); @@ -56,7 +57,13 @@ export const ApplicationForm = ({ pollId }: IApplicationFormProps): JSX.Element const create = useCreateApplication({ onSuccess: (id: bigint) => { - clearDraft(); + if (myApps) { + myApps.push(id.toString()); + setMyApps(myApps); + } else { + setMyApps([id.toString()]); + } + router.push(`/rounds/${pollId}/applications/confirmation?id=${id.toString()}`); }, onError: (err: { message: string }) => diff --git a/packages/interface/src/features/applications/hooks/useApplicationById.ts b/packages/interface/src/features/applications/hooks/useApplicationById.ts index 2cf017c6..504e7b8e 100644 --- a/packages/interface/src/features/applications/hooks/useApplicationById.ts +++ b/packages/interface/src/features/applications/hooks/useApplicationById.ts @@ -6,3 +6,7 @@ import type { IRequest } from "~/utils/types"; export function useApplicationById(registryAddress: string, id: string): UseTRPCQueryResult { return api.applications.getById.useQuery({ registryAddress, id }); } + +export function useApplicationsByIds(registryAddress: string, ids: string[]): UseTRPCQueryResult { + return api.applications.getByIds.useQuery({ registryAddress, ids }); +} diff --git a/packages/interface/src/features/projects/components/ProjectItem.tsx b/packages/interface/src/features/projects/components/ProjectItem.tsx index db4142ca..b8451b72 100644 --- a/packages/interface/src/features/projects/components/ProjectItem.tsx +++ b/packages/interface/src/features/projects/components/ProjectItem.tsx @@ -37,7 +37,7 @@ export const ProjectItem = ({ return (
diff --git a/packages/interface/src/features/rounds/components/Projects.tsx b/packages/interface/src/features/rounds/components/Projects.tsx index 606e902f..437a8a4c 100644 --- a/packages/interface/src/features/rounds/components/Projects.tsx +++ b/packages/interface/src/features/rounds/components/Projects.tsx @@ -2,6 +2,7 @@ import clsx from "clsx"; import Link from "next/link"; import { useCallback, useMemo } from "react"; import { FiAlertCircle } from "react-icons/fi"; +import { useLocalStorage } from "react-use"; import { Hex, zeroAddress } from "viem"; import { InfiniteLoading } from "~/components/InfiniteLoading"; @@ -12,9 +13,10 @@ import { Heading } from "~/components/ui/Heading"; import { useBallot } from "~/contexts/Ballot"; import { useMaci } from "~/contexts/Maci"; import { useRound } from "~/contexts/Round"; +import { useApplicationsByIds } from "~/features/applications/hooks/useApplicationById"; import { useResults } from "~/hooks/useResults"; import { useRoundState } from "~/utils/state"; -import { ERoundState } from "~/utils/types"; +import { ERoundState, MY_APPS_KEY, type IRecipient } from "~/utils/types"; import { ProjectItem, ProjectItemAwarded } from "../../projects/components/ProjectItem"; import { useSearchProjects } from "../../projects/hooks/useProjects"; @@ -43,6 +45,12 @@ export const Projects = ({ pollId = "" }: IProjectsProps): JSX.Element => { const ballot = useMemo(() => getBallot(pollId), [pollId, getBallot]); + const [myApps] = useLocalStorage(MY_APPS_KEY); + + const applications = useApplicationsByIds(round?.registryAddress ?? zeroAddress, myApps ?? []); + + const myApplications = useMemo(() => applications.data, [applications]); + const handleAction = useCallback( (projectIndex: number, projectId: string) => (e: Event) => { e.preventDefault(); @@ -119,12 +127,30 @@ export const Projects = ({ pollId = "" }: IProjectsProps): JSX.Element => {
{roundState === ERoundState.APPLICATION && ( -
- - - +
+
+ My Projects + + + + +
+ +
+ {myApplications && + myApplications.length > 0 && + myApplications.map((project) => ( + + + + ))} + + {(!myApplications || myApplications.length === 0) && ( +

Create your application by clicking the button

+ )} +
)} diff --git a/packages/interface/src/server/api/routers/applications.ts b/packages/interface/src/server/api/routers/applications.ts index dfe3d57c..022f1c7c 100644 --- a/packages/interface/src/server/api/routers/applications.ts +++ b/packages/interface/src/server/api/routers/applications.ts @@ -18,6 +18,9 @@ export const applicationsRouter = createTRPCRouter({ getById: publicProcedure .input(z.object({ registryAddress: z.string(), id: z.string() })) .query(async ({ input }) => fetchApplicationById(input.registryAddress, input.id)), + getByIds: publicProcedure + .input(z.object({ registryAddress: z.string(), ids: z.array(z.string()) })) + .query(async ({ input }) => Promise.all(input.ids.map((id) => fetchApplicationById(input.registryAddress, id)))), getByProjectId: publicProcedure .input(z.object({ registryAddress: z.string(), projectId: z.string() })) .query(async ({ input }) => fetchApplicationByProjectId(input.projectId, input.registryAddress)), diff --git a/packages/interface/src/utils/types.ts b/packages/interface/src/utils/types.ts index ab9a9a36..8df0bb6a 100644 --- a/packages/interface/src/utils/types.ts +++ b/packages/interface/src/utils/types.ts @@ -1,6 +1,8 @@ import type { IGetPollData } from "maci-cli/sdk"; import type { Address, Hex } from "viem"; +export const MY_APPS_KEY = "my-apps"; + export enum ERoundState { LOADING = "LOADING", APPLICATION = "APPLICATION",