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: Add Animation for Column, Pie and Scatter charts #940

Merged
merged 34 commits into from
May 15, 2023
Merged
Show file tree
Hide file tree
Changes from 10 commits
Commits
Show all changes
34 commits
Select commit Hold shift + click to select a range
6f713fc
refactor: Time slider -> Animation
bprusinowski Jan 19, 2023
18e7343
feat: Add Animation icon
bprusinowski Jan 19, 2023
cc40a66
feat: Use Animation icon for time slider
bprusinowski Jan 19, 2023
0f2b62b
refactor: Make label optional
bprusinowski Jan 20, 2023
9b5562d
feat: Add animation field to types and migrations
bprusinowski Jan 20, 2023
e7eb303
feat: Handle animation field changes
bprusinowski Jan 20, 2023
c63e799
chore: Remove TimeSlider UI elements
bprusinowski Jan 20, 2023
e6786ec
feat: Hide animation field behind a flag
bprusinowski Jan 20, 2023
4d2d62f
feat: Add label
bprusinowski Jan 20, 2023
2fb1ddb
feat: Consider animation time range filter
bprusinowski Jan 20, 2023
e90d7f3
refactor: Remove interactiveFiltersConfig.timeSlider in favor of Anim…
bprusinowski Jan 20, 2023
a4e56e3
chore: Merge main
bprusinowski Jan 20, 2023
02c7c6b
fix: Lint
bprusinowski Jan 20, 2023
55eee5c
fix: Only migrate timeSlider to charts that support it
bprusinowski Jan 20, 2023
1400d72
feat: Enable Animation field for Pie and Scatterplot
bprusinowski Jan 23, 2023
0b24e50
Merge branch 'main' of github.com:visualize-admin/visualization-tool …
bprusinowski May 8, 2023
76e54d5
fix: Imports
bprusinowski May 8, 2023
c04ceb6
fix: Align optional label with label (field)
bprusinowski May 8, 2023
9121a38
feat: Remove flag from Animation field
bprusinowski May 8, 2023
a753244
feat: Consolidate Add… translations
bprusinowski May 8, 2023
526120e
refactor: Move icons closer to the component that needs them
bprusinowski May 9, 2023
2910e83
refactor: Improve types
bprusinowski May 9, 2023
1f8d086
feat: Add an option to show / hide play button
bprusinowski May 9, 2023
774114b
chore: Remove unused translations
bprusinowski May 9, 2023
68fe8d9
feat: Add an option to control animation duration
bprusinowski May 9, 2023
beb6139
feat: Only show animation duration options when play button is shown
bprusinowski May 9, 2023
48cec8a
feat: Add animation types
bprusinowski May 9, 2023
73ae545
feat: Add animation to ChartConfigAdjusters
bprusinowski May 9, 2023
d0673ea
fix: Type
bprusinowski May 9, 2023
767bf9a
feat: Animate columns in column chart
bprusinowski May 9, 2023
f236df3
docs: Update CHANGELOG
bprusinowski May 9, 2023
6aac288
feat: Add a 10s option to Animation
bprusinowski May 15, 2023
8eb8833
Merge branch 'main' of github.com:visualize-admin/visualization-tool …
bprusinowski May 15, 2023
14df0b1
fix: Lint
bprusinowski May 15, 2023
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
12 changes: 11 additions & 1 deletion app/charts/chart-config-ui-options.ts
Original file line number Diff line number Diff line change
Expand Up @@ -10,9 +10,13 @@ import {
* Related to config-types.ts.
*/

type BaseEncodingFieldType = "animation";
type MapEncodingFieldType = "baseLayer" | "areaLayer" | "symbolLayer";
type XYEncodingFieldType = "x" | "y" | "segment";
export type EncodingFieldType = MapEncodingFieldType | XYEncodingFieldType;
export type EncodingFieldType =
| BaseEncodingFieldType
| MapEncodingFieldType
| XYEncodingFieldType;

export type EncodingOption =
| { field: "chartSubType" }
Expand Down Expand Up @@ -168,6 +172,12 @@ export const chartConfigOptionsUISpec: ChartSpecs = {
{ field: "useAbbreviations" },
],
},
{
field: "animation",
optional: true,
componentTypes: ["TemporalDimension"],
filters: true,
},
],
interactiveFilters: ["legend", "timeRange"],
},
Expand Down
8 changes: 2 additions & 6 deletions app/charts/shared/chart-helpers.tsx
Original file line number Diff line number Diff line change
Expand Up @@ -37,17 +37,13 @@ export const prepareQueryFilters = (
dataFilters: InteractiveFiltersState["dataFilters"]
): Filters => {
let queryFilters = filters;
const { timeSlider } = interactiveFiltersConfig || {};

if (chartType !== "table" && interactiveFiltersConfig?.dataFilters.active) {
queryFilters = { ...queryFilters, ...dataFilters };
}

queryFilters = omitBy(queryFilters, (v, k) => {
return (
(v.type === "single" && v.value === FIELD_VALUE_NONE) ||
k === timeSlider?.componentIri
);
queryFilters = omitBy(queryFilters, (v) => {
return v.type === "single" && v.value === FIELD_VALUE_NONE;
bprusinowski marked this conversation as resolved.
Show resolved Hide resolved
});

return queryFilters;
Expand Down
Original file line number Diff line number Diff line change
Expand Up @@ -74,7 +74,6 @@ export const ChartAnnotationsSelector = ({
case "dataFilters":
case "legend":
case "timeRange":
case "timeSlider":
return true;

default:
Expand Down
41 changes: 22 additions & 19 deletions app/configurator/components/chart-configurator.tsx
Original file line number Diff line number Diff line change
@@ -1,25 +1,25 @@
import { t, Trans } from "@lingui/macro";
import {
Badge,
BadgeProps,
Box,
Button,
Tooltip,
CircularProgress,
FormControlLabel,
FormControlLabelProps,
IconButton,
Menu,
MenuItem,
Switch,
Theme,
Tooltip,
Typography,
Switch,
FormControlLabel,
FormControlLabelProps,
Badge,
BadgeProps,
} from "@mui/material";
import { makeStyles } from "@mui/styles";
import isEmpty from "lodash/isEmpty";
import isEqual from "lodash/isEqual";
import sortBy from "lodash/sortBy";
import { useEffect, useRef, useState, useMemo } from "react";
import { useEffect, useMemo, useRef, useState } from "react";
import {
DragDropContext,
Draggable,
Expand Down Expand Up @@ -78,6 +78,7 @@ import { InteractiveFiltersConfigurator } from "../interactive-filters/interacti

import { TitleAndDescriptionConfigurator } from "./chart-annotator";
import { ChartTypeSelector } from "./chart-type-selector";
import { flag } from "./flag";

const DataFilterSelectGeneric = ({
dimension,
Expand Down Expand Up @@ -751,18 +752,20 @@ const ChartFields = ({
active={chartConfig.baseLayer.show}
/>
) : (
<ControlTabField
key={field}
component={
isMapConfig(chartConfig) && field === "symbolLayer"
? chartConfig.fields.symbolLayer
? component
: undefined
: component
}
value={field}
labelId={`${chartConfig.chartType}.${field}`}
/>
(field !== "animation" || flag("timeslider")) && (
<ControlTabField
key={field}
component={
isMapConfig(chartConfig) && field === "symbolLayer"
? chartConfig.fields.symbolLayer
? component
: undefined
: component
}
value={field}
labelId={`${chartConfig.chartType}.${field}`}
/>
)
);
})}
</>
Expand Down
16 changes: 9 additions & 7 deletions app/configurator/components/chart-options-selector.tsx
Original file line number Diff line number Diff line change
Expand Up @@ -219,14 +219,13 @@ const EncodingOptionsPanel = ({
const { measures, dimensions } = metaData;
const panelRef = useRef<HTMLDivElement>(null);

const getFieldLabelHint = {
const fieldLabelHint: Partial<Record<EncodingFieldType, string>> = {
x: t({ id: "controls.select.dimension", message: "Select a dimension" }),
y: t({ id: "controls.select.measure", message: "Select a measure" }),
// Empty strings for optional encodings.
baseLayer: "",
areaLayer: "",
symbolLayer: "",
segment: "",
animation: t({
id: "controls.select.dimension",
message: "Select a dimension",
}),
};

useEffect(() => {
Expand Down Expand Up @@ -308,7 +307,7 @@ const EncodingOptionsPanel = ({
<ControlSectionContent gap="none">
<ChartFieldField
field={encoding.field}
label={getFieldLabelHint[encoding.field]}
label={fieldLabelHint[encoding.field]}
optional={encoding.optional}
options={options}
/>
Expand Down Expand Up @@ -377,6 +376,7 @@ const EncodingOptionsPanel = ({
}
/>
)}

{optionsByField["showStandardError"] && hasStandardError && (
<ControlSection>
<SectionTitle iconName="eye">
Expand All @@ -394,9 +394,11 @@ const EncodingOptionsPanel = ({
</ControlSectionContent>
</ControlSection>
)}

{optionsByField["imputationType"] && isAreaConfig(state.chartConfig) && (
<ChartImputationType state={state} disabled={!imputationNeeded} />
)}

<ChartFieldMultiFilter
state={state}
component={component}
Expand Down
12 changes: 12 additions & 0 deletions app/configurator/components/field-i18n.ts
Original file line number Diff line number Diff line change
Expand Up @@ -14,6 +14,10 @@ const fieldLabels = {
id: "controls.axis.vertical",
message: "Vertical axis",
}),
"controls.animation": defineMessage({
id: "controls.animation",
message: "Animation",
}),
"controls.color": defineMessage({ id: "controls.color", message: "Color" }),
"controls.title": defineMessage({ id: "controls.title", message: "Title" }),
"controls.description": defineMessage({
Expand Down Expand Up @@ -174,6 +178,14 @@ export function getFieldLabel(field: string): string {
case "bar.y":
case "y":
return i18n._(fieldLabels["controls.axis.vertical"]);
case "bar.animation":
case "column.animation":
case "line.animation":
case "area.animation":
case "scatterplot.animation":
case "pie.animation":
case "animation":
return i18n._(fieldLabels["controls.animation"]);
case "bar.segment":
case "column.segment":
case "line.segment":
Expand Down
14 changes: 6 additions & 8 deletions app/configurator/components/field.tsx
Original file line number Diff line number Diff line change
Expand Up @@ -36,19 +36,17 @@ import {
getTimeIntervalWithProps,
} from "@/configurator/components/ui-helpers";
import {
isMultiFilterFieldChecked,
Option,
useActiveFieldField,
useChartFieldField,
useChartOptionBooleanField,
useChartOptionRadioField,
useChartOptionSelectField,
useChartOptionSliderField,
useMetaField,
useSingleFilterField,
} from "@/configurator/config-form";
import {
isMultiFilterFieldChecked,
useChartOptionBooleanField,
useChartOptionSelectField,
useMultiFilterContext,
useSingleFilterField,
useSingleFilterSelect,
} from "@/configurator/config-form";
import {
Expand Down Expand Up @@ -685,13 +683,13 @@ export const ColorPickerField = ({
};

export const ChartFieldField = ({
label,
label = "",
field,
options,
optional,
disabled,
}: {
label: string;
label?: string;
field: string;
options: Option[];
optional?: boolean;
Expand Down
4 changes: 2 additions & 2 deletions app/configurator/components/ui-helpers.ts
Original file line number Diff line number Diff line change
Expand Up @@ -188,8 +188,8 @@ export const getIconName = (name: string): IconName => {
return "tableColumnTimeHidden";
case "time":
return "time";
case "play":
return "play";
case "animation":
return "animation";

default:
return "table";
Expand Down
28 changes: 23 additions & 5 deletions app/configurator/config-types.ts
Original file line number Diff line number Diff line change
Expand Up @@ -168,6 +168,9 @@ const GenericSegmentField = t.intersection([
]);
export type GenericSegmentField = t.TypeOf<typeof GenericSegmentField>;

const AnimationField = GenericField;
export type AnimationField = t.TypeOf<typeof AnimationField>;

const SortingField = t.partial({
sorting: t.type({
sortingType: SortingType,
Expand All @@ -192,6 +195,7 @@ const ColumnFields = t.intersection([
}),
t.partial({
segment: ColumnSegmentField,
animation: AnimationField,
}),
]);
const ColumnConfig = t.type(
Expand Down Expand Up @@ -279,6 +283,7 @@ const ScatterPlotFields = t.intersection([
}),
t.partial({
segment: ScatterPlotSegmentField,
animation: AnimationField,
}),
]);
const ScatterPlotConfig = t.type(
Expand All @@ -297,11 +302,14 @@ export type ScatterPlotConfig = t.TypeOf<typeof ScatterPlotConfig>;
const PieSegmentField = t.intersection([GenericSegmentField, SortingField]);
export type PieSegmentField = t.TypeOf<typeof PieSegmentField>;

const PieFields = t.type({
y: GenericField,
// FIXME: "segment" should be "x" for consistency
segment: PieSegmentField,
});
const PieFields = t.intersection([
t.type({
y: GenericField,
// FIXME: "segment" should be "x" for consistency
segment: PieSegmentField,
}),
t.partial({ animation: AnimationField }),
]);
const PieConfig = t.type(
{
version: t.string,
Expand Down Expand Up @@ -627,6 +635,16 @@ export const isSegmentInConfig = (
return !isTableConfig(chartConfig) && !isMapConfig(chartConfig);
};

export const isAnimationInConfig = (
chartConfig: ChartConfig
): chartConfig is ColumnConfig | ScatterPlotConfig | PieConfig => {
return (
chartConfig.chartType === "column" ||
chartConfig.chartType === "scatterplot" ||
chartConfig.chartType === "pie"
);
};

export const isColorFieldInConfig = (
chartConfig: ChartConfig
): chartConfig is MapConfig => {
Expand Down
Loading