@buildcanada/charts 0.1.0

This diff represents the content of publicly available package versions that have been released to one of the supported registries. The information contained in this diff is provided for informational purposes only and reflects changes between package versions as they appear in their respective public registries.
Files changed (404) hide show
  1. package/LICENSE.md +8 -0
  2. package/README.md +113 -0
  3. package/package.json +137 -0
  4. package/src/components/BodyPortal/BodyPortal.tsx +40 -0
  5. package/src/components/Button/Button.scss +110 -0
  6. package/src/components/Button/Button.tsx +101 -0
  7. package/src/components/Checkbox.scss +93 -0
  8. package/src/components/Checkbox.tsx +47 -0
  9. package/src/components/ExpandableToggle/ExpandableToggle.scss +123 -0
  10. package/src/components/ExpandableToggle/ExpandableToggle.tsx +60 -0
  11. package/src/components/GrapherTabIcon.tsx +156 -0
  12. package/src/components/GrapherTrendArrow.scss +16 -0
  13. package/src/components/GrapherTrendArrow.tsx +30 -0
  14. package/src/components/Halo/Halo.tsx +44 -0
  15. package/src/components/LabeledSwitch/LabeledSwitch.scss +109 -0
  16. package/src/components/LabeledSwitch/LabeledSwitch.tsx +62 -0
  17. package/src/components/MarkdownTextWrap/MarkdownTextWrap.tsx +1173 -0
  18. package/src/components/OverlayHeader.scss +18 -0
  19. package/src/components/OverlayHeader.tsx +29 -0
  20. package/src/components/RadioButton.scss +69 -0
  21. package/src/components/RadioButton.tsx +42 -0
  22. package/src/components/SimpleMarkdownText.tsx +89 -0
  23. package/src/components/TextInput.scss +17 -0
  24. package/src/components/TextInput.tsx +19 -0
  25. package/src/components/TextWrap/TextWrap.tsx +361 -0
  26. package/src/components/TextWrap/TextWrapUtils.ts +32 -0
  27. package/src/components/closeButton/CloseButton.scss +40 -0
  28. package/src/components/closeButton/CloseButton.tsx +27 -0
  29. package/src/components/index.ts +70 -0
  30. package/src/components/loadingIndicator/LoadingIndicator.scss +40 -0
  31. package/src/components/loadingIndicator/LoadingIndicator.tsx +28 -0
  32. package/src/components/markdown/remarkPlainLinks.ts +36 -0
  33. package/src/components/reactUtil.ts +20 -0
  34. package/src/components/stubs/CodeSnippet.tsx +19 -0
  35. package/src/components/stubs/DataCitation.tsx +16 -0
  36. package/src/components/stubs/IndicatorKeyData.tsx +45 -0
  37. package/src/components/stubs/IndicatorProcessing.tsx +15 -0
  38. package/src/components/stubs/IndicatorSources.tsx +15 -0
  39. package/src/components/styles/colors.scss +113 -0
  40. package/src/components/styles/mixins.scss +630 -0
  41. package/src/components/styles/typography.scss +579 -0
  42. package/src/components/styles/util.scss +89 -0
  43. package/src/components/styles/variables.scss +208 -0
  44. package/src/config/ChartsConfig.ts +163 -0
  45. package/src/config/ChartsProvider.tsx +157 -0
  46. package/src/config/index.ts +20 -0
  47. package/src/core-table/CoreTable.ts +1355 -0
  48. package/src/core-table/CoreTableColumns.ts +973 -0
  49. package/src/core-table/CoreTableUtils.ts +793 -0
  50. package/src/core-table/ErrorValues.ts +73 -0
  51. package/src/core-table/OwidTable.ts +1175 -0
  52. package/src/core-table/OwidTableSynthesizers.ts +272 -0
  53. package/src/core-table/OwidTableUtil.ts +76 -0
  54. package/src/core-table/Transforms.ts +484 -0
  55. package/src/core-table/index.ts +82 -0
  56. package/src/explorer/ColumnGrammar.ts +217 -0
  57. package/src/explorer/Explorer.sample.ts +212 -0
  58. package/src/explorer/Explorer.scss +148 -0
  59. package/src/explorer/Explorer.tsx +1283 -0
  60. package/src/explorer/ExplorerConstants.ts +85 -0
  61. package/src/explorer/ExplorerControls.scss +156 -0
  62. package/src/explorer/ExplorerControls.tsx +210 -0
  63. package/src/explorer/ExplorerDecisionMatrix.ts +471 -0
  64. package/src/explorer/ExplorerGrammar.ts +161 -0
  65. package/src/explorer/ExplorerProgram.ts +568 -0
  66. package/src/explorer/ExplorerUtils.ts +59 -0
  67. package/src/explorer/GrapherGrammar.ts +387 -0
  68. package/src/explorer/gridLang/GrammarUtils.ts +121 -0
  69. package/src/explorer/gridLang/GridCell.ts +298 -0
  70. package/src/explorer/gridLang/GridLangConstants.ts +255 -0
  71. package/src/explorer/gridLang/GridProgram.ts +311 -0
  72. package/src/explorer/gridLang/readme.md +17 -0
  73. package/src/explorer/index.ts +69 -0
  74. package/src/explorer/readme.md +19 -0
  75. package/src/explorer/urlMigrations/CO2UrlMigration.ts +46 -0
  76. package/src/explorer/urlMigrations/CovidUrlMigration.ts +37 -0
  77. package/src/explorer/urlMigrations/EnergyUrlMigration.ts +41 -0
  78. package/src/explorer/urlMigrations/ExplorerPageUrlMigrationSpec.ts +12 -0
  79. package/src/explorer/urlMigrations/ExplorerUrlMigrationUtils.ts +45 -0
  80. package/src/explorer/urlMigrations/ExplorerUrlMigrations.ts +33 -0
  81. package/src/explorer/urlMigrations/LegacyCovidUrlMigration.ts +144 -0
  82. package/src/explorer/urlMigrations/readme.md +39 -0
  83. package/src/grapher/axis/Axis.ts +973 -0
  84. package/src/grapher/axis/AxisConfig.ts +179 -0
  85. package/src/grapher/axis/AxisViews.tsx +597 -0
  86. package/src/grapher/barCharts/DiscreteBarChart.tsx +728 -0
  87. package/src/grapher/barCharts/DiscreteBarChartConstants.ts +60 -0
  88. package/src/grapher/barCharts/DiscreteBarChartHelpers.ts +338 -0
  89. package/src/grapher/barCharts/DiscreteBarChartState.ts +354 -0
  90. package/src/grapher/barCharts/DiscreteBarChartThumbnail.tsx +34 -0
  91. package/src/grapher/captionedChart/CaptionedChart.scss +61 -0
  92. package/src/grapher/captionedChart/CaptionedChart.tsx +523 -0
  93. package/src/grapher/captionedChart/Logos.tsx +141 -0
  94. package/src/grapher/captionedChart/LogosSVG.tsx +16 -0
  95. package/src/grapher/captionedChart/StaticChartRasterizer.tsx +178 -0
  96. package/src/grapher/captionedChart/assets/buildcanada-logo-square.svg +15 -0
  97. package/src/grapher/captionedChart/assets/buildcanada-logo.svg +15 -0
  98. package/src/grapher/captionedChart/assets/canadaspends.svg +7 -0
  99. package/src/grapher/captionedChart/readme.md +14 -0
  100. package/src/grapher/chart/Chart.tsx +62 -0
  101. package/src/grapher/chart/ChartAreaContent.tsx +172 -0
  102. package/src/grapher/chart/ChartDimension.ts +121 -0
  103. package/src/grapher/chart/ChartInterface.ts +83 -0
  104. package/src/grapher/chart/ChartManager.ts +113 -0
  105. package/src/grapher/chart/ChartTabs.ts +178 -0
  106. package/src/grapher/chart/ChartTypeMap.tsx +158 -0
  107. package/src/grapher/chart/ChartTypeSwitcher.tsx +26 -0
  108. package/src/grapher/chart/ChartUtils.tsx +364 -0
  109. package/src/grapher/chart/DimensionSlot.ts +45 -0
  110. package/src/grapher/chart/StaticChartWrapper.tsx +94 -0
  111. package/src/grapher/chart/guidedChartUtils.ts +82 -0
  112. package/src/grapher/color/BinningStrategies.ts +484 -0
  113. package/src/grapher/color/BinningStrategyEqualSizeBins.ts +132 -0
  114. package/src/grapher/color/BinningStrategyLogarithmic.ts +121 -0
  115. package/src/grapher/color/CategoricalColorAssigner.ts +97 -0
  116. package/src/grapher/color/ColorBrewerSchemes.ts +80 -0
  117. package/src/grapher/color/ColorConstants.ts +20 -0
  118. package/src/grapher/color/ColorScale.ts +339 -0
  119. package/src/grapher/color/ColorScaleBin.ts +147 -0
  120. package/src/grapher/color/ColorScaleConfig.ts +204 -0
  121. package/src/grapher/color/ColorScheme.ts +137 -0
  122. package/src/grapher/color/ColorSchemes.ts +149 -0
  123. package/src/grapher/color/ColorUtils.ts +86 -0
  124. package/src/grapher/color/CustomSchemes.ts +1772 -0
  125. package/src/grapher/color/readme.md +84 -0
  126. package/src/grapher/comparisonLine/ComparisonLine.tsx +31 -0
  127. package/src/grapher/comparisonLine/ComparisonLineConstants.ts +11 -0
  128. package/src/grapher/comparisonLine/ComparisonLineGenerator.ts +60 -0
  129. package/src/grapher/comparisonLine/ComparisonLineHelpers.ts +10 -0
  130. package/src/grapher/comparisonLine/CustomComparisonLine.tsx +159 -0
  131. package/src/grapher/comparisonLine/VerticalComparisonLine.tsx +208 -0
  132. package/src/grapher/controls/ActionButtons.scss +97 -0
  133. package/src/grapher/controls/ActionButtons.tsx +453 -0
  134. package/src/grapher/controls/CommandPalette.scss +50 -0
  135. package/src/grapher/controls/CommandPalette.tsx +74 -0
  136. package/src/grapher/controls/ContentSwitchers.scss +93 -0
  137. package/src/grapher/controls/ContentSwitchers.tsx +238 -0
  138. package/src/grapher/controls/Controls.scss +158 -0
  139. package/src/grapher/controls/DataTableFilterDropdown.scss +7 -0
  140. package/src/grapher/controls/DataTableFilterDropdown.tsx +168 -0
  141. package/src/grapher/controls/DataTableSearchField.scss +3 -0
  142. package/src/grapher/controls/DataTableSearchField.tsx +76 -0
  143. package/src/grapher/controls/Dropdown.scss +252 -0
  144. package/src/grapher/controls/Dropdown.tsx +235 -0
  145. package/src/grapher/controls/EntitySelectionToggle.tsx +135 -0
  146. package/src/grapher/controls/MapRegionDropdown.scss +3 -0
  147. package/src/grapher/controls/MapRegionDropdown.tsx +104 -0
  148. package/src/grapher/controls/MapResetButton.tsx +115 -0
  149. package/src/grapher/controls/MapZoomDropdown.scss +9 -0
  150. package/src/grapher/controls/MapZoomDropdown.tsx +270 -0
  151. package/src/grapher/controls/MapZoomToSelectionButton.tsx +87 -0
  152. package/src/grapher/controls/SearchField.scss +78 -0
  153. package/src/grapher/controls/SearchField.tsx +63 -0
  154. package/src/grapher/controls/SettingsMenu.scss +191 -0
  155. package/src/grapher/controls/SettingsMenu.tsx +399 -0
  156. package/src/grapher/controls/ShareMenu.scss +58 -0
  157. package/src/grapher/controls/ShareMenu.tsx +304 -0
  158. package/src/grapher/controls/SortIcon.tsx +39 -0
  159. package/src/grapher/controls/VerticalScrollContainer.tsx +263 -0
  160. package/src/grapher/controls/controlsRow/ControlsRow.tsx +168 -0
  161. package/src/grapher/controls/dropdown-icons.scss +4 -0
  162. package/src/grapher/controls/entityPicker/EntityPicker.scss +255 -0
  163. package/src/grapher/controls/entityPicker/EntityPicker.tsx +816 -0
  164. package/src/grapher/controls/entityPicker/EntityPickerConstants.ts +23 -0
  165. package/src/grapher/controls/globalEntitySelector/GlobalEntitySelector.scss +129 -0
  166. package/src/grapher/controls/globalEntitySelector/GlobalEntitySelector.tsx +463 -0
  167. package/src/grapher/controls/globalEntitySelector/GlobalEntitySelectorConstants.ts +3 -0
  168. package/src/grapher/controls/globalEntitySelector/readme.md +17 -0
  169. package/src/grapher/controls/settings/AbsRelToggle.tsx +64 -0
  170. package/src/grapher/controls/settings/AxisScaleToggle.tsx +53 -0
  171. package/src/grapher/controls/settings/FacetStrategySelector.tsx +110 -0
  172. package/src/grapher/controls/settings/FacetYDomainToggle.tsx +51 -0
  173. package/src/grapher/controls/settings/NoDataAreaToggle.tsx +38 -0
  174. package/src/grapher/controls/settings/ZoomToggle.tsx +36 -0
  175. package/src/grapher/core/EntitiesByRegionType.ts +174 -0
  176. package/src/grapher/core/EntityCodes.ts +19 -0
  177. package/src/grapher/core/EntityUrlBuilder.ts +200 -0
  178. package/src/grapher/core/FetchingGrapher.tsx +156 -0
  179. package/src/grapher/core/Grapher.tsx +760 -0
  180. package/src/grapher/core/GrapherAnalytics.ts +229 -0
  181. package/src/grapher/core/GrapherConstants.ts +173 -0
  182. package/src/grapher/core/GrapherState.tsx +3659 -0
  183. package/src/grapher/core/GrapherUrl.ts +184 -0
  184. package/src/grapher/core/GrapherUrlMigrations.ts +29 -0
  185. package/src/grapher/core/GrapherUseHelpers.tsx +147 -0
  186. package/src/grapher/core/LegacyToOwidTable.ts +841 -0
  187. package/src/grapher/core/grapher.entry.ts +5 -0
  188. package/src/grapher/core/grapher.scss +257 -0
  189. package/src/grapher/core/loadGrapherTableHelpers.ts +116 -0
  190. package/src/grapher/core/loadVariable.ts +104 -0
  191. package/src/grapher/core/relatedQuestion.ts +12 -0
  192. package/src/grapher/core/typography.scss +206 -0
  193. package/src/grapher/dataTable/DataTable.sample.ts +206 -0
  194. package/src/grapher/dataTable/DataTable.scss +249 -0
  195. package/src/grapher/dataTable/DataTable.tsx +1332 -0
  196. package/src/grapher/dataTable/DataTableConstants.ts +186 -0
  197. package/src/grapher/entitySelector/EntitySelector.scss +255 -0
  198. package/src/grapher/entitySelector/EntitySelector.tsx +1838 -0
  199. package/src/grapher/facet/FacetChart.tsx +943 -0
  200. package/src/grapher/facet/FacetChartConstants.ts +24 -0
  201. package/src/grapher/facet/FacetChartUtils.ts +51 -0
  202. package/src/grapher/facet/FacetMap.tsx +604 -0
  203. package/src/grapher/facet/FacetMapConstants.ts +23 -0
  204. package/src/grapher/facet/readme.md +13 -0
  205. package/src/grapher/focus/FocusArray.ts +79 -0
  206. package/src/grapher/footer/Footer.scss +63 -0
  207. package/src/grapher/footer/Footer.tsx +809 -0
  208. package/src/grapher/footer/FooterManager.ts +44 -0
  209. package/src/grapher/fullScreen/FullScreen.scss +11 -0
  210. package/src/grapher/fullScreen/FullScreen.tsx +61 -0
  211. package/src/grapher/header/Header.scss +35 -0
  212. package/src/grapher/header/Header.tsx +372 -0
  213. package/src/grapher/header/HeaderManager.ts +28 -0
  214. package/src/grapher/index.ts +157 -0
  215. package/src/grapher/interaction/InteractionState.ts +60 -0
  216. package/src/grapher/legend/HorizontalColorLegends.tsx +923 -0
  217. package/src/grapher/legend/LegendInteractionState.ts +40 -0
  218. package/src/grapher/legend/VerticalColorLegend.tsx +295 -0
  219. package/src/grapher/lineCharts/LineChart.tsx +968 -0
  220. package/src/grapher/lineCharts/LineChartConstants.ts +89 -0
  221. package/src/grapher/lineCharts/LineChartHelpers.ts +184 -0
  222. package/src/grapher/lineCharts/LineChartState.ts +394 -0
  223. package/src/grapher/lineCharts/LineChartThumbnail.tsx +437 -0
  224. package/src/grapher/lineCharts/Lines.tsx +258 -0
  225. package/src/grapher/lineLegend/LineLegend.tsx +723 -0
  226. package/src/grapher/lineLegend/LineLegendConstants.ts +9 -0
  227. package/src/grapher/lineLegend/LineLegendFilterAlgorithms.ts +143 -0
  228. package/src/grapher/lineLegend/LineLegendHelpers.ts +253 -0
  229. package/src/grapher/lineLegend/LineLegendTypes.ts +32 -0
  230. package/src/grapher/mapCharts/CanadaTopology.ts +17922 -0
  231. package/src/grapher/mapCharts/ChoroplethGlobe.tsx +949 -0
  232. package/src/grapher/mapCharts/ChoroplethMap.tsx +662 -0
  233. package/src/grapher/mapCharts/GeoFeatures.ts +184 -0
  234. package/src/grapher/mapCharts/GlobeController.ts +496 -0
  235. package/src/grapher/mapCharts/MapAnnotationPlacements.json +1040 -0
  236. package/src/grapher/mapCharts/MapAnnotationPlacements.ts +31 -0
  237. package/src/grapher/mapCharts/MapAnnotations.ts +723 -0
  238. package/src/grapher/mapCharts/MapChart.sample.ts +59 -0
  239. package/src/grapher/mapCharts/MapChart.scss +5 -0
  240. package/src/grapher/mapCharts/MapChart.tsx +720 -0
  241. package/src/grapher/mapCharts/MapChartConstants.ts +260 -0
  242. package/src/grapher/mapCharts/MapChartState.ts +416 -0
  243. package/src/grapher/mapCharts/MapChartThumbnail.tsx +25 -0
  244. package/src/grapher/mapCharts/MapComponents.tsx +338 -0
  245. package/src/grapher/mapCharts/MapConfig.ts +156 -0
  246. package/src/grapher/mapCharts/MapHelpers.ts +181 -0
  247. package/src/grapher/mapCharts/MapProjections.ts +49 -0
  248. package/src/grapher/mapCharts/MapSparkline.tsx +257 -0
  249. package/src/grapher/mapCharts/MapTooltip.scss +49 -0
  250. package/src/grapher/mapCharts/MapTooltip.tsx +409 -0
  251. package/src/grapher/mapCharts/MapTopology.ts +1766 -0
  252. package/src/grapher/mapCharts/d3-bboxCollide.js +204 -0
  253. package/src/grapher/mapCharts/d3-geo-projection.ts +198 -0
  254. package/src/grapher/modal/DownloadIcons.tsx +39 -0
  255. package/src/grapher/modal/DownloadModal.scss +300 -0
  256. package/src/grapher/modal/DownloadModal.tsx +1226 -0
  257. package/src/grapher/modal/EmbedModal.scss +40 -0
  258. package/src/grapher/modal/EmbedModal.tsx +160 -0
  259. package/src/grapher/modal/EntitySelectorModal.tsx +59 -0
  260. package/src/grapher/modal/Modal.scss +31 -0
  261. package/src/grapher/modal/Modal.tsx +90 -0
  262. package/src/grapher/modal/ModalHeader.scss +12 -0
  263. package/src/grapher/modal/ModalHeader.tsx +16 -0
  264. package/src/grapher/modal/SourcesDescriptions.scss +87 -0
  265. package/src/grapher/modal/SourcesDescriptions.tsx +89 -0
  266. package/src/grapher/modal/SourcesKeyDataTable.scss +49 -0
  267. package/src/grapher/modal/SourcesKeyDataTable.tsx +87 -0
  268. package/src/grapher/modal/SourcesModal.scss +301 -0
  269. package/src/grapher/modal/SourcesModal.tsx +568 -0
  270. package/src/grapher/noDataModal/NoDataModal.tsx +125 -0
  271. package/src/grapher/scatterCharts/ConnectedScatterLegend.tsx +143 -0
  272. package/src/grapher/scatterCharts/MultiColorPolyline.tsx +129 -0
  273. package/src/grapher/scatterCharts/NoDataSection.scss +14 -0
  274. package/src/grapher/scatterCharts/NoDataSection.tsx +56 -0
  275. package/src/grapher/scatterCharts/ScatterPlotChart.tsx +792 -0
  276. package/src/grapher/scatterCharts/ScatterPlotChartConstants.ts +157 -0
  277. package/src/grapher/scatterCharts/ScatterPlotChartState.ts +678 -0
  278. package/src/grapher/scatterCharts/ScatterPlotChartThumbnail.tsx +155 -0
  279. package/src/grapher/scatterCharts/ScatterPlotTooltip.tsx +560 -0
  280. package/src/grapher/scatterCharts/ScatterPoints.tsx +153 -0
  281. package/src/grapher/scatterCharts/ScatterPointsWithLabels.tsx +708 -0
  282. package/src/grapher/scatterCharts/ScatterSizeLegend.tsx +327 -0
  283. package/src/grapher/scatterCharts/ScatterUtils.ts +265 -0
  284. package/src/grapher/scatterCharts/Triangle.tsx +41 -0
  285. package/src/grapher/schema/README.md +33 -0
  286. package/src/grapher/schema/defaultGrapherConfig.ts +100 -0
  287. package/src/grapher/schema/grapher-schema.009.yaml +781 -0
  288. package/src/grapher/schema/migrations/helpers.ts +58 -0
  289. package/src/grapher/schema/migrations/migrate.ts +75 -0
  290. package/src/grapher/schema/migrations/migrations.ts +158 -0
  291. package/src/grapher/selection/MapSelectionArray.ts +99 -0
  292. package/src/grapher/selection/SelectionArray.ts +71 -0
  293. package/src/grapher/selection/readme.md +16 -0
  294. package/src/grapher/sidePanel/SidePanel.scss +10 -0
  295. package/src/grapher/sidePanel/SidePanel.tsx +23 -0
  296. package/src/grapher/slideInDrawer/SlideInDrawer.scss +57 -0
  297. package/src/grapher/slideInDrawer/SlideInDrawer.tsx +125 -0
  298. package/src/grapher/slideshowController/SlideShowController.tsx +43 -0
  299. package/src/grapher/slideshowController/readme.md +7 -0
  300. package/src/grapher/slopeCharts/MarkX.tsx +45 -0
  301. package/src/grapher/slopeCharts/Slope.tsx +102 -0
  302. package/src/grapher/slopeCharts/SlopeChart.tsx +1152 -0
  303. package/src/grapher/slopeCharts/SlopeChartConstants.ts +33 -0
  304. package/src/grapher/slopeCharts/SlopeChartHelpers.ts +73 -0
  305. package/src/grapher/slopeCharts/SlopeChartState.ts +392 -0
  306. package/src/grapher/slopeCharts/SlopeChartThumbnail.tsx +368 -0
  307. package/src/grapher/stackedCharts/AbstractStackedChartState.ts +370 -0
  308. package/src/grapher/stackedCharts/MarimekkoBars.tsx +190 -0
  309. package/src/grapher/stackedCharts/MarimekkoBarsForOneEntity.tsx +168 -0
  310. package/src/grapher/stackedCharts/MarimekkoChart.tsx +1144 -0
  311. package/src/grapher/stackedCharts/MarimekkoChartConstants.ts +112 -0
  312. package/src/grapher/stackedCharts/MarimekkoChartHelpers.ts +21 -0
  313. package/src/grapher/stackedCharts/MarimekkoChartState.ts +465 -0
  314. package/src/grapher/stackedCharts/MarimekkoChartThumbnail.tsx +168 -0
  315. package/src/grapher/stackedCharts/MarimekkoInternalLabels.tsx +124 -0
  316. package/src/grapher/stackedCharts/StackedAreaChart.tsx +678 -0
  317. package/src/grapher/stackedCharts/StackedAreaChartState.ts +34 -0
  318. package/src/grapher/stackedCharts/StackedAreaChartThumbnail.tsx +215 -0
  319. package/src/grapher/stackedCharts/StackedAreas.tsx +223 -0
  320. package/src/grapher/stackedCharts/StackedBarChart.tsx +619 -0
  321. package/src/grapher/stackedCharts/StackedBarChartState.ts +80 -0
  322. package/src/grapher/stackedCharts/StackedBarChartThumbnail.tsx +220 -0
  323. package/src/grapher/stackedCharts/StackedBarSegment.tsx +87 -0
  324. package/src/grapher/stackedCharts/StackedBars.tsx +102 -0
  325. package/src/grapher/stackedCharts/StackedConstants.ts +109 -0
  326. package/src/grapher/stackedCharts/StackedDiscreteBarChart.tsx +270 -0
  327. package/src/grapher/stackedCharts/StackedDiscreteBarChartState.ts +296 -0
  328. package/src/grapher/stackedCharts/StackedDiscreteBarChartThumbnail.tsx +27 -0
  329. package/src/grapher/stackedCharts/StackedDiscreteBars.tsx +648 -0
  330. package/src/grapher/stackedCharts/StackedUtils.ts +142 -0
  331. package/src/grapher/tabs/Tabs.scss +169 -0
  332. package/src/grapher/tabs/Tabs.tsx +54 -0
  333. package/src/grapher/tabs/TabsWithDropdown.scss +62 -0
  334. package/src/grapher/tabs/TabsWithDropdown.tsx +114 -0
  335. package/src/grapher/testData/OwidTestData.sample.ts +273 -0
  336. package/src/grapher/testData/OwidTestData.ts +64 -0
  337. package/src/grapher/timeline/TimelineComponent.scss +139 -0
  338. package/src/grapher/timeline/TimelineComponent.tsx +658 -0
  339. package/src/grapher/timeline/TimelineController.ts +368 -0
  340. package/src/grapher/timeline/readme.md +7 -0
  341. package/src/grapher/tooltip/Tooltip.scss +510 -0
  342. package/src/grapher/tooltip/Tooltip.tsx +294 -0
  343. package/src/grapher/tooltip/TooltipContents.tsx +383 -0
  344. package/src/grapher/tooltip/TooltipProps.ts +123 -0
  345. package/src/grapher/tooltip/TooltipState.ts +81 -0
  346. package/src/grapher/verticalLabels/VerticalLabels.tsx +31 -0
  347. package/src/grapher/verticalLabels/VerticalLabelsState.ts +154 -0
  348. package/src/index.ts +226 -0
  349. package/src/styles/charts.scss +15 -0
  350. package/src/types/NominalType.ts +30 -0
  351. package/src/types/OwidOrigin.ts +18 -0
  352. package/src/types/OwidSource.ts +9 -0
  353. package/src/types/OwidVariable.ts +133 -0
  354. package/src/types/OwidVariableDisplayConfigInterface.ts +49 -0
  355. package/src/types/analyticsTypes.ts +54 -0
  356. package/src/types/dbTypes/Tags.ts +11 -0
  357. package/src/types/domainTypes/Archive.ts +139 -0
  358. package/src/types/domainTypes/Author.ts +28 -0
  359. package/src/types/domainTypes/ContentGraph.ts +76 -0
  360. package/src/types/domainTypes/CoreTableTypes.ts +305 -0
  361. package/src/types/domainTypes/DeployStatus.ts +23 -0
  362. package/src/types/domainTypes/Layout.ts +34 -0
  363. package/src/types/domainTypes/Posts.ts +34 -0
  364. package/src/types/domainTypes/Search.ts +299 -0
  365. package/src/types/domainTypes/Site.ts +8 -0
  366. package/src/types/domainTypes/StaticViz.ts +64 -0
  367. package/src/types/domainTypes/Toc.ts +11 -0
  368. package/src/types/domainTypes/Tombstone.ts +19 -0
  369. package/src/types/domainTypes/Various.ts +79 -0
  370. package/src/types/gdocTypes/Gdoc.ts +280 -0
  371. package/src/types/grapherTypes/BinningStrategyTypes.ts +46 -0
  372. package/src/types/grapherTypes/GrapherConstants.ts +53 -0
  373. package/src/types/grapherTypes/GrapherTypes.ts +743 -0
  374. package/src/types/index.ts +316 -0
  375. package/src/types/wordpressTypes/WordpressTypes.ts +9 -0
  376. package/src/utils/Bounds.ts +439 -0
  377. package/src/utils/BrowserUtils.ts +12 -0
  378. package/src/utils/FuzzySearch.ts +74 -0
  379. package/src/utils/MultiDimDataPageConfig.ts +31 -0
  380. package/src/utils/OwidVariable.ts +82 -0
  381. package/src/utils/PointVector.ts +97 -0
  382. package/src/utils/PromiseCache.ts +36 -0
  383. package/src/utils/PromiseSwitcher.ts +52 -0
  384. package/src/utils/TimeBounds.ts +130 -0
  385. package/src/utils/Tippy.tsx +57 -0
  386. package/src/utils/Util.ts +2369 -0
  387. package/src/utils/archival/archivalDate.ts +48 -0
  388. package/src/utils/dayjs.ts +32 -0
  389. package/src/utils/formatValue.ts +242 -0
  390. package/src/utils/grapherConfigUtils.ts +81 -0
  391. package/src/utils/image.ts +225 -0
  392. package/src/utils/index.ts +318 -0
  393. package/src/utils/isPresent.ts +5 -0
  394. package/src/utils/metadataHelpers.ts +329 -0
  395. package/src/utils/persistable/Persistable.ts +82 -0
  396. package/src/utils/persistable/readme.md +50 -0
  397. package/src/utils/regions.json +5635 -0
  398. package/src/utils/regions.ts +463 -0
  399. package/src/utils/serializers.ts +16 -0
  400. package/src/utils/string.ts +42 -0
  401. package/src/utils/urls/Url.ts +195 -0
  402. package/src/utils/urls/UrlMigration.ts +10 -0
  403. package/src/utils/urls/UrlUtils.ts +54 -0
  404. package/src/utils/urls/readme.md +90 -0
