Skip to content
New issue

Have a question about this project? Sign up for a free GitHub account to open an issue and contact its maintainers and the community.

By clicking “Sign up for GitHub”, you agree to our terms of service and privacy statement. We’ll occasionally send you account related emails.

Already on GitHub? Sign in to your account

feat: Categorical dashboard filters #1677

Merged
merged 22 commits into from
Aug 13, 2024
Merged
Show file tree
Hide file tree
Changes from 1 commit
Commits
Show all changes
22 commits
Select commit Hold shift + click to select a range
9ab3d69
refactor: Set the ground for dashboards data filters
bprusinowski Jul 24, 2024
e246b8a
feat: Display dashboard data filters in the options panel
bprusinowski Jul 24, 2024
7fca689
refactor: Do not mirror configurator dashboard filters in dashboard f…
bprusinowski Jul 24, 2024
42a5eea
feat: Add initial dashboard data filters logic
bprusinowski Jul 25, 2024
bec48ee
fix: Accessing with value
bprusinowski Jul 25, 2024
dbc18ab
chore: Remove console.log
bprusinowski Jul 25, 2024
1c42987
fix: useSyncInteractiveFilters
bprusinowski Jul 25, 2024
24df881
feat: Disable reloading of interactive filters when dashboard filters…
bprusinowski Jul 25, 2024
966810b
feat: Sort dashboard categorical filters
bprusinowski Jul 25, 2024
5a7f5ac
feat: Add a way to set override dashboard filters
bprusinowski Jul 29, 2024
940c287
fix: Maintaining the interactive filters state in tab layout
bprusinowski Jul 29, 2024
b55a51e
fix: Opening of metadata panel from global interactive filters
bprusinowski Jul 29, 2024
e6f6378
refactor: Extract helper
ptbrowne Aug 2, 2024
daf43d9
refactor: Use fromEntries techniques to create new obj
ptbrowne Aug 2, 2024
bb9c7ba
refactor
ptbrowne Aug 5, 2024
2db20b3
refactor: Extract snapshot function
ptbrowne Aug 5, 2024
c88732f
refactor: Save snapshot directly returns restore function
ptbrowne Aug 5, 2024
5b82112
test: Add test for restorer
ptbrowne Aug 5, 2024
0c100f9
refactor: Extract function closer to domain
ptbrowne Aug 5, 2024
ec5a130
refactor: rename for clarity
ptbrowne Aug 12, 2024
f86a7c2
refactor: Import order
ptbrowne Aug 12, 2024
e620968
fix: When changing dashboard config default value, cube query filters…
ptbrowne Aug 12, 2024
File filter

Filter by extension

Filter by extension

