diff --git a/packages/interface/src/components/ImageUpload.tsx b/packages/interface/src/components/ImageUpload.tsx index a914a0fb..c4dee714 100644 --- a/packages/interface/src/components/ImageUpload.tsx +++ b/packages/interface/src/components/ImageUpload.tsx @@ -1,7 +1,7 @@ import { useMutation } from "@tanstack/react-query"; import clsx from "clsx"; import { ImageIcon } from "lucide-react"; -import { type ComponentProps, useRef } from "react"; +import { type ComponentProps, useRef, useCallback } from "react"; import { Controller, useFormContext } from "react-hook-form"; import { toast } from "sonner"; @@ -33,17 +33,18 @@ export const ImageUpload = ({ }, }); + const onClick = useCallback(() => { + ref.current?.click(); + }, []); + return ( ( // eslint-disable-next-line jsx-a11y/no-static-element-interactions, jsx-a11y/click-events-have-key-events -
ref.current?.click()} - > - +
+
{ const [file] = event.target.files ?? []; if (file) { diff --git a/packages/interface/src/components/Info.tsx b/packages/interface/src/components/Info.tsx index 983ffc53..90886680 100644 --- a/packages/interface/src/components/Info.tsx +++ b/packages/interface/src/components/Info.tsx @@ -65,7 +65,7 @@ export const Info = ({ size, roundId, showVotingInfo = false }: IInfoProps): JSX {showVotingInfo && (
- + {roundState === ERoundState.VOTING && }
diff --git a/packages/interface/src/components/RoundInfo.tsx b/packages/interface/src/components/RoundInfo.tsx index 6281c17b..5448fbf2 100644 --- a/packages/interface/src/components/RoundInfo.tsx +++ b/packages/interface/src/components/RoundInfo.tsx @@ -1,18 +1,16 @@ -import Image from "next/image"; - import { Heading } from "~/components/ui/Heading"; -import { config } from "~/config"; interface IRoundInfoProps { roundId: string; + roundLogo?: string; } -export const RoundInfo = ({ roundId }: IRoundInfoProps): JSX.Element => ( +export const RoundInfo = ({ roundId, roundLogo = undefined }: IRoundInfoProps): JSX.Element => (

Round

- {config.roundLogo && round logo} + {roundLogo && round logo} {roundId} diff --git a/packages/interface/src/features/applications/components/ApplicationSteps.tsx b/packages/interface/src/components/Steps.tsx similarity index 68% rename from packages/interface/src/features/applications/components/ApplicationSteps.tsx rename to packages/interface/src/components/Steps.tsx index 6f5cd096..126d7ada 100644 --- a/packages/interface/src/features/applications/components/ApplicationSteps.tsx +++ b/packages/interface/src/components/Steps.tsx @@ -10,13 +10,17 @@ export enum EStepState { interface IStepCategoryProps { title: string; progress: EStepState; + isLast?: boolean; } -interface IApplicationStepsProps { +interface IStepsProps { step: number; + stepNames: string[]; } -const StepCategory = ({ title, progress }: IStepCategoryProps): JSX.Element => ( +const Interline = (): JSX.Element =>
; + +const StepCategory = ({ title, progress, isLast = false }: IStepCategoryProps): JSX.Element => (
{progress === EStepState.ACTIVE && ( circle-check-blue @@ -29,21 +33,15 @@ const StepCategory = ({ title, progress }: IStepCategoryProps): JSX.Element => ( {progress === EStepState.DEFAULT &&
}

{title}

+ + {!isLast && }
); -const Interline = (): JSX.Element =>
; - -export const ApplicationSteps = ({ step }: IApplicationStepsProps): JSX.Element => ( +export const Steps = ({ step, stepNames }: IStepsProps): JSX.Element => (
- - - - - - - - - + {stepNames.map((name, i) => ( + + ))}
); diff --git a/packages/interface/src/components/ui/Form.tsx b/packages/interface/src/components/ui/Form.tsx index 133f72eb..4025208f 100644 --- a/packages/interface/src/components/ui/Form.tsx +++ b/packages/interface/src/components/ui/Form.tsx @@ -66,6 +66,13 @@ export const ErrorMessage = createComponent("div", tv({ base: "pt-1 text-xs text export const Textarea = createComponent("textarea", tv({ base: [...inputBase, "w-full"] })); +// eslint-disable-next-line react/display-name +export const DateInput = forwardRef(({ ...props }: ComponentPropsWithRef, ref) => ( + + + +)); + export const SearchInput = forwardRef(({ ...props }: ComponentPropsWithRef, ref) => ( diff --git a/packages/interface/src/contexts/Round.tsx b/packages/interface/src/contexts/Round.tsx index a5415337..944ae5cc 100644 --- a/packages/interface/src/contexts/Round.tsx +++ b/packages/interface/src/contexts/Round.tsx @@ -1,4 +1,4 @@ -import React, { createContext, useContext, useMemo, useCallback } from "react"; +import React, { createContext, useContext, useMemo, useCallback, useState, useEffect } from "react"; import type { RoundContextType, RoundProviderProps } from "./types"; import type { Round } from "~/features/rounds/types"; @@ -6,28 +6,48 @@ import type { Round } from "~/features/rounds/types"; export const RoundContext = createContext(undefined); export const RoundProvider: React.FC = ({ children }: RoundProviderProps) => { - const rounds = [ - { - roundId: "open-rpgf-1", - description: "This is the description of this round, please add your own description.", - startsAt: 1723477832000, - registrationEndsAt: 1723487832000, - votingEndsAt: 1724009826000, - tallyURL: "https://upblxu2duoxmkobt.public.blob.vercel-storage.com/tally.json", - }, - ]; + const [rounds, setRounds] = useState(undefined); const getRound = useCallback( - (roundId: string): Round | undefined => rounds.find((round) => round.roundId === roundId), + (roundId: string): Round | undefined => (rounds ? rounds.find((round) => round.roundId === roundId) : undefined), [rounds], ); + const addRound = useCallback( + (round: Round): void => { + if (!rounds) { + setRounds([round]); + } else { + setRounds([...rounds, round]); + } + }, + [rounds, setRounds], + ); + + useEffect(() => { + const storageData = localStorage.getItem("rounds"); + + if (storageData) { + // eslint-disable-next-line @typescript-eslint/no-unsafe-assignment + const storedRounds = JSON.parse(storageData); + // eslint-disable-next-line @typescript-eslint/no-unsafe-argument + setRounds(storedRounds); + } + }, []); + + useEffect(() => { + if (rounds) { + localStorage.setItem("rounds", JSON.stringify(rounds)); + } + }, [rounds]); + const value = useMemo( () => ({ rounds, getRound, + addRound, }), - [rounds, getRound], + [rounds, getRound, addRound], ); return {children}; diff --git a/packages/interface/src/contexts/types.ts b/packages/interface/src/contexts/types.ts index 34483b73..865822a2 100644 --- a/packages/interface/src/contexts/types.ts +++ b/packages/interface/src/contexts/types.ts @@ -49,8 +49,9 @@ export interface BallotProviderProps { } export interface RoundContextType { - rounds: Round[]; + rounds: Round[] | undefined; getRound: (roundId: string) => Round | undefined; + addRound: (round: Round) => void; } export interface RoundProviderProps { diff --git a/packages/interface/src/features/admin/components/DeployContracts.tsx b/packages/interface/src/features/admin/components/DeployContracts.tsx new file mode 100644 index 00000000..d04c46a2 --- /dev/null +++ b/packages/interface/src/features/admin/components/DeployContracts.tsx @@ -0,0 +1 @@ +export const DeployContracts = (): JSX.Element =>
This is Deploy Contracts
; diff --git a/packages/interface/src/features/admin/components/DeployRounds.tsx b/packages/interface/src/features/admin/components/DeployRounds.tsx new file mode 100644 index 00000000..db795ef0 --- /dev/null +++ b/packages/interface/src/features/admin/components/DeployRounds.tsx @@ -0,0 +1,147 @@ +import { useRouter } from "next/router"; +import { useState, useCallback } from "react"; +import { toast } from "sonner"; +import { useAccount } from "wagmi"; + +import { ImageUpload } from "~/components/ImageUpload"; +import { Steps } from "~/components/Steps"; +import { Form, FormSection, FormControl, Textarea, Select, DateInput } from "~/components/ui/Form"; +import { Heading } from "~/components/ui/Heading"; +import { Input } from "~/components/ui/Input"; +import { RoundSchema, votingStrategyTypes } from "~/features/rounds/types"; +import { useIsCorrectNetwork } from "~/hooks/useIsCorrectNetwork"; + +import { useDeployRound } from "../hooks/useDeployRound"; + +import { DeployRoundsButtons, EDeployRoundsStep } from "./DeployRoundsButtons"; +import { ReviewDeployRoundDetails } from "./ReviewDeployRoundDetails"; + +export const DeployRounds = (): JSX.Element => { + const router = useRouter(); + + const { isCorrectNetwork, correctNetwork } = useIsCorrectNetwork(); + + const { address } = useAccount(); + + const [step, setStep] = useState(EDeployRoundsStep.CONFIGURE); + + const handleNextStep = useCallback(() => { + if (step === EDeployRoundsStep.CONFIGURE) { + setStep(EDeployRoundsStep.REVIEW); + } + }, [step, setStep]); + + const handleBackStep = useCallback(() => { + if (step === EDeployRoundsStep.REVIEW) { + setStep(EDeployRoundsStep.CONFIGURE); + } + }, [step, setStep]); + + const create = useDeployRound({ + onSuccess: () => { + router.push(`/`); + }, + onError: (err: Error) => + toast.error("Round deploy error", { + description: err.message, + }), + }); + + const { error: createError } = create; + + return ( +
+ Deploy Round Contracts + +

These round contracts specify the features for this round.

+ +
+ + +
{ + create.mutate(round); + }} + > + + + + + +
+ +