@@ -0,0 +1,658 @@
1
+ import * as _ from "lodash-es"
2
+ import * as React from "react"
3
+ import { select } from "d3-selection"
4
+ import cx from "classnames"
5
+ import {
6
+ getRelativeMouse,
7
+ isMobile,
8
+ Bounds,
9
+ Time,
10
+ Tippy,
11
+ } from "../../utils/index.js"
12
+ import { observable, computed, action, makeObservable } from "mobx"
13
+ import { observer } from "mobx-react"
14
+ import { faPlay, faPause } from "@fortawesome/free-solid-svg-icons"
15
+ import {
16
+ TimelineController,
17
+ TimelineManager,
18
+ TimelineDragTarget,
19
+ } from "./TimelineController"
20
+ import { ActionButton } from "../controls/ActionButtons"
21
+ import {
22
+ DEFAULT_GRAPHER_BOUNDS,
23
+ GRAPHER_FRAME_PADDING_HORIZONTAL,
24
+ GRAPHER_TIMELINE_CLASS,
25
+ } from "../core/GrapherConstants.js"
26
+
27
+ export const TIMELINE_HEIGHT = 32 // Keep in sync with $timelineHeight in TimelineComponent.scss
28
+
29
+ const HANDLE_DIAMETER = 20 // Keep in sync with $handle-diameter in TimelineComponent.scss
30
+ const HANDLE_TOOLTIP_FADE_TIME_MS = 2000
31
+
32
+ enum MarkerType {
33
+ Start = "startMarker",
34
+ End = "endMarker",
35
+ }
36
+
37
+ interface TimelineComponentProps {
38
+ timelineController: TimelineController
39
+ maxWidth?: number
40
+ }
41
+
42
+ @observer
43
+ export class TimelineComponent extends React.Component<TimelineComponentProps> {
44
+ base = React.createRef<HTMLDivElement>()
45
+
46
+ constructor(props: TimelineComponentProps) {
47
+ super(props)
48
+
49
+ makeObservable<
50
+ TimelineComponent,
51
+ | "startTooltipVisible"
52
+ | "endTooltipVisible"
53
+ | "lastUpdatedTooltip"
54
+ | "hoverTime"
55
+ >(this, {
56
+ startTooltipVisible: observable,
57
+ endTooltipVisible: observable,
58
+ lastUpdatedTooltip: observable,
59
+ hoverTime: observable,
60
+ })
61
+ }
62
+
63
+ /** Currently hovered time */
64
+ private hoverTime?: Time
65
+
66
+ @computed protected get maxWidth(): number {
67
+ return this.props.maxWidth ?? DEFAULT_GRAPHER_BOUNDS.width
68
+ }
69
+
70
+ @computed private get dragTarget(): TimelineDragTarget | undefined {
71
+ return this.manager.timelineDragTarget
72
+ }
73
+
74
+ @computed private get isDragging(): boolean {
75
+ return !!this.dragTarget
76
+ }
77
+
78
+ @computed private get manager(): TimelineManager {
79
+ return this.props.timelineController.manager
80
+ }
81
+
82
+ @computed private get controller(): TimelineController {
83
+ return this.props.timelineController
84
+ }
85
+
86
+ private get sliderBounds(): Bounds {
87
+ return this.slider
88
+ ? Bounds.fromRect(this.slider.getBoundingClientRect())
89
+ : new Bounds(0, 0, 100, 100)
90
+ }
91
+
92
+ private slider?: Element | HTMLElement | null
93
+ private playButton?: Element | HTMLElement | null
94
+
95
+ private getInputTimeFromMouse(
96
+ event: MouseEvent | TouchEvent
97
+ ): number | undefined {
98
+ const { minTime, maxTime } = this.controller
99
+ if (!this.slider) return
100
+ const mouseX = getRelativeMouse(this.slider, event).x
101
+
102
+ const fracWidth = mouseX / this.sliderBounds.width
103
+ return minTime + fracWidth * (maxTime - minTime)
104
+ }
105
+
106
+ @action.bound private onDrag(inputTime: number): void {
107
+ this.controller.onDrag()
108
+ this.manager.timelineDragTarget = this.controller.dragHandleToTime(
109
+ this.dragTarget!,
110
+ inputTime
111
+ )
112
+ this.showTooltips()
113
+ }
114
+
115
+ @action.bound private showTooltips(): void {
116
+ this.hideStartTooltip.cancel()
117
+ this.hideEndTooltip.cancel()
118
+ this.startTooltipVisible = true
119
+ this.endTooltipVisible = true
120
+
121
+ if (this.dragTarget === "start")
122
+ this.lastUpdatedTooltip = MarkerType.Start
123
+ if (this.dragTarget === "end") this.lastUpdatedTooltip = MarkerType.End
124
+ if (this.manager.startHandleTimeBound > this.manager.endHandleTimeBound)
125
+ this.lastUpdatedTooltip =
126
+ this.lastUpdatedTooltip === MarkerType.Start
127
+ ? MarkerType.End
128
+ : MarkerType.Start
129
+ }
130
+
131
+ private getDragTarget(
132
+ inputTime: number,
133
+ isStartMarker: boolean,
134
+ isEndMarker: boolean
135
+ ): TimelineDragTarget {
136
+ const { startHandleTimeBound, endHandleTimeBound } = this.manager
137
+
138
+ if (
139
+ startHandleTimeBound === endHandleTimeBound &&
140
+ (isStartMarker || isEndMarker)
141
+ )
142
+ return "both"
143
+ else if (isStartMarker || inputTime <= startHandleTimeBound)
144
+ return "start"
145
+ else if (isEndMarker || inputTime >= endHandleTimeBound) return "end"
146
+ return "both"
147
+ }
148
+
149
+ @action.bound private onMouseDown(event: MouseEvent | TouchEvent): void {
150
+ this.manager.onTimelineClick?.()
151
+
152
+ // Immediately hide the hover time handle
153
+ this.hoverTime = undefined
154
+
155
+ const targetEl = select(event.target as Element)
156
+
157
+ const inputTime = this.getInputTimeFromMouse(event)
158
+ if (!inputTime) return
159
+
160
+ this.manager.timelineDragTarget = this.getDragTarget(
161
+ inputTime,
162
+ targetEl.classed(MarkerType.Start),
163
+ targetEl.classed(MarkerType.End)
164
+ )
165
+
166
+ if (this.dragTarget === "both")
167
+ this.controller.setDragOffsets(inputTime)
168
+
169
+ this.onDrag(inputTime)
170
+
171
+ event.preventDefault()
172
+ }
173
+
174
+ private queuedDrag?: boolean
175
+ @action.bound private onMouseMove(event: MouseEvent | TouchEvent): void {
176
+ const { dragTarget } = this
177
+ if (!dragTarget) return
178
+ if (this.queuedDrag) return
179
+
180
+ this.queuedDrag = true
181
+ const inputTime = this.getInputTimeFromMouse(event)
182
+ if (inputTime) this.onDrag(inputTime)
183
+ this.queuedDrag = false
184
+ }
185
+
186
+ @action.bound private onMouseUp(): void {
187
+ this.manager.timelineDragTarget = undefined
188
+
189
+ if (this.manager.isPlaying) return
190
+
191
+ if (isMobile()) {
192
+ if (this.startTooltipVisible) this.hideStartTooltip()
193
+ if (this.endTooltipVisible) this.hideEndTooltip()
194
+ } else if (!this.mouseHoveringOverTimeline) {
195
+ this.startTooltipVisible = false
196
+ this.endTooltipVisible = false
197
+ }
198
+ }
199
+
200
+ @computed private get areBothHandlesVisible(): boolean {
201
+ return this.controller.startTime !== this.controller.endTime
202
+ }
203
+
204
+ @computed private get shouldShowHoverTimeHandle(): boolean {
205
+ return (
206
+ !this.manager.isSingleTimeSelectionActive &&
207
+ !this.isDragging &&
208
+ !this.areBothHandlesVisible
209
+ )
210
+ }
211
+
212
+ private setHoverTime(event: MouseEvent | TouchEvent): void {
213
+ if (!this.shouldShowHoverTimeHandle) return
214
+
215
+ const inputTime = this.getInputTimeFromMouse(event)
216
+ if (!inputTime) return
217
+
218
+ if (!this.slider) return
219
+ const mouseX = getRelativeMouse(this.slider, event).x
220
+ const startX =
221
+ this.controller.startTimeProgress * this.slider.clientWidth
222
+
223
+ // Hide the hover handle when the mouse is positioned directly over
224
+ // the existing time handle
225
+ if (Math.abs(mouseX - startX) < HANDLE_DIAMETER) {
226
+ this.hoverTime = undefined
227
+ return
228
+ }
229
+
230
+ const timeBound = this.controller.getTimeBoundFromDrag(inputTime)
231
+ if (!Number.isFinite(timeBound)) return
232
+
233
+ this.hoverTime = timeBound
234
+ }
235
+
236
+ @computed private get hoverTimeProgress(): number | undefined {
237
+ if (this.hoverTime === undefined) return undefined
238
+ return this.controller.calculateProgress(this.hoverTime)
239
+ }
240
+
241
+ private mouseHoveringOverTimeline: boolean = false
242
+ @action.bound private onMouseOverSlider(event: MouseEvent): void {
243
+ this.mouseHoveringOverTimeline = true
244
+
245
+ this.hideStartTooltip.cancel()
246
+ this.startTooltipVisible = true
247
+
248
+ this.hideEndTooltip.cancel()
249
+ this.endTooltipVisible = true
250
+
251
+ this.setHoverTime(event)
252
+ }
253
+
254
+ @action.bound private onMouseMoveSlider(event: MouseEvent): void {
255
+ this.setHoverTime(event)
256
+ }
257
+
258
+ @action.bound private onMouseLeaveSlider(): void {
259
+ if (!this.manager.isPlaying && !this.isDragging) {
260
+ this.startTooltipVisible = false
261
+ this.endTooltipVisible = false
262
+ }
263
+ this.mouseHoveringOverTimeline = false
264
+ this.hoverTime = undefined
265
+ }
266
+
267
+ private hideStartTooltip = _.debounce(() => {
268
+ this.startTooltipVisible = false
269
+ }, HANDLE_TOOLTIP_FADE_TIME_MS)
270
+ private hideEndTooltip = _.debounce(() => {
271
+ this.endTooltipVisible = false
272
+ }, HANDLE_TOOLTIP_FADE_TIME_MS)
273
+
274
+ @action.bound private onPlayTouchEnd(evt: Event): void {
275
+ evt.preventDefault()
276
+ evt.stopPropagation()
277
+ void this.controller.togglePlay()
278
+ }
279
+
280
+ @action.bound private onSliderTouchStart(event: Event): void {
281
+ this.onMouseDown(event as TouchEvent)
282
+ }
283
+
284
+ @computed private get showPlayLabel(): boolean {
285
+ const labelWidth = Bounds.forText("Play time-lapse", {
286
+ fontSize: 13,
287
+ }).width
288
+ return labelWidth < 0.1 * this.maxWidth
289
+ }
290
+
291
+ override componentDidMount(): void {
292
+ const current = this.base.current
293
+
294
+ if (current) {
295
+ this.slider = current.querySelector(".slider")
296
+ this.playButton = current.querySelector(".play")
297
+ }
298
+
299
+ document.documentElement.addEventListener("mouseup", this.onMouseUp)
300
+ document.documentElement.addEventListener("mouseleave", this.onMouseUp)
301
+ document.documentElement.addEventListener("mousemove", this.onMouseMove)
302
+ document.documentElement.addEventListener("touchend", this.onMouseUp)
303
+ document.documentElement.addEventListener("touchmove", this.onMouseMove)
304
+ this.slider?.addEventListener("touchstart", this.onSliderTouchStart, {
305
+ passive: false,
306
+ })
307
+ this.playButton?.addEventListener("touchend", this.onPlayTouchEnd, {
308
+ passive: false,
309
+ })
310
+ }
311
+
312
+ override componentWillUnmount(): void {
313
+ document.documentElement.removeEventListener("mouseup", this.onMouseUp)
314
+ document.documentElement.removeEventListener(
315
+ "mouseleave",
316
+ this.onMouseUp
317
+ )
318
+ document.documentElement.removeEventListener(
319
+ "mousemove",
320
+ this.onMouseMove
321
+ )
322
+ document.documentElement.removeEventListener("touchend", this.onMouseUp)
323
+ document.documentElement.removeEventListener(
324
+ "touchmove",
325
+ this.onMouseMove
326
+ )
327
+ this.slider?.removeEventListener("touchstart", this.onSliderTouchStart)
328
+ this.playButton?.removeEventListener("touchend", this.onPlayTouchEnd)
329
+ }
330
+
331
+ private formatTime(time: number): string {
332
+ return this.manager.formatTimeFn
333
+ ? this.manager.formatTimeFn(time)
334
+ : time.toString()
335
+ }
336
+
337
+ private timelineEdgeMarker(
338
+ markerType: "start" | "end"
339
+ ): React.ReactElement {
340
+ const { controller } = this
341
+ const time =
342
+ markerType === "start" ? controller.minTime : controller.maxTime
343
+ return (
344
+ <button
345
+ className="date clickable"
346
+ type="button"
347
+ onClick={action((): void =>
348
+ markerType === "start"
349
+ ? controller.resetStartToMin()
350
+ : controller.resetEndToMax()
351
+ )}
352
+ onMouseEnter={action(() => {
353
+ if (this.shouldShowHoverTimeHandle) this.hoverTime = time
354
+ })}
355
+ onMouseLeave={action(() => {
356
+ this.hoverTime = undefined
357
+ })}
358
+ >
359
+ {this.formatTime(time)}
360
+ </button>
361
+ )
362
+ }
363
+
364
+ private startTooltipVisible: boolean = false
365
+ private endTooltipVisible: boolean = false
366
+ private lastUpdatedTooltip?: MarkerType
367
+
368
+ @action.bound private togglePlay(): void {
369
+ void this.controller.togglePlay()
370
+ }
371
+
372
+ @action.bound updateStartTimeOnKeyDown(key: string): void {
373
+ const { controller } = this
374
+ if (key === "Home") {
375
+ controller.resetStartToMin()
376
+ } else if (key === "End") {
377
+ controller.setStartToMax()
378
+ } else if (key === "ArrowLeft" || key === "ArrowDown") {
379
+ controller.decreaseStartTime()
380
+ } else if (key === "ArrowRight" || key === "ArrowUp") {
381
+ controller.increaseStartTime()
382
+ } else if (key === "PageUp") {
383
+ controller.increaseStartTimeByLargeStep()
384
+ } else if (key === "PageDown") {
385
+ controller.decreaseStartTimeByLargeStep()
386
+ }
387
+ }
388
+
389
+ @action.bound updateEndTimeOnKeyDown(key: string): void {
390
+ const { controller } = this
391
+ if (key === "Home") {
392
+ controller.setEndToMin()
393
+ } else if (key === "End") {
394
+ controller.resetEndToMax()
395
+ } else if (key === "ArrowLeft" || key === "ArrowDown") {
396
+ controller.decreaseEndTime()
397
+ } else if (key === "ArrowRight" || key === "ArrowUp") {
398
+ controller.increaseEndTime()
399
+ } else if (key === "PageUp") {
400
+ controller.increaseEndTimeByLargeStep()
401
+ } else if (key === "PageDown") {
402
+ controller.decreaseEndTimeByLargeStep()
403
+ }
404
+ }
405
+
406
+ convertToTime(time: number): number {
407
+ if (time === -Infinity) return this.controller.minTime
408
+ if (time === +Infinity) return this.controller.maxTime
409
+ return time
410
+ }
411
+
412
+ override render(): React.ReactElement {
413
+ const { manager, controller, hoverTime } = this
414
+ const {
415
+ startTimeProgress,
416
+ endTimeProgress,
417
+ minTime,
418
+ maxTime,
419
+ startTime,
420
+ endTime,
421
+ } = controller
422
+
423
+ const formattedMinTime = this.formatTime(minTime)
424
+ const formattedMaxTime = this.formatTime(maxTime)
425
+ const formattedStartTime = this.formatTime(startTime)
426
+ const formattedEndTime = this.formatTime(endTime)
427
+ const formattedHoverTime =
428
+ hoverTime !== undefined ? this.formatTime(hoverTime) : undefined
429
+
430
+ return (
431
+ <div
432
+ ref={this.base}
433
+ className={cx(GRAPHER_TIMELINE_CLASS, {
434
+ hover: this.mouseHoveringOverTimeline,
435
+ })}
436
+ style={{ padding: `0 ${GRAPHER_FRAME_PADDING_HORIZONTAL}px` }}
437
+ role="group"
438
+ aria-label="Timeline controls"
439
+ onMouseOver={action((event) =>
440
+ this.onMouseOverSlider(event.nativeEvent)
441
+ )}
442
+ onMouseLeave={this.onMouseLeaveSlider}
443
+ onMouseMove={action((event) =>
444
+ this.onMouseMoveSlider(event.nativeEvent)
445
+ )}
446
+ >
447
+ {!this.manager.disablePlay && (
448
+ <ActionButton
449
+ dataTrackNote={
450
+ manager.isPlaying
451
+ ? "timeline_pause"
452
+ : "timeline_play"
453
+ }
454
+ onMouseDown={(e): void => e.stopPropagation()}
455
+ onClick={this.togglePlay}
456
+ showLabel={this.showPlayLabel}
457
+ label={
458
+ (manager.isPlaying ? "Pause" : "Play") +
459
+ " time-lapse"
460
+ }
461
+ icon={manager.isPlaying ? faPause : faPlay}
462
+ isActive={manager.isPlaying}
463
+ style={{ minWidth: TIMELINE_HEIGHT }}
464
+ />
465
+ )}
466
+ {this.timelineEdgeMarker("start")}
467
+ <div
468
+ className="slider clickable"
469
+ role="group"
470
+ aria-label="Timeline slider"
471
+ onMouseDown={(event) => this.onMouseDown(event.nativeEvent)}
472
+ >
473
+ <TimelineHandle
474
+ type={MarkerType.Start}
475
+ ariaLabel="Start time"
476
+ offsetPercent={startTimeProgress * 100}
477
+ formattedMinTime={formattedMinTime}
478
+ formattedMaxTime={formattedMaxTime}
479
+ formattedCurrTime={formattedStartTime}
480
+ tooltipVisible={this.startTooltipVisible}
481
+ tooltipZIndex={
482
+ this.lastUpdatedTooltip === MarkerType.Start ? 2 : 1
483
+ }
484
+ onKeyDown={action((e) => {
485
+ // Prevent scrolling
486
+ if (
487
+ e.key === "Home" ||
488
+ e.key === "End" ||
489
+ e.key === "PageUp" ||
490
+ e.key === "PageDown"
491
+ )
492
+ e.preventDefault()
493
+
494
+ this.updateStartTimeOnKeyDown(e.key)
495
+ })}
496
+ onFocus={action(() => {
497
+ this.showTooltips()
498
+ })}
499
+ onBlur={action(() => {
500
+ this.startTooltipVisible = false
501
+ this.endTooltipVisible = false
502
+ })}
503
+ />
504
+ <TimelineInterval
505
+ startTimeProgress={startTimeProgress}
506
+ endTimeProgress={endTimeProgress}
507
+ />
508
+ {this.hoverTimeProgress !== undefined && (
509
+ <TimelineInterval
510
+ className="interval-hover"
511
+ startTimeProgress={Math.min(
512
+ startTimeProgress,
513
+ this.hoverTimeProgress
514
+ )}
515
+ endTimeProgress={Math.max(
516
+ startTimeProgress,
517
+ this.hoverTimeProgress
518
+ )}
519
+ ariaHidden={true}
520
+ />
521
+ )}
522
+ <TimelineHandle
523
+ type={MarkerType.End}
524
+ ariaLabel="End time"
525
+ offsetPercent={endTimeProgress * 100}
526
+ formattedMinTime={formattedMinTime}
527
+ formattedMaxTime={formattedMaxTime}
528
+ formattedCurrTime={formattedEndTime}
529
+ tooltipVisible={this.endTooltipVisible}
530
+ tooltipZIndex={
531
+ this.lastUpdatedTooltip === MarkerType.End ? 2 : 1
532
+ }
533
+ onKeyDown={action((e) => {
534
+ // prevent browser to scroll to the top or bottom of the page
535
+ if (
536
+ e.key === "Home" ||
537
+ e.key === "End" ||
538
+ e.key === "PageUp" ||
539
+ e.key === "PageDown"
540
+ )
541
+ e.preventDefault()
542
+
543
+ this.updateEndTimeOnKeyDown(e.key)
544
+ })}
545
+ onFocus={action(() => {
546
+ this.showTooltips()
547
+ })}
548
+ onBlur={action(() => {
549
+ this.startTooltipVisible = false
550
+ this.endTooltipVisible = false
551
+ })}
552
+ />
553
+ {this.hoverTime !== undefined &&
554
+ this.hoverTimeProgress !== undefined && (
555
+ <TimelineHandle
556
+ type="hoverMarker"
557
+ offsetPercent={this.hoverTimeProgress * 100}
558
+ formattedMinTime={formattedMinTime}
559
+ formattedMaxTime={formattedMaxTime}
560
+ formattedCurrTime={formattedHoverTime!}
561
+ tooltipVisible={true}
562
+ tooltipZIndex={3}
563
+ />
564
+ )}
565
+ </div>
566
+ {this.timelineEdgeMarker("end")}
567
+ </div>
568
+ )
569
+ }
570
+ }
571
+
572
+ const TimelineHandle = ({
573
+ type,
574
+ ariaLabel,
575
+ offsetPercent,
576
+ formattedMinTime,
577
+ formattedMaxTime,
578
+ formattedCurrTime,
579
+ tooltipVisible,
580
+ tooltipZIndex,
581
+ onKeyDown,
582
+ onFocus,
583
+ onBlur,
584
+ }: {
585
+ type: MarkerType | "hoverMarker"
586
+ ariaLabel?: string
587
+ offsetPercent: number
588
+ formattedMinTime: string
589
+ formattedMaxTime: string
590
+ formattedCurrTime: string
591
+ tooltipVisible: boolean
592
+ tooltipZIndex: number
593
+ onKeyDown?: React.KeyboardEventHandler<HTMLDivElement>
594
+ onFocus?: React.FocusEventHandler<HTMLDivElement>
595
+ onBlur?: React.FocusEventHandler<HTMLDivElement>
596
+ }): React.ReactElement => {
597
+ const isInteractive = type !== "hoverMarker"
598
+ return (
599
+ // @ts-expect-error aria-value* fields expect a number, but if we're dealing with daily data,
600
+ // the numeric representation of a date is meaningless, so we pass the formatted date string instead.
601
+ <div
602
+ className={cx("handle", type)}
603
+ style={{ left: `${offsetPercent}%`, zIndex: tooltipZIndex }}
604
+ role="slider"
605
+ tabIndex={isInteractive ? 0 : -1}
606
+ aria-valuemin={castToNumberIfPossible(formattedMinTime)}
607
+ aria-valuenow={castToNumberIfPossible(formattedCurrTime)}
608
+ aria-valuemax={castToNumberIfPossible(formattedMaxTime)}
609
+ aria-valuetext={formattedCurrTime}
610
+ aria-label={ariaLabel}
611
+ aria-orientation="horizontal"
612
+ onFocus={onFocus}
613
+ onBlur={onBlur}
614
+ onKeyDown={onKeyDown}
615
+ >
616
+ <Tippy
617
+ content={formattedCurrTime}
618
+ theme="grapher-dark"
619
+ placement="top"
620
+ visible={tooltipVisible}
621
+ arrow={true}
622
+ >
623
+ <div className="icon" />
624
+ </Tippy>
625
+ </div>
626
+ )
627
+ }
628
+
629
+ const TimelineInterval = ({
630
+ startTimeProgress,
631
+ endTimeProgress,
632
+ className,
633
+ ariaHidden,
634
+ }: {
635
+ startTimeProgress: number
636
+ endTimeProgress: number
637
+ className?: string
638
+ ariaHidden?: boolean
639
+ }): React.ReactElement => {
640
+ const left = startTimeProgress * 100
641
+ const right = 100 - endTimeProgress * 100
642
+ return (
643
+ <div
644
+ className={cx("interval", className)}
645
+ style={{ left: `${left}%`, right: `${right}%` }}
646
+ role="presentation"
647
+ aria-hidden={ariaHidden}
648
+ />
649
+ )
650
+ }
651
+
652
+ function castToNumberIfPossible(s: string): string | number {
653
+ return isNumber(s) ? +s : s
654
+ }
655
+
656
+ function isNumber(s: string): boolean {
657
+ return /^\d+$/.test(s)
658
+ }