Skip to content

Commit

Permalink
🎉 (entity selector) sort by external indicators (#3466)
Browse files Browse the repository at this point in the history
[Cycle 2024.2: Entity selector](#3349) | [Designs](https://www.figma.com/file/X5mOEX8zULS6qyHocUYdmh/Grapher-UI?type=design&node-id=2523%3A6266&mode=design&t=7edFp79OOjz6RENz-1)

## Summary

Offers to sort by "Population" and "GDP per capita", even if the chart doesn't include population or GDP per capita indicators.

## Details

- If the chart has a Population or GDP per capita indicator, then we re-use that data
- If we need to download additional indicators, then we do that on demand, i.e. selecting "Population" or "GDP per capita" triggers their download
- Indicator IDs for population and GDP per capita are hard-coded (but the data team might come up with a [better solution](owid/etl#2508)) 
- "Population" and "GDP per capita" are only offered for selection if entities are detected to include countries or regions
    - This is done by checking whether any of the available entities are listed in the [regions.json](https://github.com/owid/owid-grapher/blob/master/packages/%40ourworldindata/utils/src/regions.json) file
    - Testing the available entities against the `regions.json` file is not perfect since the default population and GDP per capita indicators that we are using have data for a few entities that are not listed in `regions.json` (see details below)
    - However, we only need a single matching entity to trigger sorting by population or GDP per capita, so in practice this works well
    - If we wanted to be more correct here, we could also download population and GDP per capita metadata when the entity selector is opened and then check the actual population/GDP per capita entities against the entities that are available for the chart

<details><summary>Entities of the population or GDP per capita indicator that are not included in the `regions.json` file</summary>

- For the population indicator:
      - Africa (UN)
      - Asia (UN)
      - Europe (UN)
      - High-income countries
      - Latin America and the Caribbean (UN)
      - Low-income countries
      - Lower-middle-income countries
      - Northern America (UN)
      - Oceania (UN)
      - Upper-middle-income countries
- For the GDP per capita indicator:
      - East Asia and Pacific (WB)
      - Europe and Central Asia (WB)
      - High-income countries
      - Latin America and Caribbean (WB)
      - Low-income countries
      - Lower-middle-income countries
      - Middle East and North Africa (WB)
      - Middle-income countries
      - North America (WB)
      - South Asia (WB)
      - Sub-Saharan Africa (WB)
      - Upper-middle-income countries

</details> 

## SVG tester

The SVG tester fails due to the changes in #3373
  • Loading branch information
sophiamersmann authored May 3, 2024
1 parent a6e32bf commit e186b8e
Show file tree
Hide file tree
Showing 8 changed files with 372 additions and 72 deletions.
11 changes: 8 additions & 3 deletions adminSiteClient/EditorBasicTab.tsx
Original file line number Diff line number Diff line change
Expand Up @@ -13,7 +13,12 @@ import {
EntitySelectionMode,
StackMode,
} from "@ourworldindata/types"
import { DimensionSlot, WorldEntityName } from "@ourworldindata/grapher"
import {
DimensionSlot,
WorldEntityName,
CONTINENTS_INDICATOR_ID,
POPULATION_INDICATOR_ID_USED_IN_ADMIN,
} from "@ourworldindata/grapher"
import {
DimensionProperty,
moveArrayItemToIndex,
Expand Down Expand Up @@ -317,7 +322,7 @@ export class EditorBasicTab extends React.Component<{ editor: ChartEditor }> {
)
if (!hasColor)
grapher.addDimension({
variableId: 123, // "Countries Continents"
variableId: CONTINENTS_INDICATOR_ID,
property: DimensionProperty.color,
})
}
Expand All @@ -329,7 +334,7 @@ export class EditorBasicTab extends React.Component<{ editor: ChartEditor }> {
)
if (!hasSize)
grapher.addDimension({
variableId: 597929, // "Population (various sources, 2023.1)"
variableId: POPULATION_INDICATOR_ID_USED_IN_ADMIN,
property: DimensionProperty.size,
})
}
Expand Down
32 changes: 6 additions & 26 deletions packages/@ourworldindata/grapher/src/core/Grapher.tsx
Original file line number Diff line number Diff line change
Expand Up @@ -118,8 +118,6 @@ import {
DEFAULT_GRAPHER_CONFIG_SCHEMA,
DEFAULT_GRAPHER_WIDTH,
DEFAULT_GRAPHER_HEIGHT,
getVariableDataRoute,
getVariableMetadataRoute,
DEFAULT_GRAPHER_FRAME_PADDING,
DEFAULT_GRAPHER_ENTITY_TYPE,
DEFAULT_GRAPHER_ENTITY_TYPE_PLURAL,
Expand All @@ -128,7 +126,10 @@ import {
GRAPHER_LIGHT_TEXT,
GRAPHER_LOADED_EVENT_NAME,
GRAPHER_DRAWER_ID,
isContinentsVariableId,
isPopulationVariableId,
} from "../core/GrapherConstants"
import { loadVariableDataAndMetadata } from "./loadVariable"
import Cookies from "js-cookie"
import {
ChartDimension,
Expand Down Expand Up @@ -253,21 +254,9 @@ async function loadVariablesDataSite(
variableIds: number[],
dataApiUrl: string
): Promise<MultipleOwidVariableDataDimensionsMap> {
const loadVariableDataPromises = variableIds.map(async (variableId) => {
const dataPromise = fetch(getVariableDataRoute(dataApiUrl, variableId))
const metadataPromise = fetch(
getVariableMetadataRoute(dataApiUrl, variableId)
)
const [dataResponse, metadataResponse] = await Promise.all([
dataPromise,
metadataPromise,
])
if (!dataResponse.ok) throw new Error(dataResponse.statusText)
if (!metadataResponse.ok) throw new Error(metadataResponse.statusText)
const data = await dataResponse.json()
const metadata = await metadataResponse.json()
return { data, metadata }
})
const loadVariableDataPromises = variableIds.map((variableId) =>
loadVariableDataAndMetadata(variableId, dataApiUrl)
)
const variablesData: OwidVariableDataMetadataDimensions[] =
await Promise.all(loadVariableDataPromises)
const variablesDataMap = new Map(
Expand Down Expand Up @@ -1577,22 +1566,13 @@ export class Grapher
this
const columnSlugs: string[] = []

// "Countries Continent"
const isContinentsVariableId = (id: string): boolean => id === "123"

// exclude "Countries Continent" if it's used as the color dimension in a scatter plot, slope chart etc.
if (
colorColumnSlug !== undefined &&
!isContinentsVariableId(colorColumnSlug)
)
columnSlugs.push(colorColumnSlug)

const isPopulationVariableId = (id: string): boolean =>
id === "525709" || // "Population (historical + projections), Gapminder, HYDE & UN"
id === "525711" || // "Population (historical estimates), Gapminder, HYDE & UN"
id === "597929" || // "Population (various sources, 2023.1)"
id === "597930" // "Population (various sources, 2023.1)"

if (xColumnSlug !== undefined) {
// exclude population variable if it's used as the x dimension in a marimekko
if (!isMarimekko || !isPopulationVariableId(xColumnSlug))
Expand Down
38 changes: 16 additions & 22 deletions packages/@ourworldindata/grapher/src/core/GrapherConstants.ts
Original file line number Diff line number Diff line change
Expand Up @@ -53,28 +53,22 @@ export const ThereWasAProblemLoadingThisChart = `There was a problem loading thi

export const WorldEntityName = "World"

export const getVariableDataRoute = (
dataApiUrl: string,
variableId: number
): string => {
if (dataApiUrl.includes("v1/indicators/")) {
// fetching from Data API, e.g. https://api.ourworldindata.org/v1/indicators/123.data.json
return `${dataApiUrl}${variableId}.data.json`
} else {
throw new Error(`dataApiUrl format not supported: ${dataApiUrl}`)
}
}

export const getVariableMetadataRoute = (
dataApiUrl: string,
variableId: number
): string => {
if (dataApiUrl.includes("v1/indicators/")) {
// fetching from Data API, e.g. https://api.ourworldindata.org/v1/indicators/123.metadata.json
return `${dataApiUrl}${variableId}.metadata.json`
} else {
throw new Error(`dataApiUrl format not supported: ${dataApiUrl}`)
}
export const CONTINENTS_INDICATOR_ID = 123 // "Countries Continent"
export const POPULATION_INDICATOR_ID_USED_IN_ADMIN = 597929 // "Population (various sources, 2023.1)"
export const POPULATION_INDICATOR_ID_USED_IN_ENTITY_SELECTOR = 597930 // "Population (various sources, 2023.1)"
export const GDP_PER_CAPITA_INDICATOR_ID_USED_IN_ENTITY_SELECTOR = 735665 // "World Development Indicators - World Bank (2023.05.11)"

export const isContinentsVariableId = (id: string | number): boolean =>
id.toString() === CONTINENTS_INDICATOR_ID.toString()

export const isPopulationVariableId = (id: string | number): boolean => {
const idString = id.toString()
return (
idString === "525709" || // "Population (historical + projections), Gapminder, HYDE & UN"
idString === "525711" || // "Population (historical estimates), Gapminder, HYDE & UN"
idString === "597929" || // "Population (various sources, 2023.1)"
idString === "597930" // "Population (various sources, 2023.1)"
)
}

export enum Patterns {
Expand Down
78 changes: 77 additions & 1 deletion packages/@ourworldindata/grapher/src/core/LegacyToOwidTable.ts
Original file line number Diff line number Diff line change
Expand Up @@ -9,6 +9,7 @@ import {
OwidColumnDef,
LegacyGrapherInterface,
OwidVariableDimensions,
OwidVariableDataMetadataDimensions,
} from "@ourworldindata/types"
import {
OwidTable,
Expand Down Expand Up @@ -37,6 +38,7 @@ import {
OwidChartDimensionInterface,
OwidVariableType,
} from "@ourworldindata/utils"
import { isContinentsVariableId } from "./GrapherConstants"

export const legacyToOwidTableAndDimensions = (
json: MultipleOwidVariableDataDimensionsMap,
Expand Down Expand Up @@ -613,7 +615,7 @@ const columnDefFromOwidVariable = (
} = variable

// Without this the much used var 123 appears as "Countries Continent". We could rename in Grapher but not sure the effects of that.
const isContinent = variable.id === 123
const isContinent = isContinentsVariableId(variable.id)
const name = isContinent ? "Continent" : variable.name

// The column's type
Expand Down Expand Up @@ -738,3 +740,77 @@ const annotationsToMap = (annotations: string): Map<string, string> => {
})
return entityAnnotationsMap
}

/**
* Loads a single variable into an OwidTable.
*/
export function buildVariableTable(
variable: OwidVariableDataMetadataDimensions
): OwidTable {
const entityMeta = variable.metadata.dimensions.entities.values
const entityMetaById: OwidEntityKey = Object.fromEntries(
entityMeta.map((entity) => [entity.id.toString(), entity])
)

// Base column defs, present in all OwidTables
const baseColumnDefs: Map<ColumnSlug, CoreColumnDef> = new Map(
StandardOwidColumnDefs.map((def) => [def.slug, def])
)

const columnDefs = new Map(baseColumnDefs)

// Time column
const timeColumnDef = timeColumnDefFromOwidVariable(variable.metadata)
columnDefs.set(timeColumnDef.slug, timeColumnDef)

// Value column
const valueColumnDef = columnDefFromOwidVariable(variable.metadata)
// Because database columns can contain mixed types, we want to avoid
// parsing for Grapher data until we fix that.
valueColumnDef.skipParsing = true
columnDefs.set(valueColumnDef.slug, valueColumnDef)

// Column values

const times = timeColumnValuesFromOwidVariable(
variable.metadata,
variable.data
)
const entityIds = variable.data.entities ?? []
const entityNames = entityIds.map(
// if entityMetaById[id] does not exist, then we don't have entity
// from variable metadata in MySQL. This can happen because we take
// data from S3 and metadata from MySQL. After we unify it, it should
// no longer be a problem
(id) => entityMetaById[id]?.name ?? id.toString()
)
// see comment above about entityMetaById[id]
const entityCodes = entityIds.map((id) => entityMetaById[id]?.code)

// If there is a conversionFactor, apply it.
let values = variable.data.values || []
const conversionFactor = valueColumnDef.display?.conversionFactor
if (conversionFactor !== undefined) {
values = values.map((value) =>
isNumber(value) ? value * conversionFactor : value
)

// If a non-int conversion factor is applied to an integer column,
// we end up with a numeric column.
if (
valueColumnDef.type === ColumnTypeNames.Integer &&
!isInteger(conversionFactor)
)
valueColumnDef.type = ColumnTypeNames.Numeric
}

const columnStore: { [key: string]: any[] } = {
[OwidTableSlugs.entityId]: entityIds,
[OwidTableSlugs.entityCode]: entityCodes,
[OwidTableSlugs.entityName]: entityNames,
[timeColumnDef.slug]: times,
[valueColumnDef.slug]: values,
}

return new OwidTable(columnStore, Array.from(columnDefs.values()))
}
44 changes: 44 additions & 0 deletions packages/@ourworldindata/grapher/src/core/loadVariable.ts
Original file line number Diff line number Diff line change
@@ -0,0 +1,44 @@
import { OwidVariableDataMetadataDimensions } from "@ourworldindata/types"

export const getVariableDataRoute = (
dataApiUrl: string,
variableId: number
): string => {
if (dataApiUrl.includes("v1/indicators/")) {
// fetching from Data API, e.g. https://api.ourworldindata.org/v1/indicators/123.data.json
return `${dataApiUrl}${variableId}.data.json`
} else {
throw new Error(`dataApiUrl format not supported: ${dataApiUrl}`)
}
}

export const getVariableMetadataRoute = (
dataApiUrl: string,
variableId: number
): string => {
if (dataApiUrl.includes("v1/indicators/")) {
// fetching from Data API, e.g. https://api.ourworldindata.org/v1/indicators/123.metadata.json
return `${dataApiUrl}${variableId}.metadata.json`
} else {
throw new Error(`dataApiUrl format not supported: ${dataApiUrl}`)
}
}

export async function loadVariableDataAndMetadata(
variableId: number,
dataApiUrl: string
): Promise<OwidVariableDataMetadataDimensions> {
const dataPromise = fetch(getVariableDataRoute(dataApiUrl, variableId))
const metadataPromise = fetch(
getVariableMetadataRoute(dataApiUrl, variableId)
)
const [dataResponse, metadataResponse] = await Promise.all([
dataPromise,
metadataPromise,
])
if (!dataResponse.ok) throw new Error(dataResponse.statusText)
if (!metadataResponse.ok) throw new Error(metadataResponse.statusText)
const data = await dataResponse.json()
const metadata = await metadataResponse.json()
return { data, metadata }
}
Loading

0 comments on commit e186b8e

Please sign in to comment.