Conversations
Failed to load comments.
Loading
Jump to
Jump to file
Failed to load files.
Loading
Diff view
Diff view
7 changes: 4 additions & 3 deletions app/charts/column/chart-column.tsx
Original file line number Diff line number Diff line change
Expand Up @@ -25,8 +25,9 @@ import {
import { Tooltip } from "@/charts/shared/interaction/tooltip";
import { LegendColor } from "@/charts/shared/legend-color";
import { ColumnConfig, useChartConfigFilters } from "@/config-types";
import { hasChartConfigs } from "@/configurator";
import { TimeSlider } from "@/configurator/interactive-filters/time-slider";
import { useDashboardInteractiveFilters } from "@/stores/interactive-filters";
import { useConfiguratorState } from "@/src";

import { ChartProps, VisualizationProps } from "../shared/ChartProps";

Expand All @@ -40,10 +41,10 @@ const ChartColumns = memo((props: ChartProps<ColumnConfig>) => {
const { chartConfig, dimensions } = props;
const { fields, interactiveFiltersConfig } = chartConfig;
const filters = useChartConfigFilters(chartConfig);
const dashboardFilters = useDashboardInteractiveFilters();
const [{ dashboardFilters }] = useConfiguratorState(hasChartConfigs);
const showTimeBrush = shouldShowBrush(
interactiveFiltersConfig,
dashboardFilters.timeRange
dashboardFilters?.timeRange
);
return (
<>
Expand Down
7 changes: 4 additions & 3 deletions app/charts/combo/chart-combo-line-column.tsx
Original file line number Diff line number Diff line change
Expand Up @@ -12,7 +12,8 @@ import { HoverDotMultiple } from "@/charts/shared/interaction/hover-dots-multipl
import { Ruler } from "@/charts/shared/interaction/ruler";
import { Tooltip } from "@/charts/shared/interaction/tooltip";
import { ComboLineColumnConfig } from "@/config-types";
import { useDashboardInteractiveFilters } from "@/stores/interactive-filters";
import { hasChartConfigs } from "@/configurator";
import { useConfiguratorState } from "@/src";

import { ChartProps, VisualizationProps } from "../shared/ChartProps";

Expand All @@ -26,7 +27,7 @@ const ChartComboLineColumn = memo(
(props: ChartProps<ComboLineColumnConfig>) => {
const { chartConfig } = props;
const { interactiveFiltersConfig } = chartConfig;
const dashboardFilters = useDashboardInteractiveFilters();
const [{ dashboardFilters }] = useConfiguratorState(hasChartConfigs);
return (
<ComboLineColumnChart {...props}>
<ChartContainer>
Expand All @@ -38,7 +39,7 @@ const ChartComboLineColumn = memo(
<InteractionColumns />
{shouldShowBrush(
interactiveFiltersConfig,
dashboardFilters.timeRange
dashboardFilters?.timeRange
) && <BrushTime />}
</ChartSvg>
<HoverDotMultiple />
Expand Down
7 changes: 4 additions & 3 deletions app/charts/combo/chart-combo-line-dual.tsx
Original file line number Diff line number Diff line change
Expand Up @@ -12,7 +12,8 @@ import { Ruler } from "@/charts/shared/interaction/ruler";
import { Tooltip } from "@/charts/shared/interaction/tooltip";
import { InteractionHorizontal } from "@/charts/shared/overlay-horizontal";
import { ComboLineDualConfig } from "@/config-types";
import { useDashboardInteractiveFilters } from "@/stores/interactive-filters";
import { hasChartConfigs } from "@/configurator";
import { useConfiguratorState } from "@/src";

import { ChartProps, VisualizationProps } from "../shared/ChartProps";

Expand All @@ -25,7 +26,7 @@ export const ChartComboLineDualVisualization = (
const ChartComboLineDual = memo((props: ChartProps<ComboLineDualConfig>) => {
const { chartConfig } = props;
const { interactiveFiltersConfig } = chartConfig;
const dashboardFilters = useDashboardInteractiveFilters();
const [{ dashboardFilters }] = useConfiguratorState(hasChartConfigs);
return (
<ComboLineDualChart {...props}>
<ChartContainer>
Expand All @@ -37,7 +38,7 @@ const ChartComboLineDual = memo((props: ChartProps<ComboLineDualConfig>) => {
<InteractionHorizontal />
{shouldShowBrush(
interactiveFiltersConfig,
dashboardFilters.timeRange
dashboardFilters?.timeRange
) && <BrushTime />}
</ChartSvg>
<HoverDotMultiple />
Expand Down
7 changes: 4 additions & 3 deletions app/charts/combo/chart-combo-line-single.tsx
Original file line number Diff line number Diff line change
Expand Up @@ -12,7 +12,8 @@ import { Ruler } from "@/charts/shared/interaction/ruler";
import { Tooltip } from "@/charts/shared/interaction/tooltip";
import { InteractionHorizontal } from "@/charts/shared/overlay-horizontal";
import { ComboLineSingleConfig } from "@/config-types";
import { useDashboardInteractiveFilters } from "@/stores/interactive-filters";
import { hasChartConfigs } from "@/configurator";
import { useConfiguratorState } from "@/src";

import { ChartProps, VisualizationProps } from "../shared/ChartProps";

Expand All @@ -26,7 +27,7 @@ const ChartComboLineSingle = memo(
(props: ChartProps<ComboLineSingleConfig>) => {
const { chartConfig } = props;
const { interactiveFiltersConfig } = chartConfig;
const dashboardFilters = useDashboardInteractiveFilters();
const [{ dashboardFilters }] = useConfiguratorState(hasChartConfigs);
return (
<ComboLineSingleChart {...props}>
<ChartContainer>
Expand All @@ -36,7 +37,7 @@ const ChartComboLineSingle = memo(
<InteractionHorizontal />
{shouldShowBrush(
interactiveFiltersConfig,
dashboardFilters.timeRange
dashboardFilters?.timeRange
) && <BrushTime />}
</ChartSvg>
<HoverDotMultiple />
Expand Down
6 changes: 3 additions & 3 deletions app/charts/line/chart-lines.tsx
Original file line number Diff line number Diff line change
Expand Up @@ -17,7 +17,7 @@ import { Tooltip } from "@/charts/shared/interaction/tooltip";
import { LegendColor } from "@/charts/shared/legend-color";
import { InteractionHorizontal } from "@/charts/shared/overlay-horizontal";
import { LineConfig } from "@/config-types";
import { useDashboardInteractiveFilters } from "@/stores/interactive-filters";
import { hasChartConfigs, useConfiguratorState } from "@/configurator";

import { ChartProps, VisualizationProps } from "../shared/ChartProps";

Expand All @@ -30,7 +30,7 @@ export const ChartLinesVisualization = (
const ChartLines = memo((props: ChartProps<LineConfig>) => {
const { chartConfig } = props;
const { fields, interactiveFiltersConfig } = chartConfig;
const dashboardFilters = useDashboardInteractiveFilters();
const [{ dashboardFilters }] = useConfiguratorState(hasChartConfigs);
return (
<LineChart {...props}>
<ChartContainer>
Expand All @@ -41,7 +41,7 @@ const ChartLines = memo((props: ChartProps<LineConfig>) => {
<InteractionHorizontal />
{shouldShowBrush(
interactiveFiltersConfig,
dashboardFilters.timeRange
dashboardFilters?.timeRange
) && <BrushTime />}
</ChartSvg>

Expand Down
8 changes: 4 additions & 4 deletions app/charts/shared/chart-dimensions.tsx
Original file line number Diff line number Diff line change
Expand Up @@ -9,11 +9,11 @@ import { Bounds, Margins } from "@/charts/shared/use-size";
import { CHART_GRID_MIN_HEIGHT } from "@/components/react-grid";
import {
ChartConfig,
DashboardFiltersConfig,
hasChartConfigs,
isLayoutingFreeCanvas,
useConfiguratorState,
} from "@/configurator";
import { useDashboardInteractiveFilters } from "@/stores/interactive-filters";
import { getTextWidth } from "@/utils/get-text-width";

type ComputeChartPaddingProps = {
Expand All @@ -29,7 +29,7 @@ type ComputeChartPaddingProps = {

const computeChartPadding = (
props: ComputeChartPaddingProps & {
dashboardFilters: ReturnType<typeof useDashboardInteractiveFilters>;
dashboardFilters: DashboardFiltersConfig | undefined;
}
) => {
const {
Expand Down Expand Up @@ -61,7 +61,7 @@ const computeChartPadding = (
);

let bottom =
(!dashboardFilters.timeRange?.active &&
(!dashboardFilters?.timeRange.active &&
!!interactiveFiltersConfig?.timeRange.active) ||
animationPresent
? BRUSH_BOTTOM_SPACE
Expand All @@ -87,7 +87,7 @@ export const useChartPadding = (props: ComputeChartPaddingProps) => {
bandDomain,
normalize,
} = props;
const dashboardFilters = useDashboardInteractiveFilters();
const [{ dashboardFilters }] = useConfiguratorState(hasChartConfigs);
return useMemo(() => {
return computeChartPadding({
yScale,
Expand Down
13 changes: 6 additions & 7 deletions app/charts/shared/chart-state.ts
Original file line number Diff line number Diff line change
Expand Up @@ -32,6 +32,8 @@ import {
GenericField,
InteractiveFiltersConfig,
getAnimationField,
hasChartConfigs,
useConfiguratorState,
} from "@/configurator";
import {
parseDate,
Expand All @@ -57,10 +59,7 @@ import {
} from "@/domain/data";
import { Has } from "@/domain/types";
import { ScaleType, TimeUnit } from "@/graphql/resolver-types";
import {
useChartInteractiveFilters,
useDashboardInteractiveFilters,
} from "@/stores/interactive-filters";
import { useChartInteractiveFilters } from "@/stores/interactive-filters";

export type ChartState =
| AreasState
Expand Down Expand Up @@ -508,13 +507,13 @@ export const useChartData = (
// interactive time range
const interactiveFromTime = timeRange.from?.getTime();
const interactiveToTime = timeRange.to?.getTime();
const dashboardFilters = useDashboardInteractiveFilters();
const [{ dashboardFilters }] = useConfiguratorState(hasChartConfigs);
const interactiveTimeRangeFilters = useMemo(() => {
const interactiveTimeRangeFilter: ValuePredicate | null =
getXAsDate &&
interactiveFromTime &&
interactiveToTime &&
(interactiveTimeRange?.active || dashboardFilters.timeRange?.active)
(interactiveTimeRange?.active || dashboardFilters?.timeRange.active)
? (d: Observation) => {
const time = getXAsDate(d).getTime();
return time >= interactiveFromTime && time <= interactiveToTime;
Expand All @@ -523,7 +522,7 @@ export const useChartData = (

return interactiveTimeRangeFilter ? [interactiveTimeRangeFilter] : [];
}, [
dashboardFilters.timeRange,
dashboardFilters?.timeRange,
getXAsDate,
interactiveFromTime,
interactiveToTime,
Expand Down
27 changes: 12 additions & 15 deletions app/components/dashboard-interactive-filters.tsx
Original file line number Diff line number Diff line change
@@ -1,16 +1,13 @@
import {
Collapse,
Slider,
sliderClasses,
useEventCallback,
} from "@mui/material";
import { Slider, sliderClasses, useEventCallback } from "@mui/material";
import { Theme } from "@mui/material/styles";
import { makeStyles } from "@mui/styles";
import { useEffect, useMemo, useState } from "react";

import {
DashboardTimeRangeFilter,
hasChartConfigs,
InteractiveFiltersTimeRange,
useConfiguratorState,
} from "@/configurator";
import {
timeUnitToFormatter,
Expand Down Expand Up @@ -181,17 +178,17 @@ const DashboardTimeRangeSlider = ({
};

export const DashboardInteractiveFilters = () => {
const { timeRange } = useDashboardInteractiveFilters();
return timeRange?.active ? (
<Collapse in={timeRange.active}>
<div>
const [{ dashboardFilters }] = useConfiguratorState(hasChartConfigs);
return (
<div>
{dashboardFilters?.timeRange.active ? (
<DashboardTimeRangeSlider
filter={timeRange}
mounted={timeRange.active}
filter={dashboardFilters.timeRange}
mounted={dashboardFilters.timeRange.active}
/>
</div>
</Collapse>
) : null;
) : null}
</div>
);
};

function stepFromTimeUnit(timeUnit: TimeUnit | undefined) {
Expand Down
9 changes: 3 additions & 6 deletions app/configurator/components/layout-configurator.tsx
Original file line number Diff line number Diff line change
Expand Up @@ -115,12 +115,9 @@ const LayoutLayoutConfigurator = () => {
const LayoutSharedFiltersConfigurator = () => {
const [state, dispatch] = useConfiguratorState(isLayouting);
const { layout } = state;
const {
timeRange,
potentialTimeRangeFilterIris,
dataFilters,
potentialDataFilterIris,
} = useDashboardInteractiveFilters();
const { potentialTimeRangeFilterIris, potentialDataFilterIris } =
useDashboardInteractiveFilters();
const { timeRange, dataFilters } = state.dashboardFilters ?? {};

const locale = useLocale();
const [{ data }] = useConfigsCubeComponents({
Expand Down
12 changes: 5 additions & 7 deletions app/configurator/interactive-filters/time-slider.tsx
Original file line number Diff line number Diff line change
Expand Up @@ -16,6 +16,7 @@ import { TableChartState } from "@/charts/table/table-state";
import { Slider as GenericSlider } from "@/components/form";
import { AnimationField, Filters, SortingField } from "@/config-types";
import { parseDate } from "@/configurator/components/ui-helpers";
import { hasChartConfigs } from "@/configurator/configurator-state";
import {
Dimension,
isTemporalDimension,
Expand All @@ -25,10 +26,8 @@ import {
import { truthy } from "@/domain/types";
import { useTimeFormatUnit } from "@/formatters";
import { Icon } from "@/icons";
import {
useChartInteractiveFilters,
useDashboardInteractiveFilters,
} from "@/stores/interactive-filters";
import { useConfiguratorState } from "@/src";
import { useChartInteractiveFilters } from "@/stores/interactive-filters";
import { Timeline, TimelineProps } from "@/utils/observables";
import {
getSortingOrders,
Expand Down Expand Up @@ -67,15 +66,14 @@ export const TimeSlider = (props: TimeSliderProps) => {
dimensions,
} = props;

const [state] = useConfiguratorState(hasChartConfigs);
const dimension = useMemo(() => {
return dimensions.find((d) => d.iri === componentIri);
}, [componentIri, dimensions]);
const temporal = isTemporalDimension(dimension);
const temporalEntity = isTemporalEntityDimension(dimension);
const temporalOrdinal = isTemporalOrdinalDimension(dimension);

const dashboardFilters = useDashboardInteractiveFilters();

if (!(temporal || temporalEntity || temporalOrdinal)) {
throw new Error("You can only use TimeSlider with temporal dimensions!");
}
Expand Down Expand Up @@ -144,7 +142,7 @@ export const TimeSlider = (props: TimeSliderProps) => {
return new Timeline(timelineProps);
}, [timelineProps]);

if (dashboardFilters.timeRange?.active) {
if (state.dashboardFilters?.timeRange.active) {
return null;
}

Expand Down
20 changes: 1 addition & 19 deletions app/stores/interactive-filters.tsx
Original file line number Diff line number Diff line change
Expand Up @@ -6,11 +6,7 @@ import { getChartSpec } from "@/charts/chart-config-ui-options";
import {
CalculationType,
ChartConfig,
DashboardDataFiltersConfig,
DashboardTimeRangeFilter,
FilterValueSingle,
hasChartConfigs,
useConfiguratorState,
} from "@/configurator";
import { truthy } from "@/domain/types";
import { getOriginalIris, isJoinById } from "@/graphql/join";
Expand Down Expand Up @@ -147,9 +143,7 @@ type InteractiveFiltersContextValue = [
const InteractiveFiltersContext = createContext<
| {
potentialTimeRangeFilterIris: string[];
timeRange: DashboardTimeRangeFilter | undefined;
potentialDataFilterIris: string[];
dataFilters: DashboardDataFiltersConfig | undefined;
stores: Record<ChartConfig["key"], InteractiveFiltersContextValue>;
}
| undefined
Expand Down Expand Up @@ -211,7 +205,6 @@ export const InteractiveFiltersProvider = ({
}: React.PropsWithChildren<{
chartConfigs: ChartConfig[];
}>) => {
const [state] = useConfiguratorState(hasChartConfigs);
const storeRefs = useRef<Record<ChartConfig["key"], StoreApi<State>>>({});

const potentialTimeRangeFilterIris = useMemo(() => {
Expand Down Expand Up @@ -239,24 +232,13 @@ export const InteractiveFiltersProvider = ({
);
}, [chartConfigs]);

const timeRange = state.dashboardFilters?.timeRange;
const dataFilters = state.dashboardFilters?.dataFilters;

const ctxValue = useMemo(
() => ({
potentialTimeRangeFilterIris,
timeRange,
potentialDataFilterIris,
dataFilters,
stores,
}),
[
potentialTimeRangeFilterIris,
timeRange,
potentialDataFilterIris,
dataFilters,
stores,
]
[potentialTimeRangeFilterIris, potentialDataFilterIris, stores]
bprusinowski marked this conversation as resolved.
Show resolved Hide resolved
);

return (
Expand Down