matterviz 0.3.5 → 0.3.6

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 (855) hide show
  1. package/.vscode/launch.json +13 -0
  2. package/.vscodeignore +7 -0
  3. package/dist/assets/STLExporter-BpTH3YHE.js +8 -0
  4. package/dist/assets/browser-DdDecX_W.js +1 -0
  5. package/dist/assets/export-qgn-H9y6.js +2 -0
  6. package/dist/assets/main-DiKYzti2.css +1 -0
  7. package/dist/assets/moyo_wasm_bg-0ocwg7xY.wasm +0 -0
  8. package/dist/extension.js +31293 -0
  9. package/dist/src/lib/FilePicker.svelte +360 -0
  10. package/dist/src/lib/MillerIndexInput.svelte +66 -0
  11. package/dist/src/lib/api/mp.ts +26 -0
  12. package/dist/src/lib/api/optimade.ts +204 -0
  13. package/dist/src/lib/brillouin/BrillouinZone.svelte +549 -0
  14. package/dist/src/lib/brillouin/BrillouinZoneControls.svelte +144 -0
  15. package/dist/src/lib/brillouin/BrillouinZoneExportPane.svelte +146 -0
  16. package/dist/src/lib/brillouin/BrillouinZoneInfoPane.svelte +146 -0
  17. package/dist/src/lib/brillouin/BrillouinZoneScene.svelte +476 -0
  18. package/dist/src/lib/brillouin/BrillouinZoneTooltip.svelte +92 -0
  19. package/dist/src/lib/brillouin/compute.ts +529 -0
  20. package/dist/src/lib/brillouin/index.ts +8 -0
  21. package/dist/src/lib/brillouin/types.ts +51 -0
  22. package/dist/src/lib/chempot-diagram/ChemPotDiagram.svelte +327 -0
  23. package/dist/src/lib/chempot-diagram/ChemPotDiagram2D.svelte +846 -0
  24. package/dist/src/lib/chempot-diagram/ChemPotDiagram3D.svelte +3193 -0
  25. package/dist/src/lib/chempot-diagram/async-compute.svelte.ts +94 -0
  26. package/dist/src/lib/chempot-diagram/chempot-worker.ts +11 -0
  27. package/dist/src/lib/chempot-diagram/color.ts +42 -0
  28. package/dist/src/lib/chempot-diagram/compute.ts +1014 -0
  29. package/dist/src/lib/chempot-diagram/index.ts +6 -0
  30. package/dist/src/lib/chempot-diagram/pointer.ts +56 -0
  31. package/dist/src/lib/chempot-diagram/temperature.ts +77 -0
  32. package/dist/src/lib/chempot-diagram/types.ts +130 -0
  33. package/dist/src/lib/colors/index.ts +249 -0
  34. package/dist/src/lib/composition/BarChart.svelte +297 -0
  35. package/dist/src/lib/composition/BubbleChart.svelte +218 -0
  36. package/dist/src/lib/composition/Composition.svelte +165 -0
  37. package/dist/src/lib/composition/Formula.svelte +268 -0
  38. package/dist/src/lib/composition/FormulaFilter.svelte +1257 -0
  39. package/dist/src/lib/composition/PieChart.svelte +323 -0
  40. package/dist/src/lib/composition/format.ts +155 -0
  41. package/dist/src/lib/composition/index.ts +37 -0
  42. package/dist/src/lib/composition/parse.ts +605 -0
  43. package/dist/src/lib/constants.ts +134 -0
  44. package/dist/src/lib/controls.ts +42 -0
  45. package/dist/src/lib/convex-hull/ConvexHull.svelte +157 -0
  46. package/dist/src/lib/convex-hull/ConvexHull2D.svelte +825 -0
  47. package/dist/src/lib/convex-hull/ConvexHull3D.svelte +1801 -0
  48. package/dist/src/lib/convex-hull/ConvexHull4D.svelte +1398 -0
  49. package/dist/src/lib/convex-hull/ConvexHullControls.svelte +535 -0
  50. package/dist/src/lib/convex-hull/ConvexHullInfoPane.svelte +125 -0
  51. package/dist/src/lib/convex-hull/ConvexHullStats.svelte +929 -0
  52. package/dist/src/lib/convex-hull/ConvexHullTooltip.svelte +131 -0
  53. package/dist/src/lib/convex-hull/GasPressureControls.svelte +247 -0
  54. package/dist/src/lib/convex-hull/StructurePopup.svelte +151 -0
  55. package/dist/src/lib/convex-hull/TemperatureSlider.svelte +140 -0
  56. package/dist/src/lib/convex-hull/barycentric-coords.ts +246 -0
  57. package/dist/src/lib/convex-hull/demo-temperature.ts +63 -0
  58. package/dist/src/lib/convex-hull/gas-thermodynamics.ts +405 -0
  59. package/dist/src/lib/convex-hull/helpers.ts +932 -0
  60. package/dist/src/lib/convex-hull/index.ts +202 -0
  61. package/dist/src/lib/convex-hull/thermodynamics.ts +2192 -0
  62. package/dist/src/lib/convex-hull/types.ts +267 -0
  63. package/dist/src/lib/coordination/CoordinationBarPlot.svelte +311 -0
  64. package/dist/src/lib/coordination/calc-coordination.ts +93 -0
  65. package/dist/src/lib/coordination/index.ts +9 -0
  66. package/dist/src/lib/effects.svelte.ts +48 -0
  67. package/dist/src/lib/element/BohrAtom.svelte +147 -0
  68. package/dist/src/lib/element/ElementHeading.svelte +26 -0
  69. package/dist/src/lib/element/ElementPhoto.svelte +57 -0
  70. package/dist/src/lib/element/ElementStats.svelte +80 -0
  71. package/dist/src/lib/element/ElementTile.svelte +484 -0
  72. package/dist/src/lib/element/data.json.gz.d.ts +4 -0
  73. package/dist/src/lib/element/data.ts +14 -0
  74. package/dist/src/lib/element/index.ts +8 -0
  75. package/dist/src/lib/element/types.ts +62 -0
  76. package/dist/src/lib/feedback/ClickFeedback.svelte +58 -0
  77. package/dist/src/lib/feedback/DragOverlay.svelte +42 -0
  78. package/dist/src/lib/feedback/index.ts +4 -0
  79. package/dist/src/lib/fermi-surface/FermiSlice.svelte +189 -0
  80. package/dist/src/lib/fermi-surface/FermiSurface.svelte +600 -0
  81. package/dist/src/lib/fermi-surface/FermiSurfaceControls.svelte +448 -0
  82. package/dist/src/lib/fermi-surface/FermiSurfaceScene.svelte +794 -0
  83. package/dist/src/lib/fermi-surface/FermiSurfaceTooltip.svelte +111 -0
  84. package/dist/src/lib/fermi-surface/compute.ts +728 -0
  85. package/dist/src/lib/fermi-surface/constants.ts +32 -0
  86. package/dist/src/lib/fermi-surface/export.ts +64 -0
  87. package/dist/src/lib/fermi-surface/index.ts +14 -0
  88. package/dist/src/lib/fermi-surface/marching-cubes.ts +3 -0
  89. package/dist/src/lib/fermi-surface/parse.ts +574 -0
  90. package/dist/src/lib/fermi-surface/symmetry.ts +56 -0
  91. package/dist/src/lib/fermi-surface/types.ts +159 -0
  92. package/dist/src/lib/heatmap-matrix/HeatmapMatrix.svelte +1545 -0
  93. package/dist/src/lib/heatmap-matrix/HeatmapMatrixControls.svelte +225 -0
  94. package/dist/src/lib/heatmap-matrix/index.ts +167 -0
  95. package/dist/src/lib/heatmap-matrix/shared.ts +7 -0
  96. package/dist/src/lib/icons.ts +650 -0
  97. package/dist/src/lib/index.ts +61 -0
  98. package/dist/src/lib/io/decompress.ts +92 -0
  99. package/dist/src/lib/io/export.ts +385 -0
  100. package/dist/src/lib/io/fetch.ts +46 -0
  101. package/dist/src/lib/io/file-drop.ts +51 -0
  102. package/dist/src/lib/io/index.ts +7 -0
  103. package/dist/src/lib/io/is-binary.ts +24 -0
  104. package/dist/src/lib/io/types.ts +8 -0
  105. package/dist/src/lib/io/url-drop.ts +141 -0
  106. package/dist/src/lib/isosurface/Isosurface.svelte +285 -0
  107. package/dist/src/lib/isosurface/IsosurfaceControls.svelte +277 -0
  108. package/dist/src/lib/isosurface/index.ts +7 -0
  109. package/dist/src/lib/isosurface/parse.ts +656 -0
  110. package/dist/src/lib/isosurface/slice.ts +175 -0
  111. package/dist/src/lib/isosurface/types.ts +309 -0
  112. package/dist/src/lib/labels.ts +320 -0
  113. package/dist/src/lib/layout/FullscreenToggle.svelte +50 -0
  114. package/dist/src/lib/layout/InfoCard.svelte +120 -0
  115. package/dist/src/lib/layout/InfoTag.svelte +185 -0
  116. package/dist/src/lib/layout/PropertyFilter.svelte +246 -0
  117. package/dist/src/lib/layout/SettingsSection.svelte +148 -0
  118. package/dist/src/lib/layout/SubpageGrid.svelte +82 -0
  119. package/dist/src/lib/layout/fullscreen.ts +65 -0
  120. package/dist/src/lib/layout/index.ts +11 -0
  121. package/dist/src/lib/layout/json-tree/JsonNode.svelte +548 -0
  122. package/dist/src/lib/layout/json-tree/JsonTree.svelte +1230 -0
  123. package/dist/src/lib/layout/json-tree/JsonValue.svelte +334 -0
  124. package/dist/src/lib/layout/json-tree/index.ts +3 -0
  125. package/dist/src/lib/layout/json-tree/types.ts +126 -0
  126. package/dist/src/lib/layout/json-tree/utils.ts +682 -0
  127. package/dist/src/lib/marching-cubes.ts +614 -0
  128. package/dist/src/lib/math.ts +1081 -0
  129. package/dist/src/lib/overlays/ContextMenu.svelte +162 -0
  130. package/dist/src/lib/overlays/CopyButton.svelte +45 -0
  131. package/dist/src/lib/overlays/DragControlTab.svelte +98 -0
  132. package/dist/src/lib/overlays/DraggablePane.svelte +487 -0
  133. package/dist/src/lib/overlays/InfoPaneCards.svelte +149 -0
  134. package/dist/src/lib/overlays/index.ts +3 -0
  135. package/dist/src/lib/periodic-table/PeriodicTable.svelte +469 -0
  136. package/dist/src/lib/periodic-table/PeriodicTableControls.svelte +557 -0
  137. package/dist/src/lib/periodic-table/PropertySelect.svelte +37 -0
  138. package/dist/src/lib/periodic-table/index.ts +12 -0
  139. package/dist/src/lib/phase-diagram/IsobaricBinaryPhaseDiagram.svelte +1086 -0
  140. package/dist/src/lib/phase-diagram/PhaseDiagramControls.svelte +444 -0
  141. package/dist/src/lib/phase-diagram/PhaseDiagramEditorPane.svelte +126 -0
  142. package/dist/src/lib/phase-diagram/PhaseDiagramExportPane.svelte +184 -0
  143. package/dist/src/lib/phase-diagram/PhaseDiagramTooltip.svelte +391 -0
  144. package/dist/src/lib/phase-diagram/TdbInfoPanel.svelte +203 -0
  145. package/dist/src/lib/phase-diagram/build-diagram.ts +186 -0
  146. package/dist/src/lib/phase-diagram/colors.ts +58 -0
  147. package/dist/src/lib/phase-diagram/diagram-input.ts +40 -0
  148. package/dist/src/lib/phase-diagram/index.ts +13 -0
  149. package/dist/src/lib/phase-diagram/parse.ts +348 -0
  150. package/dist/src/lib/phase-diagram/svg-to-diagram.ts +1023 -0
  151. package/dist/src/lib/phase-diagram/types.ts +144 -0
  152. package/dist/src/lib/phase-diagram/utils.ts +775 -0
  153. package/dist/src/lib/plot/AxisLabel.svelte +51 -0
  154. package/dist/src/lib/plot/BarPlot.svelte +2113 -0
  155. package/dist/src/lib/plot/BarPlotControls.svelte +66 -0
  156. package/dist/src/lib/plot/BinnedScatterPlot.svelte +1114 -0
  157. package/dist/src/lib/plot/ColorBar.svelte +721 -0
  158. package/dist/src/lib/plot/ColorScaleSelect.svelte +54 -0
  159. package/dist/src/lib/plot/ElementScatter.svelte +63 -0
  160. package/dist/src/lib/plot/FillArea.svelte +223 -0
  161. package/dist/src/lib/plot/Histogram.svelte +1558 -0
  162. package/dist/src/lib/plot/HistogramControls.svelte +212 -0
  163. package/dist/src/lib/plot/InteractiveAxisLabel.svelte +96 -0
  164. package/dist/src/lib/plot/Line.svelte +84 -0
  165. package/dist/src/lib/plot/PlotAxis.svelte +169 -0
  166. package/dist/src/lib/plot/PlotControls.svelte +537 -0
  167. package/dist/src/lib/plot/PlotLegend.svelte +569 -0
  168. package/dist/src/lib/plot/PlotTooltip.svelte +67 -0
  169. package/dist/src/lib/plot/PortalSelect.svelte +253 -0
  170. package/dist/src/lib/plot/ReferenceLine3D.svelte +156 -0
  171. package/dist/src/lib/plot/ReferencePlane.svelte +175 -0
  172. package/dist/src/lib/plot/ScatterPlot.svelte +2778 -0
  173. package/dist/src/lib/plot/ScatterPlot3D.svelte +529 -0
  174. package/dist/src/lib/plot/ScatterPlot3DControls.svelte +437 -0
  175. package/dist/src/lib/plot/ScatterPlot3DScene.svelte +912 -0
  176. package/dist/src/lib/plot/ScatterPlotControls.svelte +306 -0
  177. package/dist/src/lib/plot/ScatterPoint.svelte +182 -0
  178. package/dist/src/lib/plot/SpacegroupBarPlot.svelte +293 -0
  179. package/dist/src/lib/plot/Surface3D.svelte +197 -0
  180. package/dist/src/lib/plot/ZeroLines.svelte +97 -0
  181. package/dist/src/lib/plot/ZoomRect.svelte +23 -0
  182. package/dist/src/lib/plot/adaptive-density.ts +316 -0
  183. package/dist/src/lib/plot/auto-place.ts +184 -0
  184. package/dist/src/lib/plot/axis-utils.ts +122 -0
  185. package/dist/src/lib/plot/binned-scatter-types.ts +83 -0
  186. package/dist/src/lib/plot/data-cleaning.ts +1069 -0
  187. package/dist/src/lib/plot/data-transform.ts +69 -0
  188. package/dist/src/lib/plot/defaults.ts +9 -0
  189. package/dist/src/lib/plot/fill-utils.ts +494 -0
  190. package/dist/src/lib/plot/hover-lock.svelte.ts +60 -0
  191. package/dist/src/lib/plot/index.ts +53 -0
  192. package/dist/src/lib/plot/interactions.ts +119 -0
  193. package/dist/src/lib/plot/layout.ts +425 -0
  194. package/dist/src/lib/plot/reference-line.ts +426 -0
  195. package/dist/src/lib/plot/scales.ts +654 -0
  196. package/dist/src/lib/plot/svg.ts +23 -0
  197. package/dist/src/lib/plot/types.ts +1144 -0
  198. package/dist/src/lib/plot/utils/label-placement.ts +541 -0
  199. package/dist/src/lib/plot/utils/series-visibility.ts +140 -0
  200. package/dist/src/lib/plot/utils.ts +11 -0
  201. package/dist/src/lib/rdf/RdfPlot.svelte +247 -0
  202. package/dist/src/lib/rdf/calc-rdf.ts +167 -0
  203. package/dist/src/lib/rdf/index.ts +27 -0
  204. package/dist/src/lib/sanitize.ts +126 -0
  205. package/dist/src/lib/settings.ts +1479 -0
  206. package/dist/src/lib/spectral/Bands.svelte +1040 -0
  207. package/dist/src/lib/spectral/BandsAndDos.svelte +134 -0
  208. package/dist/src/lib/spectral/BrillouinBandsDos.svelte +252 -0
  209. package/dist/src/lib/spectral/Dos.svelte +697 -0
  210. package/dist/src/lib/spectral/helpers.ts +1381 -0
  211. package/dist/src/lib/spectral/index.ts +8 -0
  212. package/dist/src/lib/spectral/types.ts +112 -0
  213. package/dist/src/lib/state.svelte.ts +64 -0
  214. package/dist/src/lib/structure/Arrow.svelte +72 -0
  215. package/dist/src/lib/structure/AtomLegend.svelte +815 -0
  216. package/dist/src/lib/structure/Bond.svelte +140 -0
  217. package/dist/src/lib/structure/CanvasTooltip.svelte +33 -0
  218. package/dist/src/lib/structure/CellSelect.svelte +349 -0
  219. package/dist/src/lib/structure/Cylinder.svelte +45 -0
  220. package/dist/src/lib/structure/Lattice.svelte +196 -0
  221. package/dist/src/lib/structure/Structure.svelte +2248 -0
  222. package/dist/src/lib/structure/StructureControls.svelte +1273 -0
  223. package/dist/src/lib/structure/StructureExportPane.svelte +252 -0
  224. package/dist/src/lib/structure/StructureInfoPane.svelte +737 -0
  225. package/dist/src/lib/structure/StructureScene.svelte +2255 -0
  226. package/dist/src/lib/structure/atom-properties.ts +316 -0
  227. package/dist/src/lib/structure/bond-order-perception.ts +447 -0
  228. package/dist/src/lib/structure/bonding.ts +944 -0
  229. package/dist/src/lib/structure/export.ts +861 -0
  230. package/dist/src/lib/structure/index.ts +291 -0
  231. package/dist/src/lib/structure/label-placement.ts +130 -0
  232. package/dist/src/lib/structure/measure.ts +45 -0
  233. package/dist/src/lib/structure/parse.ts +1705 -0
  234. package/dist/src/lib/structure/partial-occupancy.ts +183 -0
  235. package/dist/src/lib/structure/pbc.ts +164 -0
  236. package/dist/src/lib/structure/supercell.ts +226 -0
  237. package/dist/src/lib/structure/validation.ts +11 -0
  238. package/dist/src/lib/symmetry/SymmetryStats.svelte +226 -0
  239. package/dist/src/lib/symmetry/WyckoffTable.svelte +120 -0
  240. package/dist/src/lib/symmetry/cell-transform.ts +118 -0
  241. package/dist/src/lib/symmetry/index.ts +348 -0
  242. package/dist/src/lib/symmetry/spacegroups.ts +404 -0
  243. package/dist/src/lib/table/HeatmapTable.svelte +1833 -0
  244. package/dist/src/lib/table/ToggleMenu.svelte +385 -0
  245. package/dist/src/lib/table/index.ts +139 -0
  246. package/dist/src/lib/theme/ThemeControl.svelte +53 -0
  247. package/dist/src/lib/theme/index.ts +107 -0
  248. package/dist/src/lib/theme/themes.mjs +297 -0
  249. package/dist/src/lib/time.ts +71 -0
  250. package/dist/src/lib/tooltip/TooltipContent.svelte +58 -0
  251. package/dist/src/lib/tooltip/index.ts +2 -0
  252. package/dist/src/lib/tooltip/types.ts +13 -0
  253. package/dist/src/lib/trajectory/Trajectory.svelte +1545 -0
  254. package/dist/src/lib/trajectory/TrajectoryError.svelte +128 -0
  255. package/dist/src/lib/trajectory/TrajectoryExportPane.svelte +357 -0
  256. package/dist/src/lib/trajectory/TrajectoryInfoPane.svelte +313 -0
  257. package/dist/src/lib/trajectory/constants.ts +7 -0
  258. package/dist/src/lib/trajectory/extract.ts +196 -0
  259. package/dist/src/lib/trajectory/format-detect.ts +96 -0
  260. package/dist/src/lib/trajectory/frame-reader.ts +456 -0
  261. package/dist/src/lib/trajectory/helpers.ts +217 -0
  262. package/dist/src/lib/trajectory/index.ts +218 -0
  263. package/dist/src/lib/trajectory/parse/ase.ts +109 -0
  264. package/dist/src/lib/trajectory/parse/hdf5.ts +173 -0
  265. package/dist/src/lib/trajectory/parse/index.ts +411 -0
  266. package/dist/src/lib/trajectory/parse/lammps.ts +215 -0
  267. package/dist/src/lib/trajectory/parse/vasp.ts +102 -0
  268. package/dist/src/lib/trajectory/parse/xyz.ts +143 -0
  269. package/dist/src/lib/trajectory/plotting.ts +599 -0
  270. package/dist/src/lib/trajectory/types.ts +13 -0
  271. package/dist/src/lib/utils.ts +56 -0
  272. package/dist/src/lib/xrd/XrdPlot.svelte +615 -0
  273. package/dist/src/lib/xrd/broadening.ts +130 -0
  274. package/dist/src/lib/xrd/calc-xrd.ts +397 -0
  275. package/dist/src/lib/xrd/index.ts +38 -0
  276. package/dist/src/lib/xrd/parse.ts +858 -0
  277. package/dist/webview.js +29421 -0
  278. package/icon.png +0 -0
  279. package/license +1 -1
  280. package/matterviz-0.3.2.vsix +0 -0
  281. package/matterviz-0.3.4.vsix +0 -0
  282. package/matterviz-0.3.5.vsix +0 -0
  283. package/package.json +1460 -215
  284. package/readme.md +171 -98
  285. package/scripts/sync-config.ts +101 -0
  286. package/src/declarations.d.ts +2 -0
  287. package/src/extension.ts +972 -0
  288. package/src/node-io.ts +65 -0
  289. package/src/types.ts +17 -0
  290. package/src/webview/JsonBrowser.svelte +1079 -0
  291. package/src/webview/PlotPanel.svelte +346 -0
  292. package/src/webview/detect.ts +444 -0
  293. package/src/webview/main.ts +764 -0
  294. package/src/webview/plot-utils.ts +250 -0
  295. package/test-fixtures/all-viz-types.json.gz +0 -0
  296. package/test-fixtures/plot-demo-data.json.gz +0 -0
  297. package/tests/detect.test.ts +604 -0
  298. package/tests/extension.test.ts +2041 -0
  299. package/tests/node-io.test.ts +39 -0
  300. package/tests/plot-utils.test.ts +302 -0
  301. package/tests/vite-plugin-json-gz.test.ts +114 -0
  302. package/tests/vscode-mock.ts +18 -0
  303. package/tests/webview.test.ts +231 -0
  304. package/tsconfig.json +20 -0
  305. package/vite-plugin-json-gz.ts +29 -0
  306. package/vite.config.ts +34 -0
  307. package/vite.extension.config.ts +34 -0
  308. package/dist/EmptyState.svelte.d.ts +0 -9
  309. package/dist/FilePicker.svelte +0 -360
  310. package/dist/FilePicker.svelte.d.ts +0 -17
  311. package/dist/Icon.svelte.d.ts +0 -13
  312. package/dist/MillerIndexInput.svelte +0 -66
  313. package/dist/MillerIndexInput.svelte.d.ts +0 -7
  314. package/dist/api/mp.d.ts +0 -6
  315. package/dist/api/mp.js +0 -22
  316. package/dist/api/optimade.d.ts +0 -45
  317. package/dist/api/optimade.js +0 -135
  318. package/dist/brillouin/BrillouinZone.svelte +0 -546
  319. package/dist/brillouin/BrillouinZone.svelte.d.ts +0 -83
  320. package/dist/brillouin/BrillouinZoneControls.svelte +0 -144
  321. package/dist/brillouin/BrillouinZoneControls.svelte.d.ts +0 -17
  322. package/dist/brillouin/BrillouinZoneExportPane.svelte +0 -148
  323. package/dist/brillouin/BrillouinZoneExportPane.svelte.d.ts +0 -15
  324. package/dist/brillouin/BrillouinZoneInfoPane.svelte +0 -146
  325. package/dist/brillouin/BrillouinZoneInfoPane.svelte.d.ts +0 -13
  326. package/dist/brillouin/BrillouinZoneScene.svelte +0 -476
  327. package/dist/brillouin/BrillouinZoneScene.svelte.d.ts +0 -48
  328. package/dist/brillouin/BrillouinZoneTooltip.svelte +0 -92
  329. package/dist/brillouin/BrillouinZoneTooltip.svelte.d.ts +0 -8
  330. package/dist/brillouin/compute.d.ts +0 -17
  331. package/dist/brillouin/compute.js +0 -426
  332. package/dist/brillouin/index.d.ts +0 -8
  333. package/dist/brillouin/index.js +0 -8
  334. package/dist/brillouin/types.d.ts +0 -48
  335. package/dist/brillouin/types.js +0 -1
  336. package/dist/chempot-diagram/ChemPotDiagram.svelte +0 -327
  337. package/dist/chempot-diagram/ChemPotDiagram.svelte.d.ts +0 -13
  338. package/dist/chempot-diagram/ChemPotDiagram2D.svelte +0 -847
  339. package/dist/chempot-diagram/ChemPotDiagram2D.svelte.d.ts +0 -16
  340. package/dist/chempot-diagram/ChemPotDiagram3D.svelte +0 -3194
  341. package/dist/chempot-diagram/ChemPotDiagram3D.svelte.d.ts +0 -16
  342. package/dist/chempot-diagram/ChemPotScene3D.svelte.d.ts +0 -7
  343. package/dist/chempot-diagram/async-compute.svelte.d.ts +0 -3
  344. package/dist/chempot-diagram/async-compute.svelte.js +0 -78
  345. package/dist/chempot-diagram/chempot-worker.d.ts +0 -1
  346. package/dist/chempot-diagram/chempot-worker.js +0 -11
  347. package/dist/chempot-diagram/color.d.ts +0 -10
  348. package/dist/chempot-diagram/color.js +0 -32
  349. package/dist/chempot-diagram/compute.d.ts +0 -48
  350. package/dist/chempot-diagram/compute.js +0 -812
  351. package/dist/chempot-diagram/index.d.ts +0 -6
  352. package/dist/chempot-diagram/index.js +0 -6
  353. package/dist/chempot-diagram/pointer.d.ts +0 -16
  354. package/dist/chempot-diagram/pointer.js +0 -40
  355. package/dist/chempot-diagram/temperature.d.ts +0 -15
  356. package/dist/chempot-diagram/temperature.js +0 -36
  357. package/dist/chempot-diagram/types.d.ts +0 -86
  358. package/dist/chempot-diagram/types.js +0 -28
  359. package/dist/colors/index.d.ts +0 -47
  360. package/dist/colors/index.js +0 -203
  361. package/dist/composition/BarChart.svelte +0 -297
  362. package/dist/composition/BarChart.svelte.d.ts +0 -39
  363. package/dist/composition/BubbleChart.svelte +0 -218
  364. package/dist/composition/BubbleChart.svelte.d.ts +0 -28
  365. package/dist/composition/Composition.svelte +0 -164
  366. package/dist/composition/Composition.svelte.d.ts +0 -15
  367. package/dist/composition/Formula.svelte +0 -265
  368. package/dist/composition/Formula.svelte.d.ts +0 -19
  369. package/dist/composition/FormulaFilter.svelte +0 -1259
  370. package/dist/composition/FormulaFilter.svelte.d.ts +0 -51
  371. package/dist/composition/PieChart.svelte +0 -323
  372. package/dist/composition/PieChart.svelte.d.ts +0 -37
  373. package/dist/composition/format.d.ts +0 -15
  374. package/dist/composition/format.js +0 -109
  375. package/dist/composition/index.d.ts +0 -20
  376. package/dist/composition/index.js +0 -14
  377. package/dist/composition/parse.d.ts +0 -55
  378. package/dist/composition/parse.js +0 -459
  379. package/dist/constants.d.ts +0 -29
  380. package/dist/constants.js +0 -99
  381. package/dist/controls.d.ts +0 -14
  382. package/dist/controls.js +0 -30
  383. package/dist/convex-hull/ConvexHull.svelte +0 -157
  384. package/dist/convex-hull/ConvexHull.svelte.d.ts +0 -13
  385. package/dist/convex-hull/ConvexHull2D.svelte +0 -814
  386. package/dist/convex-hull/ConvexHull2D.svelte.d.ts +0 -11
  387. package/dist/convex-hull/ConvexHull3D.svelte +0 -1790
  388. package/dist/convex-hull/ConvexHull3D.svelte.d.ts +0 -8
  389. package/dist/convex-hull/ConvexHull4D.svelte +0 -1386
  390. package/dist/convex-hull/ConvexHull4D.svelte.d.ts +0 -8
  391. package/dist/convex-hull/ConvexHullControls.svelte +0 -546
  392. package/dist/convex-hull/ConvexHullControls.svelte.d.ts +0 -48
  393. package/dist/convex-hull/ConvexHullInfoPane.svelte +0 -122
  394. package/dist/convex-hull/ConvexHullInfoPane.svelte.d.ts +0 -18
  395. package/dist/convex-hull/ConvexHullStats.svelte +0 -922
  396. package/dist/convex-hull/ConvexHullStats.svelte.d.ts +0 -15
  397. package/dist/convex-hull/ConvexHullTooltip.svelte +0 -131
  398. package/dist/convex-hull/ConvexHullTooltip.svelte.d.ts +0 -33
  399. package/dist/convex-hull/GasPressureControls.svelte +0 -247
  400. package/dist/convex-hull/GasPressureControls.svelte.d.ts +0 -11
  401. package/dist/convex-hull/StructurePopup.svelte +0 -116
  402. package/dist/convex-hull/StructurePopup.svelte.d.ts +0 -18
  403. package/dist/convex-hull/TemperatureSlider.svelte +0 -137
  404. package/dist/convex-hull/TemperatureSlider.svelte.d.ts +0 -8
  405. package/dist/convex-hull/barycentric-coords.d.ts +0 -18
  406. package/dist/convex-hull/barycentric-coords.js +0 -182
  407. package/dist/convex-hull/demo-temperature.d.ts +0 -6
  408. package/dist/convex-hull/demo-temperature.js +0 -40
  409. package/dist/convex-hull/gas-thermodynamics.d.ts +0 -16
  410. package/dist/convex-hull/gas-thermodynamics.js +0 -316
  411. package/dist/convex-hull/helpers.d.ts +0 -103
  412. package/dist/convex-hull/helpers.js +0 -689
  413. package/dist/convex-hull/index.d.ts +0 -118
  414. package/dist/convex-hull/index.js +0 -57
  415. package/dist/convex-hull/thermodynamics.d.ts +0 -66
  416. package/dist/convex-hull/thermodynamics.js +0 -1752
  417. package/dist/convex-hull/types.d.ts +0 -162
  418. package/dist/convex-hull/types.js +0 -36
  419. package/dist/coordination/CoordinationBarPlot.svelte +0 -311
  420. package/dist/coordination/CoordinationBarPlot.svelte.d.ts +0 -30
  421. package/dist/coordination/calc-coordination.d.ts +0 -15
  422. package/dist/coordination/calc-coordination.js +0 -63
  423. package/dist/coordination/index.d.ts +0 -8
  424. package/dist/coordination/index.js +0 -7
  425. package/dist/element/BohrAtom.svelte +0 -147
  426. package/dist/element/BohrAtom.svelte.d.ts +0 -20
  427. package/dist/element/ElementHeading.svelte +0 -26
  428. package/dist/element/ElementHeading.svelte.d.ts +0 -8
  429. package/dist/element/ElementPhoto.svelte +0 -57
  430. package/dist/element/ElementPhoto.svelte.d.ts +0 -9
  431. package/dist/element/ElementStats.svelte +0 -80
  432. package/dist/element/ElementStats.svelte.d.ts +0 -8
  433. package/dist/element/ElementTile.svelte +0 -484
  434. package/dist/element/ElementTile.svelte.d.ts +0 -29
  435. package/dist/element/Nucleus.svelte.d.ts +0 -17
  436. package/dist/element/data.d.ts +0 -2
  437. package/dist/element/data.js +0 -2
  438. package/dist/element/data.json.gz.d.ts +0 -2
  439. package/dist/element/index.d.ts +0 -8
  440. package/dist/element/index.js +0 -8
  441. package/dist/element/types.d.ts +0 -57
  442. package/dist/element/types.js +0 -1
  443. package/dist/feedback/ClickFeedback.svelte +0 -58
  444. package/dist/feedback/ClickFeedback.svelte.d.ts +0 -12
  445. package/dist/feedback/DragOverlay.svelte +0 -42
  446. package/dist/feedback/DragOverlay.svelte.d.ts +0 -7
  447. package/dist/feedback/Spinner.svelte.d.ts +0 -7
  448. package/dist/feedback/StatusMessage.svelte.d.ts +0 -9
  449. package/dist/feedback/index.d.ts +0 -4
  450. package/dist/feedback/index.js +0 -4
  451. package/dist/fermi-surface/FermiSlice.svelte +0 -189
  452. package/dist/fermi-surface/FermiSlice.svelte.d.ts +0 -24
  453. package/dist/fermi-surface/FermiSurface.svelte +0 -600
  454. package/dist/fermi-surface/FermiSurface.svelte.d.ts +0 -83
  455. package/dist/fermi-surface/FermiSurfaceControls.svelte +0 -452
  456. package/dist/fermi-surface/FermiSurfaceControls.svelte.d.ts +0 -35
  457. package/dist/fermi-surface/FermiSurfaceScene.svelte +0 -792
  458. package/dist/fermi-surface/FermiSurfaceScene.svelte.d.ts +0 -50
  459. package/dist/fermi-surface/FermiSurfaceTooltip.svelte +0 -111
  460. package/dist/fermi-surface/FermiSurfaceTooltip.svelte.d.ts +0 -8
  461. package/dist/fermi-surface/compute.d.ts +0 -5
  462. package/dist/fermi-surface/compute.js +0 -538
  463. package/dist/fermi-surface/constants.d.ts +0 -9
  464. package/dist/fermi-surface/constants.js +0 -27
  465. package/dist/fermi-surface/export.d.ts +0 -5
  466. package/dist/fermi-surface/export.js +0 -63
  467. package/dist/fermi-surface/index.d.ts +0 -12
  468. package/dist/fermi-surface/index.js +0 -13
  469. package/dist/fermi-surface/marching-cubes.d.ts +0 -2
  470. package/dist/fermi-surface/marching-cubes.js +0 -2
  471. package/dist/fermi-surface/parse.d.ts +0 -2
  472. package/dist/fermi-surface/parse.js +0 -495
  473. package/dist/fermi-surface/symmetry.d.ts +0 -3
  474. package/dist/fermi-surface/symmetry.js +0 -46
  475. package/dist/fermi-surface/types.d.ts +0 -113
  476. package/dist/fermi-surface/types.js +0 -4
  477. package/dist/heatmap-matrix/HeatmapMatrix.svelte +0 -1527
  478. package/dist/heatmap-matrix/HeatmapMatrix.svelte.d.ts +0 -110
  479. package/dist/heatmap-matrix/HeatmapMatrixControls.svelte +0 -225
  480. package/dist/heatmap-matrix/HeatmapMatrixControls.svelte.d.ts +0 -30
  481. package/dist/heatmap-matrix/index.d.ts +0 -53
  482. package/dist/heatmap-matrix/index.js +0 -100
  483. package/dist/heatmap-matrix/shared.d.ts +0 -2
  484. package/dist/heatmap-matrix/shared.js +0 -4
  485. package/dist/icons.d.ts +0 -569
  486. package/dist/icons.js +0 -648
  487. package/dist/index.d.ts +0 -39
  488. package/dist/index.js +0 -39
  489. package/dist/io/decompress.d.ts +0 -10
  490. package/dist/io/decompress.js +0 -74
  491. package/dist/io/export.d.ts +0 -16
  492. package/dist/io/export.js +0 -316
  493. package/dist/io/fetch.d.ts +0 -5
  494. package/dist/io/fetch.js +0 -39
  495. package/dist/io/file-drop.d.ts +0 -7
  496. package/dist/io/file-drop.js +0 -43
  497. package/dist/io/index.d.ts +0 -7
  498. package/dist/io/index.js +0 -7
  499. package/dist/io/is-binary.d.ts +0 -1
  500. package/dist/io/is-binary.js +0 -20
  501. package/dist/io/types.d.ts +0 -8
  502. package/dist/io/types.js +0 -1
  503. package/dist/io/url-drop.d.ts +0 -2
  504. package/dist/io/url-drop.js +0 -117
  505. package/dist/isosurface/Isosurface.svelte +0 -285
  506. package/dist/isosurface/Isosurface.svelte.d.ts +0 -8
  507. package/dist/isosurface/IsosurfaceControls.svelte +0 -291
  508. package/dist/isosurface/IsosurfaceControls.svelte.d.ts +0 -9
  509. package/dist/isosurface/index.d.ts +0 -5
  510. package/dist/isosurface/index.js +0 -6
  511. package/dist/isosurface/parse.d.ts +0 -6
  512. package/dist/isosurface/parse.js +0 -553
  513. package/dist/isosurface/slice.d.ts +0 -11
  514. package/dist/isosurface/slice.js +0 -140
  515. package/dist/isosurface/types.d.ts +0 -56
  516. package/dist/isosurface/types.js +0 -227
  517. package/dist/labels.d.ts +0 -53
  518. package/dist/labels.js +0 -274
  519. package/dist/layout/FullscreenToggle.svelte +0 -50
  520. package/dist/layout/FullscreenToggle.svelte.d.ts +0 -7
  521. package/dist/layout/InfoCard.svelte +0 -120
  522. package/dist/layout/InfoCard.svelte.d.ts +0 -21
  523. package/dist/layout/InfoTag.svelte +0 -183
  524. package/dist/layout/InfoTag.svelte.d.ts +0 -19
  525. package/dist/layout/PropertyFilter.svelte +0 -244
  526. package/dist/layout/PropertyFilter.svelte.d.ts +0 -24
  527. package/dist/layout/SettingsSection.svelte +0 -148
  528. package/dist/layout/SettingsSection.svelte.d.ts +0 -17
  529. package/dist/layout/SubpageGrid.svelte +0 -82
  530. package/dist/layout/SubpageGrid.svelte.d.ts +0 -14
  531. package/dist/layout/fullscreen.d.ts +0 -9
  532. package/dist/layout/fullscreen.js +0 -53
  533. package/dist/layout/index.d.ts +0 -10
  534. package/dist/layout/index.js +0 -8
  535. package/dist/layout/json-tree/JsonNode.svelte +0 -548
  536. package/dist/layout/json-tree/JsonNode.svelte.d.ts +0 -11
  537. package/dist/layout/json-tree/JsonTree.svelte +0 -1222
  538. package/dist/layout/json-tree/JsonTree.svelte.d.ts +0 -6
  539. package/dist/layout/json-tree/JsonValue.svelte +0 -334
  540. package/dist/layout/json-tree/JsonValue.svelte.d.ts +0 -9
  541. package/dist/layout/json-tree/index.d.ts +0 -3
  542. package/dist/layout/json-tree/index.js +0 -3
  543. package/dist/layout/json-tree/types.d.ts +0 -73
  544. package/dist/layout/json-tree/types.js +0 -3
  545. package/dist/layout/json-tree/utils.d.ts +0 -29
  546. package/dist/layout/json-tree/utils.js +0 -649
  547. package/dist/marching-cubes.d.ts +0 -14
  548. package/dist/marching-cubes.js +0 -542
  549. package/dist/math.d.ts +0 -91
  550. package/dist/math.js +0 -896
  551. package/dist/overlays/ContextMenu.svelte +0 -162
  552. package/dist/overlays/ContextMenu.svelte.d.ts +0 -25
  553. package/dist/overlays/CopyButton.svelte +0 -45
  554. package/dist/overlays/CopyButton.svelte.d.ts +0 -8
  555. package/dist/overlays/DraggablePane.svelte +0 -564
  556. package/dist/overlays/DraggablePane.svelte.d.ts +0 -36
  557. package/dist/overlays/InfoPaneCards.svelte +0 -149
  558. package/dist/overlays/InfoPaneCards.svelte.d.ts +0 -22
  559. package/dist/overlays/index.d.ts +0 -2
  560. package/dist/overlays/index.js +0 -2
  561. package/dist/periodic-table/PeriodicTable.svelte +0 -469
  562. package/dist/periodic-table/PeriodicTable.svelte.d.ts +0 -55
  563. package/dist/periodic-table/PeriodicTableControls.svelte +0 -557
  564. package/dist/periodic-table/PeriodicTableControls.svelte.d.ts +0 -24
  565. package/dist/periodic-table/PropertySelect.svelte +0 -37
  566. package/dist/periodic-table/PropertySelect.svelte.d.ts +0 -13
  567. package/dist/periodic-table/TableInset.svelte.d.ts +0 -9
  568. package/dist/periodic-table/index.d.ts +0 -10
  569. package/dist/periodic-table/index.js +0 -4
  570. package/dist/phase-diagram/IsobaricBinaryPhaseDiagram.svelte +0 -1084
  571. package/dist/phase-diagram/IsobaricBinaryPhaseDiagram.svelte.d.ts +0 -44
  572. package/dist/phase-diagram/PhaseDiagramControls.svelte +0 -449
  573. package/dist/phase-diagram/PhaseDiagramControls.svelte.d.ts +0 -30
  574. package/dist/phase-diagram/PhaseDiagramEditorPane.svelte +0 -126
  575. package/dist/phase-diagram/PhaseDiagramEditorPane.svelte.d.ts +0 -15
  576. package/dist/phase-diagram/PhaseDiagramExportPane.svelte +0 -192
  577. package/dist/phase-diagram/PhaseDiagramExportPane.svelte.d.ts +0 -19
  578. package/dist/phase-diagram/PhaseDiagramTooltip.svelte +0 -392
  579. package/dist/phase-diagram/PhaseDiagramTooltip.svelte.d.ts +0 -16
  580. package/dist/phase-diagram/TdbInfoPanel.svelte +0 -203
  581. package/dist/phase-diagram/TdbInfoPanel.svelte.d.ts +0 -12
  582. package/dist/phase-diagram/build-diagram.d.ts +0 -11
  583. package/dist/phase-diagram/build-diagram.js +0 -167
  584. package/dist/phase-diagram/colors.d.ts +0 -35
  585. package/dist/phase-diagram/colors.js +0 -51
  586. package/dist/phase-diagram/diagram-input.d.ts +0 -33
  587. package/dist/phase-diagram/diagram-input.js +0 -3
  588. package/dist/phase-diagram/index.d.ts +0 -13
  589. package/dist/phase-diagram/index.js +0 -13
  590. package/dist/phase-diagram/parse.d.ts +0 -55
  591. package/dist/phase-diagram/parse.js +0 -276
  592. package/dist/phase-diagram/svg-to-diagram.d.ts +0 -2
  593. package/dist/phase-diagram/svg-to-diagram.js +0 -867
  594. package/dist/phase-diagram/types.d.ts +0 -99
  595. package/dist/phase-diagram/types.js +0 -1
  596. package/dist/phase-diagram/utils.d.ts +0 -118
  597. package/dist/phase-diagram/utils.js +0 -606
  598. package/dist/plot/AxisLabel.svelte +0 -51
  599. package/dist/plot/AxisLabel.svelte.d.ts +0 -16
  600. package/dist/plot/BarPlot.svelte +0 -2265
  601. package/dist/plot/BarPlot.svelte.d.ts +0 -83
  602. package/dist/plot/BarPlotControls.svelte +0 -66
  603. package/dist/plot/BarPlotControls.svelte.d.ts +0 -18
  604. package/dist/plot/ColorBar.svelte +0 -719
  605. package/dist/plot/ColorBar.svelte.d.ts +0 -31
  606. package/dist/plot/ColorScaleSelect.svelte +0 -54
  607. package/dist/plot/ColorScaleSelect.svelte.d.ts +0 -15
  608. package/dist/plot/ElementScatter.svelte +0 -63
  609. package/dist/plot/ElementScatter.svelte.d.ts +0 -14
  610. package/dist/plot/FillArea.svelte +0 -225
  611. package/dist/plot/FillArea.svelte.d.ts +0 -21
  612. package/dist/plot/Histogram.svelte +0 -1672
  613. package/dist/plot/Histogram.svelte.d.ts +0 -50
  614. package/dist/plot/HistogramControls.svelte +0 -212
  615. package/dist/plot/HistogramControls.svelte.d.ts +0 -22
  616. package/dist/plot/InteractiveAxisLabel.svelte +0 -94
  617. package/dist/plot/InteractiveAxisLabel.svelte.d.ts +0 -14
  618. package/dist/plot/Line.svelte +0 -84
  619. package/dist/plot/Line.svelte.d.ts +0 -15
  620. package/dist/plot/PlotControls.svelte +0 -537
  621. package/dist/plot/PlotControls.svelte.d.ts +0 -4
  622. package/dist/plot/PlotLegend.svelte +0 -569
  623. package/dist/plot/PlotLegend.svelte.d.ts +0 -29
  624. package/dist/plot/PlotTooltip.svelte +0 -67
  625. package/dist/plot/PlotTooltip.svelte.d.ts +0 -17
  626. package/dist/plot/PortalSelect.svelte +0 -253
  627. package/dist/plot/PortalSelect.svelte.d.ts +0 -16
  628. package/dist/plot/ReferenceLine.svelte.d.ts +0 -20
  629. package/dist/plot/ReferenceLine3D.svelte +0 -154
  630. package/dist/plot/ReferenceLine3D.svelte.d.ts +0 -14
  631. package/dist/plot/ReferencePlane.svelte +0 -178
  632. package/dist/plot/ReferencePlane.svelte.d.ts +0 -14
  633. package/dist/plot/ScatterPlot.svelte +0 -2845
  634. package/dist/plot/ScatterPlot.svelte.d.ts +0 -93
  635. package/dist/plot/ScatterPlot3D.svelte +0 -502
  636. package/dist/plot/ScatterPlot3D.svelte.d.ts +0 -94
  637. package/dist/plot/ScatterPlot3DControls.svelte +0 -437
  638. package/dist/plot/ScatterPlot3DControls.svelte.d.ts +0 -20
  639. package/dist/plot/ScatterPlot3DScene.svelte +0 -912
  640. package/dist/plot/ScatterPlot3DScene.svelte.d.ts +0 -74
  641. package/dist/plot/ScatterPlotControls.svelte +0 -307
  642. package/dist/plot/ScatterPlotControls.svelte.d.ts +0 -17
  643. package/dist/plot/ScatterPoint.svelte +0 -191
  644. package/dist/plot/ScatterPoint.svelte.d.ts +0 -21
  645. package/dist/plot/SpacegroupBarPlot.svelte +0 -293
  646. package/dist/plot/SpacegroupBarPlot.svelte.d.ts +0 -9
  647. package/dist/plot/Surface3D.svelte +0 -200
  648. package/dist/plot/Surface3D.svelte.d.ts +0 -13
  649. package/dist/plot/ZeroLines.svelte +0 -96
  650. package/dist/plot/ZeroLines.svelte.d.ts +0 -32
  651. package/dist/plot/ZoomRect.svelte +0 -23
  652. package/dist/plot/ZoomRect.svelte.d.ts +0 -8
  653. package/dist/plot/axis-utils.d.ts +0 -19
  654. package/dist/plot/axis-utils.js +0 -80
  655. package/dist/plot/data-cleaning.d.ts +0 -37
  656. package/dist/plot/data-cleaning.js +0 -855
  657. package/dist/plot/data-transform.d.ts +0 -16
  658. package/dist/plot/data-transform.js +0 -45
  659. package/dist/plot/defaults.d.ts +0 -19
  660. package/dist/plot/defaults.js +0 -9
  661. package/dist/plot/fill-utils.d.ts +0 -51
  662. package/dist/plot/fill-utils.js +0 -337
  663. package/dist/plot/hover-lock.svelte.d.ts +0 -14
  664. package/dist/plot/hover-lock.svelte.js +0 -46
  665. package/dist/plot/index.d.ts +0 -37
  666. package/dist/plot/index.js +0 -37
  667. package/dist/plot/interactions.d.ts +0 -12
  668. package/dist/plot/interactions.js +0 -100
  669. package/dist/plot/layout.d.ts +0 -60
  670. package/dist/plot/layout.js +0 -230
  671. package/dist/plot/reference-line.d.ts +0 -60
  672. package/dist/plot/reference-line.js +0 -316
  673. package/dist/plot/scales.d.ts +0 -48
  674. package/dist/plot/scales.js +0 -484
  675. package/dist/plot/svg.d.ts +0 -1
  676. package/dist/plot/svg.js +0 -11
  677. package/dist/plot/types.d.ts +0 -859
  678. package/dist/plot/types.js +0 -103
  679. package/dist/plot/utils/label-placement.d.ts +0 -47
  680. package/dist/plot/utils/label-placement.js +0 -256
  681. package/dist/plot/utils/series-visibility.d.ts +0 -9
  682. package/dist/plot/utils/series-visibility.js +0 -67
  683. package/dist/plot/utils.d.ts +0 -1
  684. package/dist/plot/utils.js +0 -14
  685. package/dist/rdf/RdfPlot.svelte +0 -247
  686. package/dist/rdf/RdfPlot.svelte.d.ts +0 -27
  687. package/dist/rdf/calc-rdf.d.ts +0 -4
  688. package/dist/rdf/calc-rdf.js +0 -111
  689. package/dist/rdf/index.d.ts +0 -23
  690. package/dist/rdf/index.js +0 -2
  691. package/dist/sanitize.d.ts +0 -4
  692. package/dist/sanitize.js +0 -114
  693. package/dist/settings.d.ts +0 -255
  694. package/dist/settings.js +0 -1132
  695. package/dist/spectral/Bands.svelte +0 -1040
  696. package/dist/spectral/Bands.svelte.d.ts +0 -40
  697. package/dist/spectral/BandsAndDos.svelte +0 -128
  698. package/dist/spectral/BandsAndDos.svelte.d.ts +0 -18
  699. package/dist/spectral/BrillouinBandsDos.svelte +0 -248
  700. package/dist/spectral/BrillouinBandsDos.svelte.d.ts +0 -20
  701. package/dist/spectral/Dos.svelte +0 -697
  702. package/dist/spectral/Dos.svelte.d.ts +0 -29
  703. package/dist/spectral/helpers.d.ts +0 -117
  704. package/dist/spectral/helpers.js +0 -1023
  705. package/dist/spectral/index.d.ts +0 -6
  706. package/dist/spectral/index.js +0 -7
  707. package/dist/spectral/types.d.ts +0 -84
  708. package/dist/spectral/types.js +0 -2
  709. package/dist/state.svelte.d.ts +0 -25
  710. package/dist/state.svelte.js +0 -45
  711. package/dist/structure/Arrow.svelte +0 -72
  712. package/dist/structure/Arrow.svelte.d.ts +0 -15
  713. package/dist/structure/AtomLegend.svelte +0 -798
  714. package/dist/structure/AtomLegend.svelte.d.ts +0 -34
  715. package/dist/structure/Bond.svelte +0 -140
  716. package/dist/structure/Bond.svelte.d.ts +0 -9
  717. package/dist/structure/CanvasTooltip.svelte +0 -33
  718. package/dist/structure/CanvasTooltip.svelte.d.ts +0 -12
  719. package/dist/structure/CellSelect.svelte +0 -351
  720. package/dist/structure/CellSelect.svelte.d.ts +0 -13
  721. package/dist/structure/Cylinder.svelte +0 -45
  722. package/dist/structure/Cylinder.svelte.d.ts +0 -10
  723. package/dist/structure/Lattice.svelte +0 -196
  724. package/dist/structure/Lattice.svelte.d.ts +0 -17
  725. package/dist/structure/Structure.svelte +0 -1999
  726. package/dist/structure/Structure.svelte.d.ts +0 -87
  727. package/dist/structure/StructureControls.svelte +0 -1298
  728. package/dist/structure/StructureControls.svelte.d.ts +0 -31
  729. package/dist/structure/StructureExportPane.svelte +0 -251
  730. package/dist/structure/StructureExportPane.svelte.d.ts +0 -17
  731. package/dist/structure/StructureInfoPane.svelte +0 -735
  732. package/dist/structure/StructureInfoPane.svelte.d.ts +0 -19
  733. package/dist/structure/StructureScene.svelte +0 -1905
  734. package/dist/structure/StructureScene.svelte.d.ts +0 -108
  735. package/dist/structure/atom-properties.d.ts +0 -37
  736. package/dist/structure/atom-properties.js +0 -200
  737. package/dist/structure/bond-order-perception.d.ts +0 -13
  738. package/dist/structure/bond-order-perception.js +0 -367
  739. package/dist/structure/bonding.d.ts +0 -42
  740. package/dist/structure/bonding.js +0 -525
  741. package/dist/structure/export.d.ts +0 -20
  742. package/dist/structure/export.js +0 -727
  743. package/dist/structure/index.d.ts +0 -125
  744. package/dist/structure/index.js +0 -171
  745. package/dist/structure/label-placement.d.ts +0 -14
  746. package/dist/structure/label-placement.js +0 -72
  747. package/dist/structure/measure.d.ts +0 -6
  748. package/dist/structure/measure.js +0 -29
  749. package/dist/structure/parse.d.ts +0 -66
  750. package/dist/structure/parse.js +0 -1363
  751. package/dist/structure/partial-occupancy.d.ts +0 -25
  752. package/dist/structure/partial-occupancy.js +0 -99
  753. package/dist/structure/pbc.d.ts +0 -9
  754. package/dist/structure/pbc.js +0 -123
  755. package/dist/structure/supercell.d.ts +0 -8
  756. package/dist/structure/supercell.js +0 -170
  757. package/dist/structure/validation.d.ts +0 -2
  758. package/dist/structure/validation.js +0 -10
  759. package/dist/symmetry/SymmetryStats.svelte +0 -226
  760. package/dist/symmetry/SymmetryStats.svelte.d.ts +0 -21
  761. package/dist/symmetry/WyckoffTable.svelte +0 -113
  762. package/dist/symmetry/WyckoffTable.svelte.d.ts +0 -11
  763. package/dist/symmetry/cell-transform.d.ts +0 -12
  764. package/dist/symmetry/cell-transform.js +0 -91
  765. package/dist/symmetry/index.d.ts +0 -43
  766. package/dist/symmetry/index.js +0 -229
  767. package/dist/symmetry/spacegroups.d.ts +0 -9
  768. package/dist/symmetry/spacegroups.js +0 -394
  769. package/dist/table/HeatmapTable.svelte +0 -1854
  770. package/dist/table/HeatmapTable.svelte.d.ts +0 -49
  771. package/dist/table/ToggleMenu.svelte +0 -376
  772. package/dist/table/ToggleMenu.svelte.d.ts +0 -11
  773. package/dist/table/index.d.ts +0 -74
  774. package/dist/table/index.js +0 -38
  775. package/dist/theme/ThemeControl.svelte +0 -53
  776. package/dist/theme/ThemeControl.svelte.d.ts +0 -9
  777. package/dist/theme/index.d.ts +0 -29
  778. package/dist/theme/index.js +0 -79
  779. package/dist/theme/themes.mjs +0 -285
  780. package/dist/time.d.ts +0 -4
  781. package/dist/time.js +0 -70
  782. package/dist/tooltip/TooltipContent.svelte +0 -58
  783. package/dist/tooltip/TooltipContent.svelte.d.ts +0 -31
  784. package/dist/tooltip/index.d.ts +0 -2
  785. package/dist/tooltip/index.js +0 -2
  786. package/dist/tooltip/types.d.ts +0 -8
  787. package/dist/tooltip/types.js +0 -1
  788. package/dist/trajectory/Trajectory.svelte +0 -1517
  789. package/dist/trajectory/Trajectory.svelte.d.ts +0 -77
  790. package/dist/trajectory/TrajectoryError.svelte +0 -128
  791. package/dist/trajectory/TrajectoryError.svelte.d.ts +0 -13
  792. package/dist/trajectory/TrajectoryExportPane.svelte +0 -357
  793. package/dist/trajectory/TrajectoryExportPane.svelte.d.ts +0 -17
  794. package/dist/trajectory/TrajectoryInfoPane.svelte +0 -313
  795. package/dist/trajectory/TrajectoryInfoPane.svelte.d.ts +0 -17
  796. package/dist/trajectory/constants.d.ts +0 -6
  797. package/dist/trajectory/constants.js +0 -7
  798. package/dist/trajectory/extract.d.ts +0 -5
  799. package/dist/trajectory/extract.js +0 -162
  800. package/dist/trajectory/format-detect.d.ts +0 -9
  801. package/dist/trajectory/format-detect.js +0 -76
  802. package/dist/trajectory/frame-reader.d.ts +0 -17
  803. package/dist/trajectory/frame-reader.js +0 -332
  804. package/dist/trajectory/helpers.d.ts +0 -14
  805. package/dist/trajectory/helpers.js +0 -172
  806. package/dist/trajectory/index.d.ts +0 -63
  807. package/dist/trajectory/index.js +0 -126
  808. package/dist/trajectory/parse/ase.d.ts +0 -2
  809. package/dist/trajectory/parse/ase.js +0 -73
  810. package/dist/trajectory/parse/hdf5.d.ts +0 -2
  811. package/dist/trajectory/parse/hdf5.js +0 -127
  812. package/dist/trajectory/parse/index.d.ts +0 -12
  813. package/dist/trajectory/parse/index.js +0 -299
  814. package/dist/trajectory/parse/lammps.d.ts +0 -5
  815. package/dist/trajectory/parse/lammps.js +0 -179
  816. package/dist/trajectory/parse/vasp.d.ts +0 -2
  817. package/dist/trajectory/parse/vasp.js +0 -68
  818. package/dist/trajectory/parse/xyz.d.ts +0 -2
  819. package/dist/trajectory/parse/xyz.js +0 -110
  820. package/dist/trajectory/plotting.d.ts +0 -28
  821. package/dist/trajectory/plotting.js +0 -423
  822. package/dist/trajectory/types.d.ts +0 -11
  823. package/dist/trajectory/types.js +0 -1
  824. package/dist/utils.d.ts +0 -5
  825. package/dist/utils.js +0 -36
  826. package/dist/xrd/XrdPlot.svelte +0 -615
  827. package/dist/xrd/XrdPlot.svelte.d.ts +0 -28
  828. package/dist/xrd/broadening.d.ts +0 -20
  829. package/dist/xrd/broadening.js +0 -97
  830. package/dist/xrd/calc-xrd.d.ts +0 -37
  831. package/dist/xrd/calc-xrd.js +0 -337
  832. package/dist/xrd/index.d.ts +0 -37
  833. package/dist/xrd/index.js +0 -4
  834. package/dist/xrd/parse.d.ts +0 -13
  835. package/dist/xrd/parse.js +0 -749
  836. /package/dist/{EmptyState.svelte → src/lib/EmptyState.svelte} +0 -0
  837. /package/dist/{Icon.svelte → src/lib/Icon.svelte} +0 -0
  838. /package/dist/{app.css → src/lib/app.css} +0 -0
  839. /package/dist/{chempot-diagram → src/lib/chempot-diagram}/ChemPotScene3D.svelte +0 -0
  840. /package/dist/{colors → src/lib/colors}/alloy-colors.json +0 -0
  841. /package/dist/{colors → src/lib/colors}/dark-mode-colors.json +0 -0
  842. /package/dist/{colors → src/lib/colors}/jmol-colors.json +0 -0
  843. /package/dist/{colors → src/lib/colors}/muted-colors.json +0 -0
  844. /package/dist/{colors → src/lib/colors}/pastel-colors.json +0 -0
  845. /package/dist/{colors → src/lib/colors}/vesta-colors.json +0 -0
  846. /package/dist/{element → src/lib/element}/Nucleus.svelte +0 -0
  847. /package/dist/{element → src/lib/element}/data.json +0 -0
  848. /package/dist/{element → src/lib/element}/data.json.gz +0 -0
  849. /package/dist/{element → src/lib/element}/data.schema.json +0 -0
  850. /package/dist/{element-image-urls.json → src/lib/element-image-urls.json} +0 -0
  851. /package/dist/{feedback → src/lib/feedback}/Spinner.svelte +0 -0
  852. /package/dist/{feedback → src/lib/feedback}/StatusMessage.svelte +0 -0
  853. /package/dist/{periodic-table → src/lib/periodic-table}/TableInset.svelte +0 -0
  854. /package/dist/{plot → src/lib/plot}/ReferenceLine.svelte +0 -0
  855. /package/dist/{xrd → src/lib/xrd}/atomic_scattering_params.json +0 -0
@@ -0,0 +1,1558 @@
1
+ <script lang="ts">
2
+ import { format_value } from '$lib/labels'
3
+ import { FullscreenToggle, set_fullscreen_bg } from '$lib/layout'
4
+ import type {
5
+ AxisLoadError,
6
+ BarStyle,
7
+ DataLoaderFn,
8
+ HistogramHandlerProps,
9
+ PanConfig,
10
+ RefLine,
11
+ RefLineEvent,
12
+ } from '$lib/plot'
13
+ import {
14
+ compute_element_placement,
15
+ HistogramControls,
16
+ PlotAxis,
17
+ PlotLegend,
18
+ ReferenceLine,
19
+ } from '$lib/plot'
20
+ import type { AxisChangeState } from '$lib/plot/axis-utils'
21
+ import { create_axis_change_handler } from '$lib/plot/axis-utils'
22
+ import { extract_series_color, prepare_legend_data } from '$lib/plot/data-transform'
23
+ import { AXIS_DEFAULTS } from '$lib/plot/defaults'
24
+ import {
25
+ create_dimension_tracker,
26
+ create_hover_lock,
27
+ } from '$lib/plot/hover-lock.svelte'
28
+ import {
29
+ get_relative_coords,
30
+ pan_range,
31
+ PINCH_ZOOM_THRESHOLD,
32
+ pixels_to_data_delta,
33
+ } from '$lib/plot/interactions'
34
+ import {
35
+ calc_auto_padding,
36
+ constrain_tooltip_position,
37
+ filter_padding,
38
+ LABEL_GAP_DEFAULT,
39
+ measure_max_tick_width,
40
+ } from '$lib/plot/layout'
41
+ import {
42
+ build_obstacles_norm,
43
+ clip_bar,
44
+ has_explicit_position,
45
+ measured_footprint,
46
+ place_decorations,
47
+ } from '$lib/plot/auto-place'
48
+ import type { IndexedRefLine } from '$lib/plot/reference-line'
49
+ import { group_ref_lines_by_z, index_ref_lines } from '$lib/plot/reference-line'
50
+ import {
51
+ create_scale,
52
+ generate_ticks,
53
+ get_nice_data_range,
54
+ get_tick_label,
55
+ } from '$lib/plot/scales'
56
+ import type {
57
+ BasePlotProps,
58
+ DataSeries,
59
+ InitialRanges,
60
+ LegendConfig,
61
+ PlotConfig,
62
+ ScaleType,
63
+ } from '$lib/plot/types'
64
+ import { get_scale_type_name } from '$lib/plot/types'
65
+ import ZeroLines from '$lib/plot/ZeroLines.svelte'
66
+ import ZoomRect from '$lib/plot/ZoomRect.svelte'
67
+ import { DEFAULTS } from '$lib/settings'
68
+ import { bin, max } from 'd3-array'
69
+ import type { Snippet } from 'svelte'
70
+ import { untrack } from 'svelte'
71
+ import type { HTMLAttributes } from 'svelte/elements'
72
+ import { Tween } from 'svelte/motion'
73
+ import type { Vec2 } from '$lib/math'
74
+ import PlotTooltip from './PlotTooltip.svelte'
75
+ import { bar_path } from './svg'
76
+
77
+ let {
78
+ series = $bindable([]),
79
+ x_axis: x_axis_init = {},
80
+ x2_axis: x2_axis_init = {},
81
+ y_axis: y_axis_init = {},
82
+ y2_axis: y2_axis_init = {},
83
+ display: display_init = DEFAULTS.histogram.display,
84
+ x_range = [null, null],
85
+ x2_range = [null, null],
86
+ y_range = [null, null],
87
+ y2_range = [null, null],
88
+ range_padding = 0.05,
89
+ padding = { t: 20, b: 60, l: 60, r: 20 },
90
+ bins = $bindable(100),
91
+ show_legend = $bindable(true),
92
+ legend = {},
93
+ bar: bar_init = {},
94
+ selected_property = $bindable(``),
95
+ mode = $bindable(`single`),
96
+ tooltip,
97
+ hovered = $bindable(false),
98
+ change = () => {},
99
+ on_bar_click,
100
+ on_bar_hover,
101
+ ref_lines = $bindable([]),
102
+ on_ref_line_click,
103
+ on_ref_line_hover,
104
+ show_controls = $bindable(true),
105
+ controls_open = $bindable(false),
106
+ on_series_toggle = () => {},
107
+ controls_toggle_props,
108
+ controls_pane_props,
109
+ fullscreen = $bindable(false),
110
+ fullscreen_toggle = true,
111
+ children,
112
+ header_controls,
113
+ controls_extra,
114
+ data_loader,
115
+ on_axis_change,
116
+ on_error,
117
+ pan = {},
118
+ ...rest
119
+ }: HTMLAttributes<HTMLDivElement> & BasePlotProps & PlotConfig & {
120
+ series: DataSeries[]
121
+ // Component-specific props
122
+ bins?: number
123
+ show_legend?: boolean
124
+ legend?: LegendConfig | null
125
+ bar?: BarStyle
126
+ selected_property?: string
127
+ mode?: `single` | `overlay`
128
+ tooltip?: Snippet<[HistogramHandlerProps]>
129
+ header_controls?: Snippet<
130
+ [{ height: number; width: number; fullscreen: boolean }]
131
+ >
132
+ controls_extra?: Snippet<[Required<PlotConfig>]>
133
+ change?: (data: { value: number; count: number; property: string } | null) => void
134
+ on_bar_click?: (
135
+ data: {
136
+ value: number
137
+ count: number
138
+ property: string
139
+ event: MouseEvent | KeyboardEvent
140
+ },
141
+ ) => void
142
+ on_bar_hover?: (
143
+ data:
144
+ | { value: number; count: number; property: string; event: MouseEvent }
145
+ | null,
146
+ ) => void
147
+ ref_lines?: RefLine[]
148
+ on_ref_line_click?: (event: RefLineEvent) => void
149
+ on_ref_line_hover?: (event: RefLineEvent | null) => void
150
+ on_series_toggle?: (series_idx: number) => void
151
+ // Interactive axis props
152
+ data_loader?: DataLoaderFn
153
+ on_axis_change?: (
154
+ axis: `x` | `x2` | `y` | `y2`,
155
+ key: string,
156
+ new_series: DataSeries[],
157
+ ) => void
158
+ on_error?: (error: AxisLoadError) => void
159
+ pan?: PanConfig
160
+ } = $props()
161
+
162
+ // Local state for controls (initialized from props, owned by this component)
163
+ // Include key AXIS_DEFAULTS props (range, ticks, scale_type) that PlotControls needs
164
+ // Using $state because these have bindings in HistogramControls/PlotControls
165
+ // untrack() explicitly captures initial prop values (intentional - props provide initial config)
166
+ const { format: _, ...axis_state_defaults } = AXIS_DEFAULTS // Exclude format (has component-specific default)
167
+ let bar = $state(untrack(() => ({ ...DEFAULTS.histogram.bar, ...bar_init })))
168
+ let x_axis = $state(untrack(() => ({ ...axis_state_defaults, ...x_axis_init })))
169
+ // x2-axis needs different default label_shift for top-side positioning
170
+ let x2_axis = $state(untrack(() => ({
171
+ ...axis_state_defaults,
172
+ label_shift: { x: 0, y: 40 },
173
+ ...x2_axis_init,
174
+ })))
175
+ let y_axis = $state(untrack(() => ({ ...axis_state_defaults, ...y_axis_init })))
176
+ // y2-axis needs different default label_shift for right-side positioning
177
+ let y2_axis = $state(untrack(() => ({
178
+ ...axis_state_defaults,
179
+ label_shift: { x: 0, y: 60 },
180
+ ...y2_axis_init,
181
+ })))
182
+ let display = $state(
183
+ untrack(() => ({ ...DEFAULTS.histogram.display, ...display_init })),
184
+ )
185
+
186
+ // Merge component-specific defaults with local state (format comes from here, not AXIS_DEFAULTS)
187
+ const final_x_axis = $derived({ label: `Value`, format: `.2~s`, ...x_axis })
188
+ const final_x2_axis = $derived({ label: `Value`, format: `.2~s`, ...x2_axis })
189
+ const final_y_axis = $derived({ label: `Count`, format: `d`, ...y_axis })
190
+ const final_bar = $derived({ ...DEFAULTS.histogram.bar, ...bar })
191
+ const final_y2_axis = $derived({ label: `Count`, format: `d`, ...y2_axis })
192
+
193
+ // Core state
194
+ let [width, height] = $state([0, 0])
195
+ let wrapper: HTMLDivElement | undefined = $state()
196
+ let svg_element: SVGElement | null = $state(null)
197
+ let clip_path_id = `histogram-clip-${crypto?.randomUUID?.()}`
198
+ let hover_info = $state<HistogramHandlerProps | null>(null)
199
+
200
+ // Reference line hover state
201
+ let hovered_ref_line_idx = $state<number | null>(null)
202
+
203
+ // Interactive axis loading state
204
+ let axis_loading = $state<`x` | `x2` | `y` | `y2` | null>(null)
205
+
206
+ // Compute ref_lines with index and group by z-index (using shared utilities)
207
+ let indexed_ref_lines = $derived(index_ref_lines(ref_lines))
208
+ let ref_lines_by_z = $derived(group_ref_lines_by_z(indexed_ref_lines))
209
+ let tooltip_el = $state<HTMLDivElement | undefined>()
210
+ let drag_state = $state<{
211
+ start: { x: number; y: number } | null
212
+ current: { x: number; y: number } | null
213
+ bounds: DOMRect | null
214
+ }>({ start: null, current: null, bounds: null })
215
+
216
+ // Pan state
217
+ let is_focused = $state(false)
218
+ let shift_held = $state(false)
219
+ let pan_drag_state = $state<
220
+ InitialRanges & { start: { x: number; y: number } } | null
221
+ >(null)
222
+ let touch_state = $state<
223
+ InitialRanges & { start_touches: { x: number; y: number }[] } | null
224
+ >(null)
225
+
226
+ // Legend placement stability state
227
+ let legend_element = $state<HTMLDivElement | undefined>()
228
+ let hovered_legend_series_idx = $state<number | null>(null)
229
+ const legend_hover = create_hover_lock()
230
+ const dim_tracker = create_dimension_tracker()
231
+ let has_initial_legend_placement = $state(false)
232
+
233
+ // Clear pending hover lock timeout on unmount
234
+ $effect(() => () => legend_hover.cleanup())
235
+
236
+ // Derived data
237
+ type IndexedSeries = { series_data: DataSeries; series_idx: number }
238
+ let visible_series_labels = $derived(
239
+ series
240
+ .filter((series_data) => series_data.visible ?? true)
241
+ .map((series_data) => series_data.label)
242
+ .filter((label): label is string => typeof label === `string` && label.length > 0),
243
+ )
244
+ $effect(() => {
245
+ if (mode !== `single`) return
246
+ if (selected_property && visible_series_labels.includes(selected_property)) return
247
+ selected_property = visible_series_labels[0] ?? ``
248
+ })
249
+ let selected_series_entries = $derived<IndexedSeries[]>(
250
+ series
251
+ .map((series_data: DataSeries, series_idx: number) => ({ series_data, series_idx }))
252
+ .filter(({ series_data }) =>
253
+ (series_data.visible ?? true) &&
254
+ (mode !== `single` || !selected_property || series_data.label === selected_property)
255
+ ),
256
+ )
257
+ let selected_series = $derived(
258
+ selected_series_entries.map(({ series_data }) => series_data),
259
+ )
260
+
261
+ // Separate series by y-axis
262
+ let y1_series = $derived(
263
+ selected_series.filter((srs: DataSeries) => (srs.y_axis ?? `y1`) === `y1`),
264
+ )
265
+ let y2_series = $derived(
266
+ selected_series.filter((srs: DataSeries) => srs.y_axis === `y2`),
267
+ )
268
+ let x2_series = $derived(
269
+ selected_series.filter((srs: DataSeries) => srs.x_axis === `x2`),
270
+ )
271
+
272
+ let auto_ranges = $derived.by(() => {
273
+ const all_values = selected_series.flatMap((srs: DataSeries) => srs.y)
274
+ const auto_x = get_nice_data_range(
275
+ all_values.map((val) => ({ x: val, y: 0 })),
276
+ ({ x }) => x,
277
+ x_range,
278
+ final_x_axis.scale_type ?? `linear`,
279
+ range_padding,
280
+ false,
281
+ )
282
+
283
+ const x2_values = x2_series.flatMap((srs: DataSeries) => srs.y)
284
+ const auto_x2 = x2_values.length > 0
285
+ ? get_nice_data_range(
286
+ x2_values.map((val) => ({ x: val, y: 0 })),
287
+ ({ x }) => x,
288
+ x2_range,
289
+ final_x2_axis.scale_type ?? `linear`,
290
+ range_padding,
291
+ false,
292
+ )
293
+ : [0, 1] as Vec2
294
+
295
+ // Calculate y-range for a specific set of series
296
+ const calc_y_range = (
297
+ series_list: typeof selected_series,
298
+ y_limit: typeof y_range,
299
+ scale_type: ScaleType,
300
+ ): Vec2 => {
301
+ const type_name = get_scale_type_name(scale_type)
302
+ if (!series_list.length) {
303
+ const fallback = type_name === `log` ? 1 : 0
304
+ return [fallback, 1]
305
+ }
306
+ const hist = bin().domain([auto_x[0], auto_x[1]]).thresholds(bins)
307
+ const max_count = Math.max(
308
+ 0,
309
+ ...series_list.map((srs: DataSeries) =>
310
+ max(hist(srs.y), (data) => data.length) || 0
311
+ ),
312
+ )
313
+
314
+ // If there's effectively no data, avoid log-range issues (counts can't be <= 0 on log)
315
+ if (max_count <= 0) {
316
+ const fallback = type_name === `log` ? 1 : 0
317
+ return [fallback, 1]
318
+ }
319
+
320
+ const [y0, y1] = get_nice_data_range(
321
+ [{ x: 0, y: 0 }, { x: max_count, y: 0 }],
322
+ ({ x }) => x,
323
+ y_limit,
324
+ scale_type,
325
+ range_padding,
326
+ false,
327
+ )
328
+ // For log scale, minimum must be >= 1 (count can't be 0 on log)
329
+ // For linear/arcsinh, start from 0
330
+ const y_min = type_name === `log` ? Math.max(1, y0) : Math.max(0, y0)
331
+ return [y_min, y1]
332
+ }
333
+
334
+ const y1_range = calc_y_range(
335
+ y1_series,
336
+ y_range,
337
+ final_y_axis.scale_type ?? `linear`,
338
+ )
339
+ const y2_auto_range = calc_y_range(
340
+ y2_series,
341
+ y2_range,
342
+ final_y2_axis.scale_type ?? `linear`,
343
+ )
344
+
345
+ return { x: auto_x, x2: auto_x2, y: y1_range, y2: y2_auto_range }
346
+ })
347
+
348
+ // Initialize ranges
349
+ let ranges = $state({
350
+ initial: {
351
+ x: [0, 1] as Vec2,
352
+ x2: [0, 1] as Vec2,
353
+ y: [0, 1] as Vec2,
354
+ y2: [0, 1] as Vec2,
355
+ },
356
+ current: {
357
+ x: [0, 1] as Vec2,
358
+ x2: [0, 1] as Vec2,
359
+ y: [0, 1] as Vec2,
360
+ y2: [0, 1] as Vec2,
361
+ },
362
+ })
363
+
364
+ $effect(() => {
365
+ // Support one-sided range pinning: merge user range with auto range for null values
366
+ const new_x: [number, number] = final_x_axis.range
367
+ ? [
368
+ final_x_axis.range[0] ?? auto_ranges.x[0],
369
+ final_x_axis.range[1] ?? auto_ranges.x[1],
370
+ ]
371
+ : auto_ranges.x
372
+ const new_x2: [number, number] = final_x2_axis.range
373
+ ? [
374
+ final_x2_axis.range[0] ?? auto_ranges.x2[0],
375
+ final_x2_axis.range[1] ?? auto_ranges.x2[1],
376
+ ]
377
+ : auto_ranges.x2
378
+ const new_y: [number, number] = final_y_axis.range
379
+ ? [
380
+ final_y_axis.range[0] ?? auto_ranges.y[0],
381
+ final_y_axis.range[1] ?? auto_ranges.y[1],
382
+ ]
383
+ : auto_ranges.y
384
+ const new_y2: [number, number] = final_y2_axis.range
385
+ ? [
386
+ final_y2_axis.range[0] ?? auto_ranges.y2[0],
387
+ final_y2_axis.range[1] ?? auto_ranges.y2[1],
388
+ ]
389
+ : auto_ranges.y2
390
+
391
+ // Only update if the initial (data-driven) ranges changed, not when user pans
392
+ // Comparing against initial preserves user's pan/zoom state
393
+ const x_changed = new_x[0] !== ranges.initial.x[0] ||
394
+ new_x[1] !== ranges.initial.x[1]
395
+ const x2_changed = new_x2[0] !== ranges.initial.x2[0] ||
396
+ new_x2[1] !== ranges.initial.x2[1]
397
+ const y_changed = new_y[0] !== ranges.initial.y[0] ||
398
+ new_y[1] !== ranges.initial.y[1]
399
+ const y2_changed = new_y2[0] !== ranges.initial.y2[0] ||
400
+ new_y2[1] !== ranges.initial.y2[1]
401
+
402
+ if (x_changed) [ranges.initial.x, ranges.current.x] = [new_x, new_x]
403
+ if (x2_changed) [ranges.initial.x2, ranges.current.x2] = [new_x2, new_x2]
404
+ if (y_changed) [ranges.initial.y, ranges.current.y] = [new_y, new_y]
405
+ if (y2_changed) [ranges.initial.y2, ranges.current.y2] = [new_y2, new_y2]
406
+ })
407
+
408
+ // Layout: dynamic padding based on tick label widths
409
+ const default_padding = { t: 20, b: 60, l: 60, r: 20 }
410
+ // base_pad reserves space for tick labels/axis titles; pad (below) adds decoration reservations
411
+ let base_pad = $derived(filter_padding(padding, default_padding))
412
+
413
+ // Update padding based on tick label widths (untrack breaks circular dependency)
414
+ $effect(() => {
415
+ const current_ticks_x2 = untrack(() => ticks.x2)
416
+ const current_ticks_y = untrack(() => ticks.y)
417
+ const current_ticks_y2 = untrack(() => ticks.y2)
418
+
419
+ const new_pad = width && height && current_ticks_y.length
420
+ ? calc_auto_padding({
421
+ padding,
422
+ default_padding,
423
+ x2_axis: { ...final_x2_axis, tick_values: current_ticks_x2 },
424
+ y_axis: { ...final_y_axis, tick_values: current_ticks_y },
425
+ y2_axis: { ...final_y2_axis, tick_values: current_ticks_y2 },
426
+ })
427
+ : filter_padding(padding, default_padding)
428
+
429
+ // Add y2 axis label space (calc_auto_padding only accounts for tick labels)
430
+ if (
431
+ width && height && y2_series.length && current_ticks_y2.length &&
432
+ final_y2_axis.label
433
+ ) {
434
+ const inside = final_y2_axis.tick?.label?.inside ?? false
435
+ // When ticks are inside, they don't contribute to padding
436
+ const tick_shift = inside ? 0 : (final_y2_axis.tick?.label?.shift?.x ?? 0) + 8
437
+ const tick_width_contribution = inside ? 0 : tick_label_widths.y2_max
438
+ const label_thickness = Math.round(12 * 1.2)
439
+ new_pad.r = Math.max(
440
+ new_pad.r,
441
+ tick_width_contribution + LABEL_GAP_DEFAULT + tick_shift + label_thickness,
442
+ )
443
+ }
444
+
445
+ // Add x2 axis label space (mirroring y2 logic for top padding)
446
+ if (
447
+ width && height && x2_series.length && current_ticks_x2.length &&
448
+ final_x2_axis.label
449
+ ) {
450
+ const inside = final_x2_axis.tick?.label?.inside ?? false
451
+ const tick_shift = inside
452
+ ? 0
453
+ : Math.abs(final_x2_axis.tick?.label?.shift?.y ?? 0) + 8
454
+ const label_thickness = Math.round(12 * 1.2)
455
+ new_pad.t = Math.max(
456
+ new_pad.t,
457
+ tick_shift + LABEL_GAP_DEFAULT + label_thickness,
458
+ )
459
+ }
460
+
461
+ // Only update if padding actually changed
462
+ if (
463
+ base_pad.t !== new_pad.t || base_pad.b !== new_pad.b ||
464
+ base_pad.l !== new_pad.l || base_pad.r !== new_pad.r
465
+ ) base_pad = new_pad
466
+ })
467
+
468
+ const legend_footprint = $derived(measured_footprint(legend_element, { width: 120, height: 60 }))
469
+ const legend_has_explicit_pos = $derived(has_explicit_position(legend?.style))
470
+
471
+ // Obstacle field in normalized [0,1] plot coords (y=0 at top). Each filled bar is modeled as a
472
+ // vertical segment (top -> baseline) so the legend can't hide inside a tall bar. Built from
473
+ // histogram_bins (pad-independent) + ranges so the crowding decision can't see its own reservation.
474
+ const obstacles_norm = $derived.by(() => {
475
+ if (!width || !height || !histogram_bins.length) return []
476
+ const base_w = width - base_pad.l - base_pad.r
477
+ const base_h = height - base_pad.t - base_pad.b
478
+ if (base_w <= 0 || base_h <= 0) return []
479
+ const bars: { points: { x: number; y: number }[]; draws_line: boolean }[] = []
480
+ for (const hist of histogram_bins) {
481
+ const [rx0, rx1] = hist.x_axis === `x2` ? ranges.current.x2 : ranges.current.x
482
+ const [ry0, ry1] = hist.y_axis === `y2` ? ranges.current.y2 : ranges.current.y
483
+ const x_span = rx1 - rx0
484
+ const y_span = ry1 - ry0
485
+ if (!(x_span > 0) || !(y_span > 0)) continue
486
+ for (const series_bin of hist.bins) {
487
+ if (series_bin.length <= 0) continue
488
+ const x_norm = (((series_bin.x0 ?? 0) + (series_bin.x1 ?? 0)) / 2 - rx0) / x_span
489
+ const top = 1 - (series_bin.length - ry0) / y_span
490
+ const baseline = 1 + ry0 / y_span // normalized y of count=0 (bar foot)
491
+ const seg = clip_bar(true, x_norm, top, baseline)
492
+ if (seg) bars.push(seg)
493
+ }
494
+ }
495
+ return build_obstacles_norm(bars, base_w, base_h)
496
+ })
497
+
498
+ // Move the legend to the bottom margin when no interior spot avoids the bars
499
+ const decor = $derived.by(() =>
500
+ place_decorations({
501
+ base_pad,
502
+ width,
503
+ height,
504
+ obstacles_norm,
505
+ // gate on legend_element (the render signal) not legend_data, whose entries can read pad
506
+ legend: show_legend && legend != null && series.length > 1 &&
507
+ legend_element != null && !legend_has_explicit_pos
508
+ ? { footprint: legend_footprint, clearance: legend?.axis_clearance }
509
+ : null,
510
+ })
511
+ )
512
+ const pad = $derived(decor.pad)
513
+ const legend_auto_outside = $derived(decor.legend_outside)
514
+ const legend_outside_x = $derived(decor.legend_pos.x)
515
+ const legend_outside_y = $derived(decor.legend_pos.y)
516
+
517
+ // Scales and data
518
+ let scales = $derived({
519
+ x: create_scale(
520
+ final_x_axis.scale_type ?? `linear`,
521
+ ranges.current.x,
522
+ [pad.l, width - pad.r],
523
+ ),
524
+ x2: create_scale(
525
+ final_x2_axis.scale_type ?? `linear`,
526
+ ranges.current.x2,
527
+ [pad.l, width - pad.r],
528
+ ),
529
+ y: create_scale(
530
+ final_y_axis.scale_type ?? `linear`,
531
+ ranges.current.y,
532
+ [height - pad.b, pad.t],
533
+ ),
534
+ y2: create_scale(
535
+ final_y2_axis.scale_type ?? `linear`,
536
+ ranges.current.y2,
537
+ [height - pad.b, pad.t],
538
+ ),
539
+ })
540
+
541
+ // Pad-independent binning (no pixel scales) so the auto-place obstacle field can reuse it
542
+ let histogram_bins = $derived.by(() => {
543
+ if (!selected_series.length || !width || !height) return []
544
+ const hist_generator = bin()
545
+ .domain([ranges.current.x[0], ranges.current.x[1]])
546
+ .thresholds(bins)
547
+ const x2_hist_generator = x2_series.length > 0
548
+ ? bin().domain([ranges.current.x2[0], ranges.current.x2[1]]).thresholds(bins)
549
+ : null
550
+ return selected_series_entries.map(({ series_data, series_idx }) => {
551
+ const use_x2 = series_data.x_axis === `x2`
552
+ const active_hist = use_x2 && x2_hist_generator ? x2_hist_generator : hist_generator
553
+ const bins_arr = active_hist(series_data.y)
554
+ return {
555
+ id: series_data.id ?? series_idx,
556
+ series_idx,
557
+ label: series_data.label || `Series ${series_idx + 1}`,
558
+ color: selected_series.length === 1
559
+ ? final_bar.color
560
+ : extract_series_color(series_data),
561
+ bins: bins_arr,
562
+ max_count: max(bins_arr, (data) => data.length) || 0,
563
+ x_axis: series_data.x_axis,
564
+ y_axis: series_data.y_axis,
565
+ }
566
+ })
567
+ })
568
+ // Render-time data adds the pixel scales (pad-dependent)
569
+ let histogram_data = $derived(
570
+ histogram_bins.map((hist) => ({
571
+ ...hist,
572
+ x_scale: hist.x_axis === `x2` ? scales.x2 : scales.x,
573
+ y_scale: hist.y_axis === `y2` ? scales.y2 : scales.y,
574
+ })),
575
+ )
576
+
577
+ let ticks = $derived({
578
+ x: width && height
579
+ ? generate_ticks(
580
+ ranges.current.x,
581
+ final_x_axis.scale_type ?? `linear`,
582
+ final_x_axis.ticks,
583
+ scales.x,
584
+ { default_count: 8 },
585
+ )
586
+ : [],
587
+ x2: width && height && x2_series.length > 0
588
+ ? generate_ticks(
589
+ ranges.current.x2,
590
+ final_x2_axis.scale_type ?? `linear`,
591
+ final_x2_axis.ticks,
592
+ scales.x2,
593
+ { default_count: 8 },
594
+ )
595
+ : [],
596
+ y: width && height
597
+ ? generate_ticks(
598
+ ranges.current.y,
599
+ final_y_axis.scale_type ?? `linear`,
600
+ final_y_axis.ticks,
601
+ scales.y,
602
+ { default_count: 6 },
603
+ )
604
+ : [],
605
+ y2: width && height && y2_series.length > 0
606
+ ? generate_ticks(
607
+ ranges.current.y2,
608
+ final_y2_axis.scale_type ?? `linear`,
609
+ final_y2_axis.ticks,
610
+ scales.y2,
611
+ { default_count: 6 },
612
+ )
613
+ : [],
614
+ })
615
+
616
+ // Cache measured tick-label widths so expensive text measurement only runs
617
+ // when tick values/format change, not on every template rerender.
618
+ let tick_label_widths = $derived({
619
+ x2_max: measure_max_tick_width(ticks.x2, final_x2_axis.format ?? ``),
620
+ y_max: measure_max_tick_width(ticks.y, final_y_axis.format ?? ``),
621
+ y2_max: measure_max_tick_width(ticks.y2, final_y2_axis.format ?? ``),
622
+ })
623
+
624
+ let legend_data = $derived(prepare_legend_data(series))
625
+
626
+ // Collect histogram bar positions for legend placement
627
+ let hist_points_for_placement = $derived.by(() => {
628
+ if (!width || !height || !histogram_data.length) return []
629
+
630
+ const points: { x: number; y: number }[] = []
631
+
632
+ for (const { bins: series_bins, x_scale, y_scale } of histogram_data) {
633
+ for (const series_bin of series_bins) {
634
+ if (series_bin.length > 0) {
635
+ const bar_x = x_scale(((series_bin.x0 ?? 0) + (series_bin.x1 ?? 0)) / 2)
636
+ const bar_y = y_scale(series_bin.length)
637
+ if (isFinite(bar_x) && isFinite(bar_y)) {
638
+ // Add multiple points for taller bars to increase their weight
639
+ // Cap to prevent O(N·count/10) blow-ups for large counts
640
+ const weight = Math.min(20, Math.ceil(series_bin.length / 10))
641
+ for (let idx = 0; idx < weight; idx++) points.push({ x: bar_x, y: bar_y })
642
+ }
643
+ }
644
+ }
645
+ }
646
+ return points
647
+ })
648
+
649
+ // Calculate best legend placement using continuous grid sampling
650
+ let legend_placement = $derived.by(() => {
651
+ const should_place = show_legend && legend != null && series.length > 1
652
+ if (!should_place || !width || !height) return null
653
+
654
+ const plot_width = width - pad.l - pad.r
655
+ const plot_height = height - pad.t - pad.b
656
+
657
+ const result = compute_element_placement({
658
+ plot_bounds: { x: pad.l, y: pad.t, width: plot_width, height: plot_height },
659
+ element: legend_element,
660
+ element_size: { width: 120, height: 60 }, // fallback before first render
661
+ axis_clearance: legend?.axis_clearance,
662
+ exclude_rects: [],
663
+ points: hist_points_for_placement,
664
+ })
665
+
666
+ return result
667
+ })
668
+
669
+ // Tweened legend coordinates for smooth animation - create once, update target via effect
670
+ // untrack() explicitly captures initial tween config (intentional - config set once at mount)
671
+ const tweened_legend_coords = new Tween(
672
+ { x: 0, y: 0 },
673
+ untrack(() => ({ duration: 400, ...legend?.tween })),
674
+ )
675
+
676
+ // Update legend position with stability checks
677
+ $effect(() => {
678
+ if (!width || !height || !legend_placement) return
679
+
680
+ // Track dimensions for resize detection
681
+ const dims_changed = dim_tracker.has_changed(width, height)
682
+ if (dims_changed) dim_tracker.update(width, height)
683
+
684
+ // Only update if: resize occurred, OR (not hover-locked AND (responsive OR not yet initially placed))
685
+ const is_responsive = legend?.responsive ?? false
686
+ const should_update = dims_changed || (!legend_hover.is_locked.current &&
687
+ (is_responsive || !has_initial_legend_placement))
688
+
689
+ if (should_update) {
690
+ tweened_legend_coords.set(
691
+ { x: legend_placement.x, y: legend_placement.y },
692
+ // Skip animation on initial placement to avoid jump from (0, 0)
693
+ has_initial_legend_placement ? undefined : { duration: 0 },
694
+ )
695
+ // Only lock position after we have actual measured size
696
+ if (legend_element) {
697
+ has_initial_legend_placement = true
698
+ }
699
+ }
700
+ })
701
+
702
+ // Event handlers
703
+ const handle_zoom = () => {
704
+ if (!drag_state.start || !drag_state.current) return
705
+ const start_x = scales.x.invert(drag_state.start.x)
706
+ const end_x = scales.x.invert(drag_state.current.x)
707
+ const start_x2 = scales.x2.invert(drag_state.start.x)
708
+ const end_x2 = scales.x2.invert(drag_state.current.x)
709
+ const start_y = scales.y.invert(drag_state.start.y)
710
+ const end_y = scales.y.invert(drag_state.current.y)
711
+ const start_y2 = scales.y2.invert(drag_state.start.y)
712
+ const end_y2 = scales.y2.invert(drag_state.current.y)
713
+
714
+ if (typeof start_x === `number` && typeof end_x === `number`) {
715
+ const dx = Math.abs(drag_state.start.x - drag_state.current.x)
716
+ const dy = Math.abs(drag_state.start.y - drag_state.current.y)
717
+ if (dx > 5 && dy > 5) {
718
+ // Update axis ranges to trigger reactivity and prevent effect from overriding
719
+ x_axis = {
720
+ ...x_axis,
721
+ range: [Math.min(start_x, end_x), Math.max(start_x, end_x)],
722
+ }
723
+ if (x2_series.length > 0) {
724
+ x2_axis = {
725
+ ...x2_axis,
726
+ range: [Math.min(start_x2, end_x2), Math.max(start_x2, end_x2)],
727
+ }
728
+ }
729
+ y_axis = {
730
+ ...y_axis,
731
+ range: [Math.min(start_y, end_y), Math.max(start_y, end_y)],
732
+ }
733
+ y2_axis = {
734
+ ...y2_axis,
735
+ range: [Math.min(start_y2, end_y2), Math.max(start_y2, end_y2)],
736
+ }
737
+ }
738
+ }
739
+ }
740
+
741
+ const on_window_mouse_move = (evt: MouseEvent) => {
742
+ if (!drag_state.start || !drag_state.bounds) return
743
+ drag_state.current = {
744
+ x: evt.clientX - drag_state.bounds.left,
745
+ y: evt.clientY - drag_state.bounds.top,
746
+ }
747
+ }
748
+
749
+ const on_window_mouse_up = () => {
750
+ handle_zoom()
751
+ drag_state = { start: null, current: null, bounds: null }
752
+ window.removeEventListener(`mousemove`, on_window_mouse_move)
753
+ window.removeEventListener(`mouseup`, on_window_mouse_up)
754
+ document.body.style.cursor = `default`
755
+ }
756
+
757
+ // Pan drag handlers
758
+ const on_pan_move = (evt: MouseEvent) => {
759
+ if (!pan_drag_state) return
760
+ const dx = evt.clientX - pan_drag_state.start.x
761
+ const dy = evt.clientY - pan_drag_state.start.y
762
+
763
+ // Convert pixel delta to data delta (note: drag direction is inverted for natural pan feel)
764
+ const plot_width = width - pad.l - pad.r
765
+ const plot_height = height - pad.t - pad.b
766
+ const sensitivity = pan?.drag_sensitivity ?? 1
767
+
768
+ const x_delta = pixels_to_data_delta(
769
+ -dx * sensitivity,
770
+ pan_drag_state.initial_x_range,
771
+ plot_width,
772
+ )
773
+ const x2_delta = pixels_to_data_delta(
774
+ -dx * sensitivity,
775
+ pan_drag_state.initial_x2_range,
776
+ plot_width,
777
+ )
778
+ const y_delta = pixels_to_data_delta(
779
+ dy * sensitivity,
780
+ pan_drag_state.initial_y_range,
781
+ plot_height,
782
+ )
783
+ const y2_delta = pixels_to_data_delta(
784
+ dy * sensitivity,
785
+ pan_drag_state.initial_y2_range,
786
+ plot_height,
787
+ )
788
+
789
+ ranges.current.x = pan_range(pan_drag_state.initial_x_range, x_delta)
790
+ ranges.current.x2 = pan_range(pan_drag_state.initial_x2_range, x2_delta)
791
+ ranges.current.y = pan_range(pan_drag_state.initial_y_range, y_delta)
792
+ ranges.current.y2 = pan_range(pan_drag_state.initial_y2_range, y2_delta)
793
+ }
794
+
795
+ const on_pan_end = () => {
796
+ pan_drag_state = null
797
+ document.body.style.cursor = ``
798
+ window.removeEventListener(`mousemove`, on_pan_move)
799
+ window.removeEventListener(`mouseup`, on_pan_end)
800
+ }
801
+
802
+ function handle_mouse_down(evt: MouseEvent) {
803
+ const coords = get_relative_coords(evt)
804
+ if (!coords || !svg_element) return
805
+
806
+ // Check if pan is enabled and shift is held for pan mode
807
+ const pan_enabled = pan?.enabled !== false
808
+ if (pan_enabled && evt.shiftKey) {
809
+ evt.preventDefault()
810
+ pan_drag_state = {
811
+ start: { x: evt.clientX, y: evt.clientY },
812
+ initial_x_range: [...ranges.current.x] as [number, number],
813
+ initial_x2_range: [...ranges.current.x2] as [number, number],
814
+ initial_y_range: [...ranges.current.y] as [number, number],
815
+ initial_y2_range: [...ranges.current.y2] as [number, number],
816
+ }
817
+ document.body.style.cursor = `grabbing`
818
+ window.addEventListener(`mousemove`, on_pan_move)
819
+ window.addEventListener(`mouseup`, on_pan_end)
820
+ return
821
+ }
822
+
823
+ drag_state = {
824
+ start: coords,
825
+ current: coords,
826
+ bounds: svg_element.getBoundingClientRect(),
827
+ }
828
+ window.addEventListener(`mousemove`, on_window_mouse_move)
829
+ window.addEventListener(`mouseup`, on_window_mouse_up)
830
+ evt.preventDefault()
831
+ }
832
+
833
+ // Wheel handler for pan (requires focus and shift)
834
+ function handle_wheel(evt: WheelEvent) {
835
+ const pan_enabled = pan?.enabled !== false
836
+ // Only capture wheel when focused AND Shift is held
837
+ // Use shift_held state (tracked via keydown/keyup) for compatibility with synthetic events
838
+ if (!pan_enabled || !is_focused || !shift_held) return
839
+
840
+ evt.preventDefault()
841
+
842
+ // Clamp to at least 1 to avoid Infinity deltas when padding equals container size
843
+ const plot_width = Math.max(1, width - pad.l - pad.r)
844
+ const plot_height = Math.max(1, height - pad.t - pad.b)
845
+ const sensitivity = pan?.wheel_sensitivity ?? 1
846
+
847
+ // Determine pan direction based on wheel delta
848
+ const x_delta = pixels_to_data_delta(
849
+ evt.deltaX * sensitivity,
850
+ ranges.current.x,
851
+ plot_width,
852
+ )
853
+ const x2_delta = pixels_to_data_delta(
854
+ evt.deltaX * sensitivity,
855
+ ranges.current.x2,
856
+ plot_width,
857
+ )
858
+ const y_delta = pixels_to_data_delta(
859
+ evt.deltaY * sensitivity,
860
+ ranges.current.y,
861
+ plot_height,
862
+ )
863
+ const y2_delta = pixels_to_data_delta(
864
+ evt.deltaY * sensitivity,
865
+ ranges.current.y2,
866
+ plot_height,
867
+ )
868
+
869
+ if (Math.abs(evt.deltaX) > Math.abs(evt.deltaY)) {
870
+ ranges.current.x = pan_range(ranges.current.x, x_delta)
871
+ ranges.current.x2 = pan_range(ranges.current.x2, x2_delta)
872
+ } else {
873
+ ranges.current.y = pan_range(ranges.current.y, y_delta)
874
+ ranges.current.y2 = pan_range(ranges.current.y2, y2_delta)
875
+ }
876
+ }
877
+
878
+ // Touch handlers for pinch-zoom and two-finger pan
879
+ function handle_touch_start(evt: TouchEvent) {
880
+ const touch_enabled = pan?.enabled !== false && pan?.touch_enabled !== false
881
+ if (!touch_enabled || evt.touches.length !== 2) return
882
+
883
+ evt.preventDefault()
884
+ const touches = Array.from(evt.touches)
885
+ touch_state = {
886
+ start_touches: touches.map((touch) => ({ x: touch.clientX, y: touch.clientY })),
887
+ initial_x_range: [...ranges.current.x] as [number, number],
888
+ initial_x2_range: [...ranges.current.x2] as [number, number],
889
+ initial_y_range: [...ranges.current.y] as [number, number],
890
+ initial_y2_range: [...ranges.current.y2] as [number, number],
891
+ }
892
+ }
893
+
894
+ function handle_touch_move(evt: TouchEvent) {
895
+ if (!touch_state || evt.touches.length !== 2) return
896
+ evt.preventDefault()
897
+
898
+ const [t1, t2] = Array.from(evt.touches)
899
+ const [s1, s2] = touch_state.start_touches
900
+
901
+ // Calculate center movement for pan
902
+ const start_center = { x: (s1.x + s2.x) / 2, y: (s1.y + s2.y) / 2 }
903
+ const curr_center = {
904
+ x: (t1.clientX + t2.clientX) / 2,
905
+ y: (t1.clientY + t2.clientY) / 2,
906
+ }
907
+ const dx = curr_center.x - start_center.x
908
+ const dy = curr_center.y - start_center.y
909
+
910
+ // Calculate pinch scale (curr/start so spread = zoom out, pinch = zoom in)
911
+ const start_dist = Math.hypot(s2.x - s1.x, s2.y - s1.y)
912
+ // Guard against zero-distance pinch to avoid Infinity scale
913
+ if (start_dist < Number.EPSILON) return
914
+ const curr_dist = Math.hypot(t2.clientX - t1.clientX, t2.clientY - t1.clientY)
915
+ const scale = curr_dist / start_dist
916
+
917
+ // Clamp to at least 1 to avoid Infinity deltas when padding equals container size
918
+ const plot_width = Math.max(1, width - pad.l - pad.r)
919
+ const plot_height = Math.max(1, height - pad.t - pad.b)
920
+
921
+ // If scale changed significantly, treat as pinch-zoom
922
+ // Also guard against scale being too small to avoid division by zero
923
+ if (Math.abs(scale - 1) > PINCH_ZOOM_THRESHOLD && scale > Number.EPSILON) {
924
+ // Pinch zoom centered on gesture center
925
+ // Divide by scale so spread (scale > 1) = smaller span (zoom in)
926
+ const x_span = touch_state.initial_x_range[1] - touch_state.initial_x_range[0]
927
+ const x2_span = touch_state.initial_x2_range[1] -
928
+ touch_state.initial_x2_range[0]
929
+ const y_span = touch_state.initial_y_range[1] - touch_state.initial_y_range[0]
930
+ const y2_span = touch_state.initial_y2_range[1] -
931
+ touch_state.initial_y2_range[0]
932
+ const x_center =
933
+ (touch_state.initial_x_range[0] + touch_state.initial_x_range[1]) / 2
934
+ const x2_center =
935
+ (touch_state.initial_x2_range[0] + touch_state.initial_x2_range[1]) / 2
936
+ const y_center =
937
+ (touch_state.initial_y_range[0] + touch_state.initial_y_range[1]) / 2
938
+ const y2_center =
939
+ (touch_state.initial_y2_range[0] + touch_state.initial_y2_range[1]) / 2
940
+
941
+ ranges.current.x = [
942
+ x_center - x_span / scale / 2,
943
+ x_center + x_span / scale / 2,
944
+ ]
945
+ ranges.current.x2 = [
946
+ x2_center - x2_span / scale / 2,
947
+ x2_center + x2_span / scale / 2,
948
+ ]
949
+ ranges.current.y = [
950
+ y_center - y_span / scale / 2,
951
+ y_center + y_span / scale / 2,
952
+ ]
953
+ ranges.current.y2 = [
954
+ y2_center - y2_span / scale / 2,
955
+ y2_center + y2_span / scale / 2,
956
+ ]
957
+ } else {
958
+ // Pan
959
+ const x_delta = pixels_to_data_delta(
960
+ -dx,
961
+ touch_state.initial_x_range,
962
+ plot_width,
963
+ )
964
+ const x2_delta = pixels_to_data_delta(
965
+ -dx,
966
+ touch_state.initial_x2_range,
967
+ plot_width,
968
+ )
969
+ const y_delta = pixels_to_data_delta(
970
+ dy,
971
+ touch_state.initial_y_range,
972
+ plot_height,
973
+ )
974
+ const y2_delta = pixels_to_data_delta(
975
+ dy,
976
+ touch_state.initial_y2_range,
977
+ plot_height,
978
+ )
979
+ ranges.current.x = pan_range(touch_state.initial_x_range, x_delta)
980
+ ranges.current.x2 = pan_range(touch_state.initial_x2_range, x2_delta)
981
+ ranges.current.y = pan_range(touch_state.initial_y_range, y_delta)
982
+ ranges.current.y2 = pan_range(touch_state.initial_y2_range, y2_delta)
983
+ }
984
+ }
985
+
986
+ function handle_touch_end() {
987
+ touch_state = null
988
+ }
989
+
990
+ function handle_double_click() {
991
+ // Reset zoom to initial ranges (undo any pan/zoom)
992
+ ranges.current.x = [...ranges.initial.x] as [number, number]
993
+ ranges.current.x2 = [...ranges.initial.x2] as [number, number]
994
+ ranges.current.y = [...ranges.initial.y] as [number, number]
995
+ ranges.current.y2 = [...ranges.initial.y2] as [number, number]
996
+ // Also reset axis props so future data changes recalculate auto ranges
997
+ x_axis = { ...x_axis, range: [null, null] }
998
+ x2_axis = { ...x2_axis, range: [null, null] }
999
+ y_axis = { ...y_axis, range: [null, null] }
1000
+ y2_axis = { ...y2_axis, range: [null, null] }
1001
+ }
1002
+
1003
+ function handle_mouse_move(
1004
+ evt: MouseEvent,
1005
+ value: number,
1006
+ count: number,
1007
+ property: string,
1008
+ active_y_axis: `y1` | `y2` = `y1`,
1009
+ series_idx: number = 0,
1010
+ active_x_axis: `x1` | `x2` = `x1`,
1011
+ ) {
1012
+ hovered = true
1013
+ hover_info = {
1014
+ value,
1015
+ count,
1016
+ property,
1017
+ active_y_axis,
1018
+ active_x_axis,
1019
+ x: value,
1020
+ y: count,
1021
+ series_idx,
1022
+ metadata: null,
1023
+ label: property,
1024
+ x_axis: active_x_axis === `x2` ? x2_axis : x_axis,
1025
+ x2_axis,
1026
+ y_axis: active_y_axis === `y2` ? y2_axis : y_axis,
1027
+ y2_axis,
1028
+ }
1029
+ change({ value, count, property })
1030
+ on_bar_hover?.({ value, count, property, event: evt })
1031
+ }
1032
+
1033
+ function toggle_series_visibility(series_idx: number) {
1034
+ if (series_idx >= 0 && series_idx < series.length) {
1035
+ // Toggle series visibility
1036
+ series = series.map((srs: DataSeries, idx: number) => {
1037
+ if (idx === series_idx) return { ...srs, visible: !(srs.visible ?? true) }
1038
+ return srs
1039
+ })
1040
+ ;(legend?.on_toggle || on_series_toggle)(series_idx)
1041
+ }
1042
+ }
1043
+
1044
+ // Set theme-aware background when entering fullscreen
1045
+ $effect(() => {
1046
+ set_fullscreen_bg(wrapper, fullscreen, `--histogram-fullscreen-bg`)
1047
+ })
1048
+
1049
+ // State accessors for shared axis change handler
1050
+ const axis_state: AxisChangeState<DataSeries> = {
1051
+ get_axis: (axis) => {
1052
+ if (axis === `x`) return x_axis
1053
+ if (axis === `x2`) return x2_axis
1054
+ if (axis === `y`) return y_axis
1055
+ return y2_axis
1056
+ },
1057
+ set_axis: (axis, config) => {
1058
+ // Spread into existing state to preserve merged type structure
1059
+ if (axis === `x`) x_axis = { ...x_axis, ...config }
1060
+ else if (axis === `x2`) x2_axis = { ...x2_axis, ...config }
1061
+ else if (axis === `y`) y_axis = { ...y_axis, ...config }
1062
+ else y2_axis = { ...y2_axis, ...config }
1063
+ },
1064
+ get_series: () => series,
1065
+ set_series: (new_series) => (series = new_series),
1066
+ get_loading: () => axis_loading,
1067
+ set_loading: (axis) => (axis_loading = axis),
1068
+ }
1069
+
1070
+ // Create shared handler bound to this component's state
1071
+ // Using $derived so handler updates when callback props change
1072
+ const handle_axis_change = $derived(create_axis_change_handler(
1073
+ axis_state,
1074
+ data_loader,
1075
+ on_axis_change,
1076
+ on_error,
1077
+ ))
1078
+
1079
+ let auto_load_attempted = false // prevent infinite retries on failure
1080
+
1081
+ // Auto-load data if series is empty but options exist (runs once)
1082
+ $effect(() => {
1083
+ if (series.length === 0 && data_loader && !auto_load_attempted) {
1084
+ // Check x-axis first, then y-axis
1085
+ if (x_axis.options?.length) {
1086
+ auto_load_attempted = true
1087
+ const first_key = x_axis.selected_key ?? x_axis.options[0].key
1088
+ handle_axis_change(`x`, first_key).catch(() => {})
1089
+ } else if (y_axis.options?.length) {
1090
+ auto_load_attempted = true
1091
+ const first_key = y_axis.selected_key ?? y_axis.options[0].key
1092
+ handle_axis_change(`y`, first_key).catch(() => {})
1093
+ }
1094
+ }
1095
+ })
1096
+ </script>
1097
+
1098
+ {#snippet ref_lines_layer(lines: IndexedRefLine[])}
1099
+ {#each lines as line (line.id ?? line.idx)}
1100
+ <ReferenceLine
1101
+ ref_line={line}
1102
+ line_idx={line.idx}
1103
+ x_min={line.x_axis === `x2` ? ranges.current.x2[0] : ranges.current.x[0]}
1104
+ x_max={line.x_axis === `x2` ? ranges.current.x2[1] : ranges.current.x[1]}
1105
+ y_min={line.y_axis === `y2` ? ranges.current.y2[0] : ranges.current.y[0]}
1106
+ y_max={line.y_axis === `y2` ? ranges.current.y2[1] : ranges.current.y[1]}
1107
+ x_scale={scales.x}
1108
+ x2_scale={scales.x2}
1109
+ y_scale={scales.y}
1110
+ y2_scale={scales.y2}
1111
+ {clip_path_id}
1112
+ hovered_line_idx={hovered_ref_line_idx}
1113
+ on_click={(event: RefLineEvent) => {
1114
+ line.on_click?.(event)
1115
+ on_ref_line_click?.(event)
1116
+ }}
1117
+ on_hover={(event: RefLineEvent | null) => {
1118
+ hovered_ref_line_idx = event?.line_idx ?? null
1119
+ line.on_hover?.(event)
1120
+ on_ref_line_hover?.(event)
1121
+ }}
1122
+ />
1123
+ {/each}
1124
+ {/snippet}
1125
+
1126
+ <svelte:window
1127
+ onkeydown={(evt) => {
1128
+ if (evt.key === `Escape` && fullscreen) {
1129
+ evt.preventDefault()
1130
+ fullscreen = false
1131
+ }
1132
+ if (evt.key === `Shift`) shift_held = true
1133
+ }}
1134
+ onkeyup={(evt) => {
1135
+ if (evt.key === `Shift`) shift_held = false
1136
+ }}
1137
+ />
1138
+
1139
+ <div
1140
+ class="histogram"
1141
+ bind:this={wrapper}
1142
+ bind:clientWidth={width}
1143
+ bind:clientHeight={height}
1144
+ {...rest}
1145
+ class:fullscreen
1146
+ >
1147
+ {#if width && height}
1148
+ <div class="header-controls">
1149
+ {@render header_controls?.({ height, width, fullscreen })}
1150
+ {#if fullscreen_toggle}
1151
+ <FullscreenToggle bind:fullscreen />
1152
+ {/if}
1153
+ </div>
1154
+ {/if}
1155
+ <!-- svelte-ignore a11y_no_noninteractive_tabindex -->
1156
+ <!-- svelte-ignore a11y_no_noninteractive_element_interactions -->
1157
+ <svg
1158
+ bind:this={svg_element}
1159
+ role="application"
1160
+ aria-label={rest[`aria-label`] ??
1161
+ ([final_x_axis.label, final_y_axis.label].filter(Boolean).join(` vs `) ||
1162
+ `Histogram`)}
1163
+ tabindex="0"
1164
+ onfocusin={() => (is_focused = true)}
1165
+ onfocusout={() => (is_focused = false)}
1166
+ onmouseenter={() => (hovered = true)}
1167
+ onmousedown={handle_mouse_down}
1168
+ onmouseleave={() => {
1169
+ hovered = false
1170
+ hover_info = null
1171
+ on_bar_hover?.(null)
1172
+ }}
1173
+ ondblclick={handle_double_click}
1174
+ onwheel={handle_wheel}
1175
+ ontouchstart={handle_touch_start}
1176
+ ontouchmove={handle_touch_move}
1177
+ ontouchend={handle_touch_end}
1178
+ style:cursor={pan_drag_state
1179
+ ? `grabbing`
1180
+ : shift_held && pan?.enabled !== false
1181
+ ? `grab`
1182
+ : `crosshair`}
1183
+ onkeydown={(event) => {
1184
+ if (event.key === `Escape` && drag_state.start) {
1185
+ drag_state = { start: null, current: null, bounds: null }
1186
+ }
1187
+ if ([`Enter`, ` `].includes(event.key)) {
1188
+ event.preventDefault()
1189
+ handle_double_click()
1190
+ }
1191
+ }}
1192
+ >
1193
+ <!-- Define clip path for chart area -->
1194
+ <defs>
1195
+ <clipPath id={clip_path_id}>
1196
+ <rect
1197
+ x={pad.l}
1198
+ y={pad.t}
1199
+ width={width - pad.l - pad.r}
1200
+ height={height - pad.t - pad.b}
1201
+ />
1202
+ </clipPath>
1203
+ </defs>
1204
+
1205
+ <!-- Reference lines: below grid (must render first to appear behind grid) -->
1206
+ {@render ref_lines_layer(ref_lines_by_z.below_grid)}
1207
+
1208
+ <ZoomRect start={drag_state.start} current={drag_state.current} />
1209
+
1210
+ <ZeroLines
1211
+ {display}
1212
+ x_scale_fn={scales.x}
1213
+ x2_scale_fn={scales.x2}
1214
+ y_scale_fn={scales.y}
1215
+ y2_scale_fn={scales.y2}
1216
+ x_range={ranges.current.x}
1217
+ x2_range={ranges.current.x2}
1218
+ y_range={ranges.current.y}
1219
+ y2_range={ranges.current.y2}
1220
+ x_scale_type={final_x_axis.scale_type}
1221
+ x2_scale_type={final_x2_axis.scale_type}
1222
+ y_scale_type={final_y_axis.scale_type}
1223
+ y2_scale_type={final_y2_axis.scale_type}
1224
+ has_x2={x2_series.length > 0}
1225
+ has_y2={y2_series.length > 0}
1226
+ {width}
1227
+ {height}
1228
+ {pad}
1229
+ />
1230
+
1231
+ <!-- Reference lines: below lines -->
1232
+ {@render ref_lines_layer(ref_lines_by_z.below_lines)}
1233
+
1234
+ <!-- Reference lines: below points -->
1235
+ {@render ref_lines_layer(ref_lines_by_z.below_points)}
1236
+
1237
+ <!-- X-axis -->
1238
+ <PlotAxis
1239
+ side="x"
1240
+ ticks={ticks.x as number[]}
1241
+ place={scales.x}
1242
+ axis={final_x_axis}
1243
+ {pad}
1244
+ {width}
1245
+ {height}
1246
+ show_grid={display.x_grid}
1247
+ tick_label={(tick) => get_tick_label(tick, final_x_axis.ticks)}
1248
+ label_x={(pad.l + width - pad.r) / 2 + (final_x_axis.label_shift?.x ?? 0)}
1249
+ label_y={height - 10 + (final_x_axis.label_shift?.y ?? 0)}
1250
+ axis_loading={axis_loading === `x`}
1251
+ on_axis_change={(key) => handle_axis_change(`x`, key)}
1252
+ />
1253
+
1254
+ <!-- X2-axis (Top) -->
1255
+ {#if x2_series.length > 0}
1256
+ <PlotAxis
1257
+ side="x2"
1258
+ ticks={ticks.x2 as number[]}
1259
+ place={scales.x2}
1260
+ axis={final_x2_axis}
1261
+ {pad}
1262
+ {width}
1263
+ {height}
1264
+ show_grid={display.x2_grid}
1265
+ tick_label={(tick) => get_tick_label(tick, final_x2_axis.ticks)}
1266
+ label_x={(pad.l + width - pad.r) / 2 + (final_x2_axis.label_shift?.x ?? 0)}
1267
+ label_y={Math.max(12, pad.t - (final_x2_axis.label_shift?.y ?? 40))}
1268
+ axis_loading={axis_loading === `x2`}
1269
+ on_axis_change={(key) => handle_axis_change(`x2`, key)}
1270
+ />
1271
+ {/if}
1272
+
1273
+ <!-- Y-axis -->
1274
+ <PlotAxis
1275
+ side="y"
1276
+ ticks={ticks.y as number[]}
1277
+ place={scales.y}
1278
+ axis={final_y_axis}
1279
+ {pad}
1280
+ {width}
1281
+ {height}
1282
+ show_grid={display.y_grid}
1283
+ tick_label={(tick) => get_tick_label(tick, final_y_axis.ticks)}
1284
+ label_x={Math.max(
1285
+ 12,
1286
+ pad.l - (final_y_axis.tick?.label?.inside ? 0 : tick_label_widths.y_max) -
1287
+ LABEL_GAP_DEFAULT,
1288
+ ) + (final_y_axis.label_shift?.x ?? 0)}
1289
+ label_y={pad.t + (height - pad.t - pad.b) / 2 + (final_y_axis.label_shift?.y ?? 0)}
1290
+ axis_loading={axis_loading === `y`}
1291
+ on_axis_change={(key) => handle_axis_change(`y`, key)}
1292
+ />
1293
+
1294
+ <!-- Y2-axis (Right) -->
1295
+ {#if y2_series.length > 0}
1296
+ {@const y2_inside = final_y2_axis.tick?.label?.inside ?? false}
1297
+ {@const y2_tick_shift = y2_inside ? 0 : (final_y2_axis.tick?.label?.shift?.x ?? 0) + 8}
1298
+ {@const y2_tick_width = y2_inside ? 0 : tick_label_widths.y2_max}
1299
+ <PlotAxis
1300
+ side="y2"
1301
+ ticks={ticks.y2 as number[]}
1302
+ place={scales.y2}
1303
+ axis={final_y2_axis}
1304
+ {pad}
1305
+ {width}
1306
+ {height}
1307
+ show_grid={display.y2_grid}
1308
+ tick_label={(tick) => get_tick_label(tick, final_y2_axis.ticks)}
1309
+ label_x={width - pad.r + y2_tick_shift + y2_tick_width + LABEL_GAP_DEFAULT +
1310
+ (final_y2_axis.label_shift?.x ?? 0)}
1311
+ label_y={pad.t + (height - pad.t - pad.b) / 2 + (final_y2_axis.label_shift?.y ?? 0)}
1312
+ axis_loading={axis_loading === `y2`}
1313
+ on_axis_change={(key) => handle_axis_change(`y2`, key)}
1314
+ />
1315
+ {/if}
1316
+
1317
+ <!-- Histogram bars (rendered after axes so bars appear above grid lines) -->
1318
+ {#each histogram_data as
1319
+ { id, bins, color, label, x_scale, y_scale, x_axis: srs_x_axis, y_axis, series_idx },
1320
+ idx
1321
+ (id ?? idx)
1322
+ }
1323
+ <g
1324
+ class="histogram-series"
1325
+ data-series-idx={series_idx}
1326
+ opacity={hovered_legend_series_idx !== null &&
1327
+ hovered_legend_series_idx !== series_idx
1328
+ ? 0.25
1329
+ : 1}
1330
+ >
1331
+ {#each bins as bin, bin_idx (bin_idx)}
1332
+ {@const bar_x = x_scale(bin.x0!)}
1333
+ {@const bar_width = Math.max(1, Math.abs(x_scale(bin.x1!) - bar_x))}
1334
+ {@const bar_height = Math.max(0, (height - pad.b) - y_scale(bin.length))}
1335
+ {@const bar_y = y_scale(bin.length)}
1336
+ {@const value = (bin.x0! + bin.x1!) / 2}
1337
+ {#if bar_height > 0}
1338
+ <path
1339
+ d={bar_path(
1340
+ bar_x,
1341
+ bar_y,
1342
+ bar_width,
1343
+ bar_height,
1344
+ Math.min(final_bar.border_radius ?? 0, bar_width / 2, bar_height / 2),
1345
+ )}
1346
+ fill={color}
1347
+ opacity={final_bar.opacity}
1348
+ stroke={final_bar.stroke_color}
1349
+ stroke-opacity={final_bar.stroke_opacity}
1350
+ stroke-width={final_bar.stroke_width}
1351
+ role="button"
1352
+ tabindex="0"
1353
+ onmousemove={(evt) =>
1354
+ handle_mouse_move(
1355
+ evt,
1356
+ value,
1357
+ bin.length,
1358
+ label,
1359
+ (y_axis ?? `y1`) as `y1` | `y2`,
1360
+ series_idx,
1361
+ (srs_x_axis ?? `x1`) as `x1` | `x2`,
1362
+ )}
1363
+ onmouseleave={() => {
1364
+ hover_info = null
1365
+ change(null)
1366
+ on_bar_hover?.(null)
1367
+ }}
1368
+ onclick={(event) =>
1369
+ on_bar_click?.({ value, count: bin.length, property: label, event })}
1370
+ onkeydown={(event: KeyboardEvent) => {
1371
+ if ([`Enter`, ` `].includes(event.key)) {
1372
+ event.preventDefault()
1373
+ on_bar_click?.({ value, count: bin.length, property: label, event })
1374
+ }
1375
+ }}
1376
+ style:cursor={on_bar_click ? `pointer` : undefined}
1377
+ />
1378
+ {/if}
1379
+ {/each}
1380
+ </g>
1381
+ {/each}
1382
+
1383
+ <!-- Reference lines: above all -->
1384
+ {@render ref_lines_layer(ref_lines_by_z.above_all)}
1385
+ </svg>
1386
+
1387
+ <!-- Tooltip (outside SVG for proper HTML rendering) -->
1388
+ {#if hover_info}
1389
+ {@const { value, count, property, active_y_axis, active_x_axis } = hover_info}
1390
+ {@const tooltip_x = (active_x_axis === `x2` ? scales.x2 : scales.x)(value)}
1391
+ {@const tooltip_y = (active_y_axis === `y2` ? scales.y2 : scales.y)(count)}
1392
+ {@const tooltip_pos = constrain_tooltip_position(
1393
+ tooltip_x,
1394
+ tooltip_y,
1395
+ tooltip_el?.offsetWidth ?? 120,
1396
+ tooltip_el?.offsetHeight ?? (mode === `overlay` ? 60 : 40),
1397
+ width,
1398
+ height,
1399
+ { offset_x: 5, offset_y: -10 },
1400
+ )}
1401
+ <PlotTooltip
1402
+ x={tooltip_pos.x}
1403
+ y={tooltip_pos.y}
1404
+ offset={{ x: 0, y: 0 }}
1405
+ bind:wrapper={tooltip_el}
1406
+ >
1407
+ {#if tooltip}
1408
+ {@render tooltip({ ...hover_info, fullscreen })}
1409
+ {:else}
1410
+ <div>Value: {format_value(value, hover_info.x_axis.format || `.3~s`)}</div>
1411
+ <div>Count: {format_value(count, hover_info.y_axis.format || `.3~s`)}</div>
1412
+ {#if mode === `overlay`}<div>{property}</div>{/if}
1413
+ {/if}
1414
+ </PlotTooltip>
1415
+ {/if}
1416
+
1417
+ {#if show_controls}
1418
+ <HistogramControls
1419
+ toggle_props={{
1420
+ ...controls_toggle_props,
1421
+ style: `--ctrl-btn-right: var(--fullscreen-btn-offset, 30px); ${
1422
+ controls_toggle_props?.style ?? ``
1423
+ }`,
1424
+ }}
1425
+ pane_props={controls_pane_props}
1426
+ bind:show_controls
1427
+ bind:controls_open
1428
+ bind:bins
1429
+ bind:mode
1430
+ bind:show_legend
1431
+ bind:selected_property
1432
+ bind:display
1433
+ bind:bar
1434
+ bind:x_axis
1435
+ bind:x2_axis
1436
+ bind:y_axis
1437
+ bind:y2_axis
1438
+ auto_x_range={auto_ranges.x}
1439
+ auto_x2_range={auto_ranges.x2}
1440
+ auto_y_range={auto_ranges.y}
1441
+ auto_y2_range={auto_ranges.y2}
1442
+ {series}
1443
+ has_x2_points={x2_series.length > 0}
1444
+ has_y2_points={y2_series.length > 0}
1445
+ children={controls_extra}
1446
+ />
1447
+ {/if}
1448
+
1449
+ {#if show_legend && legend != null && series.length > 1}
1450
+ {@const legend_left = legend_auto_outside
1451
+ ? legend_outside_x
1452
+ : legend_placement
1453
+ ? tweened_legend_coords.current.x
1454
+ : pad.l + 10}
1455
+ {@const legend_top = legend_auto_outside
1456
+ ? legend_outside_y
1457
+ : legend_placement
1458
+ ? tweened_legend_coords.current.y
1459
+ : pad.t + 10}
1460
+ <PlotLegend
1461
+ bind:root_element={legend_element}
1462
+ {...legend}
1463
+ series_data={legend_data}
1464
+ on_toggle={legend?.on_toggle || toggle_series_visibility}
1465
+ on_hover_change={legend_hover.set_locked}
1466
+ on_item_hover={(series_idx: number | null) =>
1467
+ (hovered_legend_series_idx = series_idx != null && series_idx >= 0
1468
+ ? series_idx
1469
+ : null)}
1470
+ active_series_idx={hover_info?.series_idx ?? hovered_legend_series_idx}
1471
+ style={`
1472
+ position: absolute;
1473
+ left: ${legend_left}px;
1474
+ top: ${legend_top}px;
1475
+ pointer-events: auto;
1476
+ ${legend?.style || ``}
1477
+ `}
1478
+ />
1479
+ {/if}
1480
+
1481
+ <!-- User-provided children (e.g. for custom absolutely-positioned overlays) -->
1482
+ {@render children?.({ height, width, fullscreen })}
1483
+ </div>
1484
+
1485
+ <style>
1486
+ .histogram {
1487
+ position: relative;
1488
+ width: var(--histogram-width, 100%);
1489
+ height: var(--histogram-height, auto);
1490
+ min-height: var(--histogram-min-height, 300px);
1491
+ container-type: size; /* enable cqh for panes if explicit height is set */
1492
+ z-index: var(--histogram-z-index, auto);
1493
+ flex: var(--histogram-flex, 1);
1494
+ display: var(--histogram-display, flex);
1495
+ flex-direction: column;
1496
+ background: var(--histogram-bg, var(--plot-bg));
1497
+ border-radius: var(--histogram-border-radius, var(--border-radius, 3pt));
1498
+ }
1499
+ .histogram.fullscreen {
1500
+ position: fixed;
1501
+ top: 0;
1502
+ left: 0;
1503
+ width: 100vw !important;
1504
+ height: 100vh !important;
1505
+ /* Must be higher than Structure.svelte's --struct-buttons-z-index. */
1506
+ z-index: var(--histogram-fullscreen-z-index, var(--z-index-overlay-nav, 100000001));
1507
+ margin: 0;
1508
+ border-radius: 0;
1509
+ background: var(--histogram-fullscreen-bg, var(--histogram-bg, var(--plot-bg)));
1510
+ max-height: none !important;
1511
+ overflow: hidden;
1512
+ /* Add padding to prevent titles from being cropped at top */
1513
+ padding-top: var(--plot-fullscreen-padding-top, 2em);
1514
+ box-sizing: border-box;
1515
+ }
1516
+ .header-controls {
1517
+ position: absolute;
1518
+ top: var(--ctrl-btn-top, 5pt);
1519
+ right: var(--fullscreen-btn-right, 4px);
1520
+ z-index: var(--fullscreen-btn-z-index, 10);
1521
+ display: flex;
1522
+ align-items: center;
1523
+ gap: 8px;
1524
+ }
1525
+ .header-controls :global(.fullscreen-toggle) {
1526
+ position: static; /* Override absolute positioning since container handles it */
1527
+ opacity: 1; /* Always visible when inside header-controls, container controls visibility */
1528
+ }
1529
+ /* Hide controls and fullscreen toggles by default, show on hover */
1530
+ .histogram :global(.pane-toggle),
1531
+ .histogram .header-controls {
1532
+ opacity: 0;
1533
+ transition: opacity 0.2s, background-color 0.2s;
1534
+ }
1535
+ .histogram:hover :global(.pane-toggle),
1536
+ .histogram:hover .header-controls,
1537
+ .histogram :global(.pane-toggle:focus-visible),
1538
+ .histogram :global(.pane-toggle[aria-expanded='true']),
1539
+ .histogram .header-controls:focus-within {
1540
+ opacity: 1;
1541
+ }
1542
+ svg {
1543
+ width: var(--histogram-svg-width, 100%);
1544
+ height: var(--histogram-svg-height, 100%);
1545
+ max-height: var(--histogram-svg-max-height, 100%);
1546
+ flex: var(--histogram-svg-flex, 1);
1547
+ overflow: var(--histogram-svg-overflow, visible);
1548
+ fill: var(--text-color);
1549
+ font-weight: var(--histogram-font-weight);
1550
+ font-size: var(--histogram-font-size);
1551
+ }
1552
+ .histogram-series path {
1553
+ transition: opacity 0.2s ease;
1554
+ }
1555
+ .histogram-series path:hover {
1556
+ opacity: 1 !important;
1557
+ }
1558
+ </style>