From 8dafdefdfb83264f7a1d85c903c83a6cbb5f6849 Mon Sep 17 00:00:00 2001 From: Marcel Gerber Date: Mon, 25 Sep 2023 17:16:04 +0200 Subject: [PATCH 1/8] fix(algolia): index explorers even if there isn't any text --- baker/algolia/indexExplorersToAlgolia.ts | 7 ++++++- 1 file changed, 6 insertions(+), 1 deletion(-) diff --git a/baker/algolia/indexExplorersToAlgolia.ts b/baker/algolia/indexExplorersToAlgolia.ts index 160fafab58c..1cd7f91a494 100644 --- a/baker/algolia/indexExplorersToAlgolia.ts +++ b/baker/algolia/indexExplorersToAlgolia.ts @@ -107,9 +107,14 @@ const getExplorerRecords = async (): Promise => { [...uniqueTextTokens].join(" "), 1000 ) + + // In case we don't have any text for this explorer, we still want to index it + const textChunksForIteration = textChunks.length + ? textChunks + : [""] const explorerRecords = [] let i = 0 - for (const chunk of textChunks) { + for (const chunk of textChunksForIteration) { explorerRecords.push({ slug, title: getNullishJSONValueAsPlaintext(title), From 18521e6db5c14f971bf8aab1f6b82d9e4fc1fccc Mon Sep 17 00:00:00 2001 From: Marcel Gerber Date: Mon, 25 Sep 2023 17:18:13 +0200 Subject: [PATCH 2/8] enhance(algolia): extract text from `grapherId` explorers --- baker/algolia/indexExplorersToAlgolia.ts | 57 +++++++++++++++++++++--- 1 file changed, 51 insertions(+), 6 deletions(-) diff --git a/baker/algolia/indexExplorersToAlgolia.ts b/baker/algolia/indexExplorersToAlgolia.ts index 1cd7f91a494..49add49e324 100644 --- a/baker/algolia/indexExplorersToAlgolia.ts +++ b/baker/algolia/indexExplorersToAlgolia.ts @@ -1,13 +1,18 @@ import cheerio from "cheerio" import { isArray } from "lodash" import { match } from "ts-pattern" -import { checkIsPlainObjectWithGuard } from "@ourworldindata/utils" +import { + checkIsPlainObjectWithGuard, + identity, + keyBy, +} from "@ourworldindata/utils" import { getAlgoliaClient } from "./configureAlgolia.js" import * as db from "../../db/db.js" import { ALGOLIA_INDEXING } from "../../settings/serverSettings.js" import { Pageview } from "../../db/model/Pageview.js" import { chunkParagraphs } from "../chunk.js" import { SearchIndexName } from "../../site/search/searchTypes.js" +import { Chart } from "../../db/model/Chart.js" type ExplorerBlockColumns = { type: "columns" @@ -17,8 +22,9 @@ type ExplorerBlockColumns = { type ExplorerBlockGraphers = { type: "graphers" block: { - title: string + title?: string subtitle?: string + grapherId?: number }[] } @@ -38,7 +44,10 @@ type ExplorerRecord = { text: string } -function extractTextFromExplorer(blocksString: string): string { +function extractTextFromExplorer( + blocksString: string, + graphersUsedInExplorers: Record +): string { const blockText = new Set() const blocks = JSON.parse(blocksString) @@ -61,9 +70,28 @@ function extractTextFromExplorer(blocksString: string): string { { type: "graphers" }, (graphers: ExplorerBlockGraphers) => { graphers.block.forEach( - ({ title = "", subtitle = "" }) => { + ({ + title = "", + subtitle = "", + grapherId = undefined, + }) => { blockText.add(title) blockText.add(subtitle) + + if (grapherId !== undefined) { + const chartConfig = + graphersUsedInExplorers[grapherId] + ?.config + + if (chartConfig) { + blockText.add( + chartConfig.title ?? "" + ) + blockText.add( + chartConfig.subtitle ?? "" + ) + } + } } ) } @@ -76,7 +104,7 @@ function extractTextFromExplorer(blocksString: string): string { } } - return [...blockText].join(" ") + return [...blockText].filter(identity).join(" ") } function getNullishJSONValueAsPlaintext(value: string): string { @@ -86,6 +114,20 @@ function getNullishJSONValueAsPlaintext(value: string): string { const getExplorerRecords = async (): Promise => { const pageviews = await Pageview.getViewsByUrlObj() + // Fetch info about all charts used in explorers, as linked by the explorer_charts table + const graphersUsedInExplorers = await db + .queryMysql( + ` + SELECT DISTINCT chartId + FROM explorer_charts + ` + ) + .then((results: { chartId: number }[]) => + results.map(({ chartId }) => chartId) + ) + .then((ids) => Promise.all(ids.map((id) => Chart.findOneBy({ id })))) + .then((charts) => keyBy(charts, "id")) + const explorerRecords = await db .queryMysql( ` @@ -99,7 +141,10 @@ const getExplorerRecords = async (): Promise => { ) .then((results: ExplorerEntry[]) => results.flatMap(({ slug, title, subtitle, blocks }) => { - const textFromExplorer = extractTextFromExplorer(blocks) + const textFromExplorer = extractTextFromExplorer( + blocks, + graphersUsedInExplorers + ) const uniqueTextTokens = new Set([ ...textFromExplorer.split(" "), ]) From fa563dcc043fb37b8c22746ce04b82d8d6a26d5b Mon Sep 17 00:00:00 2001 From: Marcel Gerber Date: Mon, 25 Sep 2023 18:33:10 +0200 Subject: [PATCH 3/8] enhance(explorer): always show an explicit "sort by name" option in explorer entity picker --- explorer/Explorer.tsx | 5 +++- .../controls/entityPicker/EntityPicker.tsx | 23 ++++++++++++------- 2 files changed, 19 insertions(+), 9 deletions(-) diff --git a/explorer/Explorer.tsx b/explorer/Explorer.tsx index 155a5ba4b86..d5adebf26c6 100644 --- a/explorer/Explorer.tsx +++ b/explorer/Explorer.tsx @@ -7,6 +7,7 @@ import { extractPotentialDataSlugsFromTransform, OwidColumnDef, OwidTable, + OwidTableSlugs, SortOrder, TableSlug, } from "@ourworldindata/core-table" @@ -1036,7 +1037,9 @@ export class Explorer ]) return allColumnDefs.filter( (def) => - def.type === undefined || !discardColumnTypes.has(def.type) + (def.type === undefined || + !discardColumnTypes.has(def.type)) && + def.slug !== undefined ) } } diff --git a/packages/@ourworldindata/grapher/src/controls/entityPicker/EntityPicker.tsx b/packages/@ourworldindata/grapher/src/controls/entityPicker/EntityPicker.tsx index ee1a221c194..bc73ee45e76 100644 --- a/packages/@ourworldindata/grapher/src/controls/entityPicker/EntityPicker.tsx +++ b/packages/@ourworldindata/grapher/src/controls/entityPicker/EntityPicker.tsx @@ -21,6 +21,8 @@ import { getUserCountryInformation, regions, sortBy, + upperFirst, + compact, } from "@ourworldindata/utils" import { VerticalScrollContainer } from "../../controls/VerticalScrollContainer" import { SortIcon } from "../../controls/SortIcon" @@ -120,8 +122,16 @@ export class EntityPicker extends React.Component<{ label: string value: string | undefined }[] { - return [ + const entityNameColumn = this.grapherTable?.entityNameColumn + const entityNameColumnInPickerColumnDefs = !!this.pickerColumnDefs.find( + (col) => col.slug === entityNameColumn?.slug + ) + return compact([ { label: "Relevance", value: undefined }, + !entityNameColumnInPickerColumnDefs && { + label: upperFirst(this.manager.entityType) ?? "Name", + value: entityNameColumn?.slug, + }, ...this.pickerColumnDefs.map( ( col @@ -135,11 +145,13 @@ export class EntityPicker extends React.Component<{ } } ), - ] + ]) } private getColumn(slug: ColumnSlug | undefined): CoreColumn | undefined { if (slug === undefined) return undefined + if (slug === OwidTableSlugs.entityName) + return this.manager.grapherTable?.entityNameColumn return this.manager.entityPickerTable?.get(slug) } @@ -461,12 +473,7 @@ export class EntityPicker extends React.Component<{ } private get pickerMenu(): JSX.Element | null { - if ( - this.isDropdownMenu || - !this.manager.entityPickerColumnDefs || - this.manager.entityPickerColumnDefs.length === 0 - ) - return null + if (this.isDropdownMenu) return null return (
Sort by From d4a9e98c704b493e6b454b40fe67f982e8d93c29 Mon Sep 17 00:00:00 2001 From: Marcel Gerber Date: Mon, 25 Sep 2023 18:37:00 +0200 Subject: [PATCH 4/8] style: remove unused import --- explorer/Explorer.tsx | 1 - 1 file changed, 1 deletion(-) diff --git a/explorer/Explorer.tsx b/explorer/Explorer.tsx index d5adebf26c6..0c148acb3e1 100644 --- a/explorer/Explorer.tsx +++ b/explorer/Explorer.tsx @@ -7,7 +7,6 @@ import { extractPotentialDataSlugsFromTransform, OwidColumnDef, OwidTable, - OwidTableSlugs, SortOrder, TableSlug, } from "@ourworldindata/core-table" From 860e0c743d4ad09ff6fe2578f115808f22e46ced Mon Sep 17 00:00:00 2001 From: Ike Saunders Date: Mon, 25 Sep 2023 19:04:04 +0000 Subject: [PATCH 5/8] =?UTF-8?q?=E2=9C=A8=20Correct=20typography=20for=20R&?= =?UTF-8?q?W=20headings?= MIME-Version: 1.0 Content-Type: text/plain; charset=UTF-8 Content-Transfer-Encoding: 8bit --- site/gdocs/ResearchAndWriting.scss | 3 ++- site/gdocs/ResearchAndWriting.tsx | 9 ++++++++- 2 files changed, 10 insertions(+), 2 deletions(-) diff --git a/site/gdocs/ResearchAndWriting.scss b/site/gdocs/ResearchAndWriting.scss index b037fa47bd4..86fe3380735 100644 --- a/site/gdocs/ResearchAndWriting.scss +++ b/site/gdocs/ResearchAndWriting.scss @@ -40,15 +40,16 @@ } .research-and-writing-link { + h2, h3 { @include owid-link-90; - @include h2-bold; margin-top: 16px; margin-bottom: 8px; text-decoration: none; } &:hover { + h2, h3 { text-decoration: underline; } diff --git a/site/gdocs/ResearchAndWriting.tsx b/site/gdocs/ResearchAndWriting.tsx index 1f740bab7f9..cfec6508461 100644 --- a/site/gdocs/ResearchAndWriting.tsx +++ b/site/gdocs/ResearchAndWriting.tsx @@ -49,6 +49,12 @@ function ResearchAndWritingLinkContainer( filename = linkedDocument.content["featured-image"] || filename } + const heading = React.createElement( + isSmall ? "h3" : "h2", + { className: isSmall ? "h3-bold" : "h2-bold" }, + title + ) + return ( ) : null} -

{title}

+ {heading} {subtitle && !shouldHideSubtitle ? (

{subtitle} @@ -135,6 +141,7 @@ export function ResearchAndWriting(props: ResearchAndWritingProps) { Date: Tue, 26 Sep 2023 09:44:25 +0200 Subject: [PATCH 6/8] refactor: use map instead of for-loop --- baker/algolia/indexExplorersToAlgolia.ts | 24 +++++++++--------------- 1 file changed, 9 insertions(+), 15 deletions(-) diff --git a/baker/algolia/indexExplorersToAlgolia.ts b/baker/algolia/indexExplorersToAlgolia.ts index 49add49e324..812d9486086 100644 --- a/baker/algolia/indexExplorersToAlgolia.ts +++ b/baker/algolia/indexExplorersToAlgolia.ts @@ -157,21 +157,15 @@ const getExplorerRecords = async (): Promise => { const textChunksForIteration = textChunks.length ? textChunks : [""] - const explorerRecords = [] - let i = 0 - for (const chunk of textChunksForIteration) { - explorerRecords.push({ - slug, - title: getNullishJSONValueAsPlaintext(title), - subtitle: getNullishJSONValueAsPlaintext(subtitle), - views_7d: - pageviews[`/explorers/${slug}`]?.views_7d || 0, - text: chunk, - objectID: `${slug}-${i}`, - }) - i++ - } - return explorerRecords + + return textChunksForIteration.map((chunk, i) => ({ + slug, + title: getNullishJSONValueAsPlaintext(title), + subtitle: getNullishJSONValueAsPlaintext(subtitle), + views_7d: pageviews[`/explorers/${slug}`]?.views_7d ?? 0, + text: chunk, + objectID: `${slug}-${i}`, + })) }) ) From f168b4b635990344775e5a055733378d27d583e0 Mon Sep 17 00:00:00 2001 From: Marcel Gerber Date: Tue, 26 Sep 2023 11:01:51 +0200 Subject: [PATCH 7/8] refactor: various fixes for entity name handling --- .../controls/entityPicker/EntityPicker.tsx | 36 +++++++++++-------- 1 file changed, 22 insertions(+), 14 deletions(-) diff --git a/packages/@ourworldindata/grapher/src/controls/entityPicker/EntityPicker.tsx b/packages/@ourworldindata/grapher/src/controls/entityPicker/EntityPicker.tsx index bc73ee45e76..98179241abd 100644 --- a/packages/@ourworldindata/grapher/src/controls/entityPicker/EntityPicker.tsx +++ b/packages/@ourworldindata/grapher/src/controls/entityPicker/EntityPicker.tsx @@ -128,10 +128,11 @@ export class EntityPicker extends React.Component<{ ) return compact([ { label: "Relevance", value: undefined }, - !entityNameColumnInPickerColumnDefs && { - label: upperFirst(this.manager.entityType) ?? "Name", - value: entityNameColumn?.slug, - }, + !entityNameColumnInPickerColumnDefs && + entityNameColumn && { + label: upperFirst(this.manager.entityType) || "Name", + value: entityNameColumn?.slug, + }, ...this.pickerColumnDefs.map( ( col @@ -148,15 +149,21 @@ export class EntityPicker extends React.Component<{ ]) } - private getColumn(slug: ColumnSlug | undefined): CoreColumn | undefined { - if (slug === undefined) return undefined - if (slug === OwidTableSlugs.entityName) - return this.manager.grapherTable?.entityNameColumn - return this.manager.entityPickerTable?.get(slug) + @computed private get metricTable(): OwidTable | undefined { + if (this.metric === undefined) return undefined + + // If the slug is "entityName", then try to get it from grapherTable first, because it might + // not be present in pickerTable (for indicator-powered explorers, for example). + if ( + this.metric === OwidTableSlugs.entityName && + this.grapherTable?.has(this.metric) + ) + return this.grapherTable + return this.pickerTable } @computed private get activePickerMetricColumn(): CoreColumn | undefined { - return this.getColumn(this.metric) + return this.metricTable?.get(this.metric) } @computed private get availableEntitiesForCurrentView(): Set { @@ -190,13 +197,13 @@ export class EntityPicker extends React.Component<{ @computed private get entitiesWithMetricValue(): EntityOptionWithMetricValue[] { - const { pickerTable, selection, localEntityNames } = this + const { metricTable, selection, localEntityNames } = this const col = this.activePickerMetricColumn const entityNames = selection.availableEntityNames.slice().sort() return entityNames.map((entityName) => { const plotValue = - col && pickerTable - ? (pickerTable.getLatestValueForEntity( + col && metricTable?.has(col.slug) + ? (metricTable.getLatestValueForEntity( entityName, col.slug ) as string | number) @@ -447,7 +454,7 @@ export class EntityPicker extends React.Component<{ } @action private updateMetric(columnSlug: ColumnSlug): void { - const col = this.getColumn(columnSlug) + const col = this.pickerTable?.get(columnSlug) this.manager.setEntityPicker?.({ metric: columnSlug, @@ -465,6 +472,7 @@ export class EntityPicker extends React.Component<{ return ( // If columnSlug is undefined, we're sorting by relevance, which is (mostly) by country name. columnSlug !== undefined && + columnSlug !== OwidTableSlugs.entityName && // If the column is currently missing (not loaded yet), assume it is numeric. (col === undefined || col.isMissing || From 027561d9366e8c2f7a56b9984deab03876b5a22f Mon Sep 17 00:00:00 2001 From: Marcel Gerber Date: Tue, 26 Sep 2023 11:02:49 +0200 Subject: [PATCH 8/8] enhance: eagerly provide current grapher table as pickerTable so sorting by entity name immediately infers the correct alphabetical sort order --- explorer/Explorer.tsx | 16 ++++++++-------- 1 file changed, 8 insertions(+), 8 deletions(-) diff --git a/explorer/Explorer.tsx b/explorer/Explorer.tsx index 0c148acb3e1..4ae8052105d 100644 --- a/explorer/Explorer.tsx +++ b/explorer/Explorer.tsx @@ -952,14 +952,14 @@ export class Explorer }) private updateEntityPickerTable(): void { - if (this.entityPickerMetric) { - this.entityPickerTableIsLoading = true - this.futureEntityPickerTable.set( - this.tableLoader.get( - this.getTableSlugOfColumnSlug(this.entityPickerMetric) - ) - ) - } + // If we don't currently have a entity picker metric, then set pickerTable to the currently-used table anyways, + // so that when we start sorting by entity name we can infer that the column is a string column immediately. + const tableSlugToLoad = this.entityPickerMetric + ? this.getTableSlugOfColumnSlug(this.entityPickerMetric) + : this.explorerProgram.grapherConfig.tableSlug + + this.entityPickerTableIsLoading = true + this.futureEntityPickerTable.set(this.tableLoader.get(tableSlugToLoad)) } setEntityPicker({