diff --git a/client/src/components/legend/item/component.tsx b/client/src/components/legend/item/component.tsx index 5b600b27f..8af865f9e 100644 --- a/client/src/components/legend/item/component.tsx +++ b/client/src/components/legend/item/component.tsx @@ -1,6 +1,7 @@ import classNames from 'classnames'; import { EyeIcon, EyeOffIcon, XIcon } from '@heroicons/react/solid'; import { useCallback } from 'react'; +import { useSearchParams } from 'next/navigation'; import OpacityControl from './opacityControl'; import DragHandle from './dragHandle'; @@ -8,8 +9,6 @@ import { ComparisonToggle } from './comparisonModeToggle'; import InfoModal from './info-modal'; import Loading from 'components/loading'; -import { useAppSelector } from 'store/hooks'; -import { scenarios } from 'store/features/analysis'; import type { Dispatch } from 'react'; import type { InfoModalProps } from './info-modal'; @@ -45,7 +44,8 @@ export const LegendItem = ({ onToggle, isActive, }: LegendItemProps) => { - const { isComparisonEnabled } = useAppSelector(scenarios); + const searchParams = useSearchParams(); + const isComparisonEnabled = Boolean(searchParams.get('compareScenarioId')); const handleToggleActive = useCallback(() => { onToggle(!isActive); diff --git a/client/src/containers/analysis-sidebar/component.tsx b/client/src/containers/analysis-sidebar/component.tsx index 39a866947..ed355fda1 100644 --- a/client/src/containers/analysis-sidebar/component.tsx +++ b/client/src/containers/analysis-sidebar/component.tsx @@ -7,7 +7,6 @@ import { pickBy } from 'lodash-es'; import { useAppSelector, useAppDispatch } from 'store/hooks'; import { scenarios, - setComparisonEnabled, setCurrentScenario, setScenarioToCompare as setScenarioToCompareAction, } from 'store/features/analysis/scenarios'; @@ -64,7 +63,6 @@ const ScenariosComponent: React.FC<{ scrollref?: MutableRefObject = ({ className, }: AnalysisDynamicMetadataTypes) => { + const searchParams = useSearchParams(); + const isComparisonEnabled = Boolean(searchParams.get('compareScenarioId')); + const dispatch = useAppDispatch(); - const { currentScenario, scenarioToCompare, isComparisonEnabled } = useAppSelector(scenarios); + const { currentScenario, scenarioToCompare } = useAppSelector(scenarios); const { data: scenario } = useScenario(currentScenario); const { data: scenarioB } = useScenario(scenarioToCompare); diff --git a/client/src/containers/analysis-visualization/analysis-legend/impact-legend-item/component.tsx b/client/src/containers/analysis-visualization/analysis-legend/impact-legend-item/component.tsx index e2bf6b888..b55e06d25 100644 --- a/client/src/containers/analysis-visualization/analysis-legend/impact-legend-item/component.tsx +++ b/client/src/containers/analysis-visualization/analysis-legend/impact-legend-item/component.tsx @@ -1,8 +1,8 @@ import { useCallback, useMemo } from 'react'; +import { useSearchParams } from 'next/navigation'; import { useAppSelector, useAppDispatch, useSyncIndicators } from 'store/hooks'; import { analysisMap, setLayer } from 'store/features/analysis/map'; -import { scenarios } from 'store/features/analysis'; import LegendTypeChoropleth from 'components/legend/types/choropleth'; import LegendTypeComparative from 'components/legend/types/comparative'; import LegendItem from 'components/legend/item'; @@ -13,6 +13,9 @@ import type { Legend } from 'types'; const LAYER_ID = 'impact'; const ImpactLayer = () => { + const searchParams = useSearchParams(); + const isComparisonEnabled = Boolean(searchParams.get('compareScenarioId')); + const dispatch = useAppDispatch(); const [syncedIndicators] = useSyncIndicators(); @@ -40,8 +43,6 @@ const ImpactLayer = () => { [layer.metadata?.legend.items], ); - const { isComparisonEnabled } = useAppSelector(scenarios); - const name = useMemo(() => { if (!layer.metadata?.legend?.name) return null; diff --git a/client/src/containers/analysis-visualization/analysis-table/component.tsx b/client/src/containers/analysis-visualization/analysis-table/component.tsx deleted file mode 100644 index 26e3dafa0..000000000 --- a/client/src/containers/analysis-visualization/analysis-table/component.tsx +++ /dev/null @@ -1,569 +0,0 @@ -import { useCallback, useEffect, useMemo, useState } from 'react'; -import { DownloadIcon, InformationCircleIcon } from '@heroicons/react/outline'; -import { uniq, omit } from 'lodash-es'; -import toast from 'react-hot-toast'; -import { ArrowLeftIcon } from '@heroicons/react/solid'; - -import ComparisonCell from './comparison-cell/component'; -import ChartCell from './chart-cell'; - -import { useAppSelector, useSyncIndicators, useSyncTableDetailView } from 'store/hooks'; -import { filtersForTabularAPI } from 'store/features/analysis/selector'; -import { scenarios } from 'store/features/analysis/scenarios'; -import { useIndicators } from 'hooks/indicators'; -import { - useImpactData, - useDownloadImpactData, - useDownloadImpactActualVsScenarioData, - useDownloadImpactScenarioVsScenarioData, -} from 'hooks/impact'; -import { useImpactComparison, useImpactScenarioComparison } from 'hooks/impact/comparison'; -import AnalysisDynamicMetadata from 'containers/analysis-visualization/analysis-dynamic-metadata'; -import { Button } from 'components/button'; -import Table from 'components/table/component'; -import { formatNumber } from 'utils/number-format'; -import { DEFAULT_PAGE_SIZES } from 'components/table/pagination/constants'; -import { handleResponseError } from 'services/api'; - -import type { - ExpandedState, - PaginationState, - RowSelectionState, - SortingState, - TableState, - Table as TableType, -} from '@tanstack/react-table'; -import type { TableProps } from 'components/table/component'; -import type { ColumnDefinition } from 'components/table/column'; -import type { ChartData } from './chart-cell/types'; -import type { ComparisonMode, ImpactRowType, ImpactTableValueItem } from './types'; - -const isParentRow = ( - row: ImpactRowType, -): row is ImpactRowType => { - return 'metadata' in row; -}; - -const AnalysisTable = () => { - const [paginationState, setPaginationState] = useState({ - pageIndex: 1, - pageSize: DEFAULT_PAGE_SIZES[0], - }); - const [sortingState, setSortingState] = useState([]); - const [expandedState, setExpandedState] = useState(null); - const [rowSelectionState, setRowSelectionState] = useState({}); - const tableState: Partial = useMemo(() => { - return { - pagination: paginationState, - sorting: sortingState, - expanded: expandedState, - rowSelection: rowSelectionState, - }; - }, [expandedState, paginationState, rowSelectionState, sortingState]); - - const [syncedIndicators, setSyncedIndicators] = useSyncIndicators(); - const [syncedDetailView, setSyncedDetailView] = useSyncTableDetailView(); - - const selectedIndicators = syncedIndicators; - - const { scenarioToCompare, isComparisonEnabled, currentScenario } = useAppSelector(scenarios); - const { data: indicators } = useIndicators(undefined, { select: (data) => data?.data }); - const downloadImpactData = useDownloadImpactData({ - onSuccess: () => { - toast.success('Data was downloaded successfully'); - }, - onError: handleResponseError, - }); - - const downloadActualVsScenarioData = useDownloadImpactActualVsScenarioData({ - onSuccess: () => { - toast.success('Data was downloaded successfully'); - }, - onError: handleResponseError, - }); - - const downloadScenarioVsScenarioData = useDownloadImpactScenarioVsScenarioData({ - onSuccess: () => { - toast.success('Data was downloaded successfully'); - }, - onError: handleResponseError, - }); - - const { indicatorId, ...restFilters } = useAppSelector(filtersForTabularAPI); - - const useIsComparison = useCallback( - (table: ImpactRowType[]): table is ImpactRowType[] => { - return isComparisonEnabled && !!scenarioToCompare; - }, - [isComparisonEnabled, scenarioToCompare], - ); - - const useIsScenarioComparison = useCallback( - (table: ImpactRowType[]): table is ImpactRowType<'scenario'>[] => { - return isComparisonEnabled && !!currentScenario; - }, - [isComparisonEnabled, currentScenario], - ); - - const isEnable = - !!indicators?.length && - !!restFilters.startYear && - !!restFilters.endYear && - restFilters.endYear !== restFilters.startYear; - - const indicatorIds = useMemo(() => { - if (Array.isArray(selectedIndicators)) { - return selectedIndicators.map((id) => id); - } - - if (selectedIndicators && !Array.isArray(selectedIndicators)) { - return [selectedIndicators]; - } - - return indicators.map((indicator) => indicator.id); - }, [indicators, selectedIndicators]); - - const sortingParams = useMemo(() => { - if (!!sortingState.length) { - return { - sortingYear: Number(sortingState?.[0].id), - sortingOrder: sortingState[0].desc ? 'DESC' : 'ASC', - }; - } - return {}; - }, [sortingState]); - - const params = useMemo( - () => ({ - indicatorIds, - startYear: restFilters.startYear, - endYear: restFilters.endYear, - groupBy: restFilters.groupBy, - ...restFilters, - ...sortingParams, - scenarioId: currentScenario, - 'page[number]': paginationState.pageIndex, - 'page[size]': paginationState.pageSize, - }), - [ - currentScenario, - indicatorIds, - paginationState.pageIndex, - paginationState.pageSize, - restFilters, - sortingParams, - ], - ); - - const plainImpactData = useImpactData(params, { - enabled: !isComparisonEnabled && isEnable, - }); - - const impactActualComparisonData = useImpactComparison( - { ...omit(params, 'scenarioId'), comparedScenarioId: scenarioToCompare }, - { - enabled: isComparisonEnabled && !currentScenario && isEnable, - }, - ); - const impactScenarioComparisonData = useImpactScenarioComparison( - { - ...omit(params, 'scenarioId'), - baseScenarioId: currentScenario, - comparedScenarioId: scenarioToCompare, - }, - { - enabled: isComparisonEnabled && !!currentScenario && isEnable, - }, - ); - - const impactComparisonData = !!currentScenario - ? impactScenarioComparisonData - : impactActualComparisonData; - - const { - data: impactData, - isLoading, - isFetching, - } = useMemo(() => { - if (isComparisonEnabled && !!scenarioToCompare) return impactComparisonData; - return plainImpactData; - }, [impactComparisonData, plainImpactData, isComparisonEnabled, scenarioToCompare]); - - const { - data: { impactTable = [] }, - metadata, - } = useMemo(() => { - if (impactData) return impactData; - return { data: { impactTable: [] }, metadata: {} }; - }, [impactData]); - - const firstProjectedYear = useMemo(() => { - if (!impactTable) return null; - return impactTable[0]?.rows[0]?.values.find((value) => value.isProjected)?.year; - }, [impactTable]); - - const handleDownloadData = useCallback(async () => { - let csv = null; - // actual vs scenario - if (!currentScenario && scenarioToCompare) { - csv = await downloadActualVsScenarioData.mutateAsync({ - ...omit(params, 'page[number]', 'page[size]'), - comparedScenarioId: scenarioToCompare, - }); - } - // scenario vs scenario - else if (currentScenario && scenarioToCompare) { - csv = await downloadScenarioVsScenarioData.mutateAsync({ - ...omit(params, 'page[number]', 'page[size]', 'scenarioId'), - baseScenarioId: currentScenario, - comparedScenarioId: scenarioToCompare, - }); - } - // no scenario or comparison - else { - csv = await downloadImpactData.mutateAsync(omit(params, 'page[number]', 'page[size]')); - } - - if (csv) { - const url = window.URL.createObjectURL(new Blob([csv])); - const link = document.createElement('a'); - link.setAttribute('href', url); - link.setAttribute('download', `impact_data_${Date.now()}.csv`); - document.body.appendChild(link); - link.click(); - document.body.removeChild(link); - } - // do not pass pagination params to download data endpoint - }, [ - currentScenario, - downloadActualVsScenarioData, - downloadImpactData, - downloadScenarioVsScenarioData, - params, - scenarioToCompare, - ]); - - const handleExpandedChange = useCallback( - (table: TableType>) => { - if (!!expandedState) { - const expandedIds = Object.keys(expandedState); - const rowsToSelect = {}; - table - .getRowModel() - .rows.filter((row) => expandedIds.includes(row.id)) - .forEach((row) => { - rowsToSelect[row.id] = true; - row.originalSubRows.forEach( - (_subRow, index) => (rowsToSelect[`${row.id}.${index}`] = true), - ); - }); - setRowSelectionState(rowsToSelect); - } - }, - [expandedState], - ); - - // Years from impact table - const years = useMemo(() => { - // TODO: do we have to check all rows or is the first one guaranteed to have all years? - // const years = impactTable[0]?.yearSum?.map((sum) => sum.year); - const years = impactTable?.flatMap(({ yearSum }) => yearSum.map((sum) => sum.year)); - - // TODO: if the above is true, we don't need this - return uniq(years); - }, [impactTable]); - - const initialTableData: ImpactRowType[] = useMemo( - () => - impactTable.map(({ indicatorShortName, yearSum, rows, ...impact }) => ({ - ...impact, - children: rows, - name: indicatorShortName, - ...(yearSum && { - values: yearSum.map((sum) => ({ - ...sum, - isProjected: rows[0]?.values.find((v) => v.year === sum.year)?.isProjected, - })), - }), - })), - [impactTable], - ); - - const [tableData, setTableData] = useState[]>([]); - - useEffect(() => { - // ? a single indicator is expanded - if (syncedDetailView) { - setTableData(initialTableData.find((row) => row.indicatorId === syncedDetailView)?.children); - } - - // ? several indicators are selected - if (selectedIndicators && !syncedDetailView) { - setTableData(initialTableData.filter((row) => selectedIndicators.includes(row.indicatorId))); - } - - // ? all indicators are selected - if (!selectedIndicators?.length && !syncedDetailView) { - setTableData(initialTableData); - } - - setExpandedState(null); - setRowSelectionState({}); - }, [selectedIndicators, initialTableData, syncedDetailView]); - - const handleExitExpanded = useCallback(() => { - setExpandedState({}); - setSyncedDetailView(null); - - if (syncedIndicators?.length === 1) { - setSyncedIndicators(null); - } - }, [setSyncedDetailView, syncedIndicators, setSyncedIndicators]); - - const handleExpandRow = useCallback( - (indicatorId: string) => { - setExpandedState({}); - setSyncedDetailView(indicatorId); - }, - [setSyncedDetailView], - ); - - const isComparison = useIsComparison(tableData); - const isScenarioComparison = useIsScenarioComparison(tableData); - - const valueIsScenarioComparison = useCallback( - (value: ImpactTableValueItem): value is ImpactTableValueItem<'scenario'> => { - return isScenarioComparison && isComparison; - }, - [isComparison, isScenarioComparison], - ); - - const expanded = useMemo(() => { - return indicators.find((i) => i.id === syncedDetailView); - }, [syncedDetailView, indicators]); - - const comparisonColumn = useCallback( - (year: number): ColumnDefinition> => { - const valueIsComparison = ( - value: ImpactTableValueItem, - ): value is ImpactTableValueItem => { - return !isScenarioComparison && isComparison; - }; - - return { - header: () => {year}, - id: `${year}`, - size: 170, - align: 'center', - enableSorting: true, - cell: ({ row: { original: data, id }, table }) => { - //* The metadata is only present at the parent row, so we need to get it from there - const { rowsById } = table.getExpandedRowModel(); - const parentRowData = rowsById[id.split('.')[0]].original as unknown as ImpactRowType< - Mode, - true - >; - - const unit: string = parentRowData.metadata?.unit || expanded?.metadata?.units; - - const value = data.values?.find((value) => value.year === year); - const isComparison = valueIsComparison(value); - const isScenarioComparison = valueIsScenarioComparison(value); - - if (!isComparison && !isScenarioComparison) { - if (unit) { - return `${formatNumber(value.value)} ${unit}`; - } - return formatNumber(value?.value); - } - - if (isScenarioComparison) { - const { baseScenarioValue, comparedScenarioValue, ...rest } = value; - return ( - - ); - } - - return ( - - ); - }, - }; - }, - [expanded, isComparison, isScenarioComparison, valueIsScenarioComparison], - ); - - const baseColumns = useMemo( - (): ColumnDefinition>[] => [ - { - id: 'name', - header: () => ( -
- {!!expanded?.name ? ( - - ) : ( - - Selected Indicators - - )} -
- ), - align: 'left', - isSticky: 'left', - size: 260, - cell: ({ row: { original, depth } }) => { - const name = - isParentRow(original) && - depth === 0 && - indicators.find((i) => i.id === original.indicatorId)?.metadata?.short_name; - - return ( -
- {!expanded?.name && ( - - )} -
- {expanded?.name ? ( - original.name - ) : ( -
- {name || original.name} - {isParentRow(original) && depth === 0 && <> ({original.metadata.unit})} -
- )} - - {!expanded?.name && isParentRow(original) && ( - - )} -
-
- ); - }, - }, - { - id: 'datesRangeChart', - header: () => ( - - {years?.length ? `${years[0]}-${years[years.length - 1]}` : '-'} - - ), - className: 'px-2 mx-auto', - align: 'center', - size: 170, - cell: ({ - row: { - original: { values }, - }, - }) => { - const chartData = values as ChartData[]; - return ( -
- -
- ); - }, - }, - ...years.map((year) => comparisonColumn(year as number)), - ], - [years, expanded?.name, handleExitExpanded, indicators, handleExpandRow, comparisonColumn], - ); - - const tableProps = useMemo( - (): TableProps> & { - firstProjectedYear: number; - } => ({ - showPagination: Boolean(selectedIndicators), - paginationProps: { - totalItems: metadata.totalItems, - totalPages: metadata.totalPages, - currentPage: metadata.page, - pageSize: metadata.size, - }, - getSubRows: (row) => row.children, - state: tableState, - onSortingChange: setSortingState, - onPaginationChange: setPaginationState, - onExpandedChange: setExpandedState, - isLoading: isFetching, - enableExpanding: Boolean(syncedDetailView), - data: (tableData as ImpactRowType[]) || [], - columns: baseColumns as ColumnDefinition>[], - handleExpandedChange, - firstProjectedYear, - }), - [ - metadata, - tableState, - isFetching, - tableData, - baseColumns, - handleExpandedChange, - firstProjectedYear, - selectedIndicators, - syncedDetailView, - ], - ); - - return ( -
-
-
- -
- -
-
-
-
- - - - ); -}; - -export default AnalysisTable; diff --git a/client/src/containers/analysis-visualization/analysis-table/footer/index.tsx b/client/src/containers/analysis-visualization/analysis-table/footer/index.tsx new file mode 100644 index 000000000..0809c6a1b --- /dev/null +++ b/client/src/containers/analysis-visualization/analysis-table/footer/index.tsx @@ -0,0 +1,46 @@ +import { FC } from 'react'; +import { useSearchParams } from 'next/navigation'; + +import { Separator } from '@/components/ui/separator'; +import { Badge } from '@/components/ui/badge'; +import { useScenario } from '@/hooks/scenarios'; + +const AnalysisTableFooter: FC = () => { + const searchParams = useSearchParams(); + const compareScenarioId = searchParams.get('compareScenarioId'); + + const { data: scenarioName } = useScenario(compareScenarioId, undefined, { + select: (scenario) => scenario?.title, + enabled: Boolean(compareScenarioId), + }); + + return ( +
+
    + {Boolean(scenarioName) && ( +
  • + + {scenarioName} + + + 140 + + +70 + + + Difference +
  • + )} +
  • + + Actual data + + + 70 +
  • +
+
+ ); +}; + +export default AnalysisTableFooter; diff --git a/client/src/containers/analysis-visualization/analysis-table/index.tsx b/client/src/containers/analysis-visualization/analysis-table/index.tsx index b404d7fd4..569f233c8 100644 --- a/client/src/containers/analysis-visualization/analysis-table/index.tsx +++ b/client/src/containers/analysis-visualization/analysis-table/index.tsx @@ -1 +1,574 @@ -export { default } from './component'; +import { useCallback, useEffect, useMemo, useState } from 'react'; +import { DownloadIcon, InformationCircleIcon } from '@heroicons/react/outline'; +import { uniq, omit } from 'lodash-es'; +import toast from 'react-hot-toast'; +import { ArrowLeftIcon } from '@heroicons/react/solid'; +import { useSearchParams } from 'next/navigation'; + +import ComparisonCell from './comparison-cell/component'; +import ChartCell from './chart-cell'; +import AnalysisTableFooter from './footer'; + +import { useAppSelector, useSyncIndicators, useSyncTableDetailView } from 'store/hooks'; +import { filtersForTabularAPI } from 'store/features/analysis/selector'; +import { scenarios } from 'store/features/analysis/scenarios'; +import { useIndicators } from 'hooks/indicators'; +import { + useImpactData, + useDownloadImpactData, + useDownloadImpactActualVsScenarioData, + useDownloadImpactScenarioVsScenarioData, +} from 'hooks/impact'; +import { useImpactComparison, useImpactScenarioComparison } from 'hooks/impact/comparison'; +import AnalysisDynamicMetadata from 'containers/analysis-visualization/analysis-dynamic-metadata'; +import { Button } from 'components/button'; +import Table from 'components/table/component'; +import { formatNumber } from 'utils/number-format'; +import { DEFAULT_PAGE_SIZES } from 'components/table/pagination/constants'; +import { handleResponseError } from 'services/api'; + +import type { + ExpandedState, + PaginationState, + RowSelectionState, + SortingState, + TableState, + Table as TableType, +} from '@tanstack/react-table'; +import type { TableProps } from 'components/table/component'; +import type { ColumnDefinition } from 'components/table/column'; +import type { ChartData } from './chart-cell/types'; +import type { ComparisonMode, ImpactRowType, ImpactTableValueItem } from './types'; + +const isParentRow = ( + row: ImpactRowType, +): row is ImpactRowType => { + return 'metadata' in row; +}; + +const AnalysisTable = () => { + const searchParams = useSearchParams(); + const isComparisonEnabled = Boolean(searchParams.get('compareScenarioId')); + + const [paginationState, setPaginationState] = useState({ + pageIndex: 1, + pageSize: DEFAULT_PAGE_SIZES[0], + }); + const [sortingState, setSortingState] = useState([]); + const [expandedState, setExpandedState] = useState(null); + const [rowSelectionState, setRowSelectionState] = useState({}); + const tableState: Partial = useMemo(() => { + return { + pagination: paginationState, + sorting: sortingState, + expanded: expandedState, + rowSelection: rowSelectionState, + }; + }, [expandedState, paginationState, rowSelectionState, sortingState]); + + const [syncedIndicators, setSyncedIndicators] = useSyncIndicators(); + const [syncedDetailView, setSyncedDetailView] = useSyncTableDetailView(); + + const selectedIndicators = syncedIndicators; + + const { scenarioToCompare, currentScenario } = useAppSelector(scenarios); + const { data: indicators } = useIndicators(undefined, { select: (data) => data?.data }); + const downloadImpactData = useDownloadImpactData({ + onSuccess: () => { + toast.success('Data was downloaded successfully'); + }, + onError: handleResponseError, + }); + + const downloadActualVsScenarioData = useDownloadImpactActualVsScenarioData({ + onSuccess: () => { + toast.success('Data was downloaded successfully'); + }, + onError: handleResponseError, + }); + + const downloadScenarioVsScenarioData = useDownloadImpactScenarioVsScenarioData({ + onSuccess: () => { + toast.success('Data was downloaded successfully'); + }, + onError: handleResponseError, + }); + + const { indicatorId, ...restFilters } = useAppSelector(filtersForTabularAPI); + + const useIsComparison = useCallback( + (table: ImpactRowType[]): table is ImpactRowType[] => { + return isComparisonEnabled && !!scenarioToCompare; + }, + [isComparisonEnabled, scenarioToCompare], + ); + + const useIsScenarioComparison = useCallback( + (table: ImpactRowType[]): table is ImpactRowType<'scenario'>[] => { + return isComparisonEnabled && !!currentScenario; + }, + [isComparisonEnabled, currentScenario], + ); + + const isEnable = + !!indicators?.length && + !!restFilters.startYear && + !!restFilters.endYear && + restFilters.endYear !== restFilters.startYear; + + const indicatorIds = useMemo(() => { + if (Array.isArray(selectedIndicators)) { + return selectedIndicators.map((id) => id); + } + + if (selectedIndicators && !Array.isArray(selectedIndicators)) { + return [selectedIndicators]; + } + + return indicators.map((indicator) => indicator.id); + }, [indicators, selectedIndicators]); + + const sortingParams = useMemo(() => { + if (!!sortingState.length) { + return { + sortingYear: Number(sortingState?.[0].id), + sortingOrder: sortingState[0].desc ? 'DESC' : 'ASC', + }; + } + return {}; + }, [sortingState]); + + const params = useMemo( + () => ({ + indicatorIds, + startYear: restFilters.startYear, + endYear: restFilters.endYear, + groupBy: restFilters.groupBy, + ...restFilters, + ...sortingParams, + scenarioId: currentScenario, + 'page[number]': paginationState.pageIndex, + 'page[size]': paginationState.pageSize, + }), + [ + currentScenario, + indicatorIds, + paginationState.pageIndex, + paginationState.pageSize, + restFilters, + sortingParams, + ], + ); + + const plainImpactData = useImpactData(params, { + enabled: !isComparisonEnabled && isEnable, + }); + + const impactActualComparisonData = useImpactComparison( + { ...omit(params, 'scenarioId'), comparedScenarioId: scenarioToCompare }, + { + enabled: isComparisonEnabled && !currentScenario && isEnable, + }, + ); + const impactScenarioComparisonData = useImpactScenarioComparison( + { + ...omit(params, 'scenarioId'), + baseScenarioId: currentScenario, + comparedScenarioId: scenarioToCompare, + }, + { + enabled: isComparisonEnabled && !!currentScenario && isEnable, + }, + ); + + const impactComparisonData = !!currentScenario + ? impactScenarioComparisonData + : impactActualComparisonData; + + const { + data: impactData, + isLoading, + isFetching, + } = useMemo(() => { + if (isComparisonEnabled && !!scenarioToCompare) return impactComparisonData; + return plainImpactData; + }, [impactComparisonData, plainImpactData, isComparisonEnabled, scenarioToCompare]); + + const { + data: { impactTable = [] }, + metadata, + } = useMemo(() => { + if (impactData) return impactData; + return { data: { impactTable: [] }, metadata: {} }; + }, [impactData]); + + const firstProjectedYear = useMemo(() => { + if (!impactTable) return null; + return impactTable[0]?.rows[0]?.values.find((value) => value.isProjected)?.year; + }, [impactTable]); + + const handleDownloadData = useCallback(async () => { + let csv = null; + // actual vs scenario + if (!currentScenario && scenarioToCompare) { + csv = await downloadActualVsScenarioData.mutateAsync({ + ...omit(params, 'page[number]', 'page[size]'), + comparedScenarioId: scenarioToCompare, + }); + } + // scenario vs scenario + else if (currentScenario && scenarioToCompare) { + csv = await downloadScenarioVsScenarioData.mutateAsync({ + ...omit(params, 'page[number]', 'page[size]', 'scenarioId'), + baseScenarioId: currentScenario, + comparedScenarioId: scenarioToCompare, + }); + } + // no scenario or comparison + else { + csv = await downloadImpactData.mutateAsync(omit(params, 'page[number]', 'page[size]')); + } + + if (csv) { + const url = window.URL.createObjectURL(new Blob([csv])); + const link = document.createElement('a'); + link.setAttribute('href', url); + link.setAttribute('download', `impact_data_${Date.now()}.csv`); + document.body.appendChild(link); + link.click(); + document.body.removeChild(link); + } + // do not pass pagination params to download data endpoint + }, [ + currentScenario, + downloadActualVsScenarioData, + downloadImpactData, + downloadScenarioVsScenarioData, + params, + scenarioToCompare, + ]); + + const handleExpandedChange = useCallback( + (table: TableType>) => { + if (!!expandedState) { + const expandedIds = Object.keys(expandedState); + const rowsToSelect = {}; + table + .getRowModel() + .rows.filter((row) => expandedIds.includes(row.id)) + .forEach((row) => { + rowsToSelect[row.id] = true; + row.originalSubRows.forEach( + (_subRow, index) => (rowsToSelect[`${row.id}.${index}`] = true), + ); + }); + setRowSelectionState(rowsToSelect); + } + }, + [expandedState], + ); + + // Years from impact table + const years = useMemo(() => { + // TODO: do we have to check all rows or is the first one guaranteed to have all years? + // const years = impactTable[0]?.yearSum?.map((sum) => sum.year); + const years = impactTable?.flatMap(({ yearSum }) => yearSum.map((sum) => sum.year)); + + // TODO: if the above is true, we don't need this + return uniq(years); + }, [impactTable]); + + const initialTableData: ImpactRowType[] = useMemo( + () => + impactTable.map(({ indicatorShortName, yearSum, rows, ...impact }) => ({ + ...impact, + children: rows, + name: indicatorShortName, + ...(yearSum && { + values: yearSum.map((sum) => ({ + ...sum, + isProjected: rows[0]?.values.find((v) => v.year === sum.year)?.isProjected, + })), + }), + })), + [impactTable], + ); + + const [tableData, setTableData] = useState[]>([]); + + useEffect(() => { + // ? a single indicator is expanded + if (syncedDetailView) { + setTableData(initialTableData.find((row) => row.indicatorId === syncedDetailView)?.children); + } + + // ? several indicators are selected + if (selectedIndicators && !syncedDetailView) { + setTableData(initialTableData.filter((row) => selectedIndicators.includes(row.indicatorId))); + } + + // ? all indicators are selected + if (!selectedIndicators?.length && !syncedDetailView) { + setTableData(initialTableData); + } + + setExpandedState(null); + setRowSelectionState({}); + }, [selectedIndicators, initialTableData, syncedDetailView]); + + const handleExitExpanded = useCallback(() => { + setExpandedState({}); + setSyncedDetailView(null); + + if (syncedIndicators?.length === 1) { + setSyncedIndicators(null); + } + }, [setSyncedDetailView, syncedIndicators, setSyncedIndicators]); + + const handleExpandRow = useCallback( + (indicatorId: string) => { + setExpandedState({}); + setSyncedDetailView(indicatorId); + }, + [setSyncedDetailView], + ); + + const isComparison = useIsComparison(tableData); + const isScenarioComparison = useIsScenarioComparison(tableData); + + const valueIsScenarioComparison = useCallback( + (value: ImpactTableValueItem): value is ImpactTableValueItem<'scenario'> => { + return isScenarioComparison && isComparison; + }, + [isComparison, isScenarioComparison], + ); + + const expanded = useMemo(() => { + return indicators.find((i) => i.id === syncedDetailView); + }, [syncedDetailView, indicators]); + + const comparisonColumn = useCallback( + (year: number): ColumnDefinition> => { + const valueIsComparison = ( + value: ImpactTableValueItem, + ): value is ImpactTableValueItem => { + return !isScenarioComparison && isComparison; + }; + + return { + header: () => {year}, + id: `${year}`, + size: 170, + align: 'center', + enableSorting: true, + cell: ({ row: { original: data, id }, table }) => { + //* The metadata is only present at the parent row, so we need to get it from there + const { rowsById } = table.getExpandedRowModel(); + const parentRowData = rowsById[id.split('.')[0]].original as unknown as ImpactRowType< + Mode, + true + >; + + const unit: string = parentRowData.metadata?.unit || expanded?.metadata?.units; + + const value = data.values?.find((value) => value.year === year); + const isComparison = valueIsComparison(value); + const isScenarioComparison = valueIsScenarioComparison(value); + + if (!isComparison && !isScenarioComparison) { + if (unit) { + return `${formatNumber(value.value)} ${unit}`; + } + return formatNumber(value?.value); + } + + if (isScenarioComparison) { + const { baseScenarioValue, comparedScenarioValue, ...rest } = value; + return ( + + ); + } + + return ( + + ); + }, + }; + }, + [expanded, isComparison, isScenarioComparison, valueIsScenarioComparison], + ); + + const baseColumns = useMemo( + (): ColumnDefinition>[] => [ + { + id: 'name', + header: () => ( +
+ {!!expanded?.name ? ( + + ) : ( + + Selected Indicators + + )} +
+ ), + align: 'left', + isSticky: 'left', + size: 260, + cell: ({ row: { original, depth } }) => { + const name = + isParentRow(original) && + depth === 0 && + indicators.find((i) => i.id === original.indicatorId)?.metadata?.short_name; + + return ( +
+ {!expanded?.name && ( + + )} +
+ {expanded?.name ? ( + original.name + ) : ( +
+ {name || original.name} + {isParentRow(original) && depth === 0 && <> ({original.metadata.unit})} +
+ )} + + {!expanded?.name && isParentRow(original) && ( + + )} +
+
+ ); + }, + }, + { + id: 'datesRangeChart', + header: () => ( + + {years?.length ? `${years[0]}-${years[years.length - 1]}` : '-'} + + ), + className: 'px-2 mx-auto', + align: 'center', + size: 170, + cell: ({ + row: { + original: { values }, + }, + }) => { + const chartData = values as ChartData[]; + return ( +
+ +
+ ); + }, + }, + ...years.map((year) => comparisonColumn(year as number)), + ], + [years, expanded?.name, handleExitExpanded, indicators, handleExpandRow, comparisonColumn], + ); + + const tableProps = useMemo( + (): TableProps> & { + firstProjectedYear: number; + } => ({ + showPagination: Boolean(syncedDetailView), + paginationProps: { + totalItems: metadata.totalItems, + totalPages: metadata.totalPages, + currentPage: metadata.page, + pageSize: metadata.size, + }, + getSubRows: (row) => row.children, + state: tableState, + onSortingChange: setSortingState, + onPaginationChange: setPaginationState, + onExpandedChange: setExpandedState, + isLoading: isFetching, + enableExpanding: Boolean(syncedDetailView), + data: (tableData as ImpactRowType[]) || [], + columns: baseColumns as ColumnDefinition>[], + handleExpandedChange, + firstProjectedYear, + }), + [ + metadata, + tableState, + isFetching, + tableData, + baseColumns, + handleExpandedChange, + firstProjectedYear, + syncedDetailView, + ], + ); + + return ( +
+
+
+ +
+ +
+
+
+
+
+ + {isComparisonEnabled && } + + ); +}; + +export default AnalysisTable; diff --git a/client/src/containers/scenarios/comparison/component.tsx b/client/src/containers/scenarios/comparison/component.tsx index 02e72a01f..d10f31fcf 100644 --- a/client/src/containers/scenarios/comparison/component.tsx +++ b/client/src/containers/scenarios/comparison/component.tsx @@ -4,12 +4,8 @@ import { pickBy } from 'lodash-es'; import { useScenarios } from 'hooks/scenarios'; import { useAppDispatch } from 'store/hooks'; -import { - setComparisonEnabled, - setScenarioToCompare as setScenarioToCompareAction, -} from 'store/features/analysis/scenarios'; +import { setScenarioToCompare as setScenarioToCompareAction } from 'store/features/analysis/scenarios'; import { AutoCompleteSelect } from 'components/forms/select'; -import useEffectOnce from 'hooks/once'; import type { Option } from 'components/forms/select'; import type { Dispatch, FC } from 'react'; @@ -39,7 +35,6 @@ const ScenariosComparison: FC = () => { const handleOnChange = useCallback>( (current) => { // TODO: deprecated, we'll keep only for retro-compatibility - dispatch(setComparisonEnabled(!!current)); dispatch(setScenarioToCompareAction(current?.value || null)); push({ query: pickBy({ ...query, compareScenarioId: current?.value || null }) }, null, { @@ -51,7 +46,6 @@ const ScenariosComparison: FC = () => { const handleScenarioRemoval = useCallback(() => { // TODO: deprecated, we'll keep only for retro-compatibility - dispatch(setComparisonEnabled(false)); dispatch(setScenarioToCompareAction(null)); push({ query: pickBy({ ...query, compareScenarioId: null }) }, null, { @@ -64,7 +58,6 @@ const ScenariosComparison: FC = () => { if (selected?.value && compareScenarioId !== selected?.value) { // TO-DO: deprecated, we'll keep only for retro-compatibility dispatch(setScenarioToCompareAction(null)); - dispatch(setComparisonEnabled(false)); push( { @@ -78,11 +71,6 @@ const ScenariosComparison: FC = () => { } }, [selected, dispatch, options, compareScenarioId, push, query]); - // We consider comparison is enabled when compareScenarioId is present - useEffectOnce(() => { - if (compareScenarioId) dispatch(setComparisonEnabled(true)); - }); - return ( <> diff --git a/client/src/hooks/h3-data/index.ts b/client/src/hooks/h3-data/index.ts index 37b5bfe6f..c2e160696 100644 --- a/client/src/hooks/h3-data/index.ts +++ b/client/src/hooks/h3-data/index.ts @@ -39,7 +39,7 @@ export const useH3Data = ({ const isImpact = id === 'impact'; const filters = useAppSelector(analysisFilters); - const { currentScenario, scenarioToCompare, isComparisonEnabled } = useAppSelector(scenarios); + const { currentScenario, scenarioToCompare } = useAppSelector(scenarios); const impactParams = useMemo( () => @@ -47,11 +47,10 @@ export const useH3Data = ({ ...filters, currentScenario, scenarioToCompare, - isComparisonEnabled, materialId, startYear: year, }), - [currentScenario, filters, isComparisonEnabled, materialId, scenarioToCompare, year], + [currentScenario, filters, materialId, scenarioToCompare, year], ); const materialParams = useMemo(() => ({ materialId, year }), [materialId, year]); diff --git a/client/src/hooks/layers/impact.ts b/client/src/hooks/layers/impact.ts index 01ed58603..803cda719 100644 --- a/client/src/hooks/layers/impact.ts +++ b/client/src/hooks/layers/impact.ts @@ -1,6 +1,6 @@ import { useEffect, useMemo } from 'react'; -import { useRouter } from 'next/router'; import { omit } from 'lodash-es'; +import { useSearchParams } from 'next/navigation'; import { useIndicator } from '../indicators'; @@ -17,14 +17,15 @@ import { storeToQueryParams } from 'hooks/h3-data/utils'; import type { LegendItem as LegendItemProp } from 'types'; export const useImpactLayer = () => { + const searchParams = useSearchParams(); + const compareScenarioId = searchParams.get('compareScenarioId'); + const scenarioId = searchParams.get('scenarioId'); + const isComparisonEnabled = Boolean(compareScenarioId); + const dispatch = useAppDispatch(); const filters = useAppSelector(analysisFilters); - const { - query: { scenarioId, compareScenarioId }, - } = useRouter(); - const isComparisonEnabled = !!compareScenarioId; const { comparisonMode } = useAppSelector(scenarios); - const colorKey = !!compareScenarioId ? 'compare' : 'impact'; + const colorKey = isComparisonEnabled ? 'compare' : 'impact'; const [syncedIndicators] = useSyncIndicators(); const { @@ -36,11 +37,10 @@ export const useImpactLayer = () => { storeToQueryParams({ ...filters, indicators: syncedIndicators?.[0] ? [syncedIndicators?.[0]] : undefined, - currentScenario: scenarioId as string, - scenarioToCompare: compareScenarioId as string, - isComparisonEnabled, + currentScenario: scenarioId, + scenarioToCompare: compareScenarioId, }), - [compareScenarioId, filters, isComparisonEnabled, scenarioId, syncedIndicators], + [compareScenarioId, filters, scenarioId, syncedIndicators], ); const { year } = params; diff --git a/client/src/hooks/scenarios/index.ts b/client/src/hooks/scenarios/index.ts index 5ff888382..4c9bd99fc 100644 --- a/client/src/hooks/scenarios/index.ts +++ b/client/src/hooks/scenarios/index.ts @@ -4,11 +4,7 @@ import { useQuery, useQueryClient, useInfiniteQuery, useMutation } from '@tansta import { apiService } from 'services/api'; // types -import type { - UseQueryResult, - UseInfiniteQueryResult, - UseQueryOptions, -} from '@tanstack/react-query'; +import type { UseInfiniteQueryResult, UseQueryOptions } from '@tanstack/react-query'; import type { AxiosResponse } from 'axios'; import type { Scenario, ScenarioDTO } from 'containers/scenarios/types'; import type { APIMetadataPagination } from 'types'; @@ -24,8 +20,6 @@ type ResponseInfiniteData = UseInfiniteQueryResult< }> >; -type ResponseDataScenario = UseQueryResult; - type QueryParams = { sort?: string; include?: string; @@ -98,24 +92,23 @@ export function useInfiniteScenarios(queryParams: QueryParams): ResponseInfinite return useMemo((): ResponseInfiniteData => query, [query]); } -export function useScenario( - id?: Scenario['id'] | null, +export function useScenario( + id: Scenario['id'], queryParams?: QueryParams, -): ResponseDataScenario { - const response: ResponseDataScenario = useQuery( + queryOptions: UseQueryOptions = {}, +) { + return useQuery( ['scenario', id], () => apiService - .request({ + .request<{ data: Scenario }>({ method: 'GET', url: `/scenarios/${id}`, params: queryParams, }) .then(({ data: responseData }) => responseData.data), - { ...DEFAULT_QUERY_OPTIONS, enabled: !!id }, + { enabled: Boolean(id), ...queryOptions }, ); - - return useMemo(() => response, [response]); } export function useDeleteScenario() { diff --git a/client/src/store/features/analysis/scenarios.ts b/client/src/store/features/analysis/scenarios.ts index 3472cf25f..5f8a8ca23 100644 --- a/client/src/store/features/analysis/scenarios.ts +++ b/client/src/store/features/analysis/scenarios.ts @@ -7,7 +7,6 @@ import type { Scenario } from 'containers/scenarios/types'; export type ScenarioComparisonMode = 'relative' | 'absolute'; export type ScenariosState = { - isComparisonEnabled: boolean; comparisonMode: ScenarioComparisonMode; /** * The current scenario id @@ -27,7 +26,6 @@ export type ScenariosState = { // Define the initial state using that type export const initialState: ScenariosState = { - isComparisonEnabled: false, comparisonMode: 'absolute', currentScenario: null, scenarioToCompare: null, @@ -48,13 +46,6 @@ export const analysisScenariosSlice = createSlice({ ...state, currentScenario: action.payload, }), - setComparisonEnabled: ( - state, - action: PayloadAction, - ) => ({ - ...state, - isComparisonEnabled: action.payload, - }), setScenarioToCompare: (state, action: PayloadAction) => ({ ...state, scenarioToCompare: action.payload, @@ -80,7 +71,6 @@ export const analysisScenariosSlice = createSlice({ export const { setCurrentScenario, - setComparisonEnabled, setScenarioToCompare, setComparisonMode, setScenarioFilter,