diff --git a/.gitignore b/.gitignore index e33cb39db80..0c56cb718ec 100755 --- a/.gitignore +++ b/.gitignore @@ -44,3 +44,4 @@ dist/ .dev.vars **/tsup.config.bundled*.mjs cfstorage/ +vite.*.mjs diff --git a/adminSiteClient/EditorBasicTab.tsx b/adminSiteClient/EditorBasicTab.tsx index 04f05a9cead..1f48d420f64 100644 --- a/adminSiteClient/EditorBasicTab.tsx +++ b/adminSiteClient/EditorBasicTab.tsx @@ -13,6 +13,7 @@ import { StackMode, ALL_GRAPHER_CHART_TYPES, GrapherChartType, + GRAPHER_CHART_TYPES, } from "@ourworldindata/types" import { DimensionSlot, @@ -109,7 +110,7 @@ class DimensionSlotView< const { selection } = grapher const { availableEntityNames, availableEntityNameSet } = selection - if (grapher.isScatter || grapher.isSlopeChart || grapher.isMarimekko) { + if (grapher.isScatter || grapher.isMarimekko) { // chart types that display all entities by default shouldn't select any by default selection.clearSelection() } else if ( @@ -367,13 +368,17 @@ export class EditorBasicTab< ? [] : [value as GrapherChartType] + if (grapher.isLineChart) { + this.addSlopeChart() + } + if (grapher.isMarimekko) { grapher.hideRelativeToggle = false grapher.stackMode = StackMode.relative } - // Give scatterplots and slope charts a default color dimension if they don't have one - if (grapher.isScatter || grapher.isSlopeChart) { + // Give scatterplots a default color and size dimensions + if (grapher.isScatter) { const hasColor = grapher.dimensions.find( (d) => d.property === DimensionProperty.color ) @@ -382,10 +387,7 @@ export class EditorBasicTab< variableId: CONTINENTS_INDICATOR_ID, property: DimensionProperty.color, }) - } - // Give scatterplots a default size dimension if they don't have one - if (grapher.isScatter) { const hasSize = grapher.dimensions.find( (d) => d.property === DimensionProperty.size ) @@ -417,6 +419,32 @@ export class EditorBasicTab< ] } + private addSlopeChart(): void { + const { grapher } = this.props.editor + if (grapher.hasSlopeChart) return + grapher.chartTypes = [ + ...grapher.chartTypes, + GRAPHER_CHART_TYPES.SlopeChart, + ] + } + + private removeSlopeChart(): void { + const { grapher } = this.props.editor + grapher.chartTypes = grapher.chartTypes.filter( + (type) => type !== GRAPHER_CHART_TYPES.SlopeChart + ) + } + + @action.bound toggleSecondarySlopeChart( + shouldHaveSlopeChart: boolean + ): void { + if (shouldHaveSlopeChart) { + this.addSlopeChart() + } else { + this.removeSlopeChart() + } + } + render() { const { editor } = this.props const { grapher } = editor @@ -441,6 +469,13 @@ export class EditorBasicTab< (grapher.hasMapTab = shouldHaveMapTab) } /> + {grapher.isLineChart && ( + + )} {!isIndicatorChart && ( diff --git a/adminSiteClient/EditorFeatures.tsx b/adminSiteClient/EditorFeatures.tsx index 2d9761add1c..0fefaa9c693 100644 --- a/adminSiteClient/EditorFeatures.tsx +++ b/adminSiteClient/EditorFeatures.tsx @@ -62,6 +62,7 @@ export class EditorFeatures { @computed get hideLegend() { return ( this.grapher.isLineChart || + this.grapher.isSlopeChart || this.grapher.isStackedArea || this.grapher.isStackedDiscreteBar ) @@ -77,6 +78,7 @@ export class EditorFeatures { this.grapher.isStackedBar || this.grapher.isStackedDiscreteBar || this.grapher.isLineChart || + this.grapher.isSlopeChart || this.grapher.isScatter || this.grapher.isMarimekko ) @@ -118,9 +120,9 @@ export class EditorFeatures { return true } - // for line charts, specifying a missing data strategy only makes sense + // for line and slope charts, specifying a missing data strategy only makes sense // if there are multiple entities - if (this.grapher.isLineChart) { + if (this.grapher.isLineChart || this.grapher.isSlopeChart) { return ( this.grapher.canChangeEntity || this.grapher.canSelectMultipleEntities @@ -132,7 +134,7 @@ export class EditorFeatures { @computed get showChangeInPrefixToggle() { return ( - this.grapher.isLineChart && + (this.grapher.isLineChart || this.grapher.isSlopeChart) && (this.grapher.isRelativeMode || this.grapher.canToggleRelativeMode) ) } diff --git a/baker/updateChartEntities.ts b/baker/updateChartEntities.ts index 49e582a71d8..cbaf6a74107 100644 --- a/baker/updateChartEntities.ts +++ b/baker/updateChartEntities.ts @@ -106,7 +106,7 @@ const obtainAvailableEntitiesForGrapherConfig = async ( // In these chart types, an unselected entity is still shown const chartTypeShowsUnselectedEntities = - grapher.isScatter || grapher.isSlopeChart || grapher.isMarimekko + grapher.isScatter || grapher.isMarimekko if (canChangeEntities || chartTypeShowsUnselectedEntities) return grapher.tableForSelection.availableEntityNames as string[] diff --git a/db/migration/1732195571407-RemoveColorDimensionFromSlopeCharts.ts b/db/migration/1732195571407-RemoveColorDimensionFromSlopeCharts.ts new file mode 100644 index 00000000000..1750d7e2576 --- /dev/null +++ b/db/migration/1732195571407-RemoveColorDimensionFromSlopeCharts.ts @@ -0,0 +1,32 @@ +import { MigrationInterface, QueryRunner } from "typeorm" + +export class RemoveColorDimensionFromSlopeCharts1732195571407 + implements MigrationInterface +{ + public async up(queryRunner: QueryRunner): Promise { + // remove color dimension for all slope charts + // the y-dimension always comes first and the color dimension second, + // so it's safe to keep the first dimension only + await queryRunner.query(` + -- sql + UPDATE chart_configs + SET + patch = JSON_REPLACE(patch, '$.dimensions', JSON_ARRAY(patch -> '$.dimensions[0]')), + full = JSON_REPLACE(full, '$.dimensions', JSON_ARRAY(full -> '$.dimensions[0]')) + WHERE + chartType = 'SlopeChart' + `) + + // remove the color dimension for slope charts from the chart_dimensions table + await queryRunner.query(` + -- sql + DELETE cd FROM chart_dimensions cd + JOIN charts c ON c.id = cd.chartId + JOIN chart_configs cc ON c.configId = cc.id + WHERE cc.chartType = 'SlopeChart' AND cd.property = 'color' + `) + } + + // eslint-disable-next-line @typescript-eslint/no-empty-function + public async down(): Promise {} +} diff --git a/db/migration/1732291572062-MigrateSlopeCharts.ts b/db/migration/1732291572062-MigrateSlopeCharts.ts new file mode 100644 index 00000000000..98cb50953ac --- /dev/null +++ b/db/migration/1732291572062-MigrateSlopeCharts.ts @@ -0,0 +1,545 @@ +import { + EntitySelectionMode, + GrapherInterface, + ScaleType, +} from "@ourworldindata/types" +import { simpleMerge } from "@ourworldindata/utils" +import { MigrationInterface, QueryRunner } from "typeorm" + +export class MigrateSlopeCharts1732291572062 implements MigrationInterface { + public async up(queryRunner: QueryRunner): Promise { + const slopeCharts = await queryRunner.query(` + -- sql + SELECT c.id, cc.id AS configId, cc.patch, cc.full + FROM charts c + JOIN chart_configs cc ON cc.id = c.configId + WHERE + cc.chartType = 'SlopeChart' + AND cc.full ->> '$.isPublished' = 'true' + `) + + const configUpdatesById = new Map( + configUpdates.map(({ id, config }) => [id, config]) + ) + + for (const chart of slopeCharts) { + const migrationConfig = configUpdatesById.get(chart.id) + if (!migrationConfig) continue + + const patchConfig = JSON.parse(chart.patch) + const fullConfig = JSON.parse(chart.full) + + const newPatchConfig = simpleMerge(patchConfig, migrationConfig) + const newFullConfig = simpleMerge(fullConfig, migrationConfig) + + await queryRunner.query( + ` + -- sql + UPDATE chart_configs + SET + patch = ?, + full = ? + WHERE id = ? + `, + [ + JSON.stringify(newPatchConfig), + JSON.stringify(newFullConfig), + chart.configId, + ] + ) + } + } + + // eslint-disable-next-line @typescript-eslint/no-empty-function + public async down(): Promise {} +} + +const configUpdates: { id: number; config: GrapherInterface }[] = [ + { + id: 414, + config: { + selectedEntityNames: [ + "Colombia", + "Guatemala", + "Indonesia", + "Iran", + "Jamaica", + "Pakistan", + "Trinidad and Tobago", + "Botswana", + "Bolivia", + "Japan", + "United States", + "Sweden", + "Germany", + "Netherlands", + "Belgium", + "France", + "Ireland", + "United Kingdom", + ], + hideRelativeToggle: true, + }, + }, + { + id: 415, + config: { + selectedEntityNames: [ + "Congenital heart anomalies", + "Neonatal preterm birth", + "Neonatal encephalopathy due to birth asphyxia and trauma", + "Congenital birth defects", + "Diarrheal diseases", + "Malaria", + ], + entityType: "cause", + entityTypePlural: "causes", + hideRelativeToggle: true, + hideLegend: false, + }, + }, + { + id: 679, + config: { + selectedEntityNames: [ + "Low-income countries", + "High-income countries", + "Lower-middle-income countries", + "Upper-middle-income countries", + ], + hideRelativeToggle: true, + yAxis: { + scaleType: ScaleType.linear, + }, + }, + }, + { + id: 874, + config: { + selectedEntityNames: [ + "North America (WB)", + "South Asia (WB)", + "Europe and Central Asia (WB)", + "Latin America and Caribbean (WB)", + ], + hideRelativeToggle: true, + }, + }, + { + id: 875, + config: { + selectedEntityNames: [ + "India", + "United States", + "Indonesia", + "Pakistan", + "Nigeria", + ], + hideRelativeToggle: true, + }, + }, + { + id: 1004, + config: { + selectedEntityNames: [ + "Europe (UN)", + "Asia (UN)", + "Africa (UN)", + "Oceania (UN)", + "Northern America (UN)", + "Latin America and the Caribbean (UN)", + ], + }, + }, + { id: 1459, config: {} }, + { + id: 1975, + config: { + selectedEntityNames: [ + "North America", + "South America", + "Europe", + "Asia", + "Oceania", + "Africa", + ], + }, + }, + { + id: 2832, + config: { + selectedEntityNames: [ + "Italy", + "France", + "Finland", + "Norway", + "Estonia", + "United Kingdom", + "Spain", + "Germany", + "Belgium", + ], + hideTimeline: true, + addCountryMode: EntitySelectionMode.Disabled, + }, + }, + { + id: 2833, + config: { + selectedEntityNames: [ + "Belgium", + "Poland", + "Italy", + "Germany", + "Norway", + "Spain", + "France", + "Finland", + "United Kingdom", + "Estonia", + ], + hideTimeline: true, + addCountryMode: EntitySelectionMode.Disabled, + }, + }, + { + id: 2834, + config: { + selectedEntityNames: [ + "Belgium", + "Italy", + "Spain", + "Norway", + "France", + "Poland", + "Estonia", + "United Kingdom", + "Finland", + "Germany", + ], + addCountryMode: EntitySelectionMode.Disabled, + }, + }, + { + id: 2835, + config: { + selectedEntityNames: [ + "Estonia", + "Norway", + "Poland", + "United Kingdom", + "France", + "Finland", + "Germany", + "Belgium", + "Italy", + "Spain", + ], + hideTimeline: true, + addCountryMode: EntitySelectionMode.Disabled, + }, + }, + { + id: 2975, + config: { + selectedEntityNames: [ + "Germany", + "Poland", + "United Kingdom", + "Finland", + "Estonia", + "Spain", + "Italy", + "Norway", + "France", + ], + hideTimeline: true, + addCountryMode: EntitySelectionMode.Disabled, + }, + }, + { + id: 2976, + config: { + selectedEntityNames: [ + "Poland", + "Italy", + "Spain", + "Estonia", + "France", + "Germany", + "Belgium", + "United Kingdom", + "Norway", + "Finland", + ], + hideTimeline: true, + addCountryMode: EntitySelectionMode.Disabled, + }, + }, + { + id: 2977, + config: { + selectedEntityNames: [ + "Poland", + "Norway", + "Estonia", + "Finland", + "Germany", + "Belgium", + "United Kingdom", + "Spain", + "Italy", + "France", + ], + hideTimeline: true, + addCountryMode: EntitySelectionMode.Disabled, + }, + }, + { + id: 2978, + config: { + selectedEntityNames: [ + "Poland", + "Norway", + "Belgium", + "Estonia", + "Italy", + "Finland", + "Germany", + "United Kingdom", + "Spain", + "France", + ], + hideTimeline: true, + addCountryMode: EntitySelectionMode.Disabled, + }, + }, + { + id: 2979, + config: { + selectedEntityNames: [ + "United Kingdom", + "Estonia", + "Belgium", + "Italy", + "France", + "Spain", + "Germany", + "Poland", + "Finland", + "Norway", + ], + hideTimeline: true, + addCountryMode: EntitySelectionMode.Disabled, + }, + }, + { + id: 3249, + config: { + selectedEntityNames: [ + "France", + "Italy", + "Japan", + "Portugal", + "Germany", + "Mexico", + "Norway", + "Sweden", + "Taiwan", + "Sri Lanka", + "United Kingdom", + "United States", + ], + }, + }, + { + id: 3359, + config: { + selectedEntityNames: [ + "Mali", + "South Africa", + "Nigeria", + "Niger", + "Chad", + "Ethiopia", + "Kenya", + "Uganda", + "Rwanda", + "Burundi", + "Tanzania", + "Mozambique", + "Madagascar", + "Zambia", + "Congo", + "Democratic Republic of Congo", + "Central African Republic", + "Cameroon", + "Togo", + "Benin", + "Sierra Leone", + "Cote d'Ivoire", + "Burkina Faso", + "Guinea-Bissau", + "Papua New Guinea", + "Senegal", + "Angola", + ], + }, + }, + { + id: 3364, + config: { + selectedEntityNames: [ + "India", + "Indonesia", + "United States", + "Pakistan", + ], + }, + }, + { + id: 3433, + config: {}, + }, + { + id: 3434, + config: {}, + }, + { + id: 3580, + config: { + entityTypePlural: "species", + }, + }, + { + id: 3620, + config: { + selectedEntityNames: [ + "Low income", + "High income", + "Middle income", + "Low & middle income", + "Lower middle income", + "Upper middle income", + ], + }, + }, + { + id: 3627, + config: { + selectedEntityNames: [ + "Low income", + "High income", + "Middle income", + "Low & middle income", + "Lower middle income", + "Upper middle income", + ], + }, + }, + { + id: 4408, + config: { + selectedEntityNames: [ + "East Asia (MPD)", + "Latin America (MPD)", + "Eastern Europe (MPD)", + "Western Europe (MPD)", + "Western offshoots (MPD)", + "Sub Saharan Africa (MPD)", + "South and South East Asia (MPD)", + "Middle East and North Africa (MPD)", + "World", + ], + }, + }, + { + id: 4764, + config: { + entityTypePlural: "species", + }, + }, + { + id: 6219, + config: { + hideRelativeToggle: true, + }, + }, + { + id: 6529, + config: { + selectedEntityNames: [ + "North America", + "Europe", + "Asia", + "South America", + "Oceania", + "Africa", + ], + }, + }, + { + id: 7150, + config: { + selectedEntityNames: [ + "Ethiopia", + "Myanmar", + "Niger", + "Chad", + "Colombia", + "Indonesia", + "Nigeria", + ], + }, + }, + { + id: 7206, + config: {}, + }, + { + id: 7220, + config: {}, + }, + { + id: 7221, + config: {}, + }, + { + id: 7226, + config: {}, + }, + { + id: 7344, + config: { + selectedEntityNames: [ + "United States", + "Romania", + "France", + "United Kingdom", + "Colombia", + "Mexico", + "Japan", + ], + }, + }, + { + id: 7448, + config: { + hideRelativeToggle: true, + }, + }, + { + id: 8157, + config: { + selectedEntityNames: [ + "South Asia (WB)", + "North America (WB)", + "Sub-Saharan Africa (WB)", + "East Asia and Pacific (WB)", + "Europe and Central Asia (WB)", + "Latin America and Caribbean (WB)", + "Middle East and North Africa (WB)", + ], + }, + }, +] diff --git a/db/migration/1733163041212-EnableSlopeTabForAllLineCharts.ts b/db/migration/1733163041212-EnableSlopeTabForAllLineCharts.ts new file mode 100644 index 00000000000..77599659203 --- /dev/null +++ b/db/migration/1733163041212-EnableSlopeTabForAllLineCharts.ts @@ -0,0 +1,18 @@ +import { MigrationInterface, QueryRunner } from "typeorm" + +export class EnableSlopeTabForAllLineCharts1733163041212 + implements MigrationInterface +{ + public async up(queryRunner: QueryRunner): Promise { + await queryRunner.query(` + update chart_configs cc + join charts c on cc.id = c.configId + set + patch = JSON_SET(cc.patch, '$.chartTypes', JSON_ARRAY('LineChart', 'SlopeChart')), + full = JSON_SET(cc.full, '$.chartTypes', JSON_ARRAY('LineChart', 'SlopeChart')) + where cc.chartType = 'LineChart' + `) + } + + public async down(queryRunner: QueryRunner): Promise {} +} diff --git a/packages/@ourworldindata/components/src/TextWrap/TextWrap.tsx b/packages/@ourworldindata/components/src/TextWrap/TextWrap.tsx index 4f0ece86fa4..10a1f9daa8a 100644 --- a/packages/@ourworldindata/components/src/TextWrap/TextWrap.tsx +++ b/packages/@ourworldindata/components/src/TextWrap/TextWrap.tsx @@ -1,4 +1,4 @@ -import { max, stripHTML, Bounds, FontFamily } from "@ourworldindata/utils" +import { max, stripHTML, Bounds, FontFamily, last } from "@ourworldindata/utils" import { computed } from "mobx" import React from "react" import { Fragment, joinFragments, splitIntoFragments } from "./TextWrapUtils" @@ -11,6 +11,7 @@ interface TextWrapProps { lineHeight?: number fontSize: FontSize fontWeight?: number + firstLineOffset?: number separators?: string[] rawHtml?: boolean } @@ -80,6 +81,9 @@ export class TextWrap { @computed get separators(): string[] { return this.props.separators ?? [" "] } + @computed get firstLineOffset(): number { + return this.props.firstLineOffset ?? 0 + } // We need to take care that HTML tags are not split across lines. // Instead, we want every line to have opening and closing tags for all tags that appear. @@ -148,11 +152,18 @@ export class TextWrap { ? stripHTML(joinFragments(nextLine)) : joinFragments(nextLine) - const nextBounds = Bounds.forText(text, { + let nextBounds = Bounds.forText(text, { fontSize, fontWeight, }) + // add offset to the first line if given + if (lines.length === 0 && this.firstLineOffset) { + nextBounds = nextBounds.set({ + width: nextBounds.width + this.firstLineOffset, + }) + } + if ( startsWithNewline(fragment.text) || (nextBounds.width + 10 > maxWidth && line.length >= 1) @@ -194,16 +205,37 @@ export class TextWrap { else return lines } + @computed get lineCount(): number { + return this.lines.length + } + + @computed get singleLineHeight(): number { + return this.fontSize * this.lineHeight + } + @computed get height(): number { - const { lines, lineHeight, fontSize } = this - if (lines.length === 0) return 0 - return lines.length * lineHeight * fontSize + if (this.lineCount === 0) return 0 + return this.lineCount * this.singleLineHeight + } + + @computed get heightWithoutOffsetedLine(): number { + if (this.firstLineOffset === 0) { + return this.height + } else if (this.lineCount > 1) { + return (this.lineCount - 1) * this.singleLineHeight + } else { + return 0 + } } @computed get width(): number { return max(this.lines.map((l) => l.width)) ?? 0 } + @computed get lastLineWidth(): number { + return last(this.lines)?.width ?? 0 + } + @computed get htmlStyle(): any { const { fontSize, fontWeight, lineHeight } = this return { @@ -266,10 +298,17 @@ export class TextWrap { textProps, id, }: { textProps?: React.SVGProps; id?: string } = {} - ): React.ReactElement | null { - const { props, lines, fontSize, fontWeight, lineHeight } = this - - if (lines.length === 0) return null + ): React.ReactElement { + const { + props, + lines, + fontSize, + fontWeight, + lineHeight, + firstLineOffset, + } = this + + if (lines.length === 0) return <> const [correctedX, correctedY] = this.getPositionForSvgRendering(x, y) @@ -283,25 +322,21 @@ export class TextWrap { {...textProps} > {lines.map((line, i) => { + const x = correctedX + (i === 0 ? firstLineOffset : 0) + const y = correctedY + lineHeight * fontSize * i + if (props.rawHtml) return ( ) else return ( - + {line.text} ) diff --git a/packages/@ourworldindata/components/src/TextWrap/TextWrapGroup.test.ts b/packages/@ourworldindata/components/src/TextWrap/TextWrapGroup.test.ts new file mode 100644 index 00000000000..3b10adaf3ed --- /dev/null +++ b/packages/@ourworldindata/components/src/TextWrap/TextWrapGroup.test.ts @@ -0,0 +1,121 @@ +#! /usr/bin/env jest + +import { TextWrap } from "./TextWrap" +import { TextWrapGroup } from "./TextWrapGroup" + +const FONT_SIZE = 14 +const TEXT = "Lower middle-income countries" +const MAX_WIDTH = 150 + +const textWrap = new TextWrap({ + text: TEXT, + maxWidth: MAX_WIDTH, + fontSize: FONT_SIZE, +}) + +it("should work like TextWrap for a single fragment", () => { + const textWrapGroup = new TextWrapGroup({ + fragments: [{ text: TEXT }], + maxWidth: MAX_WIDTH, + fontSize: FONT_SIZE, + }) + + const firstTextWrap = textWrapGroup.textWraps[0] + expect(firstTextWrap.text).toEqual(textWrap.text) + expect(firstTextWrap.width).toEqual(textWrap.width) + expect(firstTextWrap.height).toEqual(textWrap.height) + expect(firstTextWrap.lines).toEqual(textWrap.lines) +}) + +it("should place fragments in-line if there is space", () => { + const textWrapGroup = new TextWrapGroup({ + fragments: [{ text: TEXT }, { text: "30 million" }], + maxWidth: MAX_WIDTH, + fontSize: FONT_SIZE, + }) + + expect(textWrapGroup.text).toEqual([TEXT, "30 million"].join(" ")) + expect(textWrapGroup.height).toEqual(textWrap.height) +}) + +it("should place the second segment in a new line if preferred", () => { + const maxWidth = 250 + const textWrapGroup = new TextWrapGroup({ + fragments: [ + { text: TEXT }, + { text: "30 million", newLine: "avoid-wrap" }, + ], + maxWidth, + fontSize: FONT_SIZE, + }) + + // 30 million should be placed in a new line, thus the group's height + // should be greater than the textWrap's height + expect(textWrapGroup.height).toBeGreaterThan( + new TextWrap({ + text: TEXT, + maxWidth, + fontSize: FONT_SIZE, + }).height + ) +}) + +it("should place the second segment in the same line if possible", () => { + const maxWidth = 1000 + const textWrapGroup = new TextWrapGroup({ + fragments: [ + { text: TEXT }, + { text: "30 million", newLine: "avoid-wrap" }, + ], + maxWidth, + fontSize: FONT_SIZE, + }) + + // since the max width is large, "30 million" fits into the same line + // as the text of the first fragmemt + expect(textWrapGroup.height).toEqual( + new TextWrap({ + text: TEXT, + maxWidth, + fontSize: FONT_SIZE, + }).height + ) +}) + +it("should place the second segment in the same line if specified", () => { + const maxWidth = 1000 + const textWrapGroup = new TextWrapGroup({ + fragments: [{ text: TEXT }, { text: "30 million", newLine: "always" }], + maxWidth, + fontSize: FONT_SIZE, + }) + + // since the max width is large, "30 million" fits into the same line + // as the text of the first fragmemt + expect(textWrapGroup.height).toBeGreaterThan( + new TextWrap({ + text: TEXT, + maxWidth, + fontSize: FONT_SIZE, + }).height + ) +}) + +it("should use all available space when one fragment exceeds the given max width", () => { + const maxWidth = 150 + const textWrap = new TextWrap({ + text: "Long-word-that-can't-be-broken-up more words", + maxWidth, + fontSize: FONT_SIZE, + }) + const textWrapGroup = new TextWrapGroup({ + fragments: [ + { text: "Long-word-that-can't-be-broken-up more words" }, + { text: "30 million" }, + ], + maxWidth, + fontSize: FONT_SIZE, + }) + expect(textWrap.width).toBeGreaterThan(maxWidth) + expect(textWrapGroup.maxWidth).toEqual(textWrap.width) +}) diff --git a/packages/@ourworldindata/components/src/TextWrap/TextWrapGroup.tsx b/packages/@ourworldindata/components/src/TextWrap/TextWrapGroup.tsx new file mode 100644 index 00000000000..5515573dd7e --- /dev/null +++ b/packages/@ourworldindata/components/src/TextWrap/TextWrapGroup.tsx @@ -0,0 +1,230 @@ +import React from "react" +import { computed } from "mobx" +import { TextWrap } from "./TextWrap" +import { splitIntoFragments } from "./TextWrapUtils" +import { Bounds, last, max } from "@ourworldindata/utils" + +interface TextWrapFragment { + text: string + fontWeight?: number + // "always" places the fragment in a new line in all cases + // "avoid-wrap" places the fragment in a new line only if the fragment would wrap otherwise + newLine?: "always" | "avoid-wrap" +} + +interface PlacedTextWrap { + textWrap: TextWrap + yOffset: number +} + +interface TextWrapGroupProps { + fragments: TextWrapFragment[] + maxWidth: number + lineHeight?: number + fontSize: number + fontWeight?: number +} + +export class TextWrapGroup { + props: TextWrapGroupProps + constructor(props: TextWrapGroupProps) { + this.props = props + } + + @computed get lineHeight(): number { + return this.props.lineHeight ?? 1.1 + } + + @computed get fontSize(): number { + return this.props.fontSize + } + + @computed get fontWeight(): number | undefined { + return this.props.fontWeight + } + + @computed get text(): string { + return this.props.fragments.map((fragment) => fragment.text).join(" ") + } + + @computed get maxWidth(): number { + const wordWidths = this.props.fragments.flatMap((fragment) => + splitIntoFragments(fragment.text).map( + ({ text }) => + Bounds.forText(text, { + fontSize: this.fontSize, + fontWeight: fragment.fontWeight ?? this.fontWeight, + }).width + ) + ) + return max([...wordWidths, this.props.maxWidth]) ?? Infinity + } + + private makeTextWrapForFragment( + fragment: TextWrapFragment, + offset = 0 + ): TextWrap { + return new TextWrap({ + text: fragment.text, + maxWidth: this.maxWidth, + lineHeight: this.lineHeight, + fontSize: this.fontSize, + fontWeight: fragment.fontWeight ?? this.fontWeight, + firstLineOffset: offset, + }) + } + + @computed get placedTextWraps(): PlacedTextWrap[] { + const { fragments } = this.props + if (fragments.length === 0) return [] + + const whitespaceWidth = Bounds.forText(" ", { + fontSize: this.fontSize, + }).width + + const textWraps: PlacedTextWrap[] = [ + { + textWrap: this.makeTextWrapForFragment(fragments[0]), + yOffset: 0, + }, + ] + + for (let i = 1; i < fragments.length; i++) { + const fragment = fragments[i] + const { textWrap: lastTextWrap, yOffset: lastYOffset } = + textWraps[i - 1] + + // x-offset for the new text wrap + const offset = lastTextWrap.lastLineWidth + whitespaceWidth + + // place the text wrap in a new line + if (fragment.newLine === "always" || offset > this.maxWidth) { + const textWrap = this.makeTextWrapForFragment(fragment) + const yOffset = lastYOffset + lastTextWrap.height + textWraps.push({ textWrap, yOffset }) + continue + } + + let textWrap = this.makeTextWrapForFragment(fragment, offset) + + let yOffset = lastYOffset + if (textWrap.firstLineOffset === 0) { + yOffset += lastTextWrap.height + } else { + yOffset += + (lastTextWrap.lineCount - 1) * lastTextWrap.singleLineHeight + } + + // some fragments are preferred to break into a new line + // instead of being wrapped + if (fragment.newLine === "avoid-wrap" && textWrap.lineCount > 1) { + textWrap = this.makeTextWrapForFragment(fragment) + yOffset += lastTextWrap.singleLineHeight + } + + textWraps.push({ textWrap, yOffset }) + } + + return textWraps + } + + @computed get textWraps(): TextWrap[] { + return this.placedTextWraps.map(({ textWrap }) => textWrap) + } + + @computed get height(): number { + if (this.placedTextWraps.length === 0) return 0 + const { textWrap, yOffset } = last(this.placedTextWraps)! + return yOffset + textWrap.height + } + + @computed get singleLineHeight(): number { + if (this.textWraps.length === 0) return 0 + return this.textWraps[0].singleLineHeight + } + + @computed get width(): number { + return max(this.textWraps.map((textWrap) => textWrap.width)) ?? 0 + } + + @computed get lines(): { + fragments: Omit[] + textWrap: TextWrap + yOffset: number + }[] { + const lines = [] + for (const { textWrap, yOffset } of this.placedTextWraps) { + for (let i = 0; i < textWrap.lineCount; i++) { + const textWrapLine = textWrap.lines[i] + const isFirstTextWrapLine = i === 0 + + const fragment = { + textWrap, + text: textWrapLine.text, + fontWeight: textWrap.fontWeight, + } + + const lastLine = last(lines) + if ( + textWrap.firstLineOffset > 0 && + isFirstTextWrapLine && + lastLine + ) { + // if the current line is offsetted, add it to the previous line + lastLine.fragments.push(fragment) + } else { + // else, push a new line + lines.push({ + textWrap, + fragments: [fragment], + yOffset: yOffset + i * textWrap.singleLineHeight, + }) + } + } + } + + return lines + } + + render( + x: number, + y: number, + { + textProps, + id, + }: { textProps?: React.SVGProps; id?: string } = {} + ): React.ReactElement { + // Alternatively, we could render each TextWrap one by one. That would + // give us a good but not pixel-perfect result since the text + // measurements are not 100% accurate. To avoid inconsistent spacing + // between text wraps, we split the text into lines and render + // the different styles as tspans within the same text element. + return ( + + {this.lines.map((line) => { + const [textX, textY] = + line.textWrap.getPositionForSvgRendering(x, y) + return ( + + {line.fragments.map((fragment, index) => ( + + {index === 0 ? "" : " "} + {fragment.text} + + ))} + + ) + })} + + ) + } +} diff --git a/packages/@ourworldindata/components/src/index.ts b/packages/@ourworldindata/components/src/index.ts index c7118e4ab21..57b17d1e8ce 100644 --- a/packages/@ourworldindata/components/src/index.ts +++ b/packages/@ourworldindata/components/src/index.ts @@ -1,4 +1,5 @@ export { TextWrap, shortenForTargetWidth } from "./TextWrap/TextWrap.js" +export { TextWrapGroup } from "./TextWrap/TextWrapGroup.js" export { MarkdownTextWrap, diff --git a/packages/@ourworldindata/core-table/src/CoreTableColumns.ts b/packages/@ourworldindata/core-table/src/CoreTableColumns.ts index 40da7cba3dc..64b355d867e 100644 --- a/packages/@ourworldindata/core-table/src/CoreTableColumns.ts +++ b/packages/@ourworldindata/core-table/src/CoreTableColumns.ts @@ -305,12 +305,22 @@ export abstract class AbstractCoreColumn { @imemo get displayName(): string { return ( this.display?.name ?? - this.def.presentation?.titlePublic ?? // this is a bit of an unusual fallback - if display.name is not given, titlePublic is the next best thing before name + // this is a bit of an unusual fallback - if display.name is not given, titlePublic is the next best thing before name + this.def.presentation?.titlePublic ?? this.name ?? "" ) } + @imemo get nonEmptyDisplayName(): string { + return ( + this.display?.name || + // this is a bit of an unusual fallback - if display.name is not given, titlePublic is the next best thing before name + this.def.presentation?.titlePublic || + this.nonEmptyName + ) + } + @imemo get titlePublicOrDisplayName(): IndicatorTitleWithFragments { return this.def.presentation?.titlePublic ? { @@ -526,14 +536,16 @@ export abstract class AbstractCoreColumn { // assumes table is sorted by time @imemo get owidRows(): OwidVariableRow[] { const entities = this.allEntityNames - const times = this.originalTimes + const times = this.allTimes const values = this.values + const originalTimes = this.originalTimes const originalValues = this.originalValues - return range(0, times.length).map((index) => { + return range(0, originalTimes.length).map((index) => { return omitUndefinedValues({ entityName: entities[index], time: times[index], value: values[index], + originalTime: originalTimes[index], originalValue: originalValues[index], }) }) @@ -552,6 +564,23 @@ export abstract class AbstractCoreColumn { return map } + // todo: remove? Should not be on CoreTable + @imemo get owidRowByEntityNameAndTime(): Map< + EntityName, + Map> + > { + const valueByEntityNameAndTime = new Map< + EntityName, + Map> + >() + this.owidRows.forEach((row) => { + if (!valueByEntityNameAndTime.has(row.entityName)) + valueByEntityNameAndTime.set(row.entityName, new Map()) + valueByEntityNameAndTime.get(row.entityName)!.set(row.time, row) + }) + return valueByEntityNameAndTime + } + // todo: remove? Should not be on CoreTable // NOTE: this uses the original times, so any tolerance is effectively unapplied. @imemo get valueByEntityNameAndOriginalTime(): Map< @@ -567,7 +596,7 @@ export abstract class AbstractCoreColumn { valueByEntityNameAndTime.set(row.entityName, new Map()) valueByEntityNameAndTime .get(row.entityName)! - .set(row.time, row.value) + .set(row.originalTime, row.value) }) return valueByEntityNameAndTime } diff --git a/packages/@ourworldindata/core-table/src/OwidTable.test.ts b/packages/@ourworldindata/core-table/src/OwidTable.test.ts index a58250a1ec7..32208310965 100755 --- a/packages/@ourworldindata/core-table/src/OwidTable.test.ts +++ b/packages/@ourworldindata/core-table/src/OwidTable.test.ts @@ -72,7 +72,7 @@ it("can parse data to Javascript data structures", () => { table.get("Population").owidRows.forEach((row) => { expect(typeof row.entityName).toBe("string") expect(row.value).toBeGreaterThan(100) - expect(row.time).toBeGreaterThan(1999) + expect(row.originalTime).toBeGreaterThan(1999) }) }) @@ -632,7 +632,7 @@ describe("tolerance", () => { }) }) -it("assigns originalTime as 'time' in owidRows", () => { +it("assigns originalTime as 'originalTime' in owidRows", () => { const csv = `gdp,year,entityName,entityId,entityCode 1000,2019,USA,, 1001,2020,UK,,` @@ -642,7 +642,7 @@ it("assigns originalTime as 'time' in owidRows", () => { expect.not.arrayContaining([ expect.objectContaining({ entityName: "USA", - time: 2020, + originalTime: 2020, value: 1000, }), ]) @@ -651,7 +651,7 @@ it("assigns originalTime as 'time' in owidRows", () => { expect.not.arrayContaining([ expect.objectContaining({ entityName: "UK", - time: 2019, + originalTime: 2019, value: 1001, }), ]) diff --git a/packages/@ourworldindata/core-table/src/OwidTable.ts b/packages/@ourworldindata/core-table/src/OwidTable.ts index c17d1e7b540..13f2d52026a 100644 --- a/packages/@ourworldindata/core-table/src/OwidTable.ts +++ b/packages/@ourworldindata/core-table/src/OwidTable.ts @@ -325,6 +325,95 @@ export class OwidTable extends CoreTable { ) } + // Drop _all rows_ for an entity if all columns have at least one invalid or missing value for that entity. + dropEntitiesThatHaveSomeMissingOrErrorValueInAllColumns( + columnSlugs: ColumnSlug[] + ): this { + const indexesByEntityName = this.rowIndicesByEntityName + const uniqTimes = new Set(this.allTimes) + + // entity names to iterate over + const entityNamesToIterateOver = new Set(indexesByEntityName.keys()) + + // set of entities we want to keep + const entityNamesToKeep = new Set() + + // total number of entities + const entityCount = entityNamesToIterateOver.size + + // helper function to generate operation name + const makeOpName = (entityNamesToKeep: Set): string => { + const entityNamesToDrop = differenceOfSets([ + this.availableEntityNameSet, + entityNamesToKeep, + ]) + const droppedEntitiesStr = + entityNamesToDrop.size > 0 + ? [...entityNamesToDrop].join(", ") + : "(None)" + return `Drop entities that have some missing or error value in all column: ${columnSlugs.join(", ")}.\nDropped entities: ${droppedEntitiesStr}` + } + + // Optimization: if there is a column that has a valid data entry for + // every entity and every time, we are done + for (let i = 0; i <= columnSlugs.length; i++) { + const slug = columnSlugs[i] + const col = this.get(slug) + + if ( + col.numValues === entityCount * uniqTimes.size && + col.numErrorValues === 0 + ) { + const entityNamesToKeep = new Set(indexesByEntityName.keys()) + + return this.columnFilter( + this.entityNameSlug, + (rowEntityName) => + entityNamesToKeep.has(rowEntityName as string), + makeOpName(entityNamesToKeep) + ) + } + } + + for (let i = 0; i <= columnSlugs.length; i++) { + const slug = columnSlugs[i] + const col = this.get(slug) + + for (const entityName of entityNamesToIterateOver) { + const indicesForEntityName = indexesByEntityName.get(entityName) + if (!indicesForEntityName) + throw new Error("Unexpected: entity not found in index map") + + // Optimization: If the column is missing values for the entity, + // we know we can't make a decision yet, so we skip this entity + if (indicesForEntityName.length < uniqTimes.size) continue + + // Optimization: We don't care about the number of valid/error + // values, we just need to know if there is at least one invalid value + const hasSomeInvalidValueForEntityInCol = + indicesForEntityName.some( + (index) => + !isNotErrorValue( + col.valuesIncludingErrorValues[index] + ) + ) + + // Optimization: If all values are valid, we know we want to keep this entity, + // so we remove it from the entities to iterate over + if (!hasSomeInvalidValueForEntityInCol) { + entityNamesToKeep.add(entityName) + entityNamesToIterateOver.delete(entityName) + } + } + } + + return this.columnFilter( + this.entityNameSlug, + (rowEntityName) => entityNamesToKeep.has(rowEntityName as string), + makeOpName(entityNamesToKeep) + ) + } + private sumsByTime(columnSlug: ColumnSlug): Map { const timeValues = this.timeColumn.values const values = this.get(columnSlug).values as number[] diff --git a/packages/@ourworldindata/explorer/src/GrapherGrammar.ts b/packages/@ourworldindata/explorer/src/GrapherGrammar.ts index 78d6a82344a..9c12109ffd0 100644 --- a/packages/@ourworldindata/explorer/src/GrapherGrammar.ts +++ b/packages/@ourworldindata/explorer/src/GrapherGrammar.ts @@ -3,6 +3,7 @@ import { ColorSchemeName, FacetAxisDomain, FacetStrategy, + GRAPHER_CHART_TYPES, GRAPHER_TAB_OPTIONS, MissingDataStrategy, StackMode, @@ -66,10 +67,11 @@ export const GrapherGrammar: Grammar = { description: `The type of chart to show such as LineChart or ScatterPlot. If set to None, then the chart tab is hidden.`, terminalOptions: toTerminalOptions([ ...ALL_GRAPHER_CHART_TYPES, + `${GRAPHER_CHART_TYPES.LineChart} ${GRAPHER_CHART_TYPES.SlopeChart}`, "None", ]), toGrapherObject: (value) => ({ - chartTypes: value === "None" ? [] : [value], + chartTypes: value === "None" ? [] : value.split(" "), }), }, grapherId: { diff --git a/packages/@ourworldindata/grapher/src/barCharts/DiscreteBarChart.tsx b/packages/@ourworldindata/grapher/src/barCharts/DiscreteBarChart.tsx index 144ed286209..242ddc08248 100644 --- a/packages/@ourworldindata/grapher/src/barCharts/DiscreteBarChart.tsx +++ b/packages/@ourworldindata/grapher/src/barCharts/DiscreteBarChart.tsx @@ -409,11 +409,14 @@ export class DiscreteBarChart {this.placedSeries.map((series) => { return ( - series.label && - series.label.render( - series.entityLabelX, - series.barY - series.label.height / 2, - { textProps: style } + series.label && ( + + {series.label.render( + series.entityLabelX, + series.barY - series.label.height / 2, + { textProps: style } + )} + ) ) })} @@ -990,6 +993,7 @@ function makeProjectedDataPattern(color: string): React.ReactElement { const size = 7 return ( { if (manager.isOnTableTab) return undefined if (manager.isOnMapTab) return GRAPHER_MAP_TYPE if (manager.isOnChartTab) { - return manager.isLineChartThatTurnedIntoDiscreteBar + return manager.isLineChartThatTurnedIntoDiscreteBarActive ? GRAPHER_CHART_TYPES.DiscreteBar : manager.activeChartType } diff --git a/packages/@ourworldindata/grapher/src/chart/ChartInterface.ts b/packages/@ourworldindata/grapher/src/chart/ChartInterface.ts index cfd81654f90..5a3e0f17003 100644 --- a/packages/@ourworldindata/grapher/src/chart/ChartInterface.ts +++ b/packages/@ourworldindata/grapher/src/chart/ChartInterface.ts @@ -17,6 +17,25 @@ export interface ChartSeries { color: Color } +export enum FocusState { + off = "off", // nothing is currently focused + active = "active", // actively focused + background = "background", // another series is actively focused +} + +export enum HoverState { + off = "off", // nothing is currently hovered + active = "active", // actively hovered + background = "background", // another series is actively hovered +} + +export interface ChartSeriesStates { + focus: FocusState + hover: HoverState +} + +export type RenderChartSeries = TChartSeries & ChartSeriesStates + export type ChartTableTransformer = (inputTable: OwidTable) => OwidTable export interface ChartInterface { diff --git a/packages/@ourworldindata/grapher/src/chart/ChartManager.ts b/packages/@ourworldindata/grapher/src/chart/ChartManager.ts index d00e2912578..83009ae205a 100644 --- a/packages/@ourworldindata/grapher/src/chart/ChartManager.ts +++ b/packages/@ourworldindata/grapher/src/chart/ChartManager.ts @@ -17,6 +17,7 @@ import { TooltipManager } from "../tooltip/TooltipProps" import { OwidTable, CoreColumn } from "@ourworldindata/core-table" import { SelectionArray } from "../selection/SelectionArray" +import { InteractionArray } from "../selection/InteractionArray" import { ColumnSlug, SortConfig, TimeBound } from "@ourworldindata/utils" import { ColorScaleBin } from "../color/ColorScaleBin" import { ColorScale } from "../color/ColorScale" @@ -63,11 +64,13 @@ export interface ChartManager { sizeColumnSlug?: ColumnSlug colorColumnSlug?: ColumnSlug + interactionArray?: InteractionArray selection?: SelectionArray | EntityName[] entityType?: string hidePoints?: boolean // for line options startHandleTimeBound?: TimeBound // for relative-to-first-year line chart + hideNoDataSection?: boolean // for slope charts // we need endTime so DiscreteBarCharts and StackedDiscreteBarCharts can // know what date the timeline is set to. and let's pass startTime in, too. @@ -78,7 +81,7 @@ export interface ChartManager { seriesStrategy?: SeriesStrategy sortConfig?: SortConfig - showNoDataArea?: boolean + showNoDataArea?: boolean // No data area in Marimekko charts externalLegendHoverBin?: ColorScaleBin | undefined disableIntroAnimation?: boolean diff --git a/packages/@ourworldindata/grapher/src/chart/ChartUtils.tsx b/packages/@ourworldindata/grapher/src/chart/ChartUtils.tsx index 375e9816c7a..33147bbcc83 100644 --- a/packages/@ourworldindata/grapher/src/chart/ChartUtils.tsx +++ b/packages/@ourworldindata/grapher/src/chart/ChartUtils.tsx @@ -1,5 +1,10 @@ import React from "react" -import { Box, getCountryByName } from "@ourworldindata/utils" +import { + areSetsEqual, + Box, + getCountryByName, + sortBy, +} from "@ourworldindata/utils" import { SeriesStrategy, EntityName, @@ -15,7 +20,9 @@ import { GRAPHER_SIDE_PANEL_CLASS, GRAPHER_TIMELINE_CLASS, GRAPHER_SETTINGS_CLASS, + validChartTypeCombinations, } from "../core/GrapherConstants" +import { FocusState, HoverState, RenderChartSeries } from "./ChartInterface.js" export const autoDetectYColumnSlugs = (manager: ChartManager): string[] => { if (manager.yColumnSlugs && manager.yColumnSlugs.length) @@ -25,11 +32,11 @@ export const autoDetectYColumnSlugs = (manager: ChartManager): string[] => { } export const getDefaultFailMessage = (manager: ChartManager): string => { - if (manager.table.rootTable.isBlank) return `No table loaded yet.` + if (manager.table.rootTable.isBlank) return `No table loaded yet` if (manager.table.rootTable.entityNameColumn.isMissing) - return `Table is missing an EntityName column.` + return `Table is missing an EntityName column` if (manager.table.rootTable.timeColumn.isMissing) - return `Table is missing a Time column.` + return `Table is missing a Time column` const yColumnSlugs = autoDetectYColumnSlugs(manager) if (!yColumnSlugs.length) return "Missing Y axis column" const selection = makeSelectionArray(manager.selection) @@ -175,3 +182,49 @@ export function mapChartTypeNameToQueryParam( return GRAPHER_TAB_QUERY_PARAMS.marimekko } } + +export function findValidChartTypeCombination( + chartTypes: GrapherChartType[] +): GrapherChartType[] | undefined { + const chartTypeSet = new Set(chartTypes) + for (const validCombination of validChartTypeCombinations) { + const validCombinationSet = new Set(validCombination) + if (areSetsEqual(chartTypeSet, validCombinationSet)) + return validCombination + } + return undefined +} + +function byFocusState( + series: RenderChartSeries +): number { + switch (series.focus) { + case FocusState.background: + return 1 + case FocusState.off: + return 2 + case FocusState.active: + return 3 + } +} + +function byHoverState( + series: RenderChartSeries +): number { + switch (series.hover) { + case HoverState.background: + return 1 + case HoverState.off: + return 2 + case HoverState.active: + return 3 + } +} + +export function byHoverThenFocusState( + series: RenderChartSeries +): number { + const hoverScore = byHoverState(series) + const focusScore = byFocusState(series) + return 10 * hoverScore + focusScore +} diff --git a/packages/@ourworldindata/grapher/src/controls/ChartIcons.tsx b/packages/@ourworldindata/grapher/src/controls/ChartIcons.tsx index 00b89c46017..0d89bfc3abd 100644 --- a/packages/@ourworldindata/grapher/src/controls/ChartIcons.tsx +++ b/packages/@ourworldindata/grapher/src/controls/ChartIcons.tsx @@ -104,7 +104,10 @@ export const chartIcons: Record = { strokeLinejoin="round" strokeWidth="1.6" > - + + + + ), diff --git a/packages/@ourworldindata/grapher/src/controls/ContentSwitchers.tsx b/packages/@ourworldindata/grapher/src/controls/ContentSwitchers.tsx index 6c0eaa289a5..848f4340514 100644 --- a/packages/@ourworldindata/grapher/src/controls/ContentSwitchers.tsx +++ b/packages/@ourworldindata/grapher/src/controls/ContentSwitchers.tsx @@ -18,6 +18,7 @@ export interface ContentSwitchersManager { activeTab?: GrapherTabName hasMultipleChartTypes?: boolean setTab: (tab: GrapherTabName) => void + onTabChange: (oldTab: GrapherTabName, newTab: GrapherTabName) => void isNarrow?: boolean isMedium?: boolean isLineChartThatTurnedIntoDiscreteBar?: boolean @@ -112,8 +113,10 @@ export class ContentSwitchers extends React.Component<{ } @action.bound setTab(tabIndex: number): void { + const oldTab = this.manager.activeTab const newTab = this.availableTabs[tabIndex] this.manager.setTab(newTab) + this.manager.onTabChange?.(oldTab!, newTab) } render(): React.ReactElement { @@ -175,9 +178,11 @@ function TabIcon({ case GRAPHER_TAB_NAMES.WorldMap: return default: - const chartIcon = isLineChartThatTurnedIntoDiscreteBar - ? chartIcons[GRAPHER_CHART_TYPES.DiscreteBar] - : chartIcons[tab] + const chartIcon = + tab === GRAPHER_TAB_NAMES.LineChart && + isLineChartThatTurnedIntoDiscreteBar + ? chartIcons[GRAPHER_CHART_TYPES.DiscreteBar] + : chartIcons[tab] return chartIcon } } @@ -193,9 +198,15 @@ function makeTabLabelText( if (tab === GRAPHER_TAB_NAMES.WorldMap) return "Map" if (!options.hasMultipleChartTypes) return "Chart" + if ( + tab === GRAPHER_TAB_NAMES.LineChart && + options.isLineChartThatTurnedIntoDiscreteBar + ) + return "Bar" + switch (tab) { case GRAPHER_TAB_NAMES.LineChart: - return options.isLineChartThatTurnedIntoDiscreteBar ? "Bar" : "Line" + return "Line" case GRAPHER_TAB_NAMES.SlopeChart: return "Slope" diff --git a/packages/@ourworldindata/grapher/src/controls/SettingsMenu.tsx b/packages/@ourworldindata/grapher/src/controls/SettingsMenu.tsx index 30cbd370147..961863e4d64 100644 --- a/packages/@ourworldindata/grapher/src/controls/SettingsMenu.tsx +++ b/packages/@ourworldindata/grapher/src/controls/SettingsMenu.tsx @@ -51,6 +51,7 @@ const { StackedDiscreteBar, StackedBar, Marimekko, + SlopeChart, } = GRAPHER_CHART_TYPES export interface SettingsMenuManager @@ -170,6 +171,7 @@ export class SettingsMenu extends React.Component<{ ScatterPlot, LineChart, Marimekko, + SlopeChart, ].includes(this.chartType as any) } @@ -193,6 +195,7 @@ export class SettingsMenu extends React.Component<{ StackedBar, StackedDiscreteBar, LineChart, + SlopeChart, ].includes(this.chartType as any) const hasProjection = filledDimensions.some( diff --git a/packages/@ourworldindata/grapher/src/controls/settings/AbsRelToggle.tsx b/packages/@ourworldindata/grapher/src/controls/settings/AbsRelToggle.tsx index b90e825960b..d24ce59e359 100644 --- a/packages/@ourworldindata/grapher/src/controls/settings/AbsRelToggle.tsx +++ b/packages/@ourworldindata/grapher/src/controls/settings/AbsRelToggle.tsx @@ -8,7 +8,7 @@ import { } from "@ourworldindata/types" import { LabeledSwitch } from "@ourworldindata/components" -const { LineChart, ScatterPlot } = GRAPHER_CHART_TYPES +const { LineChart, ScatterPlot, SlopeChart } = GRAPHER_CHART_TYPES export interface AbsRelToggleManager { stackMode?: StackMode @@ -38,7 +38,7 @@ export class AbsRelToggle extends React.Component<{ const { activeChartType } = this.manager return activeChartType === ScatterPlot ? "Show the percentage change per year over the the selected time range." - : activeChartType === LineChart + : activeChartType === LineChart || activeChartType === SlopeChart ? "Show proportional changes over time or actual values in their original units." : "Show values as their share of the total or as actual values in their original units." } diff --git a/packages/@ourworldindata/grapher/src/core/Grapher.tsx b/packages/@ourworldindata/grapher/src/core/Grapher.tsx index 99080b5f556..d970b43aef0 100644 --- a/packages/@ourworldindata/grapher/src/core/Grapher.tsx +++ b/packages/@ourworldindata/grapher/src/core/Grapher.tsx @@ -66,7 +66,6 @@ import { extractDetailsFromSyntax, omit, isTouchDevice, - areSetsEqual, } from "@ourworldindata/utils" import { MarkdownTextWrap, @@ -112,6 +111,7 @@ import { GRAPHER_TAB_NAMES, GRAPHER_TAB_QUERY_PARAMS, GrapherTabOption, + SeriesName, } from "@ourworldindata/types" import { BlankOwidTable, @@ -138,7 +138,6 @@ import { GRAPHER_FRAME_PADDING_HORIZONTAL, GRAPHER_FRAME_PADDING_VERTICAL, latestGrapherConfigSchema, - validChartTypeCombinations, GRAPHER_SQUARE_SIZE, } from "../core/GrapherConstants" import { loadVariableDataAndMetadata } from "./loadVariable" @@ -201,6 +200,7 @@ import { ScatterPlotManager } from "../scatterCharts/ScatterPlotChartConstants" import { autoDetectSeriesStrategy, autoDetectYColumnSlugs, + findValidChartTypeCombination, mapChartTypeNameToQueryParam, mapQueryParamToChartTypeName, } from "../chart/ChartUtils" @@ -223,6 +223,7 @@ import { } from "../entitySelector/EntitySelector" import { SlideInDrawer } from "../slideInDrawer/SlideInDrawer" import { BodyDiv } from "../bodyDiv/BodyDiv" +import { InteractionArray } from "../selection/InteractionArray" declare global { interface Window { @@ -434,6 +435,7 @@ export class Grapher // Initializing arrays with `undefined` ensures that empty arrays get serialised @observable selectedEntityNames?: EntityName[] = undefined + @observable focusedSeriesNames?: SeriesName[] = undefined @observable excludedEntities?: number[] = undefined /** IncludedEntities are usually empty which means use all available entities. When includedEntities is set it means "only use these entities". excludedEntities @@ -566,6 +568,7 @@ export class Grapher ) obj.selectedEntityNames = this.selection.selectedEntityNames + obj.focusedSeriesNames = this.interactionArray.focusedEntityNames deleteRuntimeAndUnchangedProps(obj, defaultObject) @@ -607,6 +610,9 @@ export class Grapher if (obj.selectedEntityNames) this.selection.setSelectedEntities(obj.selectedEntityNames) + if (obj.focusedSeriesNames) + this.interactionArray.setFocusedEntities(obj.focusedSeriesNames) + // JSON doesn't support Infinity, so we use strings instead. this.minTime = minTimeBoundFromJSONOrNegativeInfinity(obj.minTime) this.maxTime = maxTimeBoundFromJSONOrPositiveInfinity(obj.maxTime) @@ -887,7 +893,10 @@ export class Grapher ) if (this.isOnSlopeChartTab) - return table.filterByTargetTimes([startTime, endTime]) + return table.filterByTargetTimes( + [startTime, endTime], + table.get(this.yColumnSlugs[0]).tolerance + ) return table.filterByTimeRange(startTime, endTime) } @@ -1312,6 +1321,26 @@ export class Grapher } } + @action.bound onTabChange( + oldTab: GrapherTabName, + newTab: GrapherTabName + ): void { + // if switching from a line to a slope chart and the handles are + // on the same time, then automatically adjust the handles so that + // the slope chart view is meaningful + if ( + oldTab === GRAPHER_TAB_NAMES.LineChart && + newTab === GRAPHER_TAB_NAMES.SlopeChart && + this.areHandlesOnSameTime + ) { + if (this.startHandleTimeBound !== -Infinity) { + this.startHandleTimeBound = -Infinity + } else { + this.endHandleTimeBound = Infinity + } + } + } + // todo: can we remove this? // I believe these states can only occur during editing. @action.bound private ensureValidConfigWhenEditing(): void { @@ -1410,7 +1439,6 @@ export class Grapher if (this.isLineChart || this.isDiscreteBar) return [yAxis, color] else if (this.isScatter) return [yAxis, xAxis, size, color] else if (this.isMarimekko) return [yAxis, xAxis, color] - else if (this.isSlopeChart) return [yAxis, color] return [yAxis] } @@ -1526,21 +1554,31 @@ export class Grapher }) } + @computed get hasProjectedData(): boolean { + return this.inputTable.numericColumnSlugs.some( + (slug) => this.inputTable.get(slug).isProjection + ) + } + @computed get validChartTypes(): GrapherChartType[] { const { chartTypes } = this // all single-chart Graphers are valid if (chartTypes.length <= 1) return chartTypes - const chartTypeSet = new Set(chartTypes) - for (const validCombination of validChartTypeCombinations) { - const validCombinationSet = new Set(validCombination) - if (areSetsEqual(chartTypeSet, validCombinationSet)) - return validCombination - } + // find valid combination in a pre-defined list + const validChartTypes = findValidChartTypeCombination(chartTypes) // if the given combination is not valid, then ignore all but the first chart type - return chartTypes.slice(0, 1) + if (!validChartTypes) return chartTypes.slice(0, 1) + + // projected data is only supported for line charts + const isLineChart = validChartTypes[0] === GRAPHER_CHART_TYPES.LineChart + if (isLineChart && this.hasProjectedData) { + return [GRAPHER_CHART_TYPES.LineChart] + } + + return validChartTypes } @computed get validChartTypeSet(): Set { @@ -1607,7 +1645,7 @@ export class Grapher !this.hideAnnotationFieldsInTitle?.changeInPrefix return ( !this.forceHideAnnotationFieldsInTitle?.changeInPrefix && - this.isOnLineChartTab && + (this.isOnLineChartTab || this.isOnSlopeChartTab) && this.isRelativeMode && showChangeInPrefix ) @@ -1636,7 +1674,7 @@ export class Grapher if (this.shouldAddChangeInPrefixToTitle) text = "Change in " + lowerCaseFirstLetterUnlessAbbreviation(text) - if (this.shouldAddTimeSuffixToTitle) + if (this.shouldAddTimeSuffixToTitle && this.timeTitleSuffix) text = appendAnnotationField(text, this.timeTitleSuffix) return text.trim() @@ -1739,11 +1777,11 @@ export class Grapher return this.xAxis.scaleType } - @computed private get timeTitleSuffix(): string { + @computed private get timeTitleSuffix(): string | undefined { const timeColumn = this.table.timeColumn - if (timeColumn.isMissing) return "" // Do not show year until data is loaded + if (timeColumn.isMissing) return undefined // Do not show year until data is loaded const { startTime, endTime } = this - if (startTime === undefined || endTime === undefined) return "" + if (startTime === undefined || endTime === undefined) return undefined const time = startTime === endTime @@ -1935,7 +1973,7 @@ export class Grapher @computed get typeExceptWhenLineChartAndSingleTimeThenWillBeBarChart(): GrapherChartType { - return this.isLineChartThatTurnedIntoDiscreteBar + return this.isLineChartThatTurnedIntoDiscreteBarActive ? GRAPHER_CHART_TYPES.DiscreteBar : (this.activeChartType ?? GRAPHER_CHART_TYPES.LineChart) } @@ -1994,6 +2032,12 @@ export class Grapher return closestMinTime !== undefined && closestMinTime === closestMaxTime } + @computed get isLineChartThatTurnedIntoDiscreteBarActive(): boolean { + return ( + this.isOnLineChartTab && this.isLineChartThatTurnedIntoDiscreteBar + ) + } + @computed get isOnLineChartTab(): boolean { return this.activeChartType === GRAPHER_CHART_TYPES.LineChart } @@ -2027,7 +2071,7 @@ export class Grapher } @computed get supportsMultipleYColumns(): boolean { - return !(this.isScatter || this.isSlopeChart) + return !this.isScatter } @computed private get xDimension(): ChartDimension | undefined { @@ -2150,7 +2194,8 @@ export class Grapher @computed get relativeToggleLabel(): string { if (this.isOnScatterTab) return "Display average annual change" - else if (this.isOnLineChartTab) return "Display relative change" + else if (this.isOnLineChartTab || this.isOnSlopeChartTab) + return "Display relative change" return "Display relative values" } @@ -2170,6 +2215,7 @@ export class Grapher @computed get canToggleRelativeMode(): boolean { const { isOnLineChartTab, + isOnSlopeChartTab, hideRelativeToggle, areHandlesOnSameTime, yScaleType, @@ -2180,7 +2226,7 @@ export class Grapher isStackedChartSplitByMetric, } = this - if (isOnLineChartTab) + if (isOnLineChartTab || isOnSlopeChartTab) return ( !hideRelativeToggle && !areHandlesOnSameTime && @@ -2498,6 +2544,8 @@ export class Grapher this.props.table?.availableEntities ?? [] ) + interactionArray = new InteractionArray() + @computed get availableEntities(): Entity[] { return this.tableForSelection.availableEntities } @@ -3486,7 +3534,7 @@ export class Grapher } @computed get disablePlay(): boolean { - return this.isOnSlopeChartTab + return false } @computed get animationEndTime(): Time { @@ -3543,6 +3591,7 @@ export class Grapher this.hasChartTab && this.canSelectMultipleEntities && (this.isOnLineChartTab || + this.isOnSlopeChartTab || this.isOnStackedAreaTab || this.isOnStackedBarTab || this.isOnDiscreteBarTab || diff --git a/packages/@ourworldindata/grapher/src/entitySelector/EntitySelector.tsx b/packages/@ourworldindata/grapher/src/entitySelector/EntitySelector.tsx index ebce9d92057..a6590dcf0be 100644 --- a/packages/@ourworldindata/grapher/src/entitySelector/EntitySelector.tsx +++ b/packages/@ourworldindata/grapher/src/entitySelector/EntitySelector.tsx @@ -453,7 +453,7 @@ export class EntitySelector extends React.Component<{ const rows = column.owidRowsByEntityName.get(entityName) ?? [] searchableEntity[column.slug] = maxBy( rows, - (row) => row.time + (row) => row.originalTime )?.value } diff --git a/packages/@ourworldindata/grapher/src/facetChart/FacetChart.tsx b/packages/@ourworldindata/grapher/src/facetChart/FacetChart.tsx index ee2870aac03..6ae1555fb84 100644 --- a/packages/@ourworldindata/grapher/src/facetChart/FacetChart.tsx +++ b/packages/@ourworldindata/grapher/src/facetChart/FacetChart.tsx @@ -295,7 +295,9 @@ export class FacetChart return series.map((series, index) => { const { bounds } = gridBoundsArr[index] const showLegend = !this.hideFacetLegends + const hidePoints = true + const hideNoDataSection = true // NOTE: The order of overrides is important! // We need to preserve most config coming in. @@ -319,6 +321,7 @@ export class FacetChart endTime, missingDataStrategy, backgroundColor, + hideNoDataSection, ...series.manager, xAxisConfig: { ...globalXAxisConfig, @@ -373,6 +376,13 @@ export class FacetChart ) } + @computed private get isYAxisHidden(): boolean { + return ( + this.chartTypeName === GRAPHER_CHART_TYPES.SlopeChart && + this.facetCount >= SHARED_X_AXIS_MIN_FACET_COUNT + ) + } + // Only made public for testing @computed get placedSeries(): PlacedFacetSeries[] { const { intermediateChartInstances } = this @@ -495,11 +505,10 @@ export class FacetChart ...axes.x.config, }, yAxisConfig: { - hideAxis: shouldHideFacetAxis( - yAxis, - cellEdges, - sharedAxesSizes - ), + hideAxis: + this.isYAxisHidden || + shouldHideFacetAxis(yAxis, cellEdges, sharedAxesSizes), + hideGridlines: this.isYAxisHidden, ...series.manager.yAxisConfig, ...axes.y.config, }, @@ -756,7 +765,8 @@ export class FacetChart ) if (this.facetStrategy === FacetStrategy.metric && newBins.length <= 1) return [] - return newBins + const sortedBins = sortBy(newBins, (bin) => bin.label) + return sortedBins } @observable.ref private legendHoverBin: ColorScaleBin | undefined = diff --git a/packages/@ourworldindata/grapher/src/lineCharts/LineChart.test.ts b/packages/@ourworldindata/grapher/src/lineCharts/LineChart.test.ts index c825a821719..3655c9b83e3 100755 --- a/packages/@ourworldindata/grapher/src/lineCharts/LineChart.test.ts +++ b/packages/@ourworldindata/grapher/src/lineCharts/LineChart.test.ts @@ -100,7 +100,7 @@ describe("series naming in multi-column mode", () => { selection: [table.availableEntityNames[0]], } const chart = new LineChart({ manager }) - expect(chart.series[0].seriesName).not.toContain(" - ") + expect(chart.series[0].seriesName).not.toContain(" – ") }) it("combines entity and column name if only one entity is selected and multi entity selection is enabled", () => { @@ -110,7 +110,7 @@ describe("series naming in multi-column mode", () => { selection: [table.availableEntityNames[0]], } const chart = new LineChart({ manager }) - expect(chart.series[0].seriesName).toContain(" - ") + expect(chart.series[0].seriesName).toContain(" – ") }) it("combines entity and column name if multiple entities are selected and multi entity selection is disabled", () => { @@ -120,7 +120,7 @@ describe("series naming in multi-column mode", () => { selection: table.availableEntityNames, } const chart = new LineChart({ manager }) - expect(chart.series[0].seriesName).toContain(" - ") + expect(chart.series[0].seriesName).toContain(" – ") }) }) diff --git a/packages/@ourworldindata/grapher/src/lineCharts/LineChart.tsx b/packages/@ourworldindata/grapher/src/lineCharts/LineChart.tsx index 876a49b57bf..8dc6cfe25a7 100644 --- a/packages/@ourworldindata/grapher/src/lineCharts/LineChart.tsx +++ b/packages/@ourworldindata/grapher/src/lineCharts/LineChart.tsx @@ -23,7 +23,6 @@ import { AxisAlign, Color, HorizontalAlign, - PrimitiveType, makeIdForHumanConsumption, } from "@ourworldindata/utils" import { computed, action, observable } from "mobx" @@ -32,11 +31,7 @@ import { select } from "d3-selection" import { easeLinear } from "d3-ease" import { DualAxisComponent } from "../axis/AxisViews" import { DualAxis, HorizontalAxis, VerticalAxis } from "../axis/Axis" -import { - LineLegend, - LineLabelSeries, - LineLegendManager, -} from "../lineLegend/LineLegend" +import { LineLegend, LineLabelSeries } from "../lineLegend/LineLegend" import { ComparisonLine } from "../scatterCharts/ComparisonLine" import { TooltipFooterIcon } from "../tooltip/TooltipProps.js" import { @@ -57,6 +52,7 @@ import { MissingDataStrategy, ColorScaleConfigInterface, ColorSchemeName, + VerticalAlign, } from "@ourworldindata/types" import { GRAPHER_AXIS_LINE_WIDTH_THICK, @@ -66,7 +62,7 @@ import { } from "../core/GrapherConstants" import { ColorSchemes } from "../color/ColorSchemes" import { AxisConfig, AxisManager } from "../axis/AxisConfig" -import { ChartInterface } from "../chart/ChartInterface" +import { ChartInterface, FocusState, HoverState } from "../chart/ChartInterface" import { LinesProps, LineChartSeries, @@ -74,6 +70,7 @@ import { LinePoint, PlacedLineChartSeries, PlacedPoint, + RenderLineChartSeries, } from "./LineChartConstants" import { OwidTable, @@ -83,6 +80,7 @@ import { import { autoDetectSeriesStrategy, autoDetectYColumnSlugs, + byHoverThenFocusState, getDefaultFailMessage, getSeriesKey, isTargetOutsideElement, @@ -102,6 +100,14 @@ import { HorizontalColorLegendManager, HorizontalNumericColorLegend, } from "../horizontalColorLegend/HorizontalColorLegends" +import { + AnnotationsMap, + getAnnotationsForSeries, + getAnnotationsMap, + getColorKey, + getSeriesName, +} from "./LineChartHelpers" +import { InteractionArray } from "../selection/InteractionArray" const LINE_CHART_CLASS_NAME = "LineChart" @@ -120,237 +126,6 @@ const VARIABLE_COLOR_LINE_OUTLINE_WIDTH = 1.0 // legend const LEGEND_PADDING = 25 -@observer -class Lines extends React.Component { - @computed get bounds(): Bounds { - const { horizontalAxis, verticalAxis } = this.props.dualAxis - return Bounds.fromCorners( - new PointVector(horizontalAxis.range[0], verticalAxis.range[0]), - new PointVector(horizontalAxis.range[1], verticalAxis.range[1]) - ) - } - - @computed private get focusedLines(): PlacedLineChartSeries[] { - const { focusedSeriesNames } = this.props - // If nothing is focused, everything is - if (!focusedSeriesNames.length) return this.props.placedSeries - return this.props.placedSeries.filter((series) => - focusedSeriesNames.includes(series.seriesName) - ) - } - - @computed private get backgroundLines(): PlacedLineChartSeries[] { - const { focusedSeriesNames } = this.props - // if nothing is focused, everything is focused, so nothing is in the background - if (!focusedSeriesNames.length) return [] - return this.props.placedSeries.filter( - (series) => !focusedSeriesNames.includes(series.seriesName) - ) - } - - // Don't display point markers if there are very many of them for performance reasons - // Note that we're using circle elements instead of marker-mid because marker performance in Safari 10 is very poor for some reason - @computed private get hasMarkers(): boolean { - if (this.props.hidePoints) return false - return ( - sum(this.focusedLines.map((series) => series.placedPoints.length)) < - 500 - ) - } - - @computed private get markerRadius(): number { - return this.props.markerRadius ?? DEFAULT_MARKER_RADIUS - } - - @computed private get strokeWidth(): number { - return this.props.lineStrokeWidth ?? DEFAULT_STROKE_WIDTH - } - - @computed private get lineOutlineWidth(): number { - return this.props.lineOutlineWidth ?? DEFAULT_LINE_OUTLINE_WIDTH - } - - private renderPathForSeries( - series: PlacedLineChartSeries, - props: Partial> - ): React.ReactElement { - const strokeDasharray = series.isProjection ? "2,3" : undefined - return ( - [value.x, value.y]) as [ - number, - number, - ][] - )} - /> - ) - } - - private renderFocusLines(): React.ReactElement | void { - if (this.focusedLines.length === 0) return - return ( - - {this.focusedLines.map((series) => { - const strokeDasharray = series.isProjection - ? "2,3" - : undefined - return ( - - {this.renderPathForSeries(series, { - id: makeIdForHumanConsumption( - "outline", - series.seriesName - ), - stroke: - this.props.backgroundColor ?? - GRAPHER_BACKGROUND_DEFAULT, - strokeWidth: - this.strokeWidth + - this.lineOutlineWidth * 2, - })} - {this.props.multiColor ? ( - - ) : ( - this.renderPathForSeries(series, { - id: makeIdForHumanConsumption( - "line", - series.seriesName - ), - stroke: - series.placedPoints[0]?.color ?? - DEFAULT_LINE_COLOR, - }) - )} - - ) - })} - - ) - } - - private renderLineMarkers(): React.ReactElement | void { - const { horizontalAxis } = this.props.dualAxis - if (this.focusedLines.length === 0) return - return ( - - {this.focusedLines.map((series) => { - // If the series only contains one point, then we will always want to show a marker/circle - // because we can't draw a line. - const showMarkers = - (this.hasMarkers || series.placedPoints.length === 1) && - !series.isProjection - return ( - showMarkers && ( - - {series.placedPoints.map((value, index) => ( - - ))} - - ) - ) - })} - - ) - } - - private renderFocusGroups(): React.ReactElement | void { - return ( - <> - {this.renderFocusLines()} - {this.renderLineMarkers()} - - ) - } - - private renderBackgroundGroups(): React.ReactElement | void { - if (this.backgroundLines.length === 0) return - return ( - - {this.backgroundLines.map((series) => ( - - {this.renderPathForSeries(series, { - id: makeIdForHumanConsumption( - "background-line", - series.seriesName - ), - stroke: series.color, - strokeWidth: 1, - strokeOpacity: 0.3, - })} - - ))} - - ) - } - - renderStatic(): React.ReactElement { - return ( - <> - {this.renderBackgroundGroups()} - {this.renderFocusGroups()} - - ) - } - - renderInteractive(): React.ReactElement { - const { bounds } = this - return ( - - - {this.renderBackgroundGroups()} - {this.renderFocusGroups()} - - ) - } - - render(): React.ReactElement { - return this.props.isStatic - ? this.renderStatic() - : this.renderInteractive() - } -} - @observer export class LineChart extends React.Component<{ @@ -359,7 +134,6 @@ export class LineChart }> implements ChartInterface, - LineLegendManager, AxisManager, ColorScaleManager, HorizontalColorLegendManager @@ -402,18 +176,26 @@ export class LineChart // if entities with partial data are not plotted, // make sure they don't show up in the entity selector if (this.missingDataStrategy === MissingDataStrategy.hide) { - table = table.replaceNonNumericCellsWithErrorValues( - this.yColumnSlugs - ) - - table = table.dropEntitiesThatHaveNoDataInSomeColumn( - this.yColumnSlugs - ) + table = table + .replaceNonNumericCellsWithErrorValues(this.yColumnSlugs) + .dropEntitiesThatHaveNoDataInSomeColumn(this.yColumnSlugs) } return table } + @computed get selectionArray(): SelectionArray { + return makeSelectionArray(this.manager.selection) + } + + @computed get interactionArray(): InteractionArray { + return this.manager.interactionArray ?? new InteractionArray() + } + + @computed private get focusedSeriesNameSet(): Set { + return this.interactionArray.focusedEntityNameSet + } + @computed private get missingDataStrategy(): MissingDataStrategy { return this.manager.missingDataStrategy || MissingDataStrategy.auto } @@ -437,10 +219,10 @@ export class LineChart let table = this.transformedTableFromGrapher // The % growth transform cannot be applied in transformTable() because it will filter out // any rows before startHandleTimeBound and change the timeline bounds. - const { isRelativeMode, startHandleTimeBound } = this.manager - if (isRelativeMode && startHandleTimeBound !== undefined) { + const { isRelativeMode, startTime } = this.manager + if (isRelativeMode && startTime !== undefined) { table = table.toTotalGrowthForEachColumnComparedToStartTime( - startHandleTimeBound, + startTime, this.manager.yColumnSlugs ?? [] ) } @@ -544,17 +326,6 @@ export class LineChart : DEFAULT_MARKER_RADIUS } - @computed get selectionArray(): SelectionArray { - return makeSelectionArray(this.manager.selection) - } - - seriesIsBlurred(series: LineChartSeries): boolean { - return ( - this.isFocusModeActive && - !this.focusedSeriesNames.includes(series.seriesName) - ) - } - @computed get activeX(): number | undefined { return ( this.tooltipState.target?.x ?? @@ -577,11 +348,22 @@ export class LineChart y2={verticalAxis.range[1]} stroke="rgba(180,180,180,.4)" /> - {this.series.map((series) => { + {this.renderSeries.map((series) => { const value = series.points.find( (point) => point.x === activeX ) - if (!value || this.seriesIsBlurred(series)) return null + if (!value) return null + + const seriesColor = this.hasColorScale + ? darkenColorForLine( + this.getColorScaleColor(value.colorValue) + ) + : series.color + + const color = + series.focus === FocusState.background + ? "#E7E7E7" + : seriesColor return ( { const { seriesName: name, isProjection: striped } = series - const annotation = this.getAnnotationsForSeries(name) + const annotation = getAnnotationsForSeries( + this.annotationsMap, + name + ) const point = series.points.find( (point) => point.x === target.x ) const blurred = - this.seriesIsBlurred(series) || point === undefined + this.seriesFocusState(series) === + FocusState.background || point === undefined const color = blurred ? BLUR_LINE_COLOR @@ -766,24 +544,38 @@ export class LineChart this.clearHighlightedSeries() } - @computed get focusedSeriesNames(): string[] { + @action.bound onLineLegendClick(seriesName: SeriesName): void { + this.interactionArray.toggleFocus(seriesName) + } + + // TODO + @computed get highlightedSeriesName(): string | undefined { + return this.props.manager.entityYearHighlight?.entityName + } + + @computed get facetLegendHoveredSeriesNames(): string[] { const { externalLegendHoverBin } = this.manager - const focusedSeriesNames = excludeUndefined([ - this.props.manager.entityYearHighlight?.entityName, - this.hoveredSeriesName, - ]) - if (externalLegendHoverBin) { - focusedSeriesNames.push( - ...this.series - .map((s) => s.seriesName) - .filter((name) => externalLegendHoverBin.contains(name)) - ) - } - return focusedSeriesNames + if (!externalLegendHoverBin) return [] + return this.series + .map((s) => s.seriesName) + .filter((name) => externalLegendHoverBin.contains(name)) + } + + @computed get hoveredSeriesNameSet(): Set { + return new Set( + excludeUndefined([ + this.hoveredSeriesName, + ...this.facetLegendHoveredSeriesNames, + ]) + ) + } + + @computed get isHoverModeActive(): boolean { + return this.hoveredSeriesNameSet.size > 0 } @computed get isFocusModeActive(): boolean { - return this.focusedSeriesNames.length > 0 + return this.focusedSeriesNameSet.size > 0 } @action.bound onDocumentClick(e: MouseEvent): void { @@ -842,7 +634,7 @@ export class LineChart } @computed get lineLegendX(): number { - return this.bounds.right - (this.lineLegendDimensions?.width || 0) + return this.bounds.right - this.lineLegendWidth } @computed get lineLegendY(): [number, number] { @@ -875,10 +667,18 @@ export class LineChart .on("end", () => this.forceUpdate()) // Important in case bounds changes during transition } - @computed private get lineLegendDimensions(): LineLegend | undefined { - return !this.manager.showLegend - ? undefined - : new LineLegend({ manager: this }) + @computed private get lineLegendWidth(): number { + if (!this.manager.showLegend) return 0 + + // only pass props that are required to calculate + // the width to avoid circular dependencies + return LineLegend.width({ + labelSeries: this.lineLegendSeries, + maxWidth: this.maxLineLegendWidth, + fontSize: this.fontSize, + fontWeight: this.fontWeight, + verticalAlign: VerticalAlign.top, + }) } @computed get availableFacetStrategies(): FacetStrategy[] { @@ -939,13 +739,28 @@ export class LineChart backgroundColor={this.manager.backgroundColor} /> ))} - {manager.showLegend && } + {manager.showLegend && ( + + )} - > { - return this.inputTable - .getAnnotationColumnForColumn(this.yColumnSlugs[0]) - ?.getUniqueValuesGroupedBy(this.inputTable.entityNameSlug) - } - - getAnnotationsForSeries(seriesName: SeriesName): string | undefined { - const annotationsMap = this.annotationsMap - const annos = annotationsMap?.get(seriesName) - return annos - ? Array.from(annos.values()) - .filter((anno) => anno) - .join(" & ") - : undefined + @computed private get annotationsMap(): AnnotationsMap | undefined { + return getAnnotationsMap(this.inputTable, this.yColumnSlugs[0]) } @computed private get colorScheme(): ColorScheme { @@ -1196,39 +995,6 @@ export class LineChart }) } - private getSeriesName( - entityName: EntityName, - columnName: string, - entityCount: number - ): SeriesName { - if (this.seriesStrategy === SeriesStrategy.entity) { - return entityName - } - if (entityCount > 1 || this.manager.canSelectMultipleEntities) { - return `${entityName} - ${columnName}` - } else { - return columnName - } - } - - private getColorKey( - entityName: EntityName, - columnName: string, - entityCount: number - ): SeriesName { - if (this.seriesStrategy === SeriesStrategy.entity) { - return entityName - } - // If only one entity is plotted, we want to use the column colors. - // Unlike in `getSeriesName`, we don't care whether the user can select - // multiple entities, only whether more than one is plotted. - if (entityCount > 1) { - return `${entityName} - ${columnName}` - } else { - return columnName - } - } - // cache value for performance @computed private get rowIndicesByEntityName(): Map { return this.transformedTable.rowIndex([ @@ -1237,14 +1003,20 @@ export class LineChart } private constructSingleSeries( - entityName: string, - col: CoreColumn + entityName: EntityName, + column: CoreColumn ): LineChartSeries { - const { hasColorScale, transformedTable, colorColumn } = this + const { + manager: { canSelectMultipleEntities = false }, + transformedTable: { availableEntityNames }, + seriesStrategy, + hasColorScale, + colorColumn, + } = this // Construct the points - const timeValues = col.originalTimeColumn.valuesIncludingErrorValues - const values = col.valuesIncludingErrorValues + const timeValues = column.originalTimeColumn.valuesIncludingErrorValues + const values = column.valuesIncludingErrorValues const colorValues = colorColumn.valuesIncludingErrorValues // If Y and Color are the same column, we need to get rid of any duplicate rows. // Duplicates occur because Y doesn't have tolerance applied, but Color does. @@ -1269,26 +1041,34 @@ export class LineChart }) // Construct series properties - const totalEntityCount = transformedTable.availableEntityNames.length - const seriesName = this.getSeriesName( + const columnName = column.nonEmptyDisplayName + const seriesName = getSeriesName({ entityName, - col.displayName, - totalEntityCount - ) + columnName, + seriesStrategy, + availableEntityNames, + canSelectMultipleEntities, + }) + let seriesColor: Color if (hasColorScale) { const colorValue = last(points)?.colorValue seriesColor = this.getColorScaleColor(colorValue) } else { seriesColor = this.categoricalColorAssigner.assign( - this.getColorKey(entityName, col.displayName, totalEntityCount) + getColorKey({ + entityName, + columnName, + seriesStrategy, + availableEntityNames, + }) ) } return { points, seriesName, - isProjection: col.isProjection, + isProjection: column.isProjection, color: seriesColor, } } @@ -1333,9 +1113,49 @@ export class LineChart }) } + @computed get renderSeries(): RenderLineChartSeries[] { + const series: RenderLineChartSeries[] = this.placedSeries.map( + (series) => { + return { + ...series, + focus: this.seriesFocusState(series), + hover: this.seriesHoverState(series), + } + } + ) + + if (this.isFocusModeActive || this.isHoverModeActive) { + return sortBy(series, byHoverThenFocusState) + } + + return series + } + + private seriesIsFocused(series: LineChartSeries): boolean { + return this.focusedSeriesNameSet.has(series.seriesName) + } + + private seriesIsHovered(series: LineChartSeries): boolean { + return this.hoveredSeriesNameSet.has(series.seriesName) + } + + private seriesFocusState(series: LineChartSeries): FocusState { + if (!this.isFocusModeActive) return FocusState.off + return this.seriesIsFocused(series) + ? FocusState.active + : FocusState.background + } + + private seriesHoverState(series: LineChartSeries): HoverState { + if (!this.isHoverModeActive) return HoverState.off + return this.seriesIsHovered(series) + ? HoverState.active + : HoverState.background + } + // Order of the legend items on a line chart should visually correspond // to the order of the lines as the approach the legend - @computed get labelSeries(): LineLabelSeries[] { + @computed get lineLegendSeries(): LineLabelSeries[] { // If there are any projections, ignore non-projection legends // Bit of a hack let seriesToShow = this.series @@ -1350,8 +1170,13 @@ export class LineChart seriesName, // E.g. https://ourworldindata.org/grapher/size-poverty-gap-world label: !this.manager.showLegend ? "" : `${seriesName}`, - annotation: this.getAnnotationsForSeries(seriesName), + annotation: getAnnotationsForSeries( + this.annotationsMap, + seriesName + ), yValue: lastValue, + hover: this.seriesHoverState(series), + focus: this.seriesFocusState(series), } }) } @@ -1414,8 +1239,8 @@ export class LineChart return new DualAxis({ bounds: this.boundsWithoutColorLegend .padRight( - this.lineLegendDimensions - ? this.lineLegendDimensions.width + this.manager.showLegend + ? this.lineLegendWidth : this.defaultRightPadding ) // top padding leaves room for tick labels @@ -1466,3 +1291,220 @@ export class LineChart return undefined } } + +@observer +class Lines extends React.Component { + @computed get bounds(): Bounds { + const { horizontalAxis, verticalAxis } = this.props.dualAxis + return Bounds.fromCorners( + new PointVector(horizontalAxis.range[0], verticalAxis.range[0]), + new PointVector(horizontalAxis.range[1], verticalAxis.range[1]) + ) + } + + @computed private get markerRadius(): number { + return this.props.markerRadius ?? DEFAULT_MARKER_RADIUS + } + + @computed private get strokeWidth(): number { + return this.props.lineStrokeWidth ?? DEFAULT_STROKE_WIDTH + } + + @computed private get lineOutlineWidth(): number { + return this.props.lineOutlineWidth ?? DEFAULT_LINE_OUTLINE_WIDTH + } + + // Don't display point markers if there are very many of them for performance reasons + // Note that we're using circle elements instead of marker-mid because marker performance in Safari 10 is very poor for some reason + @computed private get hasMarkers(): boolean { + if (this.props.hidePoints) return false + const totalPoints = sum( + this.props.series + .filter((s) => { + const nonFocused = s.focus === FocusState.background + const hovered = s.hover === HoverState.active + return !nonFocused || hovered + }) + .map((series) => series.placedPoints.length) + ) + return totalPoints < 500 + } + + private renderPathForSeries( + series: PlacedLineChartSeries, + props: Partial> + ): React.ReactElement { + const strokeDasharray = series.isProjection ? "2,3" : undefined + return ( + [value.x, value.y]) as [ + number, + number, + ][] + )} + /> + ) + } + + private seriesHasMarkers(series: RenderLineChartSeries): boolean { + const nonFocused = series.focus === FocusState.background + const hovered = series.hover === HoverState.active + + // Don't show markers for lines in the background + if (nonFocused && !hovered) return false + + // If the series only contains one point, then we will always want to + // show a marker/circle because we can't draw a line. + return ( + (this.hasMarkers || series.placedPoints.length === 1) && + !series.isProjection + ) + } + + renderLine(series: RenderLineChartSeries): React.ReactElement { + const nonFocused = series.focus === FocusState.background + const hovered = series.hover === HoverState.active + const nonHovered = series.hover === HoverState.background + const color = + !nonFocused || hovered + ? (series.placedPoints[0]?.color ?? DEFAULT_LINE_COLOR) + : "#E7E7E7" + const strokeWidth = nonHovered ? 1 : this.strokeWidth + const strokeOpacity = nonHovered && !nonFocused ? 0.3 : 1 + // console.log(series.seriesName, series.nonHovered) + // const strokeOpacity = series.nonHovered ? 0.3 : 1 + const strokeDasharray = series.isProjection ? "2,3" : undefined + + const outline = this.renderPathForSeries(series, { + id: makeIdForHumanConsumption("outline", series.seriesName), + stroke: this.props.backgroundColor ?? GRAPHER_BACKGROUND_DEFAULT, + strokeWidth: strokeWidth + this.lineOutlineWidth * 2, + }) + + if (this.props.multiColor) { + return ( + <> + {outline} + + + ) + } + + return ( + <> + {outline} + {this.renderPathForSeries(series, { + id: makeIdForHumanConsumption("line", series.seriesName), + stroke: color, + strokeWidth, + strokeOpacity, + })} + + ) + } + + renderLineMarkers( + series: RenderLineChartSeries + ): React.ReactElement | void { + if (!this.seriesHasMarkers(series)) return + + const { horizontalAxis } = this.props.dualAxis + const nonHovered = series.hover === HoverState.background + const opacity = nonHovered ? 0.3 : 1 + + return ( + + {series.placedPoints.map((value, index) => ( + + ))} + + ) + } + + renderLineWithMarkers(series: RenderLineChartSeries): React.ReactElement { + return ( + <> + {this.renderLine(series)} + {this.renderLineMarkers(series)} + + ) + } + + renderStatic(): React.ReactElement { + return ( + <> + {this.props.series.map((series) => { + const showMarkers = this.seriesHasMarkers(series) + return ( + <> + {this.renderLine(series)} + {showMarkers && this.renderLineMarkers(series)} + + ) + })} + + ) + } + + renderInteractive(): React.ReactElement { + const { bounds } = this + return ( + + + {this.props.series.map((series) => { + const showMarkers = this.seriesHasMarkers(series) + return ( + <> + {this.renderLine(series)} + {showMarkers && this.renderLineMarkers(series)} + + ) + })} + + ) + } + + render(): React.ReactElement { + return this.props.isStatic + ? this.renderStatic() + : this.renderInteractive() + } +} diff --git a/packages/@ourworldindata/grapher/src/lineCharts/LineChartConstants.ts b/packages/@ourworldindata/grapher/src/lineCharts/LineChartConstants.ts index e7e9a01cf7e..b2428ed043b 100644 --- a/packages/@ourworldindata/grapher/src/lineCharts/LineChartConstants.ts +++ b/packages/@ourworldindata/grapher/src/lineCharts/LineChartConstants.ts @@ -1,11 +1,7 @@ import { DualAxis } from "../axis/Axis" import { ChartManager } from "../chart/ChartManager" -import { - SeriesName, - CoreValueType, - EntityYearHighlight, -} from "@ourworldindata/types" -import { ChartSeries } from "../chart/ChartInterface" +import { CoreValueType, EntityYearHighlight } from "@ourworldindata/types" +import { ChartSeries, RenderChartSeries } from "../chart/ChartInterface" import { Color } from "@ourworldindata/utils" export interface LinePoint { @@ -30,10 +26,12 @@ export interface PlacedLineChartSeries extends LineChartSeries { placedPoints: PlacedPoint[] } +export type RenderLineChartSeries = RenderChartSeries + export interface LinesProps { dualAxis: DualAxis - placedSeries: PlacedLineChartSeries[] - focusedSeriesNames: SeriesName[] + series: RenderLineChartSeries[] + isHoverModeActive: boolean hidePoints?: boolean lineStrokeWidth?: number lineOutlineWidth?: number @@ -46,5 +44,5 @@ export interface LinesProps { export interface LineChartManager extends ChartManager { entityYearHighlight?: EntityYearHighlight lineStrokeWidth?: number - canSelectMultipleEntities?: boolean + canSelectMultipleEntities?: boolean // used to pick an appropriate series name } diff --git a/packages/@ourworldindata/grapher/src/lineCharts/LineChartHelpers.ts b/packages/@ourworldindata/grapher/src/lineCharts/LineChartHelpers.ts new file mode 100644 index 00000000000..49a1f81597a --- /dev/null +++ b/packages/@ourworldindata/grapher/src/lineCharts/LineChartHelpers.ts @@ -0,0 +1,75 @@ +import { OwidTable } from "@ourworldindata/core-table" +import { + ColumnSlug, + EntityName, + PrimitiveType, + SeriesName, + SeriesStrategy, +} from "@ourworldindata/types" + +export type AnnotationsMap = Map> + +export function getSeriesName({ + entityName, + columnName, + seriesStrategy, + availableEntityNames, + canSelectMultipleEntities, +}: { + entityName: EntityName + columnName: string + seriesStrategy: SeriesStrategy + availableEntityNames: EntityName[] + canSelectMultipleEntities: boolean +}): SeriesName { + // if entities are plotted, use the entity name + if (seriesStrategy === SeriesStrategy.entity) return entityName + + // if columns are plotted, use the column name + // and prepend the entity name if multiple entities can be selected + return availableEntityNames.length > 1 || canSelectMultipleEntities + ? `${entityName} – ${columnName}` + : columnName +} + +export function getColorKey({ + entityName, + columnName, + seriesStrategy, + availableEntityNames, +}: { + entityName: EntityName + columnName: string + seriesStrategy: SeriesStrategy + availableEntityNames: EntityName[] +}): SeriesName { + // if entities are plotted, use the entity name + if (seriesStrategy === SeriesStrategy.entity) return entityName + + // If only one entity is plotted, we want to use the column colors. + // Unlike in `getSeriesName`, we don't care whether the user can select + // multiple entities, only whether more than one is plotted. + return availableEntityNames.length > 1 + ? `${entityName} - ${columnName}` + : columnName +} + +export function getAnnotationsMap( + table: OwidTable, + slug: ColumnSlug +): AnnotationsMap | undefined { + return table + .getAnnotationColumnForColumn(slug) + ?.getUniqueValuesGroupedBy(table.entityNameSlug) +} + +export function getAnnotationsForSeries( + annotationsMap: AnnotationsMap | undefined, + seriesName: SeriesName +): string | undefined { + const annotations = annotationsMap?.get(seriesName) + if (!annotations) return undefined + return Array.from(annotations.values()) + .filter((anno) => anno) + .join(" & ") +} diff --git a/packages/@ourworldindata/grapher/src/lineLegend/LineLegend.test.tsx b/packages/@ourworldindata/grapher/src/lineLegend/LineLegend.test.tsx index 594dc456220..545f319cd3f 100755 --- a/packages/@ourworldindata/grapher/src/lineLegend/LineLegend.test.tsx +++ b/packages/@ourworldindata/grapher/src/lineLegend/LineLegend.test.tsx @@ -1,9 +1,9 @@ #! /usr/bin/env jest import { AxisConfig } from "../axis/AxisConfig" -import { LineLegend, LineLegendManager } from "./LineLegend" +import { LineLegend, LineLegendProps } from "./LineLegend" -const manager: LineLegendManager = { +const props: LineLegendProps = { labelSeries: [ { seriesName: "Canada", @@ -20,13 +20,13 @@ const manager: LineLegendManager = { annotation: "Below Canada", }, ], - lineLegendX: 200, + x: 200, focusedSeriesNames: [], yAxis: new AxisConfig({ min: 0, max: 100 }).toVerticalAxis(), } it("can create a new legend", () => { - const legend = new LineLegend({ manager }) + const legend = new LineLegend(props) expect(legend.sizedLabels.length).toEqual(2) }) diff --git a/packages/@ourworldindata/grapher/src/lineLegend/LineLegend.tsx b/packages/@ourworldindata/grapher/src/lineLegend/LineLegend.tsx index 01613c1a48c..78744001983 100644 --- a/packages/@ourworldindata/grapher/src/lineLegend/LineLegend.tsx +++ b/packages/@ourworldindata/grapher/src/lineLegend/LineLegend.tsx @@ -14,35 +14,45 @@ import { last, maxBy, } from "@ourworldindata/utils" -import { TextWrap } from "@ourworldindata/components" +import { TextWrap, TextWrapGroup } from "@ourworldindata/components" import { computed } from "mobx" import { observer } from "mobx-react" import { VerticalAxis } from "../axis/Axis" -import { EntityName } from "@ourworldindata/types" +import { EntityName, VerticalAlign } from "@ourworldindata/types" import { BASE_FONT_SIZE, GRAPHER_FONT_SCALE_12 } from "../core/GrapherConstants" -import { ChartSeries } from "../chart/ChartInterface" +import { + ChartSeries, + ChartSeriesStates, + FocusState, + HoverState, +} from "../chart/ChartInterface" import { darkenColorForText } from "../color/ColorUtils" +import { AxisConfig } from "../axis/AxisConfig.js" +import { Halo } from "../halo/Halo" // Minimum vertical space between two legend items -const LEGEND_ITEM_MIN_SPACING = 2 +const LEGEND_ITEM_MIN_SPACING = 4 // Horizontal distance from the end of the chart to the start of the marker const MARKER_MARGIN = 4 // Space between the label and the annotation -const ANNOTATION_PADDING = 2 - -const LEFT_PADDING = 35 +const ANNOTATION_PADDING = 1 +const DEFAULT_CONNECTOR_LINE_WIDTH = 35 const DEFAULT_FONT_WEIGHT = 400 -export interface LineLabelSeries extends ChartSeries { +export interface LineLabelSeries + extends ChartSeries, + Partial { label: string yValue: number annotation?: string + formattedValue?: string + placeFormattedValueInNewLine?: boolean yRange?: [number, number] } interface SizedSeries extends LineLabelSeries { - textWrap: TextWrap + textWrap: TextWrapGroup annotationTextWrap?: TextWrap width: number height: number @@ -57,14 +67,6 @@ interface PlacedSeries extends SizedSeries { midY: number } -function getSeriesKey( - series: PlacedSeries, - index: number, - key: string -): string { - return `${key}-${index}-` + series.seriesName -} - function groupBounds(group: PlacedSeries[]): Bounds { const first = group[0] const last = group[group.length - 1] @@ -89,29 +91,54 @@ function stackGroupVertically( @observer class LineLabels extends React.Component<{ series: PlacedSeries[] - uniqueKey: string - needsLines: boolean - isFocus?: boolean - isStatic?: boolean + needsConnectorLines: boolean + connectorLineWidth?: number + anchor?: "start" | "end" + interactive?: boolean onClick?: (series: PlacedSeries) => void onMouseOver?: (series: PlacedSeries) => void onMouseLeave?: (series: PlacedSeries) => void }> { - @computed get markers(): { + private opacityForSeries(series: PlacedSeries): number { + const nonHovered = series.hover === HoverState.background + return !nonHovered ? 1 : 0.6 + } + + private textColorForSeries(series: PlacedSeries): string { + const nonFocused = series.focus === FocusState.background + const hovered = series.hover === HoverState.active + return !nonFocused || hovered + ? darkenColorForText(series.color) + : "#DADADA" + } + + @computed private get anchor(): "start" | "end" { + return this.props.anchor ?? "start" + } + + @computed private get connectorLineWidth(): number { + return this.props.connectorLineWidth ?? DEFAULT_CONNECTOR_LINE_WIDTH + } + + @computed private get markers(): { series: PlacedSeries labelText: { x: number; y: number } connectorLine: { x1: number; x2: number } }[] { return this.props.series.map((series) => { + const direction = this.anchor === "start" ? 1 : -1 + const markerMargin = direction * MARKER_MARGIN + const connectorLineWidth = direction * this.connectorLineWidth + const { x } = series.origBounds const connectorLine = { - x1: x + MARKER_MARGIN, - x2: x + LEFT_PADDING - MARKER_MARGIN, + x1: x + markerMargin, + x2: x + connectorLineWidth - markerMargin, } - const textX = this.props.needsLines - ? connectorLine.x2 + MARKER_MARGIN - : x + MARKER_MARGIN + const textX = this.props.needsConnectorLines + ? connectorLine.x2 + markerMargin + : x + markerMargin const textY = series.bounds.y return { @@ -122,76 +149,64 @@ class LineLabels extends React.Component<{ }) } - @computed get textOpacity(): number { - return this.props.isFocus ? 1 : 0.6 - } - - @computed get textLabels(): React.ReactElement { + @computed private get textLabels(): React.ReactElement { return ( - {this.markers.map(({ series, labelText }, index) => { - const textColor = darkenColorForText(series.color) + {this.markers.map(({ series, labelText }) => { + const textColor = this.textColorForSeries(series) return ( - + {series.textWrap.render(labelText.x, labelText.y, { textProps: { fill: textColor, - opacity: this.textOpacity, + opacity: this.opacityForSeries(series), + textAnchor: this.anchor, }, })} - + ) })} ) } - @computed get textAnnotations(): React.ReactElement | void { + @computed private get textAnnotations(): React.ReactElement | void { const markersWithAnnotations = this.markers.filter( ({ series }) => series.annotationTextWrap !== undefined ) if (!markersWithAnnotations) return return ( - {markersWithAnnotations.map(({ series, labelText }, index) => { + {markersWithAnnotations.map(({ series, labelText }) => { + if (!series.annotationTextWrap) return return ( - - {series.annotationTextWrap?.render( + + {series.annotationTextWrap.render( labelText.x, - labelText.y + series.textWrap.height, + labelText.y + + series.textWrap.height + + ANNOTATION_PADDING, { textProps: { fill: "#333", - opacity: this.textOpacity, + opacity: this.opacityForSeries(series), + textAnchor: this.anchor, style: { fontWeight: 300 }, }, } )} - + ) })} ) } - @computed get connectorLines(): React.ReactElement | void { - if (!this.props.needsLines) return + @computed private get connectorLines(): React.ReactElement | void { + if (!this.props.needsConnectorLines) return return ( - {this.markers.map(({ series, connectorLine }, index) => { - const { isFocus } = this.props + {this.markers.map(({ series, connectorLine }) => { const { x1, x2 } = connectorLine const { level, @@ -200,19 +215,18 @@ class LineLabels extends React.Component<{ bounds: { centerY: rightCenterY }, } = series + const nonFocused = series.focus === FocusState.background + const hovered = series.hover === HoverState.active + const step = (x2 - x1) / (totalLevels + 1) const markerXMid = x1 + step + level * step const d = `M${x1},${leftCenterY} H${markerXMid} V${rightCenterY} H${x2}` - const lineColor = isFocus ? "#999" : "#eee" + const lineColor = !nonFocused || hovered ? "#999" : "#eee" return ( - {this.props.series.map((series, index) => { + {this.props.series.map((series) => { + const x = + this.anchor === "start" + ? series.origBounds.x + : series.origBounds.x - series.bounds.width return ( this.props.onClick?.(series)} onMouseOver={() => this.props.onMouseOver?.(series)} onMouseLeave={() => this.props.onMouseLeave?.(series) } - onClick={() => this.props.onClick?.(series)} style={{ cursor: "default" }} > ) } } -export interface LineLegendManager { +export interface LineLegendProps { labelSeries: LineLabelSeries[] - maxLineLegendWidth?: number + yAxis?: VerticalAxis + + // positioning + x?: number + yRange?: [number, number] + maxWidth?: number + xAnchor?: "start" | "end" + verticalAlign?: VerticalAlign + + // presentation + connectorLineWidth?: number fontSize?: number fontWeight?: number - onLineLegendMouseOver?: (key: EntityName) => void - onLineLegendClick?: (key: EntityName) => void - onLineLegendMouseLeave?: () => void - focusedSeriesNames: EntityName[] - yAxis: VerticalAxis - lineLegendY?: [number, number] - lineLegendX?: number + // used to determine which series should be labelled when there is limited space seriesSortedByImportance?: EntityName[] - isStatic?: boolean + + // interactions + isStatic?: boolean // don't add interactions if true + onClick?: (key: EntityName) => void + onMouseOver?: (key: EntityName) => void + onMouseLeave?: () => void } @observer -export class LineLegend extends React.Component<{ - manager: LineLegendManager -}> { +export class LineLegend extends React.Component { + /** + * Larger than the actual width since the width of the connector lines + * is always added, even if they're not rendered. + * + * This is partly due to a circular dependency (in line and stacked area + * charts), partly to avoid jumpy layout changes (slope charts). + */ + static width(props: LineLegendProps): number { + const test = new LineLegend(props) + return test.maxLabelWidth + test.connectorLineWidth + MARKER_MARGIN + } + + static needsConnectorLines(props: LineLegendProps): boolean { + const test = new LineLegend(props) + return test.needsLines + } + + static fontSize(props: Partial): number { + const test = new LineLegend(props as LineLegendProps) + return test.fontSize + } + @computed private get fontSize(): number { - return GRAPHER_FONT_SCALE_12 * (this.manager.fontSize ?? BASE_FONT_SIZE) + return Math.max( + GRAPHER_FONT_SCALE_12 * (this.props.fontSize ?? BASE_FONT_SIZE), + 11.5 + ) } @computed private get fontWeight(): number { - return this.manager.fontWeight ?? DEFAULT_FONT_WEIGHT + return this.props.fontWeight ?? DEFAULT_FONT_WEIGHT } @computed private get maxWidth(): number { - return this.manager.maxLineLegendWidth ?? 300 + return this.props.maxWidth ?? 300 + } + + @computed private get yAxis(): VerticalAxis { + return this.props.yAxis ?? new VerticalAxis(new AxisConfig()) + } + + @computed private get connectorLineWidth(): number { + return this.props.connectorLineWidth ?? DEFAULT_CONNECTOR_LINE_WIDTH + } + + @computed private get verticalAlign(): VerticalAlign { + return this.props.verticalAlign ?? VerticalAlign.middle } @computed.struct get sizedLabels(): SizedSeries[] { const { fontSize, fontWeight, maxWidth } = this - const maxTextWidth = maxWidth - LEFT_PADDING + const maxTextWidth = maxWidth - this.connectorLineWidth const maxAnnotationWidth = Math.min(maxTextWidth, 150) - return this.manager.labelSeries.map((label) => { + return this.props.labelSeries.map((label) => { + const mainLabel = { text: label.label, fontWeight } + const valueLabel = label.formattedValue + ? { + text: label.formattedValue, + newLine: (label.placeFormattedValueInNewLine + ? "always" + : "avoid-wrap") as "always" | "avoid-wrap", + } + : undefined + const labelFragments = excludeUndefined([mainLabel, valueLabel]) + const textWrap = new TextWrapGroup({ + fragments: labelFragments, + maxWidth: maxTextWidth, + fontSize, + }) const annotationTextWrap = label.annotation ? new TextWrap({ text: label.annotation, maxWidth: maxAnnotationWidth, fontSize: fontSize * 0.9, - lineHeight: 1, }) : undefined - const textWrap = new TextWrap({ - text: label.label, - maxWidth: maxTextWidth, - fontSize, - fontWeight, - lineHeight: 1, - }) + + const annotationWidth = annotationTextWrap + ? annotationTextWrap.width + : 0 + const annotationHeight = annotationTextWrap + ? ANNOTATION_PADDING + annotationTextWrap.height + : 0 + return { ...label, textWrap, annotationTextWrap, - width: - LEFT_PADDING + - Math.max( - textWrap.width, - annotationTextWrap ? annotationTextWrap.width : 0 - ), - height: - textWrap.height + - (annotationTextWrap - ? ANNOTATION_PADDING + annotationTextWrap.height - : 0), + width: Math.max(textWrap.width, annotationWidth), + height: textWrap.height + annotationHeight, } }) } - @computed get width(): number { - if (this.sizedLabels.length === 0) return 0 - return max(this.sizedLabels.map((d) => d.width)) ?? 0 + @computed private get maxLabelWidth(): number { + const { sizedLabels = [] } = this + return max(sizedLabels.map((d) => d.width)) ?? 0 } @computed get onMouseOver(): any { - return this.manager.onLineLegendMouseOver ?? noop + return this.props.onMouseOver ?? noop } @computed get onMouseLeave(): any { - return this.manager.onLineLegendMouseLeave ?? noop + return this.props.onMouseLeave ?? noop } @computed get onClick(): any { - return this.manager.onLineLegendClick ?? noop - } - - @computed get isFocusMode(): boolean { - return this.sizedLabels.some((label) => - this.manager.focusedSeriesNames.includes(label.seriesName) - ) + return this.props.onClick ?? noop } @computed get legendX(): number { - return this.manager.lineLegendX ?? 0 + return this.props.x ?? 0 } @computed get legendY(): [number, number] { - const range = this.manager.lineLegendY ?? this.manager.yAxis.range + const range = this.props.yRange ?? this.yAxis.range return [Math.min(range[1], range[0]), Math.max(range[1], range[0])] } + private getYPositionForSeriesLabel(series: SizedSeries): number { + const y = this.yAxis.place(series.yValue) + const lineHeight = series.textWrap.singleLineHeight + switch (this.verticalAlign) { + case VerticalAlign.middle: + return y - series.height / 2 + case VerticalAlign.top: + return y - lineHeight / 2 + case VerticalAlign.bottom: + return y - series.height + lineHeight / 2 + } + } + // Naive initial placement of each mark at the target height, before collision detection @computed private get initialSeries(): PlacedSeries[] { - const { yAxis } = this.manager - const { legendX, legendY } = this + const { yAxis, legendX, legendY } = this const [legendYMin, legendYMax] = legendY return this.sizedLabels.map((label) => { - // place vertically centered at Y value + const labelHeight = label.height + const labelWidth = label.width + this.connectorLineWidth + const midY = yAxis.place(label.yValue) - const initialY = midY - label.height / 2 const origBounds = new Bounds( legendX, - initialY, - label.width, - label.height + midY - label.height / 2, + labelWidth, + labelHeight ) // ensure label doesn't go beyond the top or bottom of the chart + const initialY = this.getYPositionForSeriesLabel(label) const y = Math.min( Math.max(initialY, legendYMin), - legendYMax - label.height + legendYMax - labelHeight ) - const bounds = new Bounds(legendX, y, label.width, label.height) + const bounds = new Bounds(legendX, y, labelWidth, labelHeight) return { ...label, @@ -490,9 +562,9 @@ export class LineLegend extends React.Component<{ } @computed get sortedSeriesByImportance(): PlacedSeries[] | undefined { - if (!this.manager.seriesSortedByImportance) return undefined + if (!this.props.seriesSortedByImportance) return undefined return excludeUndefined( - this.manager.seriesSortedByImportance.map((seriesName) => + this.props.seriesSortedByImportance.map((seriesName) => this.initialSeriesByName.get(seriesName) ) ) @@ -639,67 +711,13 @@ export class LineLegend extends React.Component<{ } } - @computed private get backgroundSeries(): PlacedSeries[] { - const { focusedSeriesNames } = this.manager - const { isFocusMode } = this - return this.placedSeries.filter( - (mark) => - isFocusMode && !focusedSeriesNames.includes(mark.seriesName) - ) - } - - @computed private get focusedSeries(): PlacedSeries[] { - const { focusedSeriesNames } = this.manager - const { isFocusMode } = this - return this.placedSeries.filter( - (mark) => - !isFocusMode || focusedSeriesNames.includes(mark.seriesName) - ) - } - // Does this placement need line markers or is the position of the labels already clear? @computed private get needsLines(): boolean { return this.placedSeries.some((series) => series.totalLevels > 1) } - private renderBackground(): React.ReactElement { - return ( - - this.onMouseOver(series.seriesName) - } - onClick={(series): void => this.onClick(series.seriesName)} - /> - ) - } - - // All labels are focused by default, moved to background when mouseover of other label - private renderFocus(): React.ReactElement { - return ( - - this.onMouseOver(series.seriesName) - } - onClick={(series): void => this.onClick(series.seriesName)} - onMouseLeave={(series): void => - this.onMouseLeave(series.seriesName) - } - /> - ) - } - - @computed get manager(): LineLegendManager { - return this.props.manager + @computed private get isStatic(): boolean { + return this.props.isStatic ?? false } render(): React.ReactElement { @@ -708,8 +726,20 @@ export class LineLegend extends React.Component<{ id={makeIdForHumanConsumption("line-labels")} className="LineLabels" > - {this.renderBackground()} - {this.renderFocus()} + this.onClick(series.seriesName)} + onMouseOver={(series): void => + this.onMouseOver(series.seriesName) + } + onMouseLeave={(series): void => + this.onMouseLeave(series.seriesName) + } + /> ) } diff --git a/packages/@ourworldindata/grapher/src/mapCharts/MapChart.tsx b/packages/@ourworldindata/grapher/src/mapCharts/MapChart.tsx index f641a1b939b..a07da5438fa 100644 --- a/packages/@ourworldindata/grapher/src/mapCharts/MapChart.tsx +++ b/packages/@ourworldindata/grapher/src/mapCharts/MapChart.tsx @@ -346,12 +346,12 @@ export class MapChart return mapColumn.owidRows .map((row) => { - const { entityName, value, time } = row + const { entityName, value, originalTime } = row const color = this.colorScale.getColor(value) || "red" // todo: color fix if (!color) return undefined return { seriesName: entityName, - time, + time: originalTime, value, isSelected: selectionArray.selectedSet.has(entityName), color, diff --git a/packages/@ourworldindata/grapher/src/mapCharts/MapSparkline.tsx b/packages/@ourworldindata/grapher/src/mapCharts/MapSparkline.tsx index 7fa2a61dbfb..72977920cdb 100644 --- a/packages/@ourworldindata/grapher/src/mapCharts/MapSparkline.tsx +++ b/packages/@ourworldindata/grapher/src/mapCharts/MapSparkline.tsx @@ -118,7 +118,7 @@ export class MapSparkline extends React.Component<{ lineStrokeWidth: 2, entityYearHighlight: { entityName: this.manager.entityName, - year: this.manager.datum?.time, + year: this.manager.datum?.originalTime, }, yAxisConfig: { hideAxis: true, diff --git a/packages/@ourworldindata/grapher/src/mapCharts/MapTooltip.tsx b/packages/@ourworldindata/grapher/src/mapCharts/MapTooltip.tsx index e58b90c5484..65775d51db1 100644 --- a/packages/@ourworldindata/grapher/src/mapCharts/MapTooltip.tsx +++ b/packages/@ourworldindata/grapher/src/mapCharts/MapTooltip.tsx @@ -119,8 +119,8 @@ export class MapTooltip : targetTime?.toString() const displayDatumTime = timeColumn && datum - ? timeColumn.formatValue(datum?.time) - : (datum?.time.toString() ?? "") + ? timeColumn.formatValue(datum?.originalTime) + : (datum?.originalTime.toString() ?? "") const valueColor: string | undefined = darkenColorForHighContrastText( lineColorScale?.getColor(datum?.value) ?? "#333" ) @@ -143,7 +143,7 @@ export class MapTooltip const yColumn = this.mapTable.get(this.mapColumnSlug) const targetNotice = - datum && datum.time !== targetTime ? displayTime : undefined + datum && datum.originalTime !== targetTime ? displayTime : undefined const toleranceNotice = targetNotice ? { icon: TooltipFooterIcon.notice, diff --git a/packages/@ourworldindata/grapher/src/noDataModal/NoDataModal.tsx b/packages/@ourworldindata/grapher/src/noDataModal/NoDataModal.tsx index 8502da6e0fe..a63ea9999b1 100644 --- a/packages/@ourworldindata/grapher/src/noDataModal/NoDataModal.tsx +++ b/packages/@ourworldindata/grapher/src/noDataModal/NoDataModal.tsx @@ -30,6 +30,7 @@ export interface NoDataModalManager { export class NoDataModal extends React.Component<{ bounds?: Bounds message?: string + helpText?: string manager: NoDataModalManager }> { @computed private get bounds(): Bounds { @@ -55,11 +56,12 @@ export class NoDataModal extends React.Component<{ isStatic, } = this.manager - const helpText = canAddEntities + const defaultHelpText = canAddEntities ? `Try adding ${entityTypePlural} to display data.` : canChangeEntity ? `Try choosing ${a(entityType)} to display data.` : undefined + const helpText = this.props.helpText ?? defaultHelpText const center = bounds.centerPos const padding = 0.75 * this.fontSize diff --git a/packages/@ourworldindata/grapher/src/scatterCharts/NoDataSection.tsx b/packages/@ourworldindata/grapher/src/scatterCharts/NoDataSection.tsx index 6ccac9dd108..bf9dfb192f3 100644 --- a/packages/@ourworldindata/grapher/src/scatterCharts/NoDataSection.tsx +++ b/packages/@ourworldindata/grapher/src/scatterCharts/NoDataSection.tsx @@ -6,19 +6,19 @@ import { } from "../core/GrapherConstants" export function NoDataSection({ - entityNames, + seriesNames, bounds, baseFontSize = 16, }: { - entityNames: string[] + seriesNames: string[] bounds: Bounds baseFontSize?: number }): React.ReactElement { { - const displayedEntities = entityNames.slice(0, 5) - const numRemainingEntities = Math.max( + const displayedNames = seriesNames.slice(0, 5) + const remaining = Math.max( 0, - entityNames.length - displayedEntities.length + seriesNames.length - displayedNames.length ) return ( @@ -40,7 +40,7 @@ export function NoDataSection({ No data
    - {displayedEntities.map((entityName) => ( + {displayedNames.map((entityName) => (
  • ))}
- {numRemainingEntities > 0 && ( -
- &{" "} - {numRemainingEntities === 1 - ? "one" - : numRemainingEntities}{" "} - more -
+ {remaining > 0 && ( +
& {remaining === 1 ? "one" : remaining} more
)} ) diff --git a/packages/@ourworldindata/grapher/src/scatterCharts/ScatterPlotChart.tsx b/packages/@ourworldindata/grapher/src/scatterCharts/ScatterPlotChart.tsx index e4688cad8ee..5906adeb482 100644 --- a/packages/@ourworldindata/grapher/src/scatterCharts/ScatterPlotChart.tsx +++ b/packages/@ourworldindata/grapher/src/scatterCharts/ScatterPlotChart.tsx @@ -871,7 +871,7 @@ export class ScatterPlotChart {!this.manager.isStatic && separatorLine(noDataSectionBounds.top)} @@ -935,7 +935,9 @@ export class ScatterPlotChart timeLabel = timeRange + (isRelativeMode ? " (avg. annual change)" : "") - const columns = [xColumn, yColumn, sizeColumn] + const columns = [xColumn, yColumn, sizeColumn].filter( + (column) => !column.isMissing + ) const allRoundedToSigFigs = columns.every( (column) => column.roundsToSignificantFigures ) diff --git a/packages/@ourworldindata/grapher/src/schema/grapher-schema.006.yaml b/packages/@ourworldindata/grapher/src/schema/grapher-schema.006.yaml index 67ae88895c5..72aeb7cbac9 100644 --- a/packages/@ourworldindata/grapher/src/schema/grapher-schema.006.yaml +++ b/packages/@ourworldindata/grapher/src/schema/grapher-schema.006.yaml @@ -104,6 +104,15 @@ properties: items: type: - string + focusedSeriesNames: + type: array + description: | + The initially focused chart elements (e.g. line or bar). + Is either a list of entity or variable names. + Only works to line and slope charts for now. + items: + type: + - string baseColorScheme: type: string description: | diff --git a/packages/@ourworldindata/grapher/src/selection/InteractionArray.ts b/packages/@ourworldindata/grapher/src/selection/InteractionArray.ts new file mode 100644 index 00000000000..9ae3c8f3746 --- /dev/null +++ b/packages/@ourworldindata/grapher/src/selection/InteractionArray.ts @@ -0,0 +1,48 @@ +import { EntityName } from "@ourworldindata/types" +import { action, computed, observable } from "mobx" + +export class InteractionArray { + constructor(focusedEntityNames: EntityName[] = []) { + this.focusedEntityNames = focusedEntityNames.slice() + } + + @observable focusedEntityNames: EntityName[] + + @computed get focusedEntityNameSet(): Set { + return new Set(this.focusedEntityNames) + } + + @action.bound focusEntity(entityName: EntityName): this { + if (!this.focusedEntityNameSet.has(entityName)) + this.focusedEntityNames.push(entityName) + return this + } + + @action.bound unfocusEntity(entityName: EntityName): this { + this.focusedEntityNames = this.focusedEntityNames.filter( + (name) => name !== entityName + ) + return this + } + + @action.bound toggleFocus(entityName: EntityName): this { + return this.focusedEntityNameSet.has(entityName) + ? this.unfocusEntity(entityName) + : this.focusEntity(entityName) + } + + @action.bound clear(): void { + this.focusedEntityNames = [] + } + + @action.bound addToFocusedEntities(entityNames: EntityName[]): this { + this.focusedEntityNames = this.focusedEntityNames.concat(entityNames) + return this + } + + // Clears and sets focused entities + @action.bound setFocusedEntities(entityNames: EntityName[]): this { + this.clear() + return this.addToFocusedEntities(entityNames) + } +} diff --git a/packages/@ourworldindata/grapher/src/slopeCharts/SlopeChart.test.ts b/packages/@ourworldindata/grapher/src/slopeCharts/SlopeChart.test.ts index 7d7e4ae8d80..8fdabb7dd4e 100755 --- a/packages/@ourworldindata/grapher/src/slopeCharts/SlopeChart.test.ts +++ b/packages/@ourworldindata/grapher/src/slopeCharts/SlopeChart.test.ts @@ -1,19 +1,29 @@ #! /usr/bin/env jest -import { SlopeChart } from "./SlopeChart" +import { SlopeChart, SlopeChartManager } from "./SlopeChart" import { + ErrorValueTypes, + OwidTable, SampleColumnSlugs, + SynthesizeFruitTableWithNonPositives, SynthesizeFruitTableWithStringValues, SynthesizeGDPTable, } from "@ourworldindata/core-table" import { ChartManager } from "../chart/ChartManager" -import { DEFAULT_SLOPE_CHART_COLOR } from "./SlopeChartConstants" -import { isNumber, OwidTableSlugs } from "@ourworldindata/utils" +import { + ColumnTypeNames, + FacetStrategy, + isNumber, + ScaleType, + SeriesStrategy, +} from "@ourworldindata/utils" +import { SelectionArray } from "../selection/SelectionArray" const table = SynthesizeGDPTable({ timeRange: [2000, 2010] }) const manager: ChartManager = { table, yColumnSlug: SampleColumnSlugs.Population, + selection: table.availableEntityNames, } it("can create a new slope chart", () => { @@ -21,18 +31,32 @@ it("can create a new slope chart", () => { expect(chart.series.length).toEqual(2) }) -it("slope charts can have different colors", () => { +it("filters non-numeric values", () => { + const table = SynthesizeFruitTableWithStringValues( + { + entityCount: 2, + timeRange: [2000, 2002], + }, + 1, + 1 + ) const manager: ChartManager = { table, - yColumnSlug: SampleColumnSlugs.Population, - colorColumnSlug: OwidTableSlugs.entityName, + yColumnSlugs: [SampleColumnSlugs.Fruit], + selection: table.availableEntityNames, } const chart = new SlopeChart({ manager }) - expect(chart.series[0].color).not.toEqual(DEFAULT_SLOPE_CHART_COLOR) + expect(chart.series.length).toEqual(1) + expect( + chart.series.every( + (series) => + isNumber(series.start.value) && isNumber(series.end.value) + ) + ).toBeTruthy() }) -it("filters non-numeric values", () => { - const table = SynthesizeFruitTableWithStringValues( +it("can filter points with negative values when using a log scale", () => { + const table = SynthesizeFruitTableWithNonPositives( { entityCount: 2, timeRange: [2000, 2002], @@ -40,18 +64,200 @@ it("filters non-numeric values", () => { 1, 1 ) + const manager: ChartManager = { table, yColumnSlugs: [SampleColumnSlugs.Fruit], selection: table.availableEntityNames, } const chart = new SlopeChart({ manager }) - expect(chart.series.length).toEqual(1) - expect( - chart.series.every((series) => - series.values.every( - (value) => isNumber(value.x) && isNumber(value.y) - ) + // expect(chart.series.length).toEqual(2) + expect(chart.allYValues.length).toEqual(4) + + const logScaleManager = { + ...manager, + yAxisConfig: { + scaleType: ScaleType.log, + }, + } + const logChart = new SlopeChart({ manager: logScaleManager }) + expect(logChart.yAxis.domain[0]).toBeGreaterThan(0) + // expect(logChart.series.length).toEqual(2) + expect(logChart.allYValues.length).toEqual(2) +}) + +describe("series naming in multi-column mode", () => { + const table = SynthesizeGDPTable() + + it("only displays column name if only one entity is selected and multi entity selection is disabled", () => { + const manager = { + table, + canSelectMultipleEntities: false, + selection: [table.availableEntityNames[0]], + } + const chart = new SlopeChart({ manager }) + expect(chart.series[0].seriesName).not.toContain(" – ") + }) + + it("combines entity and column name if only one entity is selected and multi entity selection is enabled", () => { + const manager = { + table, + canSelectMultipleEntities: true, + selection: [table.availableEntityNames[0]], + } + const chart = new SlopeChart({ manager }) + expect(chart.series[0].seriesName).toContain(" – ") + }) + + it("combines entity and column name if multiple entities are selected and multi entity selection is disabled", () => { + const selection = new SelectionArray( + table.availableEntityNames, + table.availableEntities ) - ).toBeTruthy() + const manager = { + table, + canSelectMultipleEntities: false, + selection, + } + const chart = new SlopeChart({ manager }) + expect(chart.series[0].seriesName).toContain(" – ") + }) +}) + +describe("colors", () => { + const table = new OwidTable({ + entityName: ["usa", "canada", "usa", "canada"], + year: [2000, 2000, 2001, 2001], + gdp: [100, 200, 200, 300], + entityColor: ["blue", "red", "blue", "red"], + }) + const selection = ["usa", "canada"] + it("can add custom colors", () => { + const manager = { + yColumnSlugs: ["gdp"], + table, + selection, + } + const chart = new SlopeChart({ manager }) + expect(chart.series.map((series) => series.color)).toEqual([ + "blue", + "red", + ]) + }) + + it("uses column color selections when series strategy is column", () => { + const table = new OwidTable( + { + entityName: ["usa", "usa"], + year: [2000, 2001], + gdp: [100, 200], + entityColor: ["blue", "blue"], + }, + [{ slug: "gdp", color: "green", type: ColumnTypeNames.Numeric }] + ) + + const manager: ChartManager = { + yColumnSlugs: ["gdp"], + table: table, + selection, + seriesStrategy: SeriesStrategy.column, + } + const chart = new SlopeChart({ manager }) + const series = chart.series + + expect(series).toHaveLength(1) + expect(series[0].color).toEqual("green") + }) + + it("can assign colors to selected entities and preserve those colors when selection changes when using a color map", () => { + const selection = new SelectionArray(["usa", "canada"]) + const manager: ChartManager = { + yColumnSlugs: ["gdp"], + table: table.dropColumns(["entityColor"]), + selection, + seriesColorMap: new Map(), + } + const chart = new SlopeChart({ manager }) + const series = chart.series + expect(series).toHaveLength(2) + + selection.deselectEntity("usa") + + const newSeries = chart.series + expect(newSeries).toHaveLength(1) + expect(newSeries[0].color).toEqual(series[1].color) + }) + + it("uses variable colors when only one entity selected (even if multiple can be selected with controls)", () => { + const table = new OwidTable( + { + entityName: ["usa", "usa", "canada"], + year: [2000, 2001, 2000], + gdp: [100, 200, 100], + pop: [100, 200, 100], + }, + [ + { slug: "gdp", color: "green", type: ColumnTypeNames.Numeric }, + { slug: "pop", color: "orange", type: ColumnTypeNames.Numeric }, + ] + ) + + const manager: SlopeChartManager = { + yColumnSlugs: ["gdp", "pop"], + table: table, + selection: ["usa"], + seriesStrategy: SeriesStrategy.column, + facetStrategy: FacetStrategy.entity, + canSelectMultipleEntities: true, + } + const chart = new SlopeChart({ manager }) + const series = chart.series + + expect(series).toHaveLength(2) + expect(series[0].color).toEqual("green") + expect(series[1].color).toEqual("orange") + }) + + it("doesn't use variable colors if 2 variables have single entities which are different", () => { + const table = new OwidTable( + { + entityName: ["usa", "usa", "canada", "canada"], + year: [2000, 2001, 2000, 2001], + gdp: [ + 100, + 200, + ErrorValueTypes.MissingValuePlaceholder, + ErrorValueTypes.MissingValuePlaceholder, + ], + pop: [ + ErrorValueTypes.MissingValuePlaceholder, + ErrorValueTypes.MissingValuePlaceholder, + 100, + 200, + ], + }, + [ + { slug: "gdp", color: "green", type: ColumnTypeNames.Numeric }, + { slug: "pop", color: "orange", type: ColumnTypeNames.Numeric }, + ] + ) + + const selection = new SelectionArray( + ["usa", "canada"], + [{ entityName: "usa" }, { entityName: "canada" }] + ) + const manager: SlopeChartManager = { + yColumnSlugs: ["gdp", "pop"], + table: table, + selection, + seriesStrategy: SeriesStrategy.column, + canSelectMultipleEntities: true, + } + const chart = new SlopeChart({ manager }) + const series = chart.series + + expect(series).toHaveLength(2) + expect(series[0].color).not.toEqual("green") + expect(series[1].color).not.toEqual("orange") + }) }) diff --git a/packages/@ourworldindata/grapher/src/slopeCharts/SlopeChart.tsx b/packages/@ourworldindata/grapher/src/slopeCharts/SlopeChart.tsx index 8b865b803d6..5c29720ccda 100644 --- a/packages/@ourworldindata/grapher/src/slopeCharts/SlopeChart.tsx +++ b/packages/@ourworldindata/grapher/src/slopeCharts/SlopeChart.tsx @@ -2,81 +2,100 @@ import React from "react" import { Bounds, DEFAULT_BOUNDS, - intersection, - without, - uniq, - isEmpty, - last, - sortBy, - max, - getRelativeMouse, domainExtent, - minBy, exposeInstanceOnWindow, PointVector, clamp, - HorizontalAlign, - difference, makeIdForHumanConsumption, + guid, + excludeUndefined, + partition, + getRelativeMouse, + minBy, + dyFromAlign, + uniq, + sortBy, } from "@ourworldindata/utils" -import { TextWrap } from "@ourworldindata/components" import { observable, computed, action } from "mobx" import { observer } from "mobx-react" import { NoDataModal } from "../noDataModal/NoDataModal" -import { - VerticalColorLegend, - VerticalColorLegendManager, -} from "../verticalColorLegend/VerticalColorLegend" -import { ColorScale, ColorScaleManager } from "../color/ColorScale" import { BASE_FONT_SIZE, + GRAPHER_BACKGROUND_DEFAULT, GRAPHER_DARK_TEXT, - GRAPHER_FONT_SCALE_9_6, - GRAPHER_FONT_SCALE_10_5, } from "../core/GrapherConstants" import { ScaleType, - EntitySelectionMode, - Color, SeriesName, ColorSchemeName, + ColumnSlug, + MissingDataStrategy, + Time, + SeriesStrategy, + EntityName, + VerticalAlign, + FacetStrategy, } from "@ourworldindata/types" -import { ChartInterface } from "../chart/ChartInterface" +import { ChartInterface, FocusState, HoverState } from "../chart/ChartInterface" import { ChartManager } from "../chart/ChartManager" import { scaleLinear, ScaleLinear } from "d3-scale" import { select } from "d3-selection" import { - DEFAULT_SLOPE_CHART_COLOR, - LabelledSlopesProps, + PlacedSlopeChartSeries, + RawSlopeChartSeries, + RenderSlopeChartSeries, SlopeChartSeries, - SlopeChartValue, - SlopeEntryProps, } from "./SlopeChartConstants" -import { OwidTable } from "@ourworldindata/core-table" +import { CoreColumn, OwidTable } from "@ourworldindata/core-table" import { + autoDetectSeriesStrategy, autoDetectYColumnSlugs, + byHoverThenFocusState, + getDefaultFailMessage, + getShortNameForEntity, makeSelectionArray, - isElementInteractive, } from "../chart/ChartUtils" -import { AxisConfig, AxisManager } from "../axis/AxisConfig" +import { AxisConfig } from "../axis/AxisConfig" import { VerticalAxis } from "../axis/Axis" import { VerticalAxisComponent } from "../axis/AxisViews" -import { - HorizontalCategoricalColorLegend, - HorizontalColorLegendManager, -} from "../horizontalColorLegend/HorizontalColorLegends" -import { CategoricalBin, ColorScaleBin } from "../color/ColorScaleBin" import { NoDataSection } from "../scatterCharts/NoDataSection" +import { CategoricalColorAssigner } from "../color/CategoricalColorAssigner" +import { ColorScheme } from "../color/ColorScheme" +import { ColorSchemes } from "../color/ColorSchemes" +import { LineLabelSeries, LineLegend } from "../lineLegend/LineLegend" +import { + makeTooltipRoundingNotice, + makeTooltipToleranceNotice, + Tooltip, + TooltipState, + TooltipValueRange, +} from "../tooltip/Tooltip" +import { TooltipFooterIcon } from "../tooltip/TooltipProps" +import { + AnnotationsMap, + getAnnotationsForSeries, + getAnnotationsMap, + getColorKey, + getSeriesName, +} from "../lineCharts/LineChartHelpers" +import { HorizontalColorLegendManager } from "../horizontalColorLegend/HorizontalColorLegends" +import { CategoricalBin } from "../color/ColorScaleBin" +import { InteractionArray } from "../selection/InteractionArray" + +type SVGMouseOrTouchEvent = + | React.MouseEvent + | React.TouchEvent export interface SlopeChartManager extends ChartManager { - isModalOpen?: boolean + canSelectMultipleEntities?: boolean // used to pick an appropriate series name + hasTimeline?: boolean // used to filter the table for the entity selector + hideNoDataSection?: boolean } -const LABEL_SLOPE_PADDING = 8 -const LABEL_LABEL_PADDING = 2 +const TOP_PADDING = 6 // leave room for overflowing dots +const BOTTOM_PADDING = 20 // leave room for the x-axis -const TOP_PADDING = 6 -const BOTTOM_PADDING = 20 +const LINE_LEGEND_PADDING = 4 @observer export class SlopeChart @@ -84,718 +103,411 @@ export class SlopeChart bounds?: Bounds manager: SlopeChartManager }> - implements - ChartInterface, - VerticalColorLegendManager, - HorizontalColorLegendManager, - ColorScaleManager + implements ChartInterface { - // currently hovered individual series key - @observable hoverKey?: string - // currently hovered legend color - @observable hoverColor?: string + slopeAreaRef: React.RefObject = React.createRef() + defaultBaseColorScheme = ColorSchemeName.OwidDistinctLines - private hasInteractedWithChart = false + @observable hoveredSeriesName?: string + @observable tooltipState = new TooltipState<{ + series: SlopeChartSeries + }>({ fade: "immediate" }) transformTable(table: OwidTable) { - if (!table.has(this.yColumnSlug)) return table + table = table.filterByEntityNames( + this.selectionArray.selectedEntityNames + ) // TODO: remove this filter once we don't have mixed type columns in datasets - table = table.replaceNonNumericCellsWithErrorValues([this.yColumnSlug]) + table = table.replaceNonNumericCellsWithErrorValues(this.yColumnSlugs) - return table - .dropRowsWithErrorValuesForColumn(this.yColumnSlug) - .interpolateColumnWithTolerance(this.yColumnSlug) - } + if (this.isLogScale) + table = table.replaceNonPositiveCellsForLogScale(this.yColumnSlugs) - @computed get manager() { - return this.props.manager - } + this.yColumnSlugs.forEach((slug) => { + table = table.interpolateColumnWithTolerance(slug) + }) - @computed.struct get bounds() { - return this.props.bounds ?? DEFAULT_BOUNDS + return table } - @computed get isStatic(): boolean { - return this.manager.isStatic ?? false - } + transformTableForSelection(table: OwidTable): OwidTable { + table = table.replaceNonNumericCellsWithErrorValues(this.yColumnSlugs) - @computed get fontSize() { - return this.manager.fontSize ?? BASE_FONT_SIZE - } + this.yColumnSlugs.forEach((slug) => { + table = table.interpolateColumnWithTolerance(slug) + }) - @computed private get isPortrait(): boolean { - return !!(this.manager.isNarrow || this.manager.isStaticAndSmall) - } + // if time selection is disabled, then filter all entities that + // don't have data for the current time period + if (!this.manager.hasTimeline && this.startTime !== this.endTime) { + table = table + .filterByTargetTimes([this.startTime, this.endTime]) + .dropEntitiesThatHaveSomeMissingOrErrorValueInAllColumns( + this.yColumnSlugs + ) + } - @computed private get showHorizontalLegend(): boolean { - return !!(this.manager.isSemiNarrow || this.manager.isStaticAndSmall) + // if entities with partial data are not plotted, + // make sure they don't show up in the entity selector + if (this.missingDataStrategy === MissingDataStrategy.hide) { + table = table.dropEntitiesThatHaveNoDataInSomeColumn( + this.yColumnSlugs + ) + } + + return table } - // used by the component - @computed get legendItems() { - return this.colorScale.legendBins - .filter((bin) => this.colorsInUse.includes(bin.color)) - .map((bin) => { - return { - key: bin.label ?? "", - label: bin.label ?? "", - color: bin.color, - } - }) - } - - // used by the component - @computed get categoricalLegendData(): CategoricalBin[] { - return this.legendItems.map( - (legendItem, index) => - new CategoricalBin({ - ...legendItem, - index, - value: legendItem.label, - }) + @computed get transformedTableFromGrapher(): OwidTable { + return ( + this.manager.transformedTable ?? + this.transformTable(this.inputTable) ) } - @action.bound onSlopeMouseOver(slopeProps: SlopeEntryProps) { - this.hoverKey = slopeProps.seriesName + @computed get transformedTable(): OwidTable { + let table = this.transformedTableFromGrapher + // The % growth transform cannot be applied in transformTable() because it will filter out + // any rows before startTime and change the timeline bounds. + const { isRelativeMode, startTime } = this.manager + if (isRelativeMode && startTime !== undefined) { + table = table.toTotalGrowthForEachColumnComparedToStartTime( + startTime, + this.yColumnSlugs ?? [] + ) + } + return table } - @action.bound onSlopeMouseLeave() { - this.hoverKey = undefined + @computed private get manager(): SlopeChartManager { + return this.props.manager } - @action.bound onSlopeClick() { - const { hoverKey, isEntitySelectionEnabled } = this - if (!isEntitySelectionEnabled || hoverKey === undefined) { - return - } - this.hasInteractedWithChart = true - this.selectionArray.toggleSelection(hoverKey) + @computed get inputTable(): OwidTable { + return this.manager.table } - // Both legend managers accept a `onLegendMouseOver` property, but define different signatures. - // The component expects a string, - // the component expects a ColorScaleBin. - @action.bound onLegendMouseOver(binOrColor: string | ColorScaleBin) { - this.hoverColor = - typeof binOrColor === "string" ? binOrColor : binOrColor.color + @computed private get bounds(): Bounds { + return this.props.bounds ?? DEFAULT_BOUNDS } - @action.bound onLegendMouseLeave() { - this.hoverColor = undefined + private sidebarMargin = 10 + @computed private get innerBounds(): Bounds { + return this.bounds.padRight(this.sidebarWidth + this.sidebarMargin) } - @computed private get selectionArray() { - return makeSelectionArray(this.manager.selection) + @computed get fontSize() { + return this.manager.fontSize ?? BASE_FONT_SIZE } - @computed private get selectedEntityNames() { - return this.selectionArray.selectedEntityNames + @computed private get isLogScale(): boolean { + return this.yScaleType === ScaleType.log } - @computed get isEntitySelectionEnabled(): boolean { - const { manager } = this - return !!( - manager.addCountryMode !== EntitySelectionMode.Disabled && - manager.addCountryMode - ) + @computed private get missingDataStrategy(): MissingDataStrategy { + return this.manager.missingDataStrategy || MissingDataStrategy.auto } - // When the color legend is clicked, toggle selection fo all associated keys - @action.bound onLegendClick() { - const { hoverColor, isEntitySelectionEnabled } = this - if (!isEntitySelectionEnabled || hoverColor === undefined) return - - this.hasInteractedWithChart = true - - const seriesNamesToToggle = this.series - .filter((g) => g.color === hoverColor) - .map((g) => g.seriesName) - const areAllSeriesActive = - intersection(seriesNamesToToggle, this.selectedEntityNames) - .length === seriesNamesToToggle.length - if (areAllSeriesActive) - this.selectionArray.setSelectedEntities( - without(this.selectedEntityNames, ...seriesNamesToToggle) - ) - else - this.selectionArray.setSelectedEntities( - this.selectedEntityNames.concat(seriesNamesToToggle) - ) + @computed private get selectionArray() { + return makeSelectionArray(this.manager.selection) } - // Colors on the legend for which every matching group is focused - @computed get focusColors() { - const { colorsInUse } = this - return colorsInUse.filter((color) => { - const matchingSeriesNames = this.series - .filter((g) => g.color === color) - .map((g) => g.seriesName) - return ( - intersection(matchingSeriesNames, this.selectedEntityNames) - .length === matchingSeriesNames.length - ) - }) + @computed get interactionArray(): InteractionArray { + return this.manager.interactionArray ?? new InteractionArray() } - @computed get focusKeys() { - return this.selectedEntityNames + @computed private get focusedSeriesNameSet(): Set { + return this.interactionArray.focusedEntityNameSet } - // All currently hovered group keys, combining the legend and the main UI - @computed.struct get hoverKeys() { - const { hoverColor, hoverKey } = this - - const hoverKeys = - hoverColor === undefined - ? [] - : uniq( - this.series - .filter((g) => g.color === hoverColor) - .map((g) => g.seriesName) - ) - - if (hoverKey !== undefined) hoverKeys.push(hoverKey) - - return hoverKeys + @computed private get formatColumn() { + return this.yColumns[0] } - // Colors currently on the chart and not greyed out - @computed get activeColors() { - const { hoverKeys, focusKeys } = this - const activeKeys = hoverKeys.concat(focusKeys) - - if (activeKeys.length === 0) - // No hover or focus means they're all active by default - return uniq(this.series.map((g) => g.color)) - - return uniq( - this.series - .filter((g) => activeKeys.indexOf(g.seriesName) !== -1) - .map((g) => g.color) - ) + @computed private get lineStrokeWidth(): number { + return this.manager.isStaticAndSmall ? 3 : 1.5 } - // Only show colors on legend that are actually in use - @computed private get colorsInUse() { - return uniq(this.series.map((series) => series.color)) + @computed private get backgroundColor(): string { + return this.manager.backgroundColor ?? GRAPHER_BACKGROUND_DEFAULT } - @computed get legendAlign(): HorizontalAlign { - return HorizontalAlign.left + @computed private get isHoverModeActive(): boolean { + return this.hoveredSeriesNameSet.size > 0 } - @computed get verticalColorLegend(): VerticalColorLegend { - return new VerticalColorLegend({ manager: this }) + @computed private get isFocusModeActive(): boolean { + return this.focusedSeriesNameSet.size > 0 } - @computed get horizontalColorLegend(): HorizontalCategoricalColorLegend { - return new HorizontalCategoricalColorLegend({ manager: this }) + @computed private get yColumns(): CoreColumn[] { + return this.yColumnSlugs.map((slug) => this.transformedTable.get(slug)) } - @computed get legendHeight(): number { - return this.showHorizontalLegend - ? this.horizontalColorLegend.height - : this.verticalColorLegend.height + @computed protected get yColumnSlugs(): ColumnSlug[] { + return autoDetectYColumnSlugs(this.manager) } - @computed get legendWidth(): number { - return this.showHorizontalLegend - ? this.bounds.width - : this.verticalColorLegend.width + @computed private get colorScheme(): ColorScheme { + return ( + (this.manager.baseColorScheme + ? ColorSchemes.get(this.manager.baseColorScheme) + : null) ?? ColorSchemes.get(this.defaultBaseColorScheme) + ) } - @computed get maxLegendWidth(): number { - return this.showHorizontalLegend - ? this.bounds.width - : this.bounds.width * 0.5 + @computed private get startTime(): Time { + return this.transformedTable.minTime ?? 0 } - @computed private get sidebarWidth(): number { - // the min width is set to prevent the "No data" title from line breaking - return clamp(this.legendWidth, 51, this.maxLegendWidth) - } - - // correction is to account for the space taken by the legend - @computed private get innerBounds() { - const { sidebarWidth, showLegend, legendHeight } = this - let bounds = this.bounds - if (showLegend) { - bounds = this.showHorizontalLegend - ? bounds.padTop(legendHeight + 8) - : bounds.padRight(sidebarWidth + 16) - } - return bounds + @computed private get endTime(): Time { + return this.transformedTable.maxTime ?? 0 } - // verify the validity of data used to show legend - // this is for backwards compatibility with charts that were added without legend - // eg: https://ourworldindata.org/grapher/mortality-rate-improvement-by-cohort - @computed private get showLegend() { - const { colorsInUse } = this - const { legendBins } = this.colorScale - return legendBins.some((bin) => colorsInUse.includes(bin.color)) + @computed private get startX(): number { + return this.xScale(this.startTime) } - @computed - private get selectedEntitiesWithoutData(): string[] { - return difference( - this.selectedEntityNames, - this.series.map((s) => s.seriesName) - ) + @computed private get endX(): number { + return this.xScale(this.endTime) } - @computed private get noDataSection(): React.ReactElement { - const bounds = new Bounds( - this.legendX, - this.legendY + this.legendHeight + 12, - this.sidebarWidth, - this.bounds.height - this.legendHeight - 12 - ) - return ( - - ) + @computed get seriesStrategy(): SeriesStrategy { + return autoDetectSeriesStrategy(this.manager, true) } - render() { - if (this.failMessage) - return ( - - ) + @computed get availableFacetStrategies(): FacetStrategy[] { + const strategies: FacetStrategy[] = [FacetStrategy.none] - const { manager } = this.props - const { - series, - focusKeys, - hoverKeys, - innerBounds, - showLegend, - showHorizontalLegend, - selectedEntitiesWithoutData, - } = this + if (this.selectionArray.numSelectedEntities > 1) + strategies.push(FacetStrategy.entity) - const legend = showHorizontalLegend ? ( - - ) : ( - - ) + if (this.yColumns.length > 1) strategies.push(FacetStrategy.metric) - return ( - - - {showLegend && legend} - {/* only show the "No data" section if there is space */} - {showLegend && - !showHorizontalLegend && - selectedEntitiesWithoutData.length > 0 && - this.noDataSection} - - ) + return strategies } - @computed get categoryLegendY(): number { - return this.bounds.top + @computed private get categoricalColorAssigner(): CategoricalColorAssigner { + return new CategoricalColorAssigner({ + colorScheme: this.colorScheme, + invertColorScheme: this.manager.invertColorScheme, + colorMap: + this.seriesStrategy === SeriesStrategy.entity + ? this.inputTable.entityNameColorIndex + : this.inputTable.columnDisplayNameToColorMap, + autoColorMapCache: this.manager.seriesColorMap, + }) } - @computed get legendY() { - return this.bounds.top + @computed private get annotationsMap(): AnnotationsMap | undefined { + return getAnnotationsMap(this.inputTable, this.yColumnSlugs[0]) } - @computed get legendX(): number { - return this.showHorizontalLegend - ? this.bounds.left - : this.bounds.right - this.sidebarWidth - } + private constructSingleSeries( + entityName: EntityName, + column: CoreColumn + ): RawSlopeChartSeries { + const { startTime, endTime, seriesStrategy } = this + const { canSelectMultipleEntities = false } = this.manager - @computed get failMessage() { - if (this.yColumn.isMissing) return "Missing Y column" - else if (isEmpty(this.series)) return "No matching data" - return "" - } + const { availableEntityNames } = this.transformedTable + const displayEntityName = + getShortNameForEntity(entityName) ?? entityName + const columnName = column.nonEmptyDisplayName + const seriesName = getSeriesName({ + entityName: displayEntityName, + columnName, + seriesStrategy, + availableEntityNames, + canSelectMultipleEntities, + }) - colorScale = this.props.manager.colorScaleOverride ?? new ColorScale(this) + const owidRowByTime = column.owidRowByEntityNameAndTime.get(entityName) + const start = owidRowByTime?.get(startTime) + const end = owidRowByTime?.get(endTime) - @computed get colorScaleConfig() { - return this.manager.colorScale - } + const colorKey = getColorKey({ + entityName, + columnName, + seriesStrategy, + availableEntityNames, + }) + const color = this.categoricalColorAssigner.assign(colorKey) - @computed get colorScaleColumn() { - return ( - // For faceted charts, we have to get the values of inputTable before it's filtered by - // the faceting logic. - this.manager.colorScaleColumnOverride ?? this.colorColumn + const annotation = getAnnotationsForSeries( + this.annotationsMap, + seriesName ) - } - - defaultBaseColorScheme = ColorSchemeName.continents - @computed private get yColumn() { - return this.transformedTable.get(this.yColumnSlug) - } - - @computed protected get yColumnSlug() { - return autoDetectYColumnSlugs(this.manager)[0] - } - - @computed private get colorColumn() { - // NB: This is tricky. Often it seems we use the Continent variable (123) for colors, but we only have 1 year for that variable, which - // would likely get filtered by any time filtering. So we need to jump up to the root table to get the color values we want. - // We should probably refactor this as part of a bigger color refactoring. - return this.inputTable.get(this.manager.colorColumnSlug) + return { + column, + seriesName, + entityName, + color, + start, + end, + annotation, + } } - @computed get transformedTable() { - return ( - this.manager.transformedTable ?? - this.transformTable(this.inputTable) - ) - } + private isSeriesValid( + series: RawSlopeChartSeries + ): series is SlopeChartSeries { + const { + start, + end, + column: { tolerance }, + } = series - @computed get inputTable() { - return this.manager.table - } + // if the start or end value is missing, we can't draw the slope + if (start?.value === undefined || end?.value === undefined) return false - // helper method to directly get the associated color value given an Entity - // dimension data saves color a level deeper. eg: { Afghanistan => { 2015: Asia|Color }} - // this returns that data in the form { Afghanistan => Asia } - @computed private get colorBySeriesName(): Map< - SeriesName, - Color | undefined - > { - const { colorScale, colorColumn } = this - if (colorColumn.isMissing) return new Map() + // sanity check (might happen if tolerance is enabled) + if (start.originalTime >= end.originalTime) return false - const colorByEntity = new Map() + const isToleranceAppliedToStartValue = + start.originalTime !== this.startTime + const isToleranceAppliedToEndValue = end.originalTime !== this.endTime - colorColumn.valueByEntityNameAndOriginalTime.forEach( - (timeToColorMap, seriesName) => { - const values = Array.from(timeToColorMap.values()) - const key = last(values) - colorByEntity.set(seriesName, colorScale.getColor(key)) - } - ) + // if tolerance has been applied to one of the values, then we require + // a minimal distance between the original times + if (isToleranceAppliedToStartValue || isToleranceAppliedToEndValue) { + return end.originalTime - start.originalTime >= tolerance + } - return colorByEntity + return true } - // click anywhere inside the Grapher frame to dismiss the current selection - @action.bound onGrapherClick(e: Event): void { - const target = e.target as HTMLElement - const isTargetInteractive = isElementInteractive(target) + // Usually we drop rows with missing data in the transformTable function. + // But if we did that for slope charts, we wouldn't know whether a slope + // has been dropped because it actually had no data or a sibling slope had + // no data. But we need that information for the "No data" section. That's + // why the filtering happens here, so that the noDataSeries can be populated + // correctly. + private shouldSeriesBePlotted( + series: RawSlopeChartSeries + ): series is SlopeChartSeries { + if (!this.isSeriesValid(series)) return false + + // when the missing data strategy is set to "hide", we might + // choose not to plot a valid series if ( - this.isEntitySelectionEnabled && - this.hasInteractedWithChart && - !this.hoverKey && - !this.hoverColor && - !this.manager.isModalOpen && - !isTargetInteractive + this.seriesStrategy === SeriesStrategy.column && + this.missingDataStrategy === MissingDataStrategy.hide ) { - this.selectionArray.clearSelection() + const allSeriesForEntity = this.rawSeriesByEntityName.get( + series.entityName + ) + return !!allSeriesForEntity?.every((series) => + this.isSeriesValid(series) + ) } - } - @computed private get grapherElement() { - return this.manager.base?.current + return true } - componentDidMount() { - if (this.grapherElement) { - // listening to "mousedown" instead of "click" fixes a bug - // where the current selection was incorrectly dismissed - // when the user drags the slider but releases the drag outside of the timeline - this.grapherElement.addEventListener( - "mousedown", - this.onGrapherClick + @computed private get rawSeries(): RawSlopeChartSeries[] { + return this.yColumns.flatMap((column) => + this.selectionArray.selectedEntityNames.map((entityName) => + this.constructSingleSeries(entityName, column) ) - } - exposeInstanceOnWindow(this) + ) } - componentWillUnmount(): void { - if (this.grapherElement) { - this.grapherElement.removeEventListener( - "mousedown", - this.onGrapherClick - ) - } + @computed private get rawSeriesByEntityName(): Map< + SeriesName, + RawSlopeChartSeries[] + > { + const map = new Map() + this.rawSeries.forEach((series) => { + const { entityName } = series + if (!map.has(entityName)) map.set(entityName, []) + map.get(entityName)!.push(series) + }) + return map } - @computed get series() { - const column = this.yColumn - if (!column) return [] - - const { colorBySeriesName } = this - const { minTime, maxTime } = column - - const table = this.inputTable - - return column.uniqEntityNames - .map((seriesName) => { - const values: SlopeChartValue[] = [] - - const yValues = - column.valueByEntityNameAndOriginalTime.get(seriesName)! || - [] - - yValues.forEach((value, time) => { - if (time !== minTime && time !== maxTime) return - - values.push({ - x: time, - y: value, - }) - }) - - // sort values by time - const sortedValues = sortBy(values, (v) => v.x) - - const color = - table.getColorForEntityName(seriesName) ?? - colorBySeriesName.get(seriesName) ?? - DEFAULT_SLOPE_CHART_COLOR - - return { - seriesName, - color, - values: sortedValues, - } as SlopeChartSeries - }) - .filter((series) => series.values.length >= 2) + @computed get series(): SlopeChartSeries[] { + return this.rawSeries.filter((series) => + this.shouldSeriesBePlotted(series) + ) } -} -@observer -class SlopeEntry extends React.Component { - line: SVGElement | null = null + @computed private get placedSeries(): PlacedSlopeChartSeries[] { + const { yAxis, startX, endX } = this - @computed get isInBackground() { - const { isLayerMode, isHovered, isFocused } = this.props + return this.series.map((series) => { + const startY = yAxis.place(series.start.value) + const endY = yAxis.place(series.end.value) - if (!isLayerMode) return false + const startPoint = new PointVector(startX, startY) + const endPoint = new PointVector(endX, endY) - return !(isHovered || isFocused) - } - - render() { - const { - x1, - y1, - x2, - y2, - color, - hasLeftLabel, - hasRightLabel, - leftValueLabel, - leftEntityLabel, - rightValueLabel, - rightEntityLabel, - leftEntityLabelBounds, - rightEntityLabelBounds, - isFocused, - isHovered, - isMultiHoverMode, - seriesName, - } = this.props - const { isInBackground } = this - - const lineColor = isInBackground ? "#e2e2e2" : color - const labelColor = isInBackground ? "#ccc" : GRAPHER_DARK_TEXT - const opacity = isHovered ? 1 : isFocused ? 0.7 : 0.5 - const lineStrokeWidth = - isHovered && !isMultiHoverMode ? 4 : isFocused ? 3 : 2 - - const showDots = isFocused || isHovered - const showValueLabels = isFocused || isHovered - const showLeftEntityLabel = isFocused || (isHovered && isMultiHoverMode) - - const sharedLabelProps = { - fill: labelColor, - style: { cursor: "default" }, - } - - return ( - - (this.line = el)} - x1={x1} - y1={y1} - x2={x2} - y2={y2} - stroke={lineColor} - strokeWidth={lineStrokeWidth} - opacity={opacity} - /> - {showDots && ( - <> - - - - )} - {/* value label on the left */} - {hasLeftLabel && - showValueLabels && - leftValueLabel.render( - x1 - LABEL_SLOPE_PADDING, - leftEntityLabelBounds.y, - { - textProps: { - ...sharedLabelProps, - textAnchor: "end", - }, - } - )} - {/* entity label on the left */} - {hasLeftLabel && - showLeftEntityLabel && - leftEntityLabel.render( - // -2px is a minor visual correction - leftEntityLabelBounds.x - 2, - leftEntityLabelBounds.y, - { - textProps: { - ...sharedLabelProps, - textAnchor: "end", - }, - } - )} - {/* value label on the right */} - {hasRightLabel && - showValueLabels && - rightValueLabel.render( - rightEntityLabelBounds.x + - rightEntityLabel.width + - LABEL_LABEL_PADDING, - rightEntityLabelBounds.y, - { - textProps: sharedLabelProps, - } - )} - {/* entity label on the right */} - {hasRightLabel && - rightEntityLabel.render( - rightEntityLabelBounds.x, - rightEntityLabelBounds.y, - { - textProps: { - ...sharedLabelProps, - fontWeight: - isFocused || isHovered ? "bold" : undefined, - }, - } - )} - - ) + return { ...series, startPoint, endPoint } + }) } -} - -@observer -class LabelledSlopes - extends React.Component - implements AxisManager -{ - base: React.RefObject = React.createRef() - @computed private get data() { - return this.props.seriesArr + private seriesIsFocused(series: SlopeChartSeries): boolean { + return this.focusedSeriesNameSet.has(series.seriesName) } - @computed private get yColumn() { - return this.props.yColumn + private seriesIsHovered(series: SlopeChartSeries): boolean { + return this.hoveredSeriesNameSet.has(series.seriesName) } - @computed private get manager() { - return this.props.manager + private seriesFocusState(series: SlopeChartSeries): FocusState { + if (!this.isFocusModeActive) return FocusState.off + return this.seriesIsFocused(series) + ? FocusState.active + : FocusState.background } - @computed private get bounds() { - return this.props.bounds + private seriesHoverState(series: SlopeChartSeries): HoverState { + if (!this.isHoverModeActive) return HoverState.off + return this.seriesIsHovered(series) + ? HoverState.active + : HoverState.background } - @computed get fontSize() { - return this.manager.fontSize ?? BASE_FONT_SIZE - } + @computed get renderSeries(): RenderSlopeChartSeries[] { + const series = this.placedSeries.map((series) => { + return { + ...series, + focus: this.seriesFocusState(series), + hover: this.seriesHoverState(series), + } + }) - @computed private get focusedSeriesNames() { - return intersection( - this.props.focusKeys || [], - this.data.map((g) => g.seriesName) - ) - } + if (this.isFocusModeActive || this.isHoverModeActive) { + return sortBy(series, byHoverThenFocusState) + } - @computed private get hoveredSeriesNames() { - return intersection( - this.props.hoverKeys || [], - this.data.map((g) => g.seriesName) - ) + return series } - // Layered mode occurs when any entity on the chart is hovered or focused - // Then, a special "foreground" set of entities is rendered over the background - @computed private get isLayerMode() { - return ( - this.hoveredSeriesNames.length > 0 || - this.focusedSeriesNames.length > 0 || - // if the user has selected entities that are not in the chart, - // we want to move all entities into the background - (this.props.focusKeys?.length > 0 && - this.focusedSeriesNames.length === 0) - ) + @computed + private get noDataSeries(): RawSlopeChartSeries[] { + return this.rawSeries.filter((series) => !this.isSeriesValid(series)) } - @computed private get isMultiHoverMode() { - return this.hoveredSeriesNames.length > 1 - } + @computed private get showNoDataSection(): boolean { + if (this.manager.hideNoDataSection) return false - @computed get isPortrait(): boolean { - return this.props.isPortrait - } + // nothing to show if there are no series with missing data + if (this.noDataSeries.length === 0) return false - @computed private get allValues() { - return this.props.seriesArr.flatMap((g) => g.values) - } + // the No Data section is HTML and won't show up in the SVG export + if (this.manager.isStatic) return false - @computed private get xDomainDefault(): [number, number] { - return domainExtent( - this.allValues.map((v) => v.x), - ScaleType.linear + // we usually don't show the no data section if columns are plotted + // (since columns don't appear in the entity selector there is no need + // to explain that a column is missing – it just adds noise). but if + // the missing data strategy is set to hide, then we do want to give + // feedback as to why a slope is currently not rendered + return ( + this.seriesStrategy === SeriesStrategy.entity || + this.missingDataStrategy === MissingDataStrategy.hide ) } @@ -803,28 +515,19 @@ class LabelledSlopes return new AxisConfig(this.manager.yAxisConfig, this) } - @computed get yAxis(): VerticalAxis { - const axis = this.yAxisConfig.toVerticalAxis() - axis.domain = this.yDomain - axis.range = this.yRange - axis.formatColumn = this.yColumn - axis.label = "" - return axis + @computed get allYValues(): number[] { + return this.series.flatMap((series) => [ + series.start.value, + series.end.value, + ]) } - @computed private get yScaleType() { - return this.yAxisConfig.scaleType || ScaleType.linear + @computed private get yScaleType(): ScaleType { + return this.yAxisConfig.scaleType ?? ScaleType.linear } @computed private get yDomainDefault(): [number, number] { - return domainExtent( - this.allValues.map((v) => v.y), - this.yScaleType || ScaleType.linear - ) - } - - @computed private get xDomain(): [number, number] { - return this.xDomainDefault + return domainExtent(this.allYValues, this.yScaleType) } @computed private get yDomain(): [number, number] { @@ -836,37 +539,24 @@ class LabelledSlopes ] } - @computed get yRange(): [number, number] { + @computed private get yRange(): [number, number] { return this.bounds .padTop(TOP_PADDING) .padBottom(BOTTOM_PADDING) .yRange() } - @computed get yAxisWidth(): number { - return this.yAxis.width + 5 // 5px account for the tick marks + @computed get yAxis(): VerticalAxis { + const axis = this.yAxisConfig.toVerticalAxis() + axis.domain = this.yDomain + axis.range = this.yRange + axis.formatColumn = this.yColumns[0] + axis.label = "" + return axis } - @computed get xRange(): [number, number] { - // take into account the space taken by the yAxis and slope labels - const bounds = this.bounds - .padLeft(this.yAxisWidth + 4) - .padLeft(this.maxLabelWidth) - .padRight(this.maxLabelWidth) - - // pick a reasonable width based on an ideal aspect ratio - const idealAspectRatio = 0.9 - const availableWidth = bounds.width - const idealWidth = idealAspectRatio * bounds.height - const slopeWidth = this.isPortrait - ? availableWidth - : clamp(idealWidth, 220, availableWidth) - - const leftRightPadding = (availableWidth - slopeWidth) / 2 - return bounds - .padLeft(leftRightPadding) - .padRight(leftRightPadding) - .xRange() + @computed private get yAxisWidth(): number { + return this.yAxis.width } @computed private get xScale(): ScaleLinear { @@ -874,446 +564,838 @@ class LabelledSlopes return scaleLinear().domain(xDomain).range(xRange) } - @computed get maxLabelWidth(): number { - const { slopeLabels } = this - const maxLabelWidths = slopeLabels.map((slope) => { - const entityLabelWidth = slope.leftEntityLabel.width - const maxValueLabelWidth = Math.max( - slope.leftValueLabel.width, - slope.rightValueLabel.width - ) - return ( - entityLabelWidth + - maxValueLabelWidth + - LABEL_SLOPE_PADDING + - LABEL_LABEL_PADDING + @computed private get xDomain(): [number, number] { + return [this.startTime, this.endTime] + } + + @computed private get sidebarWidth(): number { + return this.showNoDataSection + ? clamp(this.bounds.width * 0.125, 60, 140) + : 0 + } + + @computed get externalLegend(): HorizontalColorLegendManager | undefined { + if (!this.manager.showLegend) { + const categoricalLegendData = this.series.map( + (series, index) => + new CategoricalBin({ + index, + value: series.seriesName, + label: series.seriesName, + color: series.color, + }) ) - }) - return max(maxLabelWidths) ?? 0 + return { categoricalLegendData } + } + return undefined } - @computed get allowedLabelWidth() { - return this.bounds.width * 0.2 + @computed get maxLineLegendWidth(): number { + return 0.25 * this.innerBounds.width } - @computed private get slopeLabels() { - const { isPortrait, yColumn, allowedLabelWidth: maxWidth } = this + @computed get lineLegendFontSize(): number { + return LineLegend.fontSize({ fontSize: this.fontSize }) + } - return this.data.map((series) => { - const text = series.seriesName - const [v1, v2] = series.values - const fontSize = - (isPortrait - ? GRAPHER_FONT_SCALE_9_6 - : GRAPHER_FONT_SCALE_10_5) * this.fontSize - const leftValueStr = yColumn.formatValueShort(v1.y) - const rightValueStr = yColumn.formatValueShort(v2.y) + @computed get lineLegendYRange(): [number, number] { + const top = this.bounds.top - // value labels - const valueLabelProps = { - maxWidth: Infinity, // no line break - fontSize, - lineHeight: 1, - } - const leftValueLabel = new TextWrap({ - text: leftValueStr, - ...valueLabelProps, - }) - const rightValueLabel = new TextWrap({ - text: rightValueStr, - ...valueLabelProps, - }) - - // entity labels - const entityLabelProps = { - ...valueLabelProps, - maxWidth, - fontWeight: 700, - separators: [" ", "-"], - } - const leftEntityLabel = new TextWrap({ - text, - ...entityLabelProps, - }) - const rightEntityLabel = new TextWrap({ - text, - ...entityLabelProps, - }) + const bottom = + this.bounds.bottom - + // leave space for the x-axis labels + BOTTOM_PADDING + + // but allow for a little extra space + this.lineLegendFontSize / 2 + return [top, bottom] + } + + @computed private get lineLegendLeftHasConnectorLines(): boolean { + // can't use this.lineLegendSeriesLeft due to a circular dependency + const lineLegendSeries = this.series.map((series) => { + const { seriesName, color, start } = series + const formattedValue = this.formatColumn.formatValueShort( + start.value + ) return { - seriesName: series.seriesName, - leftValueLabel, - leftEntityLabel, - rightValueLabel, - rightEntityLabel, + color, + seriesName, + label: formattedValue, + yValue: start.value, } }) - } - @computed private get initialSlopeData() { - const { data, slopeLabels, xScale, yAxis, yDomain } = this + return LineLegend.needsConnectorLines({ + labelSeries: lineLegendSeries, + yAxis: this.yAxis, + maxWidth: this.maxLineLegendWidth, + connectorLineWidth: this.lineLegendConnectorLinesWidth, + fontSize: this.fontSize, + isStatic: this.manager.isStatic, + }) + } - const slopeData: SlopeEntryProps[] = [] + @computed get lineLegendWidthLeft(): number { + return LineLegend.width({ + labelSeries: this.lineLegendSeriesLeft, + yAxis: this.yAxis, + maxWidth: this.maxLineLegendWidth, + connectorLineWidth: this.lineLegendConnectorLinesWidth, + fontSize: this.fontSize, + fontWeight: this.showSeriesNamesInLineLegendLeft ? 700 : undefined, + isStatic: this.manager.isStatic, + }) + } - data.forEach((series, i) => { - // Ensure values fit inside the chart - if ( - !series.values.every( - (d) => d.y >= yDomain[0] && d.y <= yDomain[1] - ) - ) - return - - const labels = slopeLabels[i] - const [v1, v2] = series.values - const [x1, x2] = [xScale(v1.x), xScale(v2.x)] - const [y1, y2] = [yAxis.place(v1.y), yAxis.place(v2.y)] - - slopeData.push({ - ...labels, - x1, - y1, - x2, - y2, - color: series.color, - seriesName: series.seriesName, - isFocused: false, - isHovered: false, - hasLeftLabel: true, - hasRightLabel: true, - } as SlopeEntryProps) + @computed get lineLegendWidthRight(): number { + return LineLegend.width({ + labelSeries: this.lineLegendSeriesRight, + yAxis: this.yAxis, + yRange: this.lineLegendYRange, + verticalAlign: VerticalAlign.top, + maxWidth: this.maxLineLegendWidth, + connectorLineWidth: this.lineLegendConnectorLinesWidth, + fontSize: this.fontSize, + fontWeight: this.manager.showLegend ? 700 : undefined, + isStatic: this.manager.isStatic, }) + } + + @computed get xRange(): [number, number] { + const lineLegendWidthLeft = + this.lineLegendWidthLeft + LINE_LEGEND_PADDING + const lineLegendWidthRight = + this.lineLegendWidthRight + LINE_LEGEND_PADDING + const chartAreaWidth = this.innerBounds.width + + // start and end value when the slopes are as wide as possible + const minStartX = + this.innerBounds.x + this.yAxisWidth + lineLegendWidthLeft + const maxEndX = this.innerBounds.right - lineLegendWidthRight + + // use all available space if the chart is narrow + if (this.manager.isSemiNarrow) { + return [minStartX, maxEndX] + } + + const offset = 0.25 + let startX = this.innerBounds.x + offset * chartAreaWidth + let endX = this.innerBounds.right - offset * chartAreaWidth - return slopeData + // make sure the start and end values are within the bounds + startX = Math.max(startX, minStartX) + endX = Math.min(endX, maxEndX) + + // pick a reasonable max width based on an ideal aspect ratio + const idealAspectRatio = 0.9 + const availableWidth = + chartAreaWidth - + this.yAxisWidth - + lineLegendWidthLeft - + lineLegendWidthRight + const idealWidth = idealAspectRatio * this.bounds.height + const maxSlopeWidth = Math.min(idealWidth, availableWidth) + + const currentSlopeWidth = endX - startX + if (currentSlopeWidth > maxSlopeWidth) { + const padding = currentSlopeWidth - maxSlopeWidth + startX += padding / 2 + endX -= padding / 2 + } + + return [startX, endX] } - @computed get backgroundGroups() { - return this.slopeData.filter( - (group) => !(group.isHovered || group.isFocused) - ) + @computed get lineLegendX(): number { + return this.xRange[1] + LINE_LEGEND_PADDING } - @computed get foregroundGroups() { - return this.slopeData.filter( - (group) => !!(group.isHovered || group.isFocused) - ) + @computed get useCompactLineLegend(): boolean { + return !!this.manager.isSemiNarrow || this.bounds.width < 400 } - // Get the final slope data with hover focusing and collision detection - @computed get slopeData(): SlopeEntryProps[] { - const { focusedSeriesNames, hoveredSeriesNames } = this + @computed get hoveredSeriesNameSet(): Set { + const hoveredSeriesNames = new Set() - let slopeData = this.initialSlopeData + // hovered series name + if (this.hoveredSeriesName) + hoveredSeriesNames.add(this.hoveredSeriesName) - slopeData = slopeData.map((slope) => { - // used for collision detection - const leftEntityLabelBounds = new Bounds( - // labels on the left are placed like this: | - slope.x1 - - LABEL_SLOPE_PADDING - - slope.leftValueLabel.width - - LABEL_LABEL_PADDING, - slope.y1 - slope.leftEntityLabel.lines[0].height / 2, - slope.leftEntityLabel.width, - slope.leftEntityLabel.height - ) - const rightEntityLabelBounds = new Bounds( - // labels on the left are placed like this: | - slope.x2 + LABEL_SLOPE_PADDING, - slope.y2 - slope.rightEntityLabel.height / 2, - slope.rightEntityLabel.width, - slope.rightEntityLabel.height - ) + // hovered legend item in the external facet legend + const { externalLegendHoverBin } = this.manager + if (externalLegendHoverBin) { + this.series + .map((s) => s.seriesName) + .filter((name) => externalLegendHoverBin?.contains(name)) + .forEach((name) => hoveredSeriesNames.add(name)) + } + + return hoveredSeriesNames + } + + /** + * If the line legend uses connector lines, then we do show the series + * name to make it clear which slope the value belongs to + */ + @computed private get showSeriesNamesInLineLegendLeft(): boolean { + return this.lineLegendLeftHasConnectorLines && !!this.manager.showLegend + } - // used to determine priority for labelling conflicts - const isFocused = focusedSeriesNames.includes(slope.seriesName) - const isHovered = hoveredSeriesNames.includes(slope.seriesName) + @computed get lineLegendSeriesLeft(): LineLabelSeries[] { + const { showSeriesNamesInLineLegendLeft: showSeriesName } = this + return this.series.map((series) => { + const { seriesName, color, start } = series + const value = this.formatColumn.formatValueShort(start.value) + const label = showSeriesName ? seriesName : value + const formattedValue = showSeriesName ? value : undefined + return { + color, + seriesName, + label, + formattedValue, + valueInNewLine: this.useCompactLineLegend, + yValue: start.value, + focus: this.seriesFocusState(series), + hover: this.seriesHoverState(series), + } + }) + } + @computed get lineLegendSeriesRight(): LineLabelSeries[] { + return this.series.map((series) => { + const { seriesName, color, end, annotation } = series + const value = this.formatColumn.formatValueShort(end.value) + const label = this.manager.showLegend ? seriesName : value + const formattedValue = this.manager.showLegend ? value : undefined return { - ...slope, - leftEntityLabelBounds, - rightEntityLabelBounds, - isFocused, - isHovered, + color, + seriesName, + label, + formattedValue, + valueInNewLine: this.useCompactLineLegend, + annotation: + this.manager.showLegend && this.useCompactLineLegend + ? undefined + : annotation, + yValue: end.value, + focus: this.seriesFocusState(series), + hover: this.seriesHoverState(series), } }) + } + + @computed private get lineLegendConnectorLinesWidth(): number { + return this.useCompactLineLegend ? 15 : 25 + } + + private playIntroAnimation() { + // Nice little intro animation + select(this.slopeAreaRef.current) + .selectAll(".slope") + .attr("stroke-dasharray", "100%") + .attr("stroke-dashoffset", "100%") + .transition() + .duration(600) + .attr("stroke-dashoffset", "0%") + } - // How to work out which of two slopes to prioritize for labelling conflicts - function chooseLabel(s1: SlopeEntryProps, s2: SlopeEntryProps) { - if (s1.isHovered && !s2.isHovered) - // Hovered slopes always have priority - return s1 - else if (!s1.isHovered && s2.isHovered) return s2 - else if (s1.isFocused && !s2.isFocused) - // Focused slopes are next in priority - return s1 - else if (!s1.isFocused && s2.isFocused) return s2 - else if (s1.hasRightLabel && !s2.hasRightLabel) - // Slopes which already have one label are prioritized for the other side - return s1 - else if (!s1.hasRightLabel && s2.hasRightLabel) return s2 - else return s1 // Equal priority, just do the first one + componentDidMount() { + exposeInstanceOnWindow(this) + + if (!this.manager.disableIntroAnimation) { + this.playIntroAnimation() } + } - // Eliminate overlapping labels, one pass for each side - slopeData.forEach((s1) => { - slopeData.forEach((s2) => { - if ( - s1 !== s2 && - s1.hasRightLabel && - s2.hasRightLabel && - // entity labels don't necessarily share the same x position. - // that's why we check for vertical intersection only - s1.rightEntityLabelBounds.hasVerticalOverlap( - s2.rightEntityLabelBounds - ) - ) { - if (chooseLabel(s1, s2) === s1) s2.hasRightLabel = false - else s1.hasRightLabel = false - } - }) - }) + private updateTooltipPosition(event: SVGMouseOrTouchEvent) { + const ref = this.manager.base?.current + if (ref) this.tooltipState.position = getRelativeMouse(ref, event) + } + + private detectHoveredSlope(event: SVGMouseOrTouchEvent) { + const ref = this.slopeAreaRef.current + if (!ref) return + + const mouse = getRelativeMouse(ref, event) + this.mouseFrame = requestAnimationFrame(() => { + if (this.placedSeries.length === 0) return - slopeData.forEach((s1) => { - slopeData.forEach((s2) => { - if ( - s1 !== s2 && - s1.hasLeftLabel && - s2.hasLeftLabel && - // entity labels don't necessarily share the same x position. - // that's why we check for vertical intersection only - s1.leftEntityLabelBounds.hasVerticalOverlap( - s2.leftEntityLabelBounds + const distanceMap = new Map() + for (const series of this.placedSeries) { + distanceMap.set( + series, + PointVector.distanceFromPointToLineSegmentSq( + mouse, + series.startPoint, + series.endPoint ) - ) { - if (chooseLabel(s1, s2) === s1) s2.hasLeftLabel = false - else s1.hasLeftLabel = false - } - }) + ) + } + + const closestSlope = minBy(this.placedSeries, (s) => + distanceMap.get(s) + )! + const distanceSq = distanceMap.get(closestSlope)! + const tolerance = 10 + const toleranceSq = tolerance * tolerance + + if (closestSlope && distanceSq < toleranceSq) { + this.onSlopeMouseOver(closestSlope) + } else { + this.onSlopeMouseLeave() + } }) + } - // Order by focus/hover for draw order - slopeData = sortBy(slopeData, (slope) => - slope.isFocused || slope.isHovered ? 1 : 0 - ) + @action.bound onLineLegendClick(seriesName: SeriesName): void { + this.interactionArray.toggleFocus(seriesName) + } + + private hoverTimer?: NodeJS.Timeout + @action.bound onLineLegendMouseOver(seriesName: SeriesName): void { + clearTimeout(this.hoverTimer) + this.hoveredSeriesName = seriesName + } + + @action.bound onLineLegendMouseLeave(): void { + clearTimeout(this.hoverTimer) + this.hoverTimer = setTimeout(() => { + // wait before clearing selection in case the mouse is moving quickly over neighboring labels + this.hoveredSeriesName = undefined + }, 200) + } - return slopeData + @action.bound onSlopeMouseOver(series: SlopeChartSeries) { + this.hoveredSeriesName = series.seriesName + this.tooltipState.target = { series } + } + + @action.bound onSlopeMouseLeave() { + this.hoveredSeriesName = undefined + this.tooltipState.target = null } mouseFrame?: number + @action.bound onMouseMove(event: SVGMouseOrTouchEvent) { + this.updateTooltipPosition(event) + this.detectHoveredSlope(event) + } + @action.bound onMouseLeave() { if (this.mouseFrame !== undefined) cancelAnimationFrame(this.mouseFrame) - if (this.props.onMouseLeave) this.props.onMouseLeave() - } - - @action.bound onMouseMove( - ev: React.MouseEvent | React.TouchEvent - ) { - if (this.base.current) { - const mouse = getRelativeMouse(this.base.current, ev.nativeEvent) - - this.mouseFrame = requestAnimationFrame(() => { - if (this.props.bounds.contains(mouse)) { - if (this.slopeData.length === 0) return - - const { x1: startX, x2: endX } = this.slopeData[0] - - // whether the mouse is over the chart area, - // the left label area, or the right label area - const mousePosition = - mouse.x < startX - ? "left" - : mouse.x > endX - ? "right" - : "chart" - - // don't track mouse movements when hovering over labels on the left - if (mousePosition === "left") { - this.props.onMouseLeave() - return - } - - const distToSlopeOrLabel = new Map< - SlopeEntryProps, - number - >() - for (const s of this.slopeData) { - // start and end point of a line - let p1: PointVector - let p2: PointVector - - if (mousePosition === "chart") { - // points define the slope line - p1 = new PointVector(s.x1, s.y1) - p2 = new PointVector(s.x2, s.y2) - } else { - const labelBox = s.rightEntityLabelBounds.toProps() - // points define a "strike-through" line that stretches from - // the end point of the slopes to the right side of the right label - const y = labelBox.y + labelBox.height / 2 - p1 = new PointVector(endX, y) - p2 = new PointVector(labelBox.x + labelBox.width, y) - } - - // calculate the distance to the slope or label - const dist = - PointVector.distanceFromPointToLineSegmentSq( - mouse, - p1, - p2 - ) - distToSlopeOrLabel.set(s, dist) - } - - const closestSlope = minBy(this.slopeData, (s) => - distToSlopeOrLabel.get(s) - ) - const distanceSq = distToSlopeOrLabel.get(closestSlope!)! - const tolerance = mousePosition === "chart" ? 20 : 10 - const toleranceSq = tolerance * tolerance - - if ( - closestSlope && - distanceSq < toleranceSq && - this.props.onMouseOver - ) { - this.props.onMouseOver(closestSlope) - } else { - this.props.onMouseLeave() - } - } - }) - } + this.onSlopeMouseLeave() } - @action.bound onClick() { - if (this.props.onClick) this.props.onClick() + @computed get failMessage(): string { + const message = getDefaultFailMessage(this.manager) + if (message) return message + else if (this.startTime === this.endTime) + return "Two time points needed for comparison" + else if (this.series.length === 0) return "No matching data" + return "" } - componentDidMount() { - if (!this.manager.disableIntroAnimation) { - this.playIntroAnimation() + @computed get helpMessage(): string | undefined { + if (this.failMessage === "Two time points needed for compariso") + return "Click or drag the timeline to select two different points in time" + return undefined + } + + @computed get renderUid(): number { + return guid() + } + + @computed get tooltip(): React.ReactElement | undefined { + const { + manager: { isRelativeMode }, + tooltipState: { target, position, fading }, + formatColumn, + startTime, + endTime, + } = this + + const { series } = target || {} + if (!series) return + + const formatTime = (time: Time) => formatColumn.formatTime(time) + + const title = series.seriesName + const titleAnnotation = series.annotation + + const actualStartTime = series.start.originalTime + const actualEndTime = series.end.originalTime + const timeRange = `${formatTime(actualStartTime)} to ${formatTime(actualEndTime)}` + const timeLabel = isRelativeMode + ? `% change between ${formatColumn.formatTime(actualStartTime)} and ${formatColumn.formatTime(actualEndTime)}` + : timeRange + + const constructTargetYearForToleranceNotice = () => { + const isStartValueOriginal = series.start.originalTime === startTime + const isEndValueOriginal = series.end.originalTime === endTime + + if (!isStartValueOriginal && !isEndValueOriginal) { + return `${formatTime(startTime)} and ${formatTime(endTime)}` + } else if (!isStartValueOriginal) { + return formatTime(startTime) + } else if (!isEndValueOriginal) { + return formatTime(endTime) + } else { + return undefined + } } + + const targetYear = constructTargetYearForToleranceNotice() + const toleranceNotice = targetYear + ? { + icon: TooltipFooterIcon.notice, + text: makeTooltipToleranceNotice(targetYear), + } + : undefined + const roundingNotice = series.column.roundsToSignificantFigures + ? { + icon: TooltipFooterIcon.none, + text: makeTooltipRoundingNotice( + [series.column.numSignificantFigures], + { plural: !isRelativeMode } + ), + } + : undefined + const footer = excludeUndefined([toleranceNotice, roundingNotice]) + + const values = isRelativeMode + ? [series.end.value] + : [series.start.value, series.end.value] + + return ( + (this.tooltipState.target = null)} + > + + + ) } - private playIntroAnimation() { - // Nice little intro animation - select(this.base.current) - .select(".slopes") - .attr("stroke-dasharray", "100%") - .attr("stroke-dashoffset", "100%") - .transition() - .attr("stroke-dashoffset", "0%") + private makeMissingDataLabel(series: RawSlopeChartSeries): string { + const { seriesName, start, end } = series + + const startTime = this.formatColumn.formatTime(this.startTime) + const endTime = this.formatColumn.formatTime(this.endTime) + + // mention the start or end value if they're missing + if (start?.value === undefined && end?.value === undefined) { + return `${seriesName} (${startTime} & ${endTime})` + } else if (start?.value === undefined) { + return `${seriesName} (${startTime})` + } else if (end?.value === undefined) { + return `${seriesName} (${endTime})` + } + + // if both values are given but the series shows up in the No Data + // section, then tolerance has been applied to one of the values + // in such a way that we decided not to render the slope after all + // (e.g. when the original times are too close to each other) + const isToleranceAppliedToStartValue = + start.originalTime !== this.startTime + const isToleranceAppliedToEndValue = end.originalTime !== this.endTime + if (isToleranceAppliedToStartValue && isToleranceAppliedToEndValue) { + return `${seriesName} (${startTime} & ${endTime})` + } else if (isToleranceAppliedToStartValue) { + return `${seriesName} (${startTime})` + } else if (isToleranceAppliedToEndValue) { + return `${seriesName} (${endTime})` + } + + return seriesName } - renderGroups(groups: SlopeEntryProps[]) { - const { isLayerMode, isMultiHoverMode } = this + private renderNoDataSection(): React.ReactElement | void { + if (!this.showNoDataSection) return - return groups.map((slope) => ( - + this.makeMissingDataLabel(series) + ) + + return ( + - )) + ) } - render() { - const { bounds, slopeData, xDomain, yAxis, yRange, onMouseMove } = this + private renderSlopes() { + return ( + + {this.renderSeries.map((series) => ( + + ))} + + ) + } - if (isEmpty(slopeData)) - return + private renderYAxis() { + return ( + <> + {!this.yAxis.hideGridlines && ( + + )} + {!this.yAxis.hideAxis && ( + + )} + + ) + } + + private renderXAxis() { + const { xDomain, yRange, startX, endX } = this - const { x1, x2 } = slopeData[0] - const [y1, y2] = yRange + const [bottom, top] = yRange return ( - - + + + + ) + } + + private renderChartArea() { + return ( + + {this.renderYAxis()} + {this.renderXAxis()} - {this.yAxis.tickLabels.map((tick) => { - const y = yAxis.place(tick.value) - return ( - - {/* grid lines connecting the chart area to the axis */} - - {/* grid lines within the chart area */} - - - ) - })} + + {this.renderSlopes()} - - - - - {this.yColumn.formatTime(xDomain[0])} - + + ) + } + + private renderLineLegendRight(): React.ReactElement { + return ( + + ) + } + + private renderLineLegendLeft(): React.ReactElement { + const uniqYValues = uniq( + this.lineLegendSeriesLeft.map((series) => series.yValue) + ) + const allSlopesStartFromZero = + uniqYValues.length === 1 && uniqYValues[0] === 0 + + // if all values have a start value of 0, show the 0-label only once + if ( + // in relative mode, all slopes start from 0% + this.manager.isRelativeMode || + allSlopesStartFromZero + ) + return ( - {this.yColumn.formatTime(xDomain[1])} + {this.formatColumn.formatValueShort(0)} - - {this.renderGroups(this.backgroundGroups)} - {this.renderGroups(this.foregroundGroups)} - + ) + + return ( + + ) + } + + private renderLineLegends(): React.ReactElement | void { + return ( + <> + {this.renderLineLegendLeft()} + {this.renderLineLegendRight()} + + ) + } + + render() { + if (this.failMessage) + return ( + <> + {this.renderYAxis()} + + + ) + + return ( + + {this.renderChartArea()} + {this.renderLineLegends()} + {this.renderNoDataSection()} + {this.tooltip} ) } } + +interface SlopeProps { + series: RenderSlopeChartSeries + dotRadius?: number + strokeWidth?: number + outlineWidth?: number + outlineStroke?: string + onMouseOver?: (series: SlopeChartSeries) => void + onMouseLeave?: () => void +} + +function Slope({ + series, + dotRadius = 2.5, + strokeWidth = 2, + outlineWidth = 0.5, + outlineStroke = "#fff", + onMouseOver, + onMouseLeave, +}: SlopeProps) { + const { seriesName, startPoint, endPoint } = series + + const background = series.focus === FocusState.background + const hovered = series.hover === HoverState.active + const muted = series.hover === HoverState.background + + const color = !background || hovered ? series.color : "#E7E7E7" + const showOutline = !muted + const opacity = muted ? 0.3 : 1 + + return ( + onMouseOver?.(series)} + onMouseLeave={() => onMouseLeave?.()} + > + {showOutline && ( + + )} + + + ) +} + +/** + * Line with two dots at the ends, drawn as a single path element. + */ +function LineWithDots({ + startPoint, + endPoint, + radius, + color, + lineWidth = 2, + opacity = 1, +}: { + startPoint: PointVector + endPoint: PointVector + radius: number + color: string + lineWidth?: number + opacity?: number +}): React.ReactElement { + const startDotPath = makeCirclePath(startPoint.x, startPoint.y, radius) + const endDotPath = makeCirclePath(endPoint.x, endPoint.y, radius) + + const linePath = makeLinePath( + startPoint.x, + startPoint.y, + endPoint.x, + endPoint.y + ) + + return ( + + ) +} + +interface GridLinesProps { + bounds: Bounds + yAxis: VerticalAxis +} + +function GridLines({ bounds, yAxis }: GridLinesProps) { + return ( + + {yAxis.tickLabels.map((tick) => { + const y = yAxis.place(tick.value) + return ( + + + + ) + })} + + ) +} + +function MarkX({ + label, + x, + top, + bottom, + fontSize, +}: { + label: string + x: number + top: number + bottom: number + fontSize: number +}) { + return ( + <> + + + {label} + + + ) +} + +const makeCirclePath = (centerX: number, centerY: number, radius: number) => { + const topX = centerX + const topY = centerY - radius + return `M ${topX},${topY} A ${radius},${radius} 0 1,1 ${topX - 0.0001},${topY}` +} + +const makeLinePath = (x1: number, y1: number, x2: number, y2: number) => { + return `M ${x1},${y1} L ${x2},${y2}` +} diff --git a/packages/@ourworldindata/grapher/src/slopeCharts/SlopeChartConstants.ts b/packages/@ourworldindata/grapher/src/slopeCharts/SlopeChartConstants.ts index bb52f727212..08b0a5a5d1e 100644 --- a/packages/@ourworldindata/grapher/src/slopeCharts/SlopeChartConstants.ts +++ b/packages/@ourworldindata/grapher/src/slopeCharts/SlopeChartConstants.ts @@ -1,62 +1,23 @@ +import { PartialBy, PointVector } from "@ourworldindata/utils" +import { EntityName, OwidVariableRow } from "@ourworldindata/types" +import { ChartSeries, RenderChartSeries } from "../chart/ChartInterface" import { CoreColumn } from "@ourworldindata/core-table" -import { ChartSeries } from "../chart/ChartInterface" -import { ChartManager } from "../chart/ChartManager" -import { ScaleType } from "@ourworldindata/types" -import { Bounds } from "@ourworldindata/utils" -import { TextWrap } from "@ourworldindata/components" - -export interface SlopeChartValue { - x: number - y: number -} export interface SlopeChartSeries extends ChartSeries { - size: number - values: SlopeChartValue[] + column: CoreColumn + entityName: EntityName + start: Pick, "value" | "originalTime"> + end: Pick, "value" | "originalTime"> + annotation?: string } -export const DEFAULT_SLOPE_CHART_COLOR = "#ff7f0e" - -export interface SlopeEntryProps extends ChartSeries { - isLayerMode: boolean - isMultiHoverMode: boolean - x1: number - y1: number - x2: number - y2: number - - hasLeftLabel: boolean - leftEntityLabel: TextWrap - leftValueLabel: TextWrap - leftEntityLabelBounds: Bounds +export type RawSlopeChartSeries = PartialBy - hasRightLabel: boolean - rightEntityLabel: TextWrap - rightEntityLabelBounds: Bounds - rightValueLabel: TextWrap - - isFocused: boolean - isHovered: boolean +export interface PlacedSlopeChartSeries extends SlopeChartSeries { + startPoint: PointVector + endPoint: PointVector } -export interface LabelledSlopesProps { - manager: ChartManager - yColumn: CoreColumn - bounds: Bounds - seriesArr: SlopeChartSeries[] - focusKeys: string[] - hoverKeys: string[] - onMouseOver: (slopeProps: SlopeEntryProps) => void - onMouseLeave: () => void - onClick: () => void - isPortrait: boolean -} +export type RenderSlopeChartSeries = RenderChartSeries -export interface SlopeAxisProps { - bounds: Bounds - orient: "left" | "right" - column: CoreColumn - scale: any - scaleType: ScaleType - fontSize: number -} +export const DEFAULT_SLOPE_CHART_COLOR = "#ff7f0e" diff --git a/packages/@ourworldindata/grapher/src/stackedCharts/AbstractStackedChart.tsx b/packages/@ourworldindata/grapher/src/stackedCharts/AbstractStackedChart.tsx index 9004f813505..172d7e0bf9c 100644 --- a/packages/@ourworldindata/grapher/src/stackedCharts/AbstractStackedChart.tsx +++ b/packages/@ourworldindata/grapher/src/stackedCharts/AbstractStackedChart.tsx @@ -405,8 +405,8 @@ export class AbstractStackedChart const pointColor = row.value > 0 ? POSITIVE_COLOR : NEGATIVE_COLOR return { - position: row.time, - time: row.time, + position: row.originalTime, + time: row.originalTime, value: row.value, valueOffset: 0, interpolated: diff --git a/packages/@ourworldindata/grapher/src/stackedCharts/MarimekkoChart.tsx b/packages/@ourworldindata/grapher/src/stackedCharts/MarimekkoChart.tsx index 9fa7f41afd6..e89b176b4da 100644 --- a/packages/@ourworldindata/grapher/src/stackedCharts/MarimekkoChart.tsx +++ b/packages/@ourworldindata/grapher/src/stackedCharts/MarimekkoChart.tsx @@ -383,7 +383,7 @@ export class MarimekkoChart col.def.color ?? colorScheme.getColors(yColumns.length)[i], points: col.owidRows.map((row) => ({ - time: row.time, + time: row.originalTime, position: row.entityName, value: row.value, valueOffset: 0, @@ -417,7 +417,7 @@ export class MarimekkoChart const points: SimplePoint[] = [] for (const row of rows) { points.push({ - time: row.time, + time: row.originalTime, value: row.value, entity: row.entityName, }) diff --git a/packages/@ourworldindata/grapher/src/stackedCharts/StackedAreaChart.tsx b/packages/@ourworldindata/grapher/src/stackedCharts/StackedAreaChart.tsx index e218c1950c9..19fd0b4f6cc 100644 --- a/packages/@ourworldindata/grapher/src/stackedCharts/StackedAreaChart.tsx +++ b/packages/@ourworldindata/grapher/src/stackedCharts/StackedAreaChart.tsx @@ -29,11 +29,7 @@ import { import { observer } from "mobx-react" import { DualAxisComponent } from "../axis/AxisViews" import { DualAxis } from "../axis/Axis" -import { - LineLabelSeries, - LineLegend, - LineLegendManager, -} from "../lineLegend/LineLegend" +import { LineLabelSeries, LineLegend } from "../lineLegend/LineLegend" import { NoDataModal } from "../noDataModal/NoDataModal" import { TooltipFooterIcon } from "../tooltip/TooltipProps.js" import { @@ -66,23 +62,30 @@ interface AreasProps extends React.SVGAttributes { onAreaMouseLeave?: () => void } +enum RenderMode { + default = "default", + focus = "focus", // hovered or focused + mute = "mute", // not hovered + background = "background", // not focused +} + const STACKED_AREA_CHART_CLASS_NAME = "StackedArea" -const AREA_OPACITY = { - DEFAULT: GRAPHER_AREA_OPACITY_DEFAULT, - FOCUS: GRAPHER_AREA_OPACITY_FOCUS, - MUTE: GRAPHER_AREA_OPACITY_MUTE, +const AREA_OPACITY: Partial> = { + default: GRAPHER_AREA_OPACITY_DEFAULT, + focus: GRAPHER_AREA_OPACITY_FOCUS, + mute: GRAPHER_AREA_OPACITY_MUTE, } -const BORDER_OPACITY = { - DEFAULT: 0.7, - HOVER: 1, - MUTE: 0.3, +const BORDER_OPACITY: Partial> = { + default: 0.7, + focus: 1, + mute: 0.3, } -const BORDER_WIDTH = { - DEFAULT: 0.5, - HOVER: 1.5, +const BORDER_WIDTH: Partial> = { + default: 0.5, + mute: 1.5, } @observer @@ -183,10 +186,10 @@ class Areas extends React.Component { } const points = [...placedPoints, ...reverse(clone(prevPoints))] const opacity = !this.isFocusModeActive - ? AREA_OPACITY.DEFAULT // normal opacity + ? AREA_OPACITY.default // normal opacity : focusedSeriesName === series.seriesName - ? AREA_OPACITY.FOCUS // hovered - : AREA_OPACITY.MUTE // non-hovered + ? AREA_OPACITY.focus // hovered + : AREA_OPACITY.mute // non-hovered return ( { return placedSeriesArr.map((placedSeries) => { const opacity = !this.isFocusModeActive - ? BORDER_OPACITY.DEFAULT // normal opacity + ? BORDER_OPACITY.default // normal opacity : focusedSeriesName === placedSeries.seriesName - ? BORDER_OPACITY.HOVER // hovered - : BORDER_OPACITY.MUTE // non-hovered + ? BORDER_OPACITY.focus // hovered + : BORDER_OPACITY.mute // non-hovered const strokeWidth = focusedSeriesName === placedSeries.seriesName - ? BORDER_WIDTH.HOVER - : BORDER_WIDTH.DEFAULT + ? BORDER_WIDTH.focus + : BORDER_WIDTH.default return ( { } @observer -export class StackedAreaChart - extends AbstractStackedChart - implements LineLegendManager -{ +export class StackedAreaChart extends AbstractStackedChart { constructor(props: AbstractStackedChartProps) { super(props) } @@ -301,7 +301,7 @@ export class StackedAreaChart }) } - @computed get labelSeries(): LineLabelSeries[] { + @computed get lineLegendSeries(): LineLabelSeries[] { const { midpoints } = this return this.series .map((series, index) => ({ @@ -310,6 +310,10 @@ export class StackedAreaChart label: series.seriesName, yValue: midpoints[index], isAllZeros: series.isAllZeros, + hovered: series.seriesName === this.hoveredSeriesName, + muted: + !!this.hoveredSeriesName && + series.seriesName !== this.hoveredSeriesName, })) .filter((series) => !series.isAllZeros) .reverse() @@ -319,9 +323,16 @@ export class StackedAreaChart return Math.min(150, this.bounds.width / 3) } - @computed get legendDimensions(): LineLegend | undefined { - if (!this.manager.showLegend) return undefined - return new LineLegend({ manager: this }) + @computed get lineLegendWidth(): number { + if (!this.manager.showLegend) return 0 + + // only pass props that are required to calculate + // the width to avoid circular dependencies + return LineLegend.width({ + labelSeries: this.lineLegendSeries, + maxWidth: this.maxLineLegendWidth, + fontSize: this.fontSize, + }) } @observable tooltipState = new TooltipState<{ @@ -348,8 +359,7 @@ export class StackedAreaChart @observable private hoverTimer?: NodeJS.Timeout @computed protected get paddingForLegendRight(): number { - const { legendDimensions } = this - return legendDimensions ? legendDimensions.width : 0 + return this.lineLegendWidth } @computed get seriesSortedByImportance(): string[] { @@ -417,7 +427,7 @@ export class StackedAreaChart return hoveredSeries?.seriesName } - @computed get focusedSeriesName(): SeriesName | undefined { + @computed get hoveredSeriesName(): SeriesName | undefined { return ( // if the chart area is hovered this.tooltipState.target?.series ?? @@ -428,11 +438,6 @@ export class StackedAreaChart ) } - // used by the line legend component - @computed get focusedSeriesNames(): string[] { - return this.focusedSeriesName ? [this.focusedSeriesName] : [] - } - @action.bound private onCursorMove( ev: React.MouseEvent | React.TouchEvent ): void { @@ -593,8 +598,8 @@ export class StackedAreaChart point?.fake ? undefined : point?.value, ] const opacity = focused - ? AREA_OPACITY.FOCUS - : AREA_OPACITY.DEFAULT + ? AREA_OPACITY.focus + : AREA_OPACITY.default const swatch = { color, opacity } return { @@ -656,7 +661,20 @@ export class StackedAreaChart renderLegend(): React.ReactElement | void { if (!this.manager.showLegend) return - return + return ( + + ) } renderStatic(): React.ReactElement { @@ -667,7 +685,7 @@ export class StackedAreaChart ) @@ -705,7 +723,7 @@ export class StackedAreaChart @@ -735,8 +753,8 @@ export class StackedAreaChart } @computed get lineLegendX(): number { - return this.legendDimensions - ? this.bounds.right - this.legendDimensions.width + return this.manager.showLegend + ? this.bounds.right - this.lineLegendWidth : 0 } diff --git a/packages/@ourworldindata/grapher/src/stackedCharts/StackedDiscreteBarChart.tsx b/packages/@ourworldindata/grapher/src/stackedCharts/StackedDiscreteBarChart.tsx index c575e65c476..b298d2ae160 100644 --- a/packages/@ourworldindata/grapher/src/stackedCharts/StackedDiscreteBarChart.tsx +++ b/packages/@ourworldindata/grapher/src/stackedCharts/StackedDiscreteBarChart.tsx @@ -1006,7 +1006,7 @@ export class StackedDiscreteBarChart col.displayName ), points: col.owidRows.map((row) => ({ - time: row.time, + time: row.originalTime, position: row.entityName, value: row.value, valueOffset: 0, diff --git a/packages/@ourworldindata/grapher/src/tooltip/Tooltip.scss b/packages/@ourworldindata/grapher/src/tooltip/Tooltip.scss index c8e74e151f4..7e75dbac713 100644 --- a/packages/@ourworldindata/grapher/src/tooltip/Tooltip.scss +++ b/packages/@ourworldindata/grapher/src/tooltip/Tooltip.scss @@ -67,6 +67,13 @@ font-size: 14px; font-weight: $bold; letter-spacing: 0; + + .annotation { + margin-left: 4px; + font-weight: normal; + color: $grey; + font-size: 0.9em; + } } .subtitle { margin: 4px 0 2px 0; diff --git a/packages/@ourworldindata/grapher/src/tooltip/Tooltip.tsx b/packages/@ourworldindata/grapher/src/tooltip/Tooltip.tsx index 5e4523480a9..01d11012823 100644 --- a/packages/@ourworldindata/grapher/src/tooltip/Tooltip.tsx +++ b/packages/@ourworldindata/grapher/src/tooltip/Tooltip.tsx @@ -103,6 +103,7 @@ class TooltipCard extends React.Component< let { id, title, + titleAnnotation, subtitle, subtitleFormat, footer, @@ -189,7 +190,14 @@ class TooltipCard extends React.Component< > {hasHeader && (
- {title &&
{title}
} + {title && ( +
+ {title}{" "} + + {titleAnnotation} + +
+ )} {subtitle && (
{timeNotice && TOOLTIP_ICON.notice} diff --git a/packages/@ourworldindata/grapher/src/tooltip/TooltipContents.tsx b/packages/@ourworldindata/grapher/src/tooltip/TooltipContents.tsx index 3078fd31d53..a3a8b7f0f1d 100644 --- a/packages/@ourworldindata/grapher/src/tooltip/TooltipContents.tsx +++ b/packages/@ourworldindata/grapher/src/tooltip/TooltipContents.tsx @@ -337,8 +337,12 @@ export function IconCircledS({ ) } -export function makeTooltipToleranceNotice(targetYear: string): string { - return `Data not available for ${targetYear}. Showing closest available data point instead` +export function makeTooltipToleranceNotice( + targetYear: string, + { plural }: { plural: boolean } = { plural: false } +): string { + const dataPoint = plural ? "data points" : "data point" + return `Data not available for ${targetYear}. Showing closest available ${dataPoint} instead` } export function makeTooltipRoundingNotice( diff --git a/packages/@ourworldindata/grapher/src/tooltip/TooltipProps.ts b/packages/@ourworldindata/grapher/src/tooltip/TooltipProps.ts index e3222a8949b..9fae113da6b 100644 --- a/packages/@ourworldindata/grapher/src/tooltip/TooltipProps.ts +++ b/packages/@ourworldindata/grapher/src/tooltip/TooltipProps.ts @@ -28,6 +28,7 @@ export interface TooltipProps { offsetXDirection?: "left" | "right" offsetYDirection?: "upward" | "downward" title?: string | number // header text + titleAnnotation?: string // rendered next to the title, but muted subtitle?: string | number // header deck subtitleFormat?: "notice" | "unit" // optional postprocessing for subtitle footer?: { icon: TooltipFooterIcon; text: string }[] diff --git a/packages/@ourworldindata/types/src/domainTypes/CoreTableTypes.ts b/packages/@ourworldindata/types/src/domainTypes/CoreTableTypes.ts index 5fa5ba92b00..a44d6d42459 100644 --- a/packages/@ourworldindata/types/src/domainTypes/CoreTableTypes.ts +++ b/packages/@ourworldindata/types/src/domainTypes/CoreTableTypes.ts @@ -300,5 +300,6 @@ export interface OwidVariableRow { entityName: EntityName time: Time value: ValueType + originalTime: Time originalValue?: ValueType } diff --git a/packages/@ourworldindata/types/src/grapherTypes/GrapherTypes.ts b/packages/@ourworldindata/types/src/grapherTypes/GrapherTypes.ts index 1d73e6a6470..cb146f77cc3 100644 --- a/packages/@ourworldindata/types/src/grapherTypes/GrapherTypes.ts +++ b/packages/@ourworldindata/types/src/grapherTypes/GrapherTypes.ts @@ -580,6 +580,7 @@ export interface GrapherInterface extends SortConfig { includedEntities?: number[] selectedEntityNames?: EntityName[] selectedEntityColors?: { [entityName: string]: string | undefined } + focusedSeriesNames?: SeriesName[] missingDataStrategy?: MissingDataStrategy hideFacetControl?: boolean facettingLabelByYVariables?: string @@ -698,6 +699,7 @@ export const grapherKeysToSerialize = [ "dimensions", "selectedEntityNames", "selectedEntityColors", + "focusedSeriesNames", "sortBy", "sortOrder", "sortColumnSlug", diff --git a/packages/@ourworldindata/utils/src/grapherConfigUtils.ts b/packages/@ourworldindata/utils/src/grapherConfigUtils.ts index 2a8db8e0120..7649db3d09f 100644 --- a/packages/@ourworldindata/utils/src/grapherConfigUtils.ts +++ b/packages/@ourworldindata/utils/src/grapherConfigUtils.ts @@ -22,6 +22,27 @@ const KEYS_EXCLUDED_FROM_INHERITANCE = [ "isPublished", ] +/** + * Simple merge function that doesn't do any Grapher-specific checks. + * + * You usually want to use `mergeGrapherConfigs` instead that implements the + * inheritance model correctly. Only use this if you know what you're doing. + */ +export function simpleMerge( + ...grapherConfigs: GrapherInterface[] +): GrapherInterface { + return mergeWith( + {}, // mergeWith mutates the first argument + ...grapherConfigs, + (_: unknown, childValue: unknown): any => { + // don't concat arrays, just use the last one + if (Array.isArray(childValue)) { + return childValue + } + } + ) +} + export function mergeGrapherConfigs( ...grapherConfigs: GrapherInterface[] ): GrapherInterface { @@ -60,16 +81,7 @@ export function mergeGrapherConfigs( return omit(config, KEYS_EXCLUDED_FROM_INHERITANCE) }) - return mergeWith( - {}, // mergeWith mutates the first argument - ...cleanedConfigs, - (_: unknown, childValue: unknown): any => { - // don't concat arrays, just use the last one - if (Array.isArray(childValue)) { - return childValue - } - } - ) + return simpleMerge(...cleanedConfigs) } export function diffGrapherConfigs( diff --git a/packages/@ourworldindata/utils/src/index.ts b/packages/@ourworldindata/utils/src/index.ts index 75f99afdd16..1e74033b5d5 100644 --- a/packages/@ourworldindata/utils/src/index.ts +++ b/packages/@ourworldindata/utils/src/index.ts @@ -335,6 +335,7 @@ export { export { isAndroid, isIOS } from "./BrowserUtils.js" export { + simpleMerge, diffGrapherConfigs, mergeGrapherConfigs, } from "./grapherConfigUtils.js" diff --git a/site/multiembedder/MultiEmbedder.tsx b/site/multiembedder/MultiEmbedder.tsx index b7b84b59516..3089954f249 100644 --- a/site/multiembedder/MultiEmbedder.tsx +++ b/site/multiembedder/MultiEmbedder.tsx @@ -19,6 +19,7 @@ import { merge, MultiDimDataPageConfig, extractMultiDimChoicesFromQueryStr, + deserializeJSONFromHTML, } from "@ourworldindata/utils" import { action } from "mobx" import React from "react" @@ -161,8 +162,10 @@ class MultiEmbedder { dataApiUrl: DATA_API_URL, } + const html = await fetchText(fullUrl) + if (isExplorer) { - const html = await fetchText(fullUrl) + // const html = await fetchText(fullUrl) const props: ExplorerProps = await buildExplorerProps( html, queryStr, @@ -173,35 +176,36 @@ class MultiEmbedder { ReactDOM.render(, figure) } else { figure.classList.remove(GRAPHER_PREVIEW_CLASS) - const url = new URL(fullUrl) - const slug = url.pathname.split("/").pop() - let configUrl - if (isMultiDim) { - const mdimConfigUrl = `${MULTI_DIM_DYNAMIC_CONFIG_URL}/${slug}.json` - const mdimJsonConfig = await fetch(mdimConfigUrl).then((res) => - res.json() - ) - const mdimConfig = - MultiDimDataPageConfig.fromObject(mdimJsonConfig) - const dimensions = extractMultiDimChoicesFromQueryStr( - url.search, - mdimConfig - ) - const view = mdimConfig.findViewByDimensions(dimensions) - if (!view) { - throw new Error( - `No view found for dimensions ${JSON.stringify( - dimensions - )}` - ) - } - configUrl = `${GRAPHER_DYNAMIC_CONFIG_URL}/by-uuid/${view.fullConfigId}.config.json` - } else { - configUrl = `${GRAPHER_DYNAMIC_CONFIG_URL}/${slug}.config.json` - } - const grapherPageConfig = await fetch(configUrl).then((res) => - res.json() - ) + const grapherPageConfig = deserializeJSONFromHTML(html) + // const url = new URL(fullUrl) + // const slug = url.pathname.split("/").pop() + // let configUrl + // if (isMultiDim) { + // const mdimConfigUrl = `${MULTI_DIM_DYNAMIC_CONFIG_URL}/${slug}.json` + // const mdimJsonConfig = await fetch(mdimConfigUrl).then((res) => + // res.json() + // ) + // const mdimConfig = + // MultiDimDataPageConfig.fromObject(mdimJsonConfig) + // const dimensions = extractMultiDimChoicesFromQueryStr( + // url.search, + // mdimConfig + // ) + // const view = mdimConfig.findViewByDimensions(dimensions) + // if (!view) { + // throw new Error( + // `No view found for dimensions ${JSON.stringify( + // dimensions + // )}` + // ) + // } + // configUrl = `${GRAPHER_DYNAMIC_CONFIG_URL}/by-uuid/${view.fullConfigId}.config.json` + // } else { + // configUrl = `${GRAPHER_DYNAMIC_CONFIG_URL}/${slug}.config.json` + // } + // const grapherPageConfig = await fetch(configUrl).then((res) => + // res.json() + // ) const figureConfigAttr = figure.getAttribute( GRAPHER_EMBEDDED_FIGURE_CONFIG_ATTR