From 9347c5e9a9c7a932abc22818afe60db0cb29301b Mon Sep 17 00:00:00 2001 From: Sophia Mersmann Date: Wed, 11 Dec 2024 17:57:40 +0100 Subject: [PATCH] =?UTF-8?q?=E2=9C=A8=20improve=20hover=20state=20of=20the?= =?UTF-8?q?=20facet=20legend?= MIME-Version: 1.0 Content-Type: text/plain; charset=UTF-8 Content-Transfer-Encoding: 8bit --- .../grapher/src/chart/ChartUtils.tsx | 16 +++++++++++-- .../grapher/src/facetChart/FacetChart.tsx | 5 ++++ .../HorizontalColorLegends.tsx | 23 +++++++++++++++---- .../grapher/src/lineCharts/LineChart.tsx | 13 +++++++++-- .../grapher/src/slopeCharts/SlopeChart.tsx | 13 +++++++++-- .../src/stackedCharts/StackedAreaChart.tsx | 17 ++++++++++++-- 6 files changed, 74 insertions(+), 13 deletions(-) diff --git a/packages/@ourworldindata/grapher/src/chart/ChartUtils.tsx b/packages/@ourworldindata/grapher/src/chart/ChartUtils.tsx index 53e43dd8524..5188506049e 100644 --- a/packages/@ourworldindata/grapher/src/chart/ChartUtils.tsx +++ b/packages/@ourworldindata/grapher/src/chart/ChartUtils.tsx @@ -206,9 +206,21 @@ export function byInteractionState(series: { export function getInteractionStateForSeries( series: ChartSeries, - activeSeriesNames: SeriesName[] + props: { + activeSeriesNames: SeriesName[] + // usually the interaction mode is active when there is + // at least one active element. But sometimes the interaction + // mode might be active although there are no active elements. + // For example, when the facet legend is hovered but a particular + // chart doesn't plot the hovered element. + isInteractionModeActive?: boolean + } ): InteractionState { + const activeSeriesNames = props.activeSeriesNames + const isInteractionModeActive = + props.isInteractionModeActive ?? activeSeriesNames.length > 0 + const active = activeSeriesNames.includes(series.seriesName) - const background = activeSeriesNames.length > 0 && !active + const background = isInteractionModeActive && !active return { active, background } } diff --git a/packages/@ourworldindata/grapher/src/facetChart/FacetChart.tsx b/packages/@ourworldindata/grapher/src/facetChart/FacetChart.tsx index cc760503661..54d821f2dc4 100644 --- a/packages/@ourworldindata/grapher/src/facetChart/FacetChart.tsx +++ b/packages/@ourworldindata/grapher/src/facetChart/FacetChart.tsx @@ -724,6 +724,11 @@ export class FacetChart return this.getExternalLegendProp("equalSizeBins") } + @computed get hoverColors(): Color[] | undefined { + if (!this.legendHoverBin) return undefined + return [this.legendHoverBin.color] + } + @computed get numericLegendData(): ColorScaleBin[] { if (!this.isNumericLegend || !this.hideFacetLegends) return [] const allBins: ColorScaleBin[] = this.externalLegends.flatMap( diff --git a/packages/@ourworldindata/grapher/src/horizontalColorLegend/HorizontalColorLegends.tsx b/packages/@ourworldindata/grapher/src/horizontalColorLegend/HorizontalColorLegends.tsx index 7ae57544011..c5ce14e262e 100644 --- a/packages/@ourworldindata/grapher/src/horizontalColorLegend/HorizontalColorLegends.tsx +++ b/packages/@ourworldindata/grapher/src/horizontalColorLegend/HorizontalColorLegends.tsx @@ -27,6 +27,7 @@ import { GRAPHER_FONT_SCALE_12, GRAPHER_FONT_SCALE_12_8, GRAPHER_FONT_SCALE_14, + GRAPHER_OPACITY_MUTE, } from "../core/GrapherConstants" import { darkenColorForLine } from "../color/ColorUtils" @@ -91,8 +92,9 @@ export interface HorizontalColorLegendManager { onLegendMouseLeave?: () => void onLegendMouseOver?: (d: ColorScaleBin) => void onLegendClick?: (d: ColorScaleBin) => void - activeColors?: string[] - focusColors?: string[] + activeColors?: string[] // inactive colors are grayed out + focusColors?: string[] // focused colors are bolded + hoverColors?: string[] // non-hovered colors are muted isStatic?: boolean } @@ -808,12 +810,15 @@ export class HorizontalCategoricalColorLegend extends HorizontalColorLegend { renderLabels(): React.ReactElement { const { manager, marks } = this - const { focusColors } = manager + const { focusColors, hoverColors = [] } = manager return ( {marks.map((mark, index) => { const isFocus = focusColors?.includes(mark.bin.color) + const isNotHovered = + hoverColors.length > 0 && + !hoverColors.includes(mark.bin.color) return ( {mark.label.text} @@ -837,12 +843,15 @@ export class HorizontalCategoricalColorLegend extends HorizontalColorLegend { renderSwatches(): React.ReactElement { const { manager, marks } = this - const { activeColors } = manager + const { activeColors, hoverColors = [] } = manager return ( {marks.map((mark, index) => { const isActive = activeColors?.includes(mark.bin.color) + const isNotHovered = + hoverColors.length > 0 && + !hoverColors.includes(mark.bin.color) const color = mark.bin.patternRef ? `url(#${mark.bin.patternRef})` @@ -851,6 +860,10 @@ export class HorizontalCategoricalColorLegend extends HorizontalColorLegend { const fill = isActive || activeColors === undefined ? color : "#ccc" + const opacity = isNotHovered + ? GRAPHER_OPACITY_MUTE + : manager.legendOpacity + return ( ) })} diff --git a/packages/@ourworldindata/grapher/src/lineCharts/LineChart.tsx b/packages/@ourworldindata/grapher/src/lineCharts/LineChart.tsx index 6f6c4e8c49f..c26fbf9ee06 100644 --- a/packages/@ourworldindata/grapher/src/lineCharts/LineChart.tsx +++ b/packages/@ourworldindata/grapher/src/lineCharts/LineChart.tsx @@ -555,7 +555,13 @@ export class LineChart } @computed get isHoverModeActive(): boolean { - return this.hoveredSeriesNames.length > 0 + return ( + this.hoveredSeriesNames.length > 0 || + // if the external legend is hovered, we want to mute + // all non-hovered series even if the chart doesn't plot + // the currently hovered series + !!this.manager.externalLegendHoverBin + ) } @computed private get hasEntityYearHighlight(): boolean { @@ -1097,7 +1103,10 @@ export class LineChart } private hoverStateForSeries(series: LineChartSeries): InteractionState { - return getInteractionStateForSeries(series, this.hoveredSeriesNames) + return getInteractionStateForSeries(series, { + isInteractionModeActive: this.isHoverModeActive, + activeSeriesNames: this.hoveredSeriesNames, + }) } @computed get renderSeries(): RenderLineChartSeries[] { diff --git a/packages/@ourworldindata/grapher/src/slopeCharts/SlopeChart.tsx b/packages/@ourworldindata/grapher/src/slopeCharts/SlopeChart.tsx index e635f866c1c..c31b447b4d4 100644 --- a/packages/@ourworldindata/grapher/src/slopeCharts/SlopeChart.tsx +++ b/packages/@ourworldindata/grapher/src/slopeCharts/SlopeChart.tsx @@ -237,7 +237,13 @@ export class SlopeChart } @computed private get isHoverModeActive(): boolean { - return this.hoveredSeriesNames.length > 0 + return ( + this.hoveredSeriesNames.length > 0 || + // if the external legend is hovered, we want to mute + // all non-hovered series even if the chart doesn't plot + // the currently hovered series + !!this.manager.externalLegendHoverBin + ) } @computed private get yColumns(): CoreColumn[] { @@ -448,7 +454,10 @@ export class SlopeChart } private hoverStateForSeries(series: SlopeChartSeries): InteractionState { - return getInteractionStateForSeries(series, this.hoveredSeriesNames) + return getInteractionStateForSeries(series, { + isInteractionModeActive: this.isHoverModeActive, + activeSeriesNames: this.hoveredSeriesNames, + }) } @computed private get renderSeries(): RenderSlopeChartSeries[] { diff --git a/packages/@ourworldindata/grapher/src/stackedCharts/StackedAreaChart.tsx b/packages/@ourworldindata/grapher/src/stackedCharts/StackedAreaChart.tsx index dae31017a7c..fde4903f9f3 100644 --- a/packages/@ourworldindata/grapher/src/stackedCharts/StackedAreaChart.tsx +++ b/packages/@ourworldindata/grapher/src/stackedCharts/StackedAreaChart.tsx @@ -301,8 +301,11 @@ export class StackedAreaChart extends AbstractStackedChart { private hoverStateForSeries( series: StackedSeries ): InteractionState { - const focusedSeriesNames = excludeUndefined([this.hoveredSeriesName]) - return getInteractionStateForSeries(series, focusedSeriesNames) + const hoveredSeriesNames = excludeUndefined([this.hoveredSeriesName]) + return getInteractionStateForSeries(series, { + isInteractionModeActive: this.isHoverModeActive, + activeSeriesNames: hoveredSeriesNames, + }) } @computed get lineLegendSeries(): LineLabelSeries[] { @@ -439,6 +442,16 @@ export class StackedAreaChart extends AbstractStackedChart { ) } + @computed get isHoverModeActive(): boolean { + return ( + !!this.hoveredSeriesName || + // if the external legend is hovered, we want to mute + // all non-hovered series even if the chart doesn't plot + // the currently hovered series + !!this.manager.externalLegendHoverBin + ) + } + @computed get hoveredSeriesNames(): string[] { return this.hoveredSeriesName ? [this.hoveredSeriesName] : [] }