diff --git a/webapp/public/locales/en/main.json b/webapp/public/locales/en/main.json index aa800ee1b5..0798174306 100644 --- a/webapp/public/locales/en/main.json +++ b/webapp/public/locales/en/main.json @@ -77,6 +77,7 @@ "global.error.failedtoretrievejobs": "Failed to retrieve job information", "global.error.failedtoretrievelogs": "Failed to retrieve job logs", "global.error.failedtoretrievedownloads": "Failed to retrieve downloads list", + "global.error.deleteFailed": "Unable to delete", "global.area.add": "Add an area", "login.error": "Failed to authenticate", "tasks.title": "Tasks", diff --git a/webapp/public/locales/fr/main.json b/webapp/public/locales/fr/main.json index 98a72e00b6..bafb48829f 100644 --- a/webapp/public/locales/fr/main.json +++ b/webapp/public/locales/fr/main.json @@ -77,6 +77,7 @@ "global.error.failedtoretrievejobs": "Échec de la récupération des tâches", "global.error.failedtoretrievelogs": "Échec de la récupération des logs", "global.error.failedtoretrievedownloads": "Échec de la récupération des exports", + "global.error.deleteFailed": "Impossible d'effectuer la suppression", "global.area.add": "Ajouter une zone", "login.error": "Échec de l'authentification", "tasks.title": "Tâches", diff --git a/webapp/src/components/App/Singlestudy/explore/Modelization/Areas/Renewables/index.tsx b/webapp/src/components/App/Singlestudy/explore/Modelization/Areas/Renewables/index.tsx index cd1a467e54..cfc2c726f5 100644 --- a/webapp/src/components/App/Singlestudy/explore/Modelization/Areas/Renewables/index.tsx +++ b/webapp/src/components/App/Singlestudy/explore/Modelization/Areas/Renewables/index.tsx @@ -114,7 +114,8 @@ function Renewables() { return createRenewableCluster(study.id, areaId, cluster); }; - const handleDeleteSelection = (ids: string[]) => { + const handleDelete = (rows: RenewableClusterWithCapacity[]) => { + const ids = rows.map((row) => row.id); return deleteRenewableClusters(study.id, areaId, ids); }; @@ -133,7 +134,7 @@ function Renewables() { columns={columns} groups={RENEWABLE_GROUPS} onCreate={handleCreateRow} - onDelete={handleDeleteSelection} + onDelete={handleDelete} onNameClick={handleNameClick} deleteConfirmationMessage={(count) => t("studies.modelization.clusters.question.delete", { count }) diff --git a/webapp/src/components/App/Singlestudy/explore/Modelization/Areas/Storages/index.tsx b/webapp/src/components/App/Singlestudy/explore/Modelization/Areas/Storages/index.tsx index 934fd0236b..b88156e214 100644 --- a/webapp/src/components/App/Singlestudy/explore/Modelization/Areas/Storages/index.tsx +++ b/webapp/src/components/App/Singlestudy/explore/Modelization/Areas/Storages/index.tsx @@ -159,7 +159,8 @@ function Storages() { return createStorage(study.id, areaId, storage); }; - const handleDeleteSelection = (ids: string[]) => { + const handleDelete = (rows: Storage[]) => { + const ids = rows.map((row) => row.id); return deleteStorages(study.id, areaId, ids); }; @@ -178,7 +179,7 @@ function Storages() { columns={columns} groups={STORAGE_GROUPS} onCreate={handleCreateRow} - onDelete={handleDeleteSelection} + onDelete={handleDelete} onNameClick={handleNameClick} deleteConfirmationMessage={(count) => t("studies.modelization.clusters.question.delete", { count }) diff --git a/webapp/src/components/App/Singlestudy/explore/Modelization/Areas/Thermal/index.tsx b/webapp/src/components/App/Singlestudy/explore/Modelization/Areas/Thermal/index.tsx index 46bd508ae3..e410897db7 100644 --- a/webapp/src/components/App/Singlestudy/explore/Modelization/Areas/Thermal/index.tsx +++ b/webapp/src/components/App/Singlestudy/explore/Modelization/Areas/Thermal/index.tsx @@ -128,7 +128,8 @@ function Thermal() { return createThermalCluster(study.id, areaId, cluster); }; - const handleDeleteSelection = (ids: string[]) => { + const handleDelete = (rows: ThermalClusterWithCapacity[]) => { + const ids = rows.map((row) => row.id); return deleteThermalClusters(study.id, areaId, ids); }; @@ -147,7 +148,7 @@ function Thermal() { columns={columns} groups={THERMAL_GROUPS} onCreate={handleCreateRow} - onDelete={handleDeleteSelection} + onDelete={handleDelete} onNameClick={handleNameClick} deleteConfirmationMessage={(count) => t("studies.modelization.clusters.question.delete", { count }) diff --git a/webapp/src/components/common/GroupedDataTable/index.tsx b/webapp/src/components/common/GroupedDataTable/index.tsx index 8014bca43b..042186f9c1 100644 --- a/webapp/src/components/common/GroupedDataTable/index.tsx +++ b/webapp/src/components/common/GroupedDataTable/index.tsx @@ -25,6 +25,9 @@ import * as R from "ramda"; import * as RA from "ramda-adjunct"; import { usePrevious } from "react-use"; import useUpdateEffectOnce from "../../../hooks/useUpdateEffectOnce"; +import { PromiseAny } from "../../../utils/tsUtils"; +import useEnqueueErrorSnackbar from "../../../hooks/useEnqueueErrorSnackbar"; +import { toError } from "../../../utils/fnUtils"; export interface GroupedDataTableProps { data: TData[]; @@ -32,7 +35,7 @@ export interface GroupedDataTableProps { columns: Array>; groups: string[] | readonly string[]; onCreate?: (values: TData) => Promise; - onDelete?: (ids: string[]) => void; + onDelete?: (rows: TData[]) => PromiseAny | void; onNameClick?: (row: MRT_Row) => void; isLoading?: boolean; deleteConfirmationMessage?: string | ((count: number) => string); @@ -60,6 +63,8 @@ function GroupedDataTable({ >(""); const [tableData, setTableData] = useState(data); const [rowSelection, setRowSelection] = useState({}); + const [isSaving, setIsSaving] = useState(false); + const enqueueErrorSnackbar = useEnqueueErrorSnackbar(); const callbacksRef = useAutoUpdateRef({ onNameClick }); const prevData = usePrevious(data); @@ -126,7 +131,7 @@ function GroupedDataTable({ expanded: true, columnPinning: { left: [GROUP_COLUMN_ID] }, }, - state: { isLoading, rowSelection }, + state: { isLoading, isSaving, rowSelection }, enableGrouping: true, enableStickyFooter: true, enableStickyHeader: true, @@ -231,24 +236,30 @@ function GroupedDataTable({ } }; - const handleDelete = () => { + const handleDelete = async () => { if (!onDelete) { return; } - const rowIndexes = Object.keys(rowSelection) - .map(Number) - // ignore groups names - .filter(Number.isInteger); + const rowsToDelete = selectedRows; - const rowIdsToDelete = rowIndexes.map((index) => tableData[index].id); - - onDelete(rowIdsToDelete); setTableData((prevTableData) => - prevTableData.filter((row) => !rowIdsToDelete.includes(row.id)), + prevTableData.filter((row) => !rowsToDelete.includes(row)), ); + setRowSelection({}); closeDialog(); + + setIsSaving(true); + + try { + await onDelete(rowsToDelete); + } catch (error) { + enqueueErrorSnackbar(t("global.error.deleteFailed"), toError(error)); + setTableData((prevTableData) => [...prevTableData, ...rowsToDelete]); + } + + setIsSaving(false); }; const handleDuplicate = async (name: string) => {