From 2de724bd188c851de86b9eff03ad9f0ce0a48999 Mon Sep 17 00:00:00 2001 From: =?UTF-8?q?Micha=C5=82=20Sza=C5=82owski?= Date: Mon, 11 Mar 2024 20:22:58 +0100 Subject: [PATCH] [#377, #432] feat: form inputs validation --- CHANGELOG.md | 17 ++- .../src/components/atoms/TextArea.tsx | 1 + .../components/molecules/Field/TextArea.tsx | 13 +- .../ChooseGovernanceActionType.tsx | 40 +++--- .../CreateGovernanceActionForm.tsx | 101 ++++++++------- .../ReviewCreatedGovernanceAction.tsx | 4 +- .../StorageInformation.tsx | 27 +++- .../StoreDataInfo.tsx | 8 +- .../WhatGovernanceActionIsAbout.tsx | 6 +- .../CreateGovernanceActionSteps/index.ts | 4 + .../src/components/organisms/index.ts | 4 - .../src/consts/governanceActionFields.ts | 122 ++++++++++++++++++ .../src/consts/governanceActionTypes.ts | 64 --------- govtool/frontend/src/consts/index.ts | 2 +- .../{index.tsx => contextProviders.tsx} | 0 govtool/frontend/src/context/index.ts | 4 + .../forms/useCreateGovernanceActionForm.ts | 24 ++-- govtool/frontend/src/i18n/locales/en.ts | 38 ++++++ .../src/pages/CreateGovernanceAction.tsx | 4 +- govtool/frontend/src/types/global.d.ts | 8 ++ .../frontend/src/types/governanceAction.ts | 46 +++++++ 21 files changed, 367 insertions(+), 170 deletions(-) rename govtool/frontend/src/components/organisms/{ => CreateGovernanceActionSteps}/ChooseGovernanceActionType.tsx (65%) rename govtool/frontend/src/components/organisms/{ => CreateGovernanceActionSteps}/CreateGovernanceActionForm.tsx (61%) rename govtool/frontend/src/components/organisms/{ => CreateGovernanceActionSteps}/ReviewCreatedGovernanceAction.tsx (98%) rename govtool/frontend/src/components/organisms/{ => CreateGovernanceActionSteps}/WhatGovernanceActionIsAbout.tsx (94%) create mode 100644 govtool/frontend/src/consts/governanceActionFields.ts delete mode 100644 govtool/frontend/src/consts/governanceActionTypes.ts rename govtool/frontend/src/context/{index.tsx => contextProviders.tsx} (100%) create mode 100644 govtool/frontend/src/context/index.ts create mode 100644 govtool/frontend/src/types/governanceAction.ts diff --git a/CHANGELOG.md b/CHANGELOG.md index 059c7167d..1b4be57e6 100644 --- a/CHANGELOG.md +++ b/CHANGELOG.md @@ -9,23 +9,27 @@ As a minor extension, we also keep a semantic version for the `UNRELEASED` changes. ## [Unreleased] + - Create GA review subbmision page [Issue 362](https://github.com/IntersectMBO/govtool/issues/362) - Create GA creation form [Issue 360](https://github.com/IntersectMBO/govtool/issues/360) - Create TextArea [Issue 110](https://github.com/IntersectMBO/govtool/issues/110) +- Choose GA type - GA Submiter [Issue 358](https://github.com/IntersectMBO/govtool/issues/358) + +- Add on-chain inputs validation [Issue 377](https://github.com/IntersectMBO/govtool/issues/377) + +### Added + +- Added `isRegisteredAsSoleVoter` and `wasRegisteredAsSoleVoter` fields to the drep/info response [Issue 212](https://github.com/IntersectMBO/govtool/issues/212) - Abandoning registration as DRep [Issue 151](https://github.com/IntersectMBO/govtool/issues/151) - Abandoning GA creation [Issue 359](https://github.com/IntersectMBO/govtool/issues/359) -- Choose GA type - GA Submiter [Issue 358](https://github.com/IntersectMBO/govtool/issues/358) -- Change step 3 components [Issue 152](https://github.com/intersectMBO/govtool/issues/152) -- Add possibility to vote on behalf of myself - Sole Voter [Issue 119](https://github.com/IntersectMBO/govtool/issues/119) - Create DRep registration page about roles [Issue 205](https://github.com/IntersectMBO/govtool/issues/205) - Create Checkbox component. Improve Field and ControlledField [Issue 177](https://github.com/IntersectMBO/govtool/pull/177) - Vitest unit tests added for utils functions [Issue 81](https://github.com/IntersectMBO/govtool/issues/81) - i18next library added to FE [Issue 80](https://github.com/IntersectMBO/govtool/issues/80) - -### Added -- Added `isRegisteredAsSoleVoter` and `wasRegisteredAsSoleVoter` fields to the drep/info response [Issue 212](https://github.com/IntersectMBO/govtool/issues/212) +- Add possibility to vote on behalf of myself - Sole Voter [Issue 119](https://github.com/IntersectMBO/govtool/issues/119) ### Fixed + - Fix drep type detection when changing metadata [Issue 333](https://github.com/IntersectMBO/govtool/issues/333) - Fix make button disble when wallet tries connect [Issue 265](https://github.com/IntersectMBO/govtool/issues/265) - Fix drep voting power calculation [Issue 231](https://github.com/IntersectMBO/govtool/issues/231) @@ -58,6 +62,7 @@ changes. - Added a grafana panel to track all the deploys on the target machines [Issue 361](https://github.com/IntersectMBO/govtool/issues/361). ### Removed + - ## [sancho-v1.0.0](https://github.com/IntersectMBO/govtool/releases/tag/sancho-v1.0.0) 2023-12-17 diff --git a/govtool/frontend/src/components/atoms/TextArea.tsx b/govtool/frontend/src/components/atoms/TextArea.tsx index 06c778a04..0e5b0eacd 100644 --- a/govtool/frontend/src/components/atoms/TextArea.tsx +++ b/govtool/frontend/src/components/atoms/TextArea.tsx @@ -55,6 +55,7 @@ export const TextArea = forwardRef( ( {label} )} - + ( /> {props?.value?.toString()?.length ?? 0}/{maxLength} diff --git a/govtool/frontend/src/components/organisms/ChooseGovernanceActionType.tsx b/govtool/frontend/src/components/organisms/CreateGovernanceActionSteps/ChooseGovernanceActionType.tsx similarity index 65% rename from govtool/frontend/src/components/organisms/ChooseGovernanceActionType.tsx rename to govtool/frontend/src/components/organisms/CreateGovernanceActionSteps/ChooseGovernanceActionType.tsx index da64375e6..e90606635 100644 --- a/govtool/frontend/src/components/organisms/ChooseGovernanceActionType.tsx +++ b/govtool/frontend/src/components/organisms/CreateGovernanceActionSteps/ChooseGovernanceActionType.tsx @@ -1,14 +1,13 @@ import { Dispatch, SetStateAction } from "react"; - import { ActionRadio, Spacer, Typography } from "@atoms"; -import { GOVERNANCE_ACTION_TYPES } from "@consts"; import { useCreateGovernanceActionForm, useScreenDimension, useTranslation, } from "@hooks"; +import { GovernanceActionType } from "@/types/governanceAction"; -import { BgCard } from "./BgCard"; +import { BgCard } from "../BgCard"; type ChooseGovernanceActionTypeProps = { setStep: Dispatch>; @@ -32,25 +31,26 @@ export const ChooseGovernanceActionType = ({ }; // TODO: Add tooltips when they will be available - const renderGovernanceActionTypes = () => { - return GOVERNANCE_ACTION_TYPES.map((type, index) => { - const isChecked = getValues("governance_action_type") === type; - return ( -
- - {index + 1 < GOVERNANCE_ACTION_TYPES.length ? : null} -
- ); - }); - }; + const renderGovernanceActionTypes = () => + Object.keys(GovernanceActionType).map( + (type, index, governanceActionTypes) => { + const isChecked = getValues("governance_action_type") === type; + return ( +
+ + {index + 1 < governanceActionTypes.length ? : null} +
+ ); + } + ); const onChangeType = (value: string) => { - setValue("governance_action_type", value); + setValue("governance_action_type", value as GovernanceActionType); }; return ( diff --git a/govtool/frontend/src/components/organisms/CreateGovernanceActionForm.tsx b/govtool/frontend/src/components/organisms/CreateGovernanceActionSteps/CreateGovernanceActionForm.tsx similarity index 61% rename from govtool/frontend/src/components/organisms/CreateGovernanceActionForm.tsx rename to govtool/frontend/src/components/organisms/CreateGovernanceActionSteps/CreateGovernanceActionForm.tsx index b48a536af..0d49018e5 100644 --- a/govtool/frontend/src/components/organisms/CreateGovernanceActionForm.tsx +++ b/govtool/frontend/src/components/organisms/CreateGovernanceActionSteps/CreateGovernanceActionForm.tsx @@ -3,26 +3,32 @@ import { useFieldArray } from "react-hook-form"; import DeleteOutlineIcon from "@mui/icons-material/DeleteOutline"; import { Button, InfoText, Spacer, Typography } from "@atoms"; -import { GOVERNANCE_ACTIONS_FIELDS } from "@consts"; +import { GOVERNANCE_ACTION_FIELDS } from "@consts"; import { useCreateGovernanceActionForm, useTranslation } from "@hooks"; import { Field } from "@molecules"; -import { BgCard } from "./BgCard"; -import { ControlledField } from "./ControlledField"; +import { BgCard } from "../BgCard"; +import { ControlledField } from "../ControlledField"; +import { GovernanceActionField } from "@/types/governanceAction"; +import { URL_REGEX } from "@/utils"; const LINK_PLACEHOLDER = "https://website.com/"; const MAX_NUMBER_OF_LINKS = 8; -type ChooseGovernanceActionTypeProps = { +type CreateGovernanceActionFormProps = { setStep: Dispatch>; }; export const CreateGovernanceActionForm = ({ setStep, -}: ChooseGovernanceActionTypeProps) => { +}: CreateGovernanceActionFormProps) => { const { t } = useTranslation(); const { control, errors, getValues, register, reset, watch } = useCreateGovernanceActionForm(); + + const isError = Object.keys(errors).length > 0; + + const type = getValues("governance_action_type"); const { append, fields: links, @@ -32,16 +38,11 @@ export const CreateGovernanceActionForm = ({ name: "links", }); - const governanceActionType = getValues("governance_action_type"); - const fields = - GOVERNANCE_ACTIONS_FIELDS.find( - (field) => field.name === governanceActionType - )?.fields ?? []; - // TODO: Replace any - const isContinueButtonDisabled = Object.keys(fields).some( - (field: any) => !watch(field) - ); + const isContinueButtonDisabled = + Object.keys(GOVERNANCE_ACTION_FIELDS[type!]).some( + (field: any) => !watch(field) + ) || isError; const onClickContinue = () => { setStep(4); @@ -53,37 +54,35 @@ export const CreateGovernanceActionForm = ({ }; const renderGovernanceActionField = () => { - return Object.entries(fields).map(([key, value]) => { - const label = - key.charAt(0).toUpperCase() + key.slice(1).replace("_", " "); - - if (value.component === "input") { - return ( - - ); - } - if (value.component === "textarea") { - return ( - - ); + return Object.entries(GOVERNANCE_ACTION_FIELDS[type!]).map( + ([key, field]) => { + const fieldProps = { + helpfulText: field.tipI18nKey ? t(field.tipI18nKey) : undefined, + key, + label: t(field.labelI18nKey), + layoutStyles: { mb: 3 }, + name: key, + placeholder: field.placeholderI18nKey + ? t(field.placeholderI18nKey) + : undefined, + rules: field.rules, + }; + + if (field.component === GovernanceActionField.Input) { + return ( + + ); + } + if (field.component === GovernanceActionField.TextArea) { + return ( + + ); + } } - }); + ); }; const addLink = useCallback(() => { @@ -102,6 +101,7 @@ export const CreateGovernanceActionForm = ({ return ( 1 ? ( ); }); @@ -136,7 +147,7 @@ export const CreateGovernanceActionForm = ({ disabled={true} helpfulText={t("forms.createGovernanceAction.typeTip")} label={t("forms.createGovernanceAction.typeLabel")} - value={governanceActionType} + value={type} /> {renderGovernanceActionField()} diff --git a/govtool/frontend/src/components/organisms/ReviewCreatedGovernanceAction.tsx b/govtool/frontend/src/components/organisms/CreateGovernanceActionSteps/ReviewCreatedGovernanceAction.tsx similarity index 98% rename from govtool/frontend/src/components/organisms/ReviewCreatedGovernanceAction.tsx rename to govtool/frontend/src/components/organisms/CreateGovernanceActionSteps/ReviewCreatedGovernanceAction.tsx index f05db801e..6afc22249 100644 --- a/govtool/frontend/src/components/organisms/ReviewCreatedGovernanceAction.tsx +++ b/govtool/frontend/src/components/organisms/CreateGovernanceActionSteps/ReviewCreatedGovernanceAction.tsx @@ -1,4 +1,3 @@ -import { Dispatch, SetStateAction } from "react"; import { Box } from "@mui/material"; import DriveFileRenameOutlineOutlinedIcon from "@mui/icons-material/DriveFileRenameOutlineOutlined"; @@ -12,7 +11,8 @@ import { import { LinkWithIcon } from "@molecules"; import { openInNewTab } from "@utils"; -import { BgCard } from "./BgCard"; +import { BgCard } from "../BgCard"; +import { Dispatch, SetStateAction } from "react"; type ReviewCreatedGovernanceActionProps = { setStep: Dispatch>; diff --git a/govtool/frontend/src/components/organisms/CreateGovernanceActionSteps/StorageInformation.tsx b/govtool/frontend/src/components/organisms/CreateGovernanceActionSteps/StorageInformation.tsx index 38e9dbbcf..41bd01112 100644 --- a/govtool/frontend/src/components/organisms/CreateGovernanceActionSteps/StorageInformation.tsx +++ b/govtool/frontend/src/components/organisms/CreateGovernanceActionSteps/StorageInformation.tsx @@ -6,13 +6,13 @@ import { Button, Spacer, Typography } from "@atoms"; import { useCreateGovernanceActionForm, useTranslation } from "@hooks"; import { Step } from "@molecules"; import { BgCard, ControlledField } from "@organisms"; -import { downloadJson, openInNewTab } from "@utils"; +import { URL_REGEX, downloadJson, openInNewTab } from "@utils"; -export const StorageInformation = ({ - setStep, -}: { +type StorageInformationProps = { setStep: Dispatch>; -}) => { +}; + +export const StorageInformation = ({ setStep }: StorageInformationProps) => { const { t } = useTranslation(); const { control, @@ -23,6 +23,7 @@ export const StorageInformation = ({ watch, } = useCreateGovernanceActionForm(); const [isJsonDownloaded, setIsJsonDownloaded] = useState(false); + // TODO: change on correct file name const fileName = getValues("governance_action_type"); @@ -36,7 +37,7 @@ export const StorageInformation = ({ const onClickBack = useCallback(() => setStep(5), []); - const onClickDowloadJson = () => { + const onClickDownloadJson = () => { const data = getValues(); const jsonBody = generateJsonBody(data); downloadJson(jsonBody, fileName); @@ -66,7 +67,7 @@ export const StorageInformation = ({ // TODO: add onClick action when available component={