diff --git a/src/components/elements/Inputs/TreeSpeciesInput/RHFSeedingTableInput.tsx b/src/components/elements/Inputs/TreeSpeciesInput/RHFSeedingTableInput.tsx index 62b487e34..447eadfbe 100644 --- a/src/components/elements/Inputs/TreeSpeciesInput/RHFSeedingTableInput.tsx +++ b/src/components/elements/Inputs/TreeSpeciesInput/RHFSeedingTableInput.tsx @@ -32,6 +32,8 @@ const RHFSeedingTableInput = (props: PropsWithChildren) {...props} title={t("Tree Species")} buttonCaptionSuffix={t("Species")} + withPreviousCounts={true} + useTaxonomicBackbone={true} value={value ?? []} onChange={onChange} collection={collection} diff --git a/src/components/elements/Inputs/TreeSpeciesInput/TreeSpeciesInput.stories.tsx b/src/components/elements/Inputs/TreeSpeciesInput/TreeSpeciesInput.stories.tsx index 6e0df7760..f3fbe7609 100644 --- a/src/components/elements/Inputs/TreeSpeciesInput/TreeSpeciesInput.stories.tsx +++ b/src/components/elements/Inputs/TreeSpeciesInput/TreeSpeciesInput.stories.tsx @@ -27,6 +27,8 @@ export const Default: Story = { label: "Tree Species Grown", description: "List the tree species that you expect to restore on this project, across all sites. Please enter the scientific name for each tree species.", + withPreviousCounts: false, + useTaxonomicBackbone: true, required: true, value: [ { diff --git a/src/components/elements/Inputs/TreeSpeciesInput/TreeSpeciesInput.tsx b/src/components/elements/Inputs/TreeSpeciesInput/TreeSpeciesInput.tsx index 01ef90959..8677e0eba 100644 --- a/src/components/elements/Inputs/TreeSpeciesInput/TreeSpeciesInput.tsx +++ b/src/components/elements/Inputs/TreeSpeciesInput/TreeSpeciesInput.tsx @@ -14,6 +14,7 @@ import { EstablishmentEntityType, useEstablishmentTrees } from "@/connections/Es import { useEntityContext } from "@/context/entity.provider"; import { useModalContext } from "@/context/modal.provider"; import { useDebounce } from "@/hooks/useDebounce"; +import { useValueChanged } from "@/hooks/useValueChanged"; import { isReportModelName } from "@/types/common"; import { updateArrayState } from "@/utils/array"; @@ -29,6 +30,8 @@ export interface TreeSpeciesInputProps extends Omit title: string; buttonCaptionSuffix: string; withNumbers?: boolean; + withPreviousCounts: boolean; + useTaxonomicBackbone: boolean; value: TreeSpeciesValue[]; onChange: (value: any[]) => void; clearErrors: () => void; @@ -101,16 +104,31 @@ const TreeSpeciesInput = (props: TreeSpeciesInputProps) => { const { entityUuid, entityName } = useEntityContext(); const isEntity = entityName != null && entityUuid != null; const isReport = isEntity && isReportModelName(entityName); - const handleBaseEntityTrees = isReport || (isEntity && ["sites", "nurseries"].includes(entityName)); + const handleBaseEntityTrees = + props.withPreviousCounts && (isReport || (isEntity && ["sites", "nurseries"].includes(entityName))); + const displayPreviousCounts = props.withPreviousCounts && isReport; const entity = (handleBaseEntityTrees ? entityName : undefined) as EstablishmentEntityType; const uuid = handleBaseEntityTrees ? entityUuid : undefined; - const [, { establishmentTrees, previousPlantingCounts }] = useEstablishmentTrees({ entity, uuid }); + const [establishmentLoaded, { establishmentTrees, previousPlantingCounts }] = useEstablishmentTrees({ entity, uuid }); + const shouldPrepopulate = value.length == 0 && Object.values(previousPlantingCounts ?? {}).length > 0; + useValueChanged(shouldPrepopulate, function () { + if (shouldPrepopulate) { + onChange( + Object.entries(previousPlantingCounts!).map(([name, previousCount]) => ({ + uuid: uuidv4(), + name, + taxon_id: previousCount.taxonId, + amount: 0 + })) + ); + } + }); const totalWithPrevious = useMemo( () => props.value.reduce( - (total, { name, amount }) => total + (amount ?? 0) + (previousPlantingCounts?.[name ?? ""] ?? 0), + (total, { name, amount }) => total + (amount ?? 0) + (previousPlantingCounts?.[name ?? ""]?.amount ?? 0), 0 ), [previousPlantingCounts, props.value] @@ -157,7 +175,7 @@ const TreeSpeciesInput = (props: TreeSpeciesInputProps) => { handleCreate?.({ uuid: uuidv4(), name: valueAutoComplete, - taxon_id: taxonId, + taxon_id: props.useTaxonomicBackbone ? taxonId : undefined, amount: props.withNumbers ? 0 : undefined }); @@ -183,7 +201,7 @@ const TreeSpeciesInput = (props: TreeSpeciesInputProps) => { handleUpdate({ ...editValue, name: valueAutoComplete, - taxon_id: findTaxonId(valueAutoComplete) + taxon_id: props.useTaxonomicBackbone ? taxonId : undefined }); setValueAutoComplete(""); @@ -202,6 +220,8 @@ const TreeSpeciesInput = (props: TreeSpeciesInputProps) => { addValue(e); }; + if (!establishmentLoaded || shouldPrepopulate) return null; + return ( { value={valueAutoComplete} onChange={e => setValueAutoComplete(e.target.value)} onSearch={async search => { + if (!props.useTaxonomicBackbone) return []; + const result = await autocompleteSearch(search); setSearchResult(result); return result; @@ -289,7 +311,10 @@ const TreeSpeciesInput = (props: TreeSpeciesInputProps) => {
-
+
{props.title} @@ -297,7 +322,7 @@ const TreeSpeciesInput = (props: TreeSpeciesInputProps) => { {props.value.length}
-
+
{isReport ? t("SPECIES PLANTED:") : t("TREES TO BE PLANTED:")} @@ -305,7 +330,7 @@ const TreeSpeciesInput = (props: TreeSpeciesInputProps) => { {props.withNumbers ? props.value.reduce((total, v) => total + (v.amount || 0), 0).toLocaleString() : "0"}
- +
{t("TOTAL PLANTED TO DATE:")} @@ -368,7 +393,7 @@ const TreeSpeciesInput = (props: TreeSpeciesInputProps) => { } >
- +
@@ -410,9 +435,9 @@ const TreeSpeciesInput = (props: TreeSpeciesInputProps) => { containerClassName="" />
- + - {(previousPlantingCounts?.[value.name ?? ""] ?? 0).toLocaleString()} + {(previousPlantingCounts?.[value.name ?? ""]?.amount ?? 0).toLocaleString()}
@@ -426,11 +451,21 @@ const TreeSpeciesInput = (props: TreeSpeciesInputProps) => { autoCompleteRef.current?.focus(); }} /> - setDeleteIndex(value.uuid ?? null)} - /> + name === value.name) == null + } + > + setDeleteIndex(value.uuid ?? null)} + /> +
)} diff --git a/src/components/extensive/WizardForm/WizardForm.stories.tsx b/src/components/extensive/WizardForm/WizardForm.stories.tsx index d351a3f2b..2a2de3ef9 100644 --- a/src/components/extensive/WizardForm/WizardForm.stories.tsx +++ b/src/components/extensive/WizardForm/WizardForm.stories.tsx @@ -187,6 +187,8 @@ const getSteps = (edit?: boolean): FormStepSchema[] => { description: "TRee species input description", fieldProps: { title: "Tree Species", + withPreviousCounts: false, + useTaxonomicBackbone: false, buttonCaptionSuffix: "Species", required: true }, @@ -204,6 +206,8 @@ const getSteps = (edit?: boolean): FormStepSchema[] => { fieldProps: { title: "Tree Species", + withPreviousCounts: false, + useTaxonomicBackbone: false, buttonCaptionSuffix: "Species", required: true, withNumbers: true diff --git a/src/generated/v3/entityService/entityServiceSchemas.ts b/src/generated/v3/entityService/entityServiceSchemas.ts index bb11f5e3f..9734a4f86 100644 --- a/src/generated/v3/entityService/entityServiceSchemas.ts +++ b/src/generated/v3/entityService/entityServiceSchemas.ts @@ -3,6 +3,17 @@ * * @version 1.0 */ +export type PreviousPlantingCountDto = { + /** + * Taxonomic ID for this tree species row + */ + taxonId: string | null; + /** + * Number of trees of this type that have been planted in all previous reports on this entity. + */ + amount: number; +}; + export type ScientificNameDto = { /** * The scientific name for this tree species @@ -20,9 +31,9 @@ export type EstablishmentsTreesDto = { /** * If the entity in this request is a report, the sum totals of previous planting by species. * - * @example {"Aster persaliens":256,"Cirsium carniolicum":1024} + * @example {"Aster persaliens":{"amount":256},"Cirsium carniolicum":{"taxonId":"wfo-0000130112","amount":1024}} */ previousPlantingCounts: { - [key: string]: number; + [key: string]: PreviousPlantingCountDto; } | null; }; diff --git a/src/helpers/customForms.ts b/src/helpers/customForms.ts index 229a11bf0..edc947569 100644 --- a/src/helpers/customForms.ts +++ b/src/helpers/customForms.ts @@ -732,7 +732,7 @@ const getFieldValidation = (question: FormQuestionRead, t: typeof useT, framewor const arrayItemShape = question.with_numbers ? yup.object({ name: yup.string().required(), - amount: yup.number().min(1).required() + amount: yup.number().min(0).required() }) : yup.object({ name: yup.string().required() diff --git a/src/pages/organization/[id]/components/edit/getEditOrganisationSteps.ts b/src/pages/organization/[id]/components/edit/getEditOrganisationSteps.ts index d58972eb0..b14bd4804 100644 --- a/src/pages/organization/[id]/components/edit/getEditOrganisationSteps.ts +++ b/src/pages/organization/[id]/components/edit/getEditOrganisationSteps.ts @@ -599,6 +599,8 @@ export const getSteps = (t: typeof useT, uuid: string): FormStepSchema[] => { ), fieldProps: { title: t("Tree Species"), + withPreviousCounts: false, + useTaxonomicBackbone: true, buttonCaptionSuffix: t("Species"), withNumbers: false } diff --git a/src/pages/organization/[id]/project-pitch/create/[pitchUUID]/getCreatePitchSteps.ts b/src/pages/organization/[id]/project-pitch/create/[pitchUUID]/getCreatePitchSteps.ts index f98dcbe16..39c190d1e 100644 --- a/src/pages/organization/[id]/project-pitch/create/[pitchUUID]/getCreatePitchSteps.ts +++ b/src/pages/organization/[id]/project-pitch/create/[pitchUUID]/getCreatePitchSteps.ts @@ -332,6 +332,8 @@ export const getSteps = (t: typeof useT, uuid: string): FormStepSchema[] => [ ), fieldProps: { title: t("Tree Species"), + withPreviousCounts: false, + useTaxonomicBackbone: true, buttonCaptionSuffix: t("Species"), withNumbers: true }