matterviz 0.3.6 → 0.3.7

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 (863) hide show
  1. package/dist/EmptyState.svelte.d.ts +9 -0
  2. package/dist/FilePicker.svelte +360 -0
  3. package/dist/FilePicker.svelte.d.ts +17 -0
  4. package/dist/Icon.svelte.d.ts +13 -0
  5. package/dist/MillerIndexInput.svelte +66 -0
  6. package/dist/MillerIndexInput.svelte.d.ts +7 -0
  7. package/dist/api/mp.d.ts +6 -0
  8. package/dist/api/mp.js +22 -0
  9. package/dist/api/optimade.d.ts +45 -0
  10. package/dist/api/optimade.js +135 -0
  11. package/dist/brillouin/BrillouinZone.svelte +549 -0
  12. package/dist/brillouin/BrillouinZone.svelte.d.ts +83 -0
  13. package/dist/brillouin/BrillouinZoneControls.svelte +144 -0
  14. package/dist/brillouin/BrillouinZoneControls.svelte.d.ts +17 -0
  15. package/dist/brillouin/BrillouinZoneExportPane.svelte +146 -0
  16. package/dist/brillouin/BrillouinZoneExportPane.svelte.d.ts +15 -0
  17. package/dist/brillouin/BrillouinZoneInfoPane.svelte +146 -0
  18. package/dist/brillouin/BrillouinZoneInfoPane.svelte.d.ts +13 -0
  19. package/dist/brillouin/BrillouinZoneScene.svelte +476 -0
  20. package/dist/brillouin/BrillouinZoneScene.svelte.d.ts +48 -0
  21. package/dist/brillouin/BrillouinZoneTooltip.svelte +92 -0
  22. package/dist/brillouin/BrillouinZoneTooltip.svelte.d.ts +8 -0
  23. package/dist/brillouin/compute.d.ts +17 -0
  24. package/dist/brillouin/compute.js +426 -0
  25. package/dist/brillouin/index.d.ts +8 -0
  26. package/dist/brillouin/index.js +7 -0
  27. package/dist/brillouin/types.d.ts +43 -0
  28. package/dist/brillouin/types.js +1 -0
  29. package/dist/chempot-diagram/ChemPotDiagram.svelte +327 -0
  30. package/dist/chempot-diagram/ChemPotDiagram.svelte.d.ts +13 -0
  31. package/dist/chempot-diagram/ChemPotDiagram2D.svelte +846 -0
  32. package/dist/chempot-diagram/ChemPotDiagram2D.svelte.d.ts +16 -0
  33. package/dist/chempot-diagram/ChemPotDiagram3D.svelte +3193 -0
  34. package/dist/chempot-diagram/ChemPotDiagram3D.svelte.d.ts +16 -0
  35. package/dist/chempot-diagram/ChemPotScene3D.svelte.d.ts +7 -0
  36. package/dist/chempot-diagram/async-compute.svelte.d.ts +3 -0
  37. package/dist/chempot-diagram/async-compute.svelte.js +78 -0
  38. package/dist/chempot-diagram/chempot-worker.d.ts +1 -0
  39. package/dist/chempot-diagram/chempot-worker.js +11 -0
  40. package/dist/chempot-diagram/color.d.ts +10 -0
  41. package/dist/chempot-diagram/color.js +32 -0
  42. package/dist/chempot-diagram/compute.d.ts +48 -0
  43. package/dist/chempot-diagram/compute.js +806 -0
  44. package/dist/chempot-diagram/index.d.ts +6 -0
  45. package/dist/chempot-diagram/index.js +6 -0
  46. package/dist/chempot-diagram/pointer.d.ts +16 -0
  47. package/dist/chempot-diagram/pointer.js +40 -0
  48. package/dist/chempot-diagram/temperature.d.ts +15 -0
  49. package/dist/chempot-diagram/temperature.js +34 -0
  50. package/dist/chempot-diagram/types.d.ts +81 -0
  51. package/dist/chempot-diagram/types.js +28 -0
  52. package/dist/colors/index.d.ts +47 -0
  53. package/dist/colors/index.js +203 -0
  54. package/dist/composition/BarChart.svelte +297 -0
  55. package/dist/composition/BarChart.svelte.d.ts +39 -0
  56. package/dist/composition/BubbleChart.svelte +218 -0
  57. package/dist/composition/BubbleChart.svelte.d.ts +28 -0
  58. package/dist/composition/Composition.svelte +165 -0
  59. package/dist/composition/Composition.svelte.d.ts +15 -0
  60. package/dist/composition/Formula.svelte +268 -0
  61. package/dist/composition/Formula.svelte.d.ts +19 -0
  62. package/dist/composition/FormulaFilter.svelte +1257 -0
  63. package/dist/composition/FormulaFilter.svelte.d.ts +51 -0
  64. package/dist/composition/PieChart.svelte +323 -0
  65. package/dist/composition/PieChart.svelte.d.ts +37 -0
  66. package/dist/composition/format.d.ts +15 -0
  67. package/dist/composition/format.js +109 -0
  68. package/dist/composition/index.d.ts +20 -0
  69. package/dist/composition/index.js +14 -0
  70. package/dist/composition/parse.d.ts +56 -0
  71. package/dist/composition/parse.js +474 -0
  72. package/dist/constants.d.ts +29 -0
  73. package/dist/constants.js +99 -0
  74. package/dist/controls.d.ts +14 -0
  75. package/dist/controls.js +30 -0
  76. package/dist/convex-hull/ConvexHull.svelte +157 -0
  77. package/dist/convex-hull/ConvexHull.svelte.d.ts +13 -0
  78. package/dist/convex-hull/ConvexHull2D.svelte +825 -0
  79. package/dist/convex-hull/ConvexHull2D.svelte.d.ts +11 -0
  80. package/dist/convex-hull/ConvexHull3D.svelte +1801 -0
  81. package/dist/convex-hull/ConvexHull3D.svelte.d.ts +8 -0
  82. package/dist/convex-hull/ConvexHull4D.svelte +1398 -0
  83. package/dist/convex-hull/ConvexHull4D.svelte.d.ts +8 -0
  84. package/dist/convex-hull/ConvexHullControls.svelte +535 -0
  85. package/dist/convex-hull/ConvexHullControls.svelte.d.ts +48 -0
  86. package/dist/convex-hull/ConvexHullInfoPane.svelte +125 -0
  87. package/dist/convex-hull/ConvexHullInfoPane.svelte.d.ts +20 -0
  88. package/dist/convex-hull/ConvexHullStats.svelte +929 -0
  89. package/dist/convex-hull/ConvexHullStats.svelte.d.ts +17 -0
  90. package/dist/convex-hull/ConvexHullTooltip.svelte +131 -0
  91. package/dist/convex-hull/ConvexHullTooltip.svelte.d.ts +33 -0
  92. package/dist/convex-hull/GasPressureControls.svelte +247 -0
  93. package/dist/convex-hull/GasPressureControls.svelte.d.ts +11 -0
  94. package/dist/convex-hull/StructurePopup.svelte +151 -0
  95. package/dist/convex-hull/StructurePopup.svelte.d.ts +18 -0
  96. package/dist/convex-hull/TemperatureSlider.svelte.d.ts +8 -0
  97. package/dist/convex-hull/barycentric-coords.d.ts +18 -0
  98. package/dist/convex-hull/barycentric-coords.js +182 -0
  99. package/dist/convex-hull/demo-temperature.d.ts +6 -0
  100. package/dist/convex-hull/demo-temperature.js +40 -0
  101. package/dist/convex-hull/gas-thermodynamics.d.ts +16 -0
  102. package/dist/convex-hull/gas-thermodynamics.js +314 -0
  103. package/dist/convex-hull/helpers.d.ts +114 -0
  104. package/dist/convex-hull/helpers.js +710 -0
  105. package/dist/convex-hull/index.d.ts +119 -0
  106. package/dist/convex-hull/index.js +58 -0
  107. package/dist/convex-hull/thermodynamics.d.ts +67 -0
  108. package/dist/convex-hull/thermodynamics.js +1752 -0
  109. package/dist/convex-hull/types.d.ts +162 -0
  110. package/dist/convex-hull/types.js +36 -0
  111. package/dist/coordination/CoordinationBarPlot.svelte +311 -0
  112. package/dist/coordination/CoordinationBarPlot.svelte.d.ts +30 -0
  113. package/dist/coordination/calc-coordination.d.ts +15 -0
  114. package/dist/coordination/calc-coordination.js +63 -0
  115. package/dist/coordination/index.d.ts +8 -0
  116. package/dist/coordination/index.js +7 -0
  117. package/dist/effects.svelte.d.ts +12 -0
  118. package/dist/effects.svelte.js +37 -0
  119. package/dist/element/BohrAtom.svelte.d.ts +20 -0
  120. package/dist/element/ElementHeading.svelte +26 -0
  121. package/dist/element/ElementHeading.svelte.d.ts +8 -0
  122. package/dist/element/ElementPhoto.svelte +57 -0
  123. package/dist/element/ElementPhoto.svelte.d.ts +9 -0
  124. package/dist/element/ElementStats.svelte +80 -0
  125. package/dist/element/ElementStats.svelte.d.ts +8 -0
  126. package/dist/element/ElementTile.svelte +484 -0
  127. package/dist/element/ElementTile.svelte.d.ts +29 -0
  128. package/dist/element/Nucleus.svelte.d.ts +17 -0
  129. package/dist/element/data.d.ts +2 -0
  130. package/dist/element/data.js +2 -0
  131. package/dist/element/index.d.ts +8 -0
  132. package/dist/element/index.js +7 -0
  133. package/dist/element/types.d.ts +57 -0
  134. package/dist/element/types.js +1 -0
  135. package/dist/feedback/ClickFeedback.svelte +58 -0
  136. package/dist/feedback/ClickFeedback.svelte.d.ts +12 -0
  137. package/dist/feedback/DragOverlay.svelte +42 -0
  138. package/dist/feedback/DragOverlay.svelte.d.ts +7 -0
  139. package/dist/feedback/Spinner.svelte.d.ts +7 -0
  140. package/dist/feedback/StatusMessage.svelte.d.ts +9 -0
  141. package/dist/feedback/index.d.ts +4 -0
  142. package/dist/feedback/index.js +4 -0
  143. package/dist/fermi-surface/FermiSlice.svelte +189 -0
  144. package/dist/fermi-surface/FermiSlice.svelte.d.ts +24 -0
  145. package/dist/fermi-surface/FermiSurface.svelte +600 -0
  146. package/dist/fermi-surface/FermiSurface.svelte.d.ts +83 -0
  147. package/dist/fermi-surface/FermiSurfaceControls.svelte +448 -0
  148. package/dist/fermi-surface/FermiSurfaceControls.svelte.d.ts +35 -0
  149. package/dist/fermi-surface/FermiSurfaceScene.svelte +794 -0
  150. package/dist/fermi-surface/FermiSurfaceScene.svelte.d.ts +50 -0
  151. package/dist/fermi-surface/FermiSurfaceTooltip.svelte +111 -0
  152. package/dist/fermi-surface/FermiSurfaceTooltip.svelte.d.ts +8 -0
  153. package/dist/fermi-surface/compute.d.ts +5 -0
  154. package/dist/fermi-surface/compute.js +538 -0
  155. package/dist/fermi-surface/constants.d.ts +9 -0
  156. package/dist/fermi-surface/constants.js +27 -0
  157. package/dist/fermi-surface/export.d.ts +5 -0
  158. package/dist/fermi-surface/export.js +50 -0
  159. package/dist/fermi-surface/index.d.ts +12 -0
  160. package/dist/fermi-surface/index.js +13 -0
  161. package/dist/fermi-surface/marching-cubes.d.ts +2 -0
  162. package/dist/fermi-surface/marching-cubes.js +2 -0
  163. package/dist/fermi-surface/parse.d.ts +2 -0
  164. package/dist/fermi-surface/parse.js +491 -0
  165. package/dist/fermi-surface/symmetry.d.ts +3 -0
  166. package/dist/fermi-surface/symmetry.js +46 -0
  167. package/dist/fermi-surface/types.d.ts +110 -0
  168. package/dist/fermi-surface/types.js +4 -0
  169. package/dist/heatmap-matrix/HeatmapMatrix.svelte +1545 -0
  170. package/dist/heatmap-matrix/HeatmapMatrix.svelte.d.ts +110 -0
  171. package/dist/heatmap-matrix/HeatmapMatrixControls.svelte +225 -0
  172. package/dist/heatmap-matrix/HeatmapMatrixControls.svelte.d.ts +30 -0
  173. package/dist/heatmap-matrix/index.d.ts +53 -0
  174. package/dist/heatmap-matrix/index.js +100 -0
  175. package/dist/heatmap-matrix/shared.d.ts +2 -0
  176. package/dist/heatmap-matrix/shared.js +4 -0
  177. package/dist/icons.d.ts +569 -0
  178. package/dist/icons.js +648 -0
  179. package/dist/index.d.ts +39 -0
  180. package/dist/index.js +39 -0
  181. package/dist/io/decompress.d.ts +11 -0
  182. package/dist/io/decompress.js +74 -0
  183. package/dist/io/export.d.ts +16 -0
  184. package/dist/io/export.js +316 -0
  185. package/dist/io/fetch.d.ts +5 -0
  186. package/dist/io/fetch.js +39 -0
  187. package/dist/io/file-drop.d.ts +7 -0
  188. package/dist/io/file-drop.js +43 -0
  189. package/dist/io/index.d.ts +7 -0
  190. package/dist/io/index.js +6 -0
  191. package/dist/io/is-binary.d.ts +1 -0
  192. package/dist/io/is-binary.js +20 -0
  193. package/dist/io/types.d.ts +8 -0
  194. package/dist/io/types.js +1 -0
  195. package/dist/io/url-drop.d.ts +2 -0
  196. package/dist/io/url-drop.js +123 -0
  197. package/dist/isosurface/Isosurface.svelte +285 -0
  198. package/dist/isosurface/Isosurface.svelte.d.ts +8 -0
  199. package/dist/isosurface/IsosurfaceControls.svelte +277 -0
  200. package/dist/isosurface/IsosurfaceControls.svelte.d.ts +9 -0
  201. package/dist/isosurface/index.d.ts +5 -0
  202. package/dist/isosurface/index.js +6 -0
  203. package/dist/isosurface/parse.d.ts +6 -0
  204. package/dist/isosurface/parse.js +553 -0
  205. package/dist/isosurface/slice.d.ts +11 -0
  206. package/dist/isosurface/slice.js +140 -0
  207. package/dist/isosurface/types.d.ts +56 -0
  208. package/dist/isosurface/types.js +227 -0
  209. package/dist/labels.d.ts +53 -0
  210. package/dist/labels.js +277 -0
  211. package/dist/layout/FullscreenToggle.svelte +50 -0
  212. package/dist/layout/FullscreenToggle.svelte.d.ts +7 -0
  213. package/dist/layout/InfoCard.svelte +120 -0
  214. package/dist/layout/InfoCard.svelte.d.ts +21 -0
  215. package/dist/layout/InfoTag.svelte +185 -0
  216. package/dist/layout/InfoTag.svelte.d.ts +19 -0
  217. package/dist/layout/PropertyFilter.svelte +246 -0
  218. package/dist/layout/PropertyFilter.svelte.d.ts +24 -0
  219. package/dist/layout/SettingsSection.svelte +148 -0
  220. package/dist/layout/SettingsSection.svelte.d.ts +17 -0
  221. package/dist/layout/SubpageGrid.svelte +82 -0
  222. package/dist/layout/SubpageGrid.svelte.d.ts +14 -0
  223. package/dist/layout/fullscreen.d.ts +9 -0
  224. package/dist/layout/fullscreen.js +53 -0
  225. package/dist/layout/index.d.ts +10 -0
  226. package/dist/layout/index.js +8 -0
  227. package/dist/layout/json-tree/JsonNode.svelte +548 -0
  228. package/dist/layout/json-tree/JsonNode.svelte.d.ts +11 -0
  229. package/dist/layout/json-tree/JsonTree.svelte +1230 -0
  230. package/dist/layout/json-tree/JsonTree.svelte.d.ts +6 -0
  231. package/dist/layout/json-tree/JsonValue.svelte.d.ts +9 -0
  232. package/dist/layout/json-tree/index.d.ts +3 -0
  233. package/dist/layout/json-tree/index.js +3 -0
  234. package/dist/layout/json-tree/types.d.ts +74 -0
  235. package/dist/layout/json-tree/types.js +2 -0
  236. package/dist/layout/json-tree/utils.d.ts +29 -0
  237. package/dist/layout/json-tree/utils.js +641 -0
  238. package/dist/marching-cubes.d.ts +14 -0
  239. package/dist/marching-cubes.js +540 -0
  240. package/dist/math.d.ts +101 -0
  241. package/dist/math.js +905 -0
  242. package/dist/overlays/ContextMenu.svelte +162 -0
  243. package/dist/overlays/ContextMenu.svelte.d.ts +25 -0
  244. package/dist/overlays/CopyButton.svelte +45 -0
  245. package/dist/overlays/CopyButton.svelte.d.ts +8 -0
  246. package/dist/overlays/DragControlTab.svelte +98 -0
  247. package/dist/overlays/DragControlTab.svelte.d.ts +8 -0
  248. package/dist/overlays/DraggablePane.svelte +487 -0
  249. package/dist/overlays/DraggablePane.svelte.d.ts +36 -0
  250. package/dist/overlays/InfoPaneCards.svelte +149 -0
  251. package/dist/overlays/InfoPaneCards.svelte.d.ts +22 -0
  252. package/dist/overlays/index.d.ts +3 -0
  253. package/dist/overlays/index.js +3 -0
  254. package/dist/periodic-table/PeriodicTable.svelte +469 -0
  255. package/dist/periodic-table/PeriodicTable.svelte.d.ts +55 -0
  256. package/dist/periodic-table/PeriodicTableControls.svelte +557 -0
  257. package/dist/periodic-table/PeriodicTableControls.svelte.d.ts +24 -0
  258. package/dist/periodic-table/PropertySelect.svelte +37 -0
  259. package/dist/periodic-table/PropertySelect.svelte.d.ts +13 -0
  260. package/dist/periodic-table/TableInset.svelte.d.ts +9 -0
  261. package/dist/periodic-table/index.d.ts +10 -0
  262. package/dist/periodic-table/index.js +4 -0
  263. package/dist/phase-diagram/IsobaricBinaryPhaseDiagram.svelte +1086 -0
  264. package/dist/phase-diagram/IsobaricBinaryPhaseDiagram.svelte.d.ts +44 -0
  265. package/dist/phase-diagram/PhaseDiagramControls.svelte +444 -0
  266. package/dist/phase-diagram/PhaseDiagramControls.svelte.d.ts +30 -0
  267. package/dist/phase-diagram/PhaseDiagramEditorPane.svelte +126 -0
  268. package/dist/phase-diagram/PhaseDiagramEditorPane.svelte.d.ts +15 -0
  269. package/dist/phase-diagram/PhaseDiagramExportPane.svelte +184 -0
  270. package/dist/phase-diagram/PhaseDiagramExportPane.svelte.d.ts +19 -0
  271. package/dist/phase-diagram/PhaseDiagramTooltip.svelte +391 -0
  272. package/dist/phase-diagram/PhaseDiagramTooltip.svelte.d.ts +16 -0
  273. package/dist/phase-diagram/TdbInfoPanel.svelte +203 -0
  274. package/dist/phase-diagram/TdbInfoPanel.svelte.d.ts +12 -0
  275. package/dist/phase-diagram/build-diagram.d.ts +11 -0
  276. package/dist/phase-diagram/build-diagram.js +160 -0
  277. package/dist/phase-diagram/colors.d.ts +35 -0
  278. package/dist/phase-diagram/colors.js +51 -0
  279. package/dist/phase-diagram/diagram-input.d.ts +29 -0
  280. package/dist/phase-diagram/diagram-input.js +3 -0
  281. package/dist/phase-diagram/index.d.ts +13 -0
  282. package/dist/phase-diagram/index.js +11 -0
  283. package/dist/phase-diagram/parse.d.ts +55 -0
  284. package/dist/phase-diagram/parse.js +272 -0
  285. package/dist/phase-diagram/svg-to-diagram.d.ts +2 -0
  286. package/dist/phase-diagram/svg-to-diagram.js +867 -0
  287. package/dist/phase-diagram/types.d.ts +93 -0
  288. package/dist/phase-diagram/types.js +1 -0
  289. package/dist/phase-diagram/utils.d.ts +118 -0
  290. package/dist/phase-diagram/utils.js +604 -0
  291. package/dist/plot/AxisLabel.svelte +51 -0
  292. package/dist/plot/AxisLabel.svelte.d.ts +16 -0
  293. package/dist/plot/BarPlot.svelte +2113 -0
  294. package/dist/plot/BarPlot.svelte.d.ts +84 -0
  295. package/dist/plot/BarPlotControls.svelte +66 -0
  296. package/dist/plot/BarPlotControls.svelte.d.ts +18 -0
  297. package/dist/plot/BinnedScatterPlot.svelte +1114 -0
  298. package/dist/plot/BinnedScatterPlot.svelte.d.ts +66 -0
  299. package/dist/plot/ColorBar.svelte +721 -0
  300. package/dist/plot/ColorBar.svelte.d.ts +31 -0
  301. package/dist/plot/ColorScaleSelect.svelte +54 -0
  302. package/dist/plot/ColorScaleSelect.svelte.d.ts +15 -0
  303. package/dist/plot/ElementScatter.svelte +63 -0
  304. package/dist/plot/ElementScatter.svelte.d.ts +14 -0
  305. package/dist/plot/FillArea.svelte.d.ts +21 -0
  306. package/dist/plot/Histogram.svelte +1558 -0
  307. package/dist/plot/Histogram.svelte.d.ts +50 -0
  308. package/dist/plot/HistogramControls.svelte +212 -0
  309. package/dist/plot/HistogramControls.svelte.d.ts +22 -0
  310. package/dist/plot/InteractiveAxisLabel.svelte +96 -0
  311. package/dist/plot/InteractiveAxisLabel.svelte.d.ts +14 -0
  312. package/dist/plot/Line.svelte +84 -0
  313. package/dist/plot/Line.svelte.d.ts +15 -0
  314. package/dist/plot/PlotAxis.svelte +169 -0
  315. package/dist/plot/PlotAxis.svelte.d.ts +24 -0
  316. package/dist/plot/PlotControls.svelte +537 -0
  317. package/dist/plot/PlotControls.svelte.d.ts +4 -0
  318. package/dist/plot/PlotLegend.svelte +569 -0
  319. package/dist/plot/PlotLegend.svelte.d.ts +29 -0
  320. package/dist/plot/PlotTooltip.svelte +67 -0
  321. package/dist/plot/PlotTooltip.svelte.d.ts +17 -0
  322. package/dist/plot/PortalSelect.svelte +253 -0
  323. package/dist/plot/PortalSelect.svelte.d.ts +16 -0
  324. package/dist/plot/ReferenceLine.svelte.d.ts +20 -0
  325. package/dist/plot/ReferenceLine3D.svelte +156 -0
  326. package/dist/plot/ReferenceLine3D.svelte.d.ts +14 -0
  327. package/dist/plot/ReferencePlane.svelte +175 -0
  328. package/dist/plot/ReferencePlane.svelte.d.ts +14 -0
  329. package/dist/plot/ScatterPlot.svelte +2778 -0
  330. package/dist/plot/ScatterPlot.svelte.d.ts +96 -0
  331. package/dist/plot/ScatterPlot3D.svelte +529 -0
  332. package/dist/plot/ScatterPlot3D.svelte.d.ts +95 -0
  333. package/dist/plot/ScatterPlot3DControls.svelte +437 -0
  334. package/dist/plot/ScatterPlot3DControls.svelte.d.ts +20 -0
  335. package/dist/plot/ScatterPlot3DScene.svelte +912 -0
  336. package/dist/plot/ScatterPlot3DScene.svelte.d.ts +74 -0
  337. package/dist/plot/ScatterPlotControls.svelte +306 -0
  338. package/dist/plot/ScatterPlotControls.svelte.d.ts +17 -0
  339. package/dist/plot/ScatterPoint.svelte +182 -0
  340. package/dist/plot/ScatterPoint.svelte.d.ts +22 -0
  341. package/dist/plot/SpacegroupBarPlot.svelte +293 -0
  342. package/dist/plot/SpacegroupBarPlot.svelte.d.ts +9 -0
  343. package/dist/plot/Surface3D.svelte +197 -0
  344. package/dist/plot/Surface3D.svelte.d.ts +13 -0
  345. package/dist/plot/ZeroLines.svelte +97 -0
  346. package/dist/plot/ZeroLines.svelte.d.ts +33 -0
  347. package/dist/plot/ZoomRect.svelte +23 -0
  348. package/dist/plot/ZoomRect.svelte.d.ts +8 -0
  349. package/dist/plot/adaptive-density.d.ts +69 -0
  350. package/dist/plot/adaptive-density.js +191 -0
  351. package/dist/plot/auto-place.d.ts +43 -0
  352. package/dist/plot/auto-place.js +122 -0
  353. package/dist/plot/axis-utils.d.ts +19 -0
  354. package/dist/plot/axis-utils.js +78 -0
  355. package/dist/plot/binned-scatter-types.d.ts +59 -0
  356. package/dist/plot/binned-scatter-types.js +1 -0
  357. package/dist/plot/data-cleaning.d.ts +37 -0
  358. package/dist/plot/data-cleaning.js +855 -0
  359. package/dist/plot/data-transform.d.ts +16 -0
  360. package/dist/plot/data-transform.js +45 -0
  361. package/dist/plot/defaults.d.ts +19 -0
  362. package/dist/plot/defaults.js +9 -0
  363. package/dist/plot/fill-utils.d.ts +46 -0
  364. package/dist/plot/fill-utils.js +322 -0
  365. package/dist/plot/hover-lock.svelte.d.ts +14 -0
  366. package/dist/plot/hover-lock.svelte.js +46 -0
  367. package/dist/plot/index.d.ts +41 -0
  368. package/dist/plot/index.js +39 -0
  369. package/dist/plot/interactions.d.ts +12 -0
  370. package/dist/plot/interactions.js +101 -0
  371. package/dist/plot/layout.d.ts +78 -0
  372. package/dist/plot/layout.js +273 -0
  373. package/dist/plot/reference-line.d.ts +60 -0
  374. package/dist/plot/reference-line.js +314 -0
  375. package/dist/plot/scales.d.ts +48 -0
  376. package/dist/plot/scales.js +481 -0
  377. package/dist/plot/svg.d.ts +1 -0
  378. package/dist/plot/svg.js +11 -0
  379. package/dist/plot/types.d.ts +831 -0
  380. package/dist/plot/types.js +99 -0
  381. package/dist/plot/utils/label-placement.d.ts +68 -0
  382. package/dist/plot/utils/label-placement.js +326 -0
  383. package/dist/plot/utils/series-visibility.d.ts +15 -0
  384. package/dist/plot/utils/series-visibility.js +85 -0
  385. package/dist/plot/utils.d.ts +1 -0
  386. package/dist/plot/utils.js +14 -0
  387. package/dist/rdf/RdfPlot.svelte +247 -0
  388. package/dist/rdf/RdfPlot.svelte.d.ts +27 -0
  389. package/dist/rdf/calc-rdf.d.ts +4 -0
  390. package/dist/rdf/calc-rdf.js +111 -0
  391. package/dist/rdf/index.d.ts +23 -0
  392. package/dist/rdf/index.js +2 -0
  393. package/dist/sanitize.d.ts +6 -0
  394. package/dist/sanitize.js +116 -0
  395. package/dist/settings.d.ts +255 -0
  396. package/dist/settings.js +1132 -0
  397. package/dist/spectral/Bands.svelte +1040 -0
  398. package/dist/spectral/Bands.svelte.d.ts +40 -0
  399. package/dist/spectral/BandsAndDos.svelte +134 -0
  400. package/dist/spectral/BandsAndDos.svelte.d.ts +18 -0
  401. package/dist/spectral/BrillouinBandsDos.svelte +252 -0
  402. package/dist/spectral/BrillouinBandsDos.svelte.d.ts +20 -0
  403. package/dist/spectral/Dos.svelte +697 -0
  404. package/dist/spectral/Dos.svelte.d.ts +29 -0
  405. package/dist/spectral/helpers.d.ts +119 -0
  406. package/dist/spectral/helpers.js +1032 -0
  407. package/dist/spectral/index.d.ts +6 -0
  408. package/dist/spectral/index.js +6 -0
  409. package/dist/spectral/types.d.ts +84 -0
  410. package/dist/spectral/types.js +2 -0
  411. package/dist/state.svelte.d.ts +25 -0
  412. package/dist/state.svelte.js +45 -0
  413. package/dist/structure/Arrow.svelte +72 -0
  414. package/dist/structure/Arrow.svelte.d.ts +15 -0
  415. package/dist/structure/AtomLegend.svelte +815 -0
  416. package/dist/structure/AtomLegend.svelte.d.ts +35 -0
  417. package/dist/structure/Bond.svelte +140 -0
  418. package/dist/structure/Bond.svelte.d.ts +9 -0
  419. package/dist/structure/CanvasTooltip.svelte +33 -0
  420. package/dist/structure/CanvasTooltip.svelte.d.ts +12 -0
  421. package/dist/structure/CellSelect.svelte +349 -0
  422. package/dist/structure/CellSelect.svelte.d.ts +13 -0
  423. package/dist/structure/Cylinder.svelte +45 -0
  424. package/dist/structure/Cylinder.svelte.d.ts +10 -0
  425. package/dist/structure/Lattice.svelte +196 -0
  426. package/dist/structure/Lattice.svelte.d.ts +17 -0
  427. package/dist/structure/Structure.svelte +2248 -0
  428. package/dist/structure/Structure.svelte.d.ts +89 -0
  429. package/dist/structure/StructureControls.svelte +1273 -0
  430. package/dist/structure/StructureControls.svelte.d.ts +31 -0
  431. package/dist/structure/StructureExportPane.svelte +252 -0
  432. package/dist/structure/StructureExportPane.svelte.d.ts +17 -0
  433. package/dist/structure/StructureInfoPane.svelte +737 -0
  434. package/dist/structure/StructureInfoPane.svelte.d.ts +19 -0
  435. package/dist/structure/StructureScene.svelte +2255 -0
  436. package/dist/structure/StructureScene.svelte.d.ts +111 -0
  437. package/dist/structure/atom-properties.d.ts +37 -0
  438. package/dist/structure/atom-properties.js +200 -0
  439. package/dist/structure/bond-order-perception.d.ts +13 -0
  440. package/dist/structure/bond-order-perception.js +384 -0
  441. package/dist/structure/bonding.d.ts +68 -0
  442. package/dist/structure/bonding.js +696 -0
  443. package/dist/structure/export.d.ts +20 -0
  444. package/dist/structure/export.js +727 -0
  445. package/dist/structure/index.d.ts +126 -0
  446. package/dist/structure/index.js +169 -0
  447. package/dist/structure/label-placement.d.ts +14 -0
  448. package/dist/structure/label-placement.js +72 -0
  449. package/dist/structure/measure.d.ts +6 -0
  450. package/dist/structure/measure.js +29 -0
  451. package/dist/structure/parse.d.ts +66 -0
  452. package/dist/structure/parse.js +1392 -0
  453. package/dist/structure/partial-occupancy.d.ts +25 -0
  454. package/dist/structure/partial-occupancy.js +99 -0
  455. package/dist/structure/pbc.d.ts +9 -0
  456. package/dist/structure/pbc.js +123 -0
  457. package/dist/structure/supercell.d.ts +8 -0
  458. package/dist/structure/supercell.js +170 -0
  459. package/dist/structure/validation.d.ts +2 -0
  460. package/dist/structure/validation.js +10 -0
  461. package/dist/symmetry/SymmetryStats.svelte +226 -0
  462. package/dist/symmetry/SymmetryStats.svelte.d.ts +21 -0
  463. package/dist/symmetry/WyckoffTable.svelte +120 -0
  464. package/dist/symmetry/WyckoffTable.svelte.d.ts +11 -0
  465. package/dist/symmetry/cell-transform.d.ts +12 -0
  466. package/dist/symmetry/cell-transform.js +91 -0
  467. package/dist/symmetry/index.d.ts +43 -0
  468. package/dist/symmetry/index.js +228 -0
  469. package/dist/symmetry/spacegroups.d.ts +9 -0
  470. package/dist/symmetry/spacegroups.js +394 -0
  471. package/dist/table/HeatmapTable.svelte +1833 -0
  472. package/dist/table/HeatmapTable.svelte.d.ts +49 -0
  473. package/dist/table/ToggleMenu.svelte +385 -0
  474. package/dist/table/ToggleMenu.svelte.d.ts +11 -0
  475. package/dist/table/index.d.ts +74 -0
  476. package/dist/table/index.js +38 -0
  477. package/dist/theme/ThemeControl.svelte +53 -0
  478. package/dist/theme/ThemeControl.svelte.d.ts +9 -0
  479. package/dist/theme/index.d.ts +29 -0
  480. package/dist/theme/index.js +79 -0
  481. package/dist/time.d.ts +4 -0
  482. package/dist/time.js +70 -0
  483. package/dist/tooltip/TooltipContent.svelte +58 -0
  484. package/dist/tooltip/TooltipContent.svelte.d.ts +31 -0
  485. package/dist/tooltip/index.d.ts +2 -0
  486. package/dist/tooltip/index.js +1 -0
  487. package/dist/tooltip/types.d.ts +8 -0
  488. package/dist/tooltip/types.js +1 -0
  489. package/dist/trajectory/Trajectory.svelte +1545 -0
  490. package/dist/trajectory/Trajectory.svelte.d.ts +77 -0
  491. package/dist/trajectory/TrajectoryError.svelte +128 -0
  492. package/dist/trajectory/TrajectoryError.svelte.d.ts +13 -0
  493. package/dist/trajectory/TrajectoryExportPane.svelte +357 -0
  494. package/dist/trajectory/TrajectoryExportPane.svelte.d.ts +17 -0
  495. package/dist/trajectory/TrajectoryInfoPane.svelte +313 -0
  496. package/dist/trajectory/TrajectoryInfoPane.svelte.d.ts +17 -0
  497. package/dist/trajectory/constants.d.ts +6 -0
  498. package/dist/trajectory/constants.js +7 -0
  499. package/dist/trajectory/extract.d.ts +5 -0
  500. package/dist/trajectory/extract.js +162 -0
  501. package/dist/trajectory/format-detect.d.ts +9 -0
  502. package/dist/trajectory/format-detect.js +76 -0
  503. package/dist/trajectory/frame-reader.d.ts +17 -0
  504. package/dist/trajectory/frame-reader.js +332 -0
  505. package/dist/trajectory/helpers.d.ts +15 -0
  506. package/dist/trajectory/helpers.js +164 -0
  507. package/dist/trajectory/index.d.ts +63 -0
  508. package/dist/trajectory/index.js +126 -0
  509. package/dist/trajectory/parse/ase.d.ts +2 -0
  510. package/dist/trajectory/parse/ase.js +73 -0
  511. package/dist/trajectory/parse/hdf5.d.ts +2 -0
  512. package/dist/trajectory/parse/hdf5.js +127 -0
  513. package/dist/trajectory/parse/index.d.ts +12 -0
  514. package/dist/trajectory/parse/index.js +298 -0
  515. package/dist/trajectory/parse/lammps.d.ts +5 -0
  516. package/dist/trajectory/parse/lammps.js +179 -0
  517. package/dist/trajectory/parse/vasp.d.ts +2 -0
  518. package/dist/trajectory/parse/vasp.js +68 -0
  519. package/dist/trajectory/parse/xyz.d.ts +2 -0
  520. package/dist/trajectory/parse/xyz.js +110 -0
  521. package/dist/trajectory/plotting.d.ts +28 -0
  522. package/dist/trajectory/plotting.js +423 -0
  523. package/dist/trajectory/types.d.ts +11 -0
  524. package/dist/trajectory/types.js +1 -0
  525. package/dist/utils.d.ts +6 -0
  526. package/dist/utils.js +45 -0
  527. package/dist/xrd/XrdPlot.svelte +615 -0
  528. package/dist/xrd/XrdPlot.svelte.d.ts +28 -0
  529. package/dist/xrd/broadening.d.ts +20 -0
  530. package/dist/xrd/broadening.js +97 -0
  531. package/dist/xrd/calc-xrd.d.ts +37 -0
  532. package/dist/xrd/calc-xrd.js +336 -0
  533. package/dist/xrd/index.d.ts +37 -0
  534. package/dist/xrd/index.js +4 -0
  535. package/dist/xrd/parse.d.ts +13 -0
  536. package/dist/xrd/parse.js +749 -0
  537. package/license +1 -1
  538. package/package.json +232 -1457
  539. package/readme.md +98 -171
  540. package/.vscode/launch.json +0 -13
  541. package/.vscodeignore +0 -7
  542. package/dist/assets/STLExporter-BpTH3YHE.js +0 -8
  543. package/dist/assets/browser-DdDecX_W.js +0 -1
  544. package/dist/assets/export-qgn-H9y6.js +0 -2
  545. package/dist/assets/main-DiKYzti2.css +0 -1
  546. package/dist/assets/moyo_wasm_bg-0ocwg7xY.wasm +0 -0
  547. package/dist/extension.js +0 -31293
  548. package/dist/src/lib/FilePicker.svelte +0 -360
  549. package/dist/src/lib/MillerIndexInput.svelte +0 -66
  550. package/dist/src/lib/api/mp.ts +0 -26
  551. package/dist/src/lib/api/optimade.ts +0 -204
  552. package/dist/src/lib/brillouin/BrillouinZone.svelte +0 -549
  553. package/dist/src/lib/brillouin/BrillouinZoneControls.svelte +0 -144
  554. package/dist/src/lib/brillouin/BrillouinZoneExportPane.svelte +0 -146
  555. package/dist/src/lib/brillouin/BrillouinZoneInfoPane.svelte +0 -146
  556. package/dist/src/lib/brillouin/BrillouinZoneScene.svelte +0 -476
  557. package/dist/src/lib/brillouin/BrillouinZoneTooltip.svelte +0 -92
  558. package/dist/src/lib/brillouin/compute.ts +0 -529
  559. package/dist/src/lib/brillouin/index.ts +0 -8
  560. package/dist/src/lib/brillouin/types.ts +0 -51
  561. package/dist/src/lib/chempot-diagram/ChemPotDiagram.svelte +0 -327
  562. package/dist/src/lib/chempot-diagram/ChemPotDiagram2D.svelte +0 -846
  563. package/dist/src/lib/chempot-diagram/ChemPotDiagram3D.svelte +0 -3193
  564. package/dist/src/lib/chempot-diagram/async-compute.svelte.ts +0 -94
  565. package/dist/src/lib/chempot-diagram/chempot-worker.ts +0 -11
  566. package/dist/src/lib/chempot-diagram/color.ts +0 -42
  567. package/dist/src/lib/chempot-diagram/compute.ts +0 -1014
  568. package/dist/src/lib/chempot-diagram/index.ts +0 -6
  569. package/dist/src/lib/chempot-diagram/pointer.ts +0 -56
  570. package/dist/src/lib/chempot-diagram/temperature.ts +0 -77
  571. package/dist/src/lib/chempot-diagram/types.ts +0 -130
  572. package/dist/src/lib/colors/index.ts +0 -249
  573. package/dist/src/lib/composition/BarChart.svelte +0 -297
  574. package/dist/src/lib/composition/BubbleChart.svelte +0 -218
  575. package/dist/src/lib/composition/Composition.svelte +0 -165
  576. package/dist/src/lib/composition/Formula.svelte +0 -268
  577. package/dist/src/lib/composition/FormulaFilter.svelte +0 -1257
  578. package/dist/src/lib/composition/PieChart.svelte +0 -323
  579. package/dist/src/lib/composition/format.ts +0 -155
  580. package/dist/src/lib/composition/index.ts +0 -37
  581. package/dist/src/lib/composition/parse.ts +0 -605
  582. package/dist/src/lib/constants.ts +0 -134
  583. package/dist/src/lib/controls.ts +0 -42
  584. package/dist/src/lib/convex-hull/ConvexHull.svelte +0 -157
  585. package/dist/src/lib/convex-hull/ConvexHull2D.svelte +0 -825
  586. package/dist/src/lib/convex-hull/ConvexHull3D.svelte +0 -1801
  587. package/dist/src/lib/convex-hull/ConvexHull4D.svelte +0 -1398
  588. package/dist/src/lib/convex-hull/ConvexHullControls.svelte +0 -535
  589. package/dist/src/lib/convex-hull/ConvexHullInfoPane.svelte +0 -125
  590. package/dist/src/lib/convex-hull/ConvexHullStats.svelte +0 -929
  591. package/dist/src/lib/convex-hull/ConvexHullTooltip.svelte +0 -131
  592. package/dist/src/lib/convex-hull/GasPressureControls.svelte +0 -247
  593. package/dist/src/lib/convex-hull/StructurePopup.svelte +0 -151
  594. package/dist/src/lib/convex-hull/barycentric-coords.ts +0 -246
  595. package/dist/src/lib/convex-hull/demo-temperature.ts +0 -63
  596. package/dist/src/lib/convex-hull/gas-thermodynamics.ts +0 -405
  597. package/dist/src/lib/convex-hull/helpers.ts +0 -932
  598. package/dist/src/lib/convex-hull/index.ts +0 -202
  599. package/dist/src/lib/convex-hull/thermodynamics.ts +0 -2192
  600. package/dist/src/lib/convex-hull/types.ts +0 -267
  601. package/dist/src/lib/coordination/CoordinationBarPlot.svelte +0 -311
  602. package/dist/src/lib/coordination/calc-coordination.ts +0 -93
  603. package/dist/src/lib/coordination/index.ts +0 -9
  604. package/dist/src/lib/effects.svelte.ts +0 -48
  605. package/dist/src/lib/element/ElementHeading.svelte +0 -26
  606. package/dist/src/lib/element/ElementPhoto.svelte +0 -57
  607. package/dist/src/lib/element/ElementStats.svelte +0 -80
  608. package/dist/src/lib/element/ElementTile.svelte +0 -484
  609. package/dist/src/lib/element/data.ts +0 -14
  610. package/dist/src/lib/element/index.ts +0 -8
  611. package/dist/src/lib/element/types.ts +0 -62
  612. package/dist/src/lib/feedback/ClickFeedback.svelte +0 -58
  613. package/dist/src/lib/feedback/DragOverlay.svelte +0 -42
  614. package/dist/src/lib/feedback/index.ts +0 -4
  615. package/dist/src/lib/fermi-surface/FermiSlice.svelte +0 -189
  616. package/dist/src/lib/fermi-surface/FermiSurface.svelte +0 -600
  617. package/dist/src/lib/fermi-surface/FermiSurfaceControls.svelte +0 -448
  618. package/dist/src/lib/fermi-surface/FermiSurfaceScene.svelte +0 -794
  619. package/dist/src/lib/fermi-surface/FermiSurfaceTooltip.svelte +0 -111
  620. package/dist/src/lib/fermi-surface/compute.ts +0 -728
  621. package/dist/src/lib/fermi-surface/constants.ts +0 -32
  622. package/dist/src/lib/fermi-surface/export.ts +0 -64
  623. package/dist/src/lib/fermi-surface/index.ts +0 -14
  624. package/dist/src/lib/fermi-surface/marching-cubes.ts +0 -3
  625. package/dist/src/lib/fermi-surface/parse.ts +0 -574
  626. package/dist/src/lib/fermi-surface/symmetry.ts +0 -56
  627. package/dist/src/lib/fermi-surface/types.ts +0 -159
  628. package/dist/src/lib/heatmap-matrix/HeatmapMatrix.svelte +0 -1545
  629. package/dist/src/lib/heatmap-matrix/HeatmapMatrixControls.svelte +0 -225
  630. package/dist/src/lib/heatmap-matrix/index.ts +0 -167
  631. package/dist/src/lib/heatmap-matrix/shared.ts +0 -7
  632. package/dist/src/lib/icons.ts +0 -650
  633. package/dist/src/lib/index.ts +0 -61
  634. package/dist/src/lib/io/decompress.ts +0 -92
  635. package/dist/src/lib/io/export.ts +0 -385
  636. package/dist/src/lib/io/fetch.ts +0 -46
  637. package/dist/src/lib/io/file-drop.ts +0 -51
  638. package/dist/src/lib/io/index.ts +0 -7
  639. package/dist/src/lib/io/is-binary.ts +0 -24
  640. package/dist/src/lib/io/types.ts +0 -8
  641. package/dist/src/lib/io/url-drop.ts +0 -141
  642. package/dist/src/lib/isosurface/Isosurface.svelte +0 -285
  643. package/dist/src/lib/isosurface/IsosurfaceControls.svelte +0 -277
  644. package/dist/src/lib/isosurface/index.ts +0 -7
  645. package/dist/src/lib/isosurface/parse.ts +0 -656
  646. package/dist/src/lib/isosurface/slice.ts +0 -175
  647. package/dist/src/lib/isosurface/types.ts +0 -309
  648. package/dist/src/lib/labels.ts +0 -320
  649. package/dist/src/lib/layout/FullscreenToggle.svelte +0 -50
  650. package/dist/src/lib/layout/InfoCard.svelte +0 -120
  651. package/dist/src/lib/layout/InfoTag.svelte +0 -185
  652. package/dist/src/lib/layout/PropertyFilter.svelte +0 -246
  653. package/dist/src/lib/layout/SettingsSection.svelte +0 -148
  654. package/dist/src/lib/layout/SubpageGrid.svelte +0 -82
  655. package/dist/src/lib/layout/fullscreen.ts +0 -65
  656. package/dist/src/lib/layout/index.ts +0 -11
  657. package/dist/src/lib/layout/json-tree/JsonNode.svelte +0 -548
  658. package/dist/src/lib/layout/json-tree/JsonTree.svelte +0 -1230
  659. package/dist/src/lib/layout/json-tree/index.ts +0 -3
  660. package/dist/src/lib/layout/json-tree/types.ts +0 -126
  661. package/dist/src/lib/layout/json-tree/utils.ts +0 -682
  662. package/dist/src/lib/marching-cubes.ts +0 -614
  663. package/dist/src/lib/math.ts +0 -1081
  664. package/dist/src/lib/overlays/ContextMenu.svelte +0 -162
  665. package/dist/src/lib/overlays/CopyButton.svelte +0 -45
  666. package/dist/src/lib/overlays/DragControlTab.svelte +0 -98
  667. package/dist/src/lib/overlays/DraggablePane.svelte +0 -487
  668. package/dist/src/lib/overlays/InfoPaneCards.svelte +0 -149
  669. package/dist/src/lib/overlays/index.ts +0 -3
  670. package/dist/src/lib/periodic-table/PeriodicTable.svelte +0 -469
  671. package/dist/src/lib/periodic-table/PeriodicTableControls.svelte +0 -557
  672. package/dist/src/lib/periodic-table/PropertySelect.svelte +0 -37
  673. package/dist/src/lib/periodic-table/index.ts +0 -12
  674. package/dist/src/lib/phase-diagram/IsobaricBinaryPhaseDiagram.svelte +0 -1086
  675. package/dist/src/lib/phase-diagram/PhaseDiagramControls.svelte +0 -444
  676. package/dist/src/lib/phase-diagram/PhaseDiagramEditorPane.svelte +0 -126
  677. package/dist/src/lib/phase-diagram/PhaseDiagramExportPane.svelte +0 -184
  678. package/dist/src/lib/phase-diagram/PhaseDiagramTooltip.svelte +0 -391
  679. package/dist/src/lib/phase-diagram/TdbInfoPanel.svelte +0 -203
  680. package/dist/src/lib/phase-diagram/build-diagram.ts +0 -186
  681. package/dist/src/lib/phase-diagram/colors.ts +0 -58
  682. package/dist/src/lib/phase-diagram/diagram-input.ts +0 -40
  683. package/dist/src/lib/phase-diagram/index.ts +0 -13
  684. package/dist/src/lib/phase-diagram/parse.ts +0 -348
  685. package/dist/src/lib/phase-diagram/svg-to-diagram.ts +0 -1023
  686. package/dist/src/lib/phase-diagram/types.ts +0 -144
  687. package/dist/src/lib/phase-diagram/utils.ts +0 -775
  688. package/dist/src/lib/plot/AxisLabel.svelte +0 -51
  689. package/dist/src/lib/plot/BarPlot.svelte +0 -2113
  690. package/dist/src/lib/plot/BarPlotControls.svelte +0 -66
  691. package/dist/src/lib/plot/BinnedScatterPlot.svelte +0 -1114
  692. package/dist/src/lib/plot/ColorBar.svelte +0 -721
  693. package/dist/src/lib/plot/ColorScaleSelect.svelte +0 -54
  694. package/dist/src/lib/plot/ElementScatter.svelte +0 -63
  695. package/dist/src/lib/plot/Histogram.svelte +0 -1558
  696. package/dist/src/lib/plot/HistogramControls.svelte +0 -212
  697. package/dist/src/lib/plot/InteractiveAxisLabel.svelte +0 -96
  698. package/dist/src/lib/plot/Line.svelte +0 -84
  699. package/dist/src/lib/plot/PlotAxis.svelte +0 -169
  700. package/dist/src/lib/plot/PlotControls.svelte +0 -537
  701. package/dist/src/lib/plot/PlotLegend.svelte +0 -569
  702. package/dist/src/lib/plot/PlotTooltip.svelte +0 -67
  703. package/dist/src/lib/plot/PortalSelect.svelte +0 -253
  704. package/dist/src/lib/plot/ReferenceLine3D.svelte +0 -156
  705. package/dist/src/lib/plot/ReferencePlane.svelte +0 -175
  706. package/dist/src/lib/plot/ScatterPlot.svelte +0 -2778
  707. package/dist/src/lib/plot/ScatterPlot3D.svelte +0 -529
  708. package/dist/src/lib/plot/ScatterPlot3DControls.svelte +0 -437
  709. package/dist/src/lib/plot/ScatterPlot3DScene.svelte +0 -912
  710. package/dist/src/lib/plot/ScatterPlotControls.svelte +0 -306
  711. package/dist/src/lib/plot/ScatterPoint.svelte +0 -182
  712. package/dist/src/lib/plot/SpacegroupBarPlot.svelte +0 -293
  713. package/dist/src/lib/plot/Surface3D.svelte +0 -197
  714. package/dist/src/lib/plot/ZeroLines.svelte +0 -97
  715. package/dist/src/lib/plot/ZoomRect.svelte +0 -23
  716. package/dist/src/lib/plot/adaptive-density.ts +0 -316
  717. package/dist/src/lib/plot/auto-place.ts +0 -184
  718. package/dist/src/lib/plot/axis-utils.ts +0 -122
  719. package/dist/src/lib/plot/binned-scatter-types.ts +0 -83
  720. package/dist/src/lib/plot/data-cleaning.ts +0 -1069
  721. package/dist/src/lib/plot/data-transform.ts +0 -69
  722. package/dist/src/lib/plot/defaults.ts +0 -9
  723. package/dist/src/lib/plot/fill-utils.ts +0 -494
  724. package/dist/src/lib/plot/hover-lock.svelte.ts +0 -60
  725. package/dist/src/lib/plot/index.ts +0 -53
  726. package/dist/src/lib/plot/interactions.ts +0 -119
  727. package/dist/src/lib/plot/layout.ts +0 -425
  728. package/dist/src/lib/plot/reference-line.ts +0 -426
  729. package/dist/src/lib/plot/scales.ts +0 -654
  730. package/dist/src/lib/plot/svg.ts +0 -23
  731. package/dist/src/lib/plot/types.ts +0 -1144
  732. package/dist/src/lib/plot/utils/label-placement.ts +0 -541
  733. package/dist/src/lib/plot/utils/series-visibility.ts +0 -140
  734. package/dist/src/lib/plot/utils.ts +0 -11
  735. package/dist/src/lib/rdf/RdfPlot.svelte +0 -247
  736. package/dist/src/lib/rdf/calc-rdf.ts +0 -167
  737. package/dist/src/lib/rdf/index.ts +0 -27
  738. package/dist/src/lib/sanitize.ts +0 -126
  739. package/dist/src/lib/settings.ts +0 -1479
  740. package/dist/src/lib/spectral/Bands.svelte +0 -1040
  741. package/dist/src/lib/spectral/BandsAndDos.svelte +0 -134
  742. package/dist/src/lib/spectral/BrillouinBandsDos.svelte +0 -252
  743. package/dist/src/lib/spectral/Dos.svelte +0 -697
  744. package/dist/src/lib/spectral/helpers.ts +0 -1381
  745. package/dist/src/lib/spectral/index.ts +0 -8
  746. package/dist/src/lib/spectral/types.ts +0 -112
  747. package/dist/src/lib/state.svelte.ts +0 -64
  748. package/dist/src/lib/structure/Arrow.svelte +0 -72
  749. package/dist/src/lib/structure/AtomLegend.svelte +0 -815
  750. package/dist/src/lib/structure/Bond.svelte +0 -140
  751. package/dist/src/lib/structure/CanvasTooltip.svelte +0 -33
  752. package/dist/src/lib/structure/CellSelect.svelte +0 -349
  753. package/dist/src/lib/structure/Cylinder.svelte +0 -45
  754. package/dist/src/lib/structure/Lattice.svelte +0 -196
  755. package/dist/src/lib/structure/Structure.svelte +0 -2248
  756. package/dist/src/lib/structure/StructureControls.svelte +0 -1273
  757. package/dist/src/lib/structure/StructureExportPane.svelte +0 -252
  758. package/dist/src/lib/structure/StructureInfoPane.svelte +0 -737
  759. package/dist/src/lib/structure/StructureScene.svelte +0 -2255
  760. package/dist/src/lib/structure/atom-properties.ts +0 -316
  761. package/dist/src/lib/structure/bond-order-perception.ts +0 -447
  762. package/dist/src/lib/structure/bonding.ts +0 -944
  763. package/dist/src/lib/structure/export.ts +0 -861
  764. package/dist/src/lib/structure/index.ts +0 -291
  765. package/dist/src/lib/structure/label-placement.ts +0 -130
  766. package/dist/src/lib/structure/measure.ts +0 -45
  767. package/dist/src/lib/structure/parse.ts +0 -1705
  768. package/dist/src/lib/structure/partial-occupancy.ts +0 -183
  769. package/dist/src/lib/structure/pbc.ts +0 -164
  770. package/dist/src/lib/structure/supercell.ts +0 -226
  771. package/dist/src/lib/structure/validation.ts +0 -11
  772. package/dist/src/lib/symmetry/SymmetryStats.svelte +0 -226
  773. package/dist/src/lib/symmetry/WyckoffTable.svelte +0 -120
  774. package/dist/src/lib/symmetry/cell-transform.ts +0 -118
  775. package/dist/src/lib/symmetry/index.ts +0 -348
  776. package/dist/src/lib/symmetry/spacegroups.ts +0 -404
  777. package/dist/src/lib/table/HeatmapTable.svelte +0 -1833
  778. package/dist/src/lib/table/ToggleMenu.svelte +0 -385
  779. package/dist/src/lib/table/index.ts +0 -139
  780. package/dist/src/lib/theme/ThemeControl.svelte +0 -53
  781. package/dist/src/lib/theme/index.ts +0 -107
  782. package/dist/src/lib/time.ts +0 -71
  783. package/dist/src/lib/tooltip/TooltipContent.svelte +0 -58
  784. package/dist/src/lib/tooltip/index.ts +0 -2
  785. package/dist/src/lib/tooltip/types.ts +0 -13
  786. package/dist/src/lib/trajectory/Trajectory.svelte +0 -1545
  787. package/dist/src/lib/trajectory/TrajectoryError.svelte +0 -128
  788. package/dist/src/lib/trajectory/TrajectoryExportPane.svelte +0 -357
  789. package/dist/src/lib/trajectory/TrajectoryInfoPane.svelte +0 -313
  790. package/dist/src/lib/trajectory/constants.ts +0 -7
  791. package/dist/src/lib/trajectory/extract.ts +0 -196
  792. package/dist/src/lib/trajectory/format-detect.ts +0 -96
  793. package/dist/src/lib/trajectory/frame-reader.ts +0 -456
  794. package/dist/src/lib/trajectory/helpers.ts +0 -217
  795. package/dist/src/lib/trajectory/index.ts +0 -218
  796. package/dist/src/lib/trajectory/parse/ase.ts +0 -109
  797. package/dist/src/lib/trajectory/parse/hdf5.ts +0 -173
  798. package/dist/src/lib/trajectory/parse/index.ts +0 -411
  799. package/dist/src/lib/trajectory/parse/lammps.ts +0 -215
  800. package/dist/src/lib/trajectory/parse/vasp.ts +0 -102
  801. package/dist/src/lib/trajectory/parse/xyz.ts +0 -143
  802. package/dist/src/lib/trajectory/plotting.ts +0 -599
  803. package/dist/src/lib/trajectory/types.ts +0 -13
  804. package/dist/src/lib/utils.ts +0 -56
  805. package/dist/src/lib/xrd/XrdPlot.svelte +0 -615
  806. package/dist/src/lib/xrd/broadening.ts +0 -130
  807. package/dist/src/lib/xrd/calc-xrd.ts +0 -397
  808. package/dist/src/lib/xrd/index.ts +0 -38
  809. package/dist/src/lib/xrd/parse.ts +0 -858
  810. package/dist/webview.js +0 -29421
  811. package/icon.png +0 -0
  812. package/matterviz-0.3.2.vsix +0 -0
  813. package/matterviz-0.3.4.vsix +0 -0
  814. package/matterviz-0.3.5.vsix +0 -0
  815. package/scripts/sync-config.ts +0 -101
  816. package/src/declarations.d.ts +0 -2
  817. package/src/extension.ts +0 -972
  818. package/src/node-io.ts +0 -65
  819. package/src/types.ts +0 -17
  820. package/src/webview/JsonBrowser.svelte +0 -1079
  821. package/src/webview/PlotPanel.svelte +0 -346
  822. package/src/webview/detect.ts +0 -444
  823. package/src/webview/main.ts +0 -764
  824. package/src/webview/plot-utils.ts +0 -250
  825. package/test-fixtures/all-viz-types.json.gz +0 -0
  826. package/test-fixtures/plot-demo-data.json.gz +0 -0
  827. package/tests/detect.test.ts +0 -604
  828. package/tests/extension.test.ts +0 -2041
  829. package/tests/node-io.test.ts +0 -39
  830. package/tests/plot-utils.test.ts +0 -302
  831. package/tests/vite-plugin-json-gz.test.ts +0 -114
  832. package/tests/vscode-mock.ts +0 -18
  833. package/tests/webview.test.ts +0 -231
  834. package/tsconfig.json +0 -20
  835. package/vite-plugin-json-gz.ts +0 -29
  836. package/vite.config.ts +0 -34
  837. package/vite.extension.config.ts +0 -34
  838. /package/dist/{src/lib/EmptyState.svelte → EmptyState.svelte} +0 -0
  839. /package/dist/{src/lib/Icon.svelte → Icon.svelte} +0 -0
  840. /package/dist/{src/lib/app.css → app.css} +0 -0
  841. /package/dist/{src/lib/chempot-diagram → chempot-diagram}/ChemPotScene3D.svelte +0 -0
  842. /package/dist/{src/lib/colors → colors}/alloy-colors.json +0 -0
  843. /package/dist/{src/lib/colors → colors}/dark-mode-colors.json +0 -0
  844. /package/dist/{src/lib/colors → colors}/jmol-colors.json +0 -0
  845. /package/dist/{src/lib/colors → colors}/muted-colors.json +0 -0
  846. /package/dist/{src/lib/colors → colors}/pastel-colors.json +0 -0
  847. /package/dist/{src/lib/colors → colors}/vesta-colors.json +0 -0
  848. /package/dist/{src/lib/convex-hull → convex-hull}/TemperatureSlider.svelte +0 -0
  849. /package/dist/{src/lib/element → element}/BohrAtom.svelte +0 -0
  850. /package/dist/{src/lib/element → element}/Nucleus.svelte +0 -0
  851. /package/dist/{src/lib/element → element}/data.json +0 -0
  852. /package/dist/{src/lib/element → element}/data.json.gz +0 -0
  853. /package/dist/{src/lib/element → element}/data.json.gz.d.ts +0 -0
  854. /package/dist/{src/lib/element → element}/data.schema.json +0 -0
  855. /package/dist/{src/lib/element-image-urls.json → element-image-urls.json} +0 -0
  856. /package/dist/{src/lib/feedback → feedback}/Spinner.svelte +0 -0
  857. /package/dist/{src/lib/feedback → feedback}/StatusMessage.svelte +0 -0
  858. /package/dist/{src/lib/layout → layout}/json-tree/JsonValue.svelte +0 -0
  859. /package/dist/{src/lib/periodic-table → periodic-table}/TableInset.svelte +0 -0
  860. /package/dist/{src/lib/plot → plot}/FillArea.svelte +0 -0
  861. /package/dist/{src/lib/plot → plot}/ReferenceLine.svelte +0 -0
  862. /package/dist/{src/lib/theme → theme}/themes.mjs +0 -0
  863. /package/dist/{src/lib/xrd → xrd}/atomic_scattering_params.json +0 -0
@@ -1,1023 +0,0 @@
1
- // SVG-to-DiagramInput converter
2
- // Parses phase diagram SVGs (matplotlib or simple/Gemini format) into DiagramInput JSON
3
- // for immediate rendering by IsobaricBinaryPhaseDiagram
4
-
5
- import type { DiagramInput, DiagramPoint, RegionInput } from './diagram-input'
6
-
7
- // Two-phase color keys to cycle through for region assignment
8
- const TWO_PHASE_COLORS = [
9
- `two_phase`,
10
- `two_phase_intermetallic`,
11
- `two_phase_fcc_liquid`,
12
- `two_phase_intermetallic_alt`,
13
- `two_phase_alt`,
14
- `two_phase_gamma`,
15
- `two_phase_mixed`,
16
- `two_phase_si`,
17
- `two_phase_bcc_liquid`,
18
- `two_phase_hcp_liquid`,
19
- `two_phase_theta_liquid`,
20
- `two_phase_eta`,
21
- ]
22
-
23
- // Round to 6 decimal places for clean floating-point output
24
- const round = (val: number): number => Math.round(val * 1e6) / 1e6
25
-
26
- // === Types ===
27
-
28
- type SvgFormat = `matplotlib` | `simple` | `mpds`
29
-
30
- interface LinearScale {
31
- to_data: (px: number) => number
32
- domain: [number, number] // [min_data, max_data]
33
- }
34
-
35
- interface Boundary {
36
- x1: number // data coordinates
37
- y1: number
38
- x2: number
39
- y2: number
40
- orientation: `horizontal` | `vertical`
41
- }
42
-
43
- interface Label {
44
- text: string // plain text, e.g. "La2NiO4 + NiO"
45
- px_x: number // pixel position
46
- px_y: number
47
- }
48
-
49
- // === Format Detection ===
50
-
51
- // Detect whether the SVG is matplotlib, MPDS, or simple format
52
- function detect_format(doc: Document): SvgFormat {
53
- // Matplotlib SVGs have xtick/ytick group IDs
54
- if (doc.querySelector(`[id^="xtick_"]`) || doc.querySelector(`[id^="ytick_"]`)) {
55
- return `matplotlib`
56
- }
57
- // MPDS SVGs (from CorelDRAW/Inkscape) have sodipodi namespace or inkscape attributes
58
- const svg_el = doc.querySelector(`svg`)
59
- if (
60
- svg_el?.getAttribute(`sodipodi:docname`) ||
61
- svg_el?.getAttribute(`inkscape:version`) ||
62
- svg_el?.getAttributeNS(`http://sodipodi.sourceforge.net/DTD/sodipodi-0.dtd`, `docname`)
63
- ) {
64
- return `mpds`
65
- }
66
- return `simple`
67
- }
68
-
69
- // === Axis Scale Extraction ===
70
-
71
- // Extract x and y axis scales from tick marks
72
- function extract_axis_scales(
73
- doc: Document,
74
- format: SvgFormat,
75
- ): { x_scale: LinearScale; y_scale: LinearScale } {
76
- const x_ticks: { px: number; value: number }[] = []
77
- const y_ticks: { px: number; value: number }[] = []
78
-
79
- if (format === `matplotlib`) {
80
- extract_matplotlib_ticks(doc, x_ticks, y_ticks)
81
- } else if (format === `mpds`) {
82
- return extract_mpds_scales(doc)
83
- } else {
84
- extract_simple_ticks(doc, x_ticks, y_ticks)
85
- }
86
-
87
- if (x_ticks.length < 2) {
88
- throw new Error(`Need at least 2 x-axis ticks, found ${x_ticks.length}`)
89
- }
90
- if (y_ticks.length < 2) {
91
- throw new Error(`Need at least 2 y-axis ticks, found ${y_ticks.length}`)
92
- }
93
-
94
- return {
95
- x_scale: build_scale(x_ticks),
96
- y_scale: build_scale(y_ticks), // y-axis inverted (SVG y down, temp up)
97
- }
98
- }
99
-
100
- // Extract ticks from matplotlib SVG (id="xtick_N", comment-based values)
101
- function extract_matplotlib_ticks(
102
- doc: Document,
103
- x_ticks: { px: number; value: number }[],
104
- y_ticks: { px: number; value: number }[],
105
- ): void {
106
- const axes: [string, string, { px: number; value: number }[]][] = [
107
- [`xtick_`, `x`, x_ticks],
108
- [`ytick_`, `y`, y_ticks],
109
- ]
110
- for (const [prefix, attr, ticks] of axes) {
111
- for (const group of Array.from(doc.querySelectorAll(`[id^="${prefix}"]`))) {
112
- const value = extract_comment_number(group)
113
- const use_el = group.querySelector(`use`)
114
- if (value !== null && use_el) {
115
- const px = parse_float_attr(use_el, attr)
116
- if (px !== null) ticks.push({ px, value })
117
- }
118
- }
119
- }
120
- }
121
-
122
- // Extract ticks from simple SVG (class-based text elements)
123
- function extract_simple_ticks(
124
- doc: Document,
125
- x_ticks: { px: number; value: number }[],
126
- y_ticks: { px: number; value: number }[],
127
- ): void {
128
- // Y-axis ticks: class="tick-text" with text-anchor: end
129
- for (const text_el of Array.from(doc.querySelectorAll(`.tick-text`))) {
130
- const value = parseFloat(text_el.textContent?.trim() ?? ``)
131
- if (isNaN(value)) continue
132
-
133
- // Find the immediately preceding sibling tick line (not just any line in parent)
134
- const tick_line = text_el.previousElementSibling
135
- if (!tick_line || !tick_line.matches(`.tick-line, line`)) continue
136
- const py = parse_float_attr(tick_line, `y1`)
137
- // Apply parent group transform if present
138
- const parent = text_el.parentElement
139
- const transform_y = get_group_translate(parent, `y`)
140
- if (py !== null) y_ticks.push({ px: py + transform_y, value })
141
- }
142
-
143
- // X-axis ticks: class="tick-text-x"
144
- for (const text_el of Array.from(doc.querySelectorAll(`.tick-text-x`))) {
145
- const value = parseFloat(text_el.textContent?.trim() ?? ``)
146
- if (isNaN(value)) continue
147
-
148
- const px_x = parse_float_attr(text_el, `x`)
149
- // Apply parent group transform if present
150
- const parent = text_el.parentElement
151
- const transform_x = get_group_translate(parent, `x`)
152
- if (px_x !== null) x_ticks.push({ px: px_x + transform_x, value })
153
- }
154
- }
155
-
156
- // Extract scales from MPDS SVGs using tick mark paths and text values
157
- // MPDS SVGs store tick marks as multi-segment paths with major ticks (longer)
158
- // and minor ticks (shorter). Text values are not positionally useful but
159
- // tell us the data range.
160
- function extract_mpds_scales(doc: Document): {
161
- x_scale: LinearScale
162
- y_scale: LinearScale
163
- } {
164
- // Extract all numeric text values to infer axis ranges
165
- const numbers: number[] = []
166
- for (const text_el of Array.from(doc.querySelectorAll(`text`))) {
167
- const val = parseFloat(text_el.textContent?.trim() ?? ``)
168
- if (!isNaN(val)) numbers.push(val)
169
- }
170
-
171
- // Separate composition (0-100 at%) from temperature (typically 100-3000)
172
- // Note: value 100 appears in both filters (valid as 100 at% and 100°C).
173
- // This is intentional — only endpoints are used for scale mapping.
174
- const comp_vals = [
175
- ...new Set(numbers.filter((value) => value >= 0 && value <= 100 && value % 10 === 0)),
176
- ].toSorted((a, b) => a - b)
177
- const temp_vals = [
178
- ...new Set(numbers.filter((value) => value >= 100 && value % 100 === 0 && value <= 3000)),
179
- ].toSorted((a, b) => a - b)
180
-
181
- if (comp_vals.length < 2 || temp_vals.length < 2) {
182
- throw new Error(
183
- `MPDS SVG: could not infer axis ranges (found ${comp_vals.length} composition, ${temp_vals.length} temperature values)`,
184
- )
185
- }
186
-
187
- // Find tick mark paths — multi-segment paths with stroke-width ~0.5
188
- // containing both major (longer) and minor (shorter) tick marks
189
- const x_major_ticks: number[] = []
190
- const y_major_ticks: number[] = []
191
-
192
- for (const path of Array.from(doc.querySelectorAll(`path`))) {
193
- const path_data = path.getAttribute(`d`) ?? ``
194
- const stroke_width = parse_stroke_width(path)
195
-
196
- // Tick mark paths have stroke-width ~0.5
197
- if (stroke_width < 0.3 || stroke_width > 1.0) continue
198
-
199
- // Parse path into absolute line segments (handles both absolute & relative commands)
200
- const segments = parse_path_segments(path_data)
201
- if (segments.length < 3) continue
202
-
203
- for (const [sx1, sy1, sx2, sy2] of segments) {
204
- const seg_dx = Math.abs(sx2 - sx1)
205
- const seg_dy = Math.abs(sy2 - sy1)
206
-
207
- // Major ticks are longer (~3.9 px), minor are shorter (~1.7 px)
208
- if (Math.hypot(seg_dx, seg_dy) < 3) continue
209
-
210
- // Horizontal tick segments → y-axis tick (x changes, y constant)
211
- if (seg_dy < 0.1 && seg_dx > 2) {
212
- y_major_ticks.push(Math.round(sy1 * 100) / 100)
213
- }
214
- // Vertical tick segments → x-axis tick (y changes, x constant)
215
- if (seg_dx < 0.1 && seg_dy > 2) {
216
- x_major_ticks.push(Math.round(sx1 * 100) / 100)
217
- }
218
- }
219
- }
220
-
221
- // Deduplicate and sort
222
- const x_ticks_sorted = [
223
- ...new Set(x_major_ticks.map((value) => Math.round(value * 10) / 10)),
224
- ].toSorted((a, b) => a - b)
225
- const y_ticks_sorted = [
226
- ...new Set(y_major_ticks.map((value) => Math.round(value * 10) / 10)),
227
- ].toSorted((a, b) => a - b)
228
-
229
- if (x_ticks_sorted.length < 2 || y_ticks_sorted.length < 2) {
230
- throw new Error(
231
- `MPDS SVG: could not find tick marks (found ${x_ticks_sorted.length} x-ticks, ${y_ticks_sorted.length} y-ticks)`,
232
- )
233
- }
234
-
235
- // Map tick positions to data values using endpoints
236
- // Only the first and last tick+value need to match for a linear scale
237
- // x-axis: composition in at% → fraction; y-axis: temperature (SVG y inverted)
238
- const x_ticks = [
239
- { px: x_ticks_sorted[0], value: comp_vals[0] / 100 },
240
- {
241
- px: x_ticks_sorted[x_ticks_sorted.length - 1],
242
- value: comp_vals[comp_vals.length - 1] / 100,
243
- },
244
- ]
245
- const y_ticks = [
246
- { px: y_ticks_sorted[0], value: temp_vals[temp_vals.length - 1] }, // top = highest temp
247
- { px: y_ticks_sorted[y_ticks_sorted.length - 1], value: temp_vals[0] }, // bottom = lowest temp
248
- ]
249
-
250
- return {
251
- x_scale: build_scale(x_ticks),
252
- y_scale: build_scale(y_ticks),
253
- }
254
- }
255
-
256
- // Build a linear scale from tick data points
257
- function build_scale(ticks: { px: number; value: number }[]): LinearScale {
258
- ticks.sort((a, b) => a.value - b.value)
259
- const first = ticks[0]
260
- const last = ticks[ticks.length - 1]
261
-
262
- const range = last.value - first.value
263
- if (range === 0) {
264
- return { to_data: () => first.value, domain: [first.value, last.value] }
265
- }
266
- const px_per_unit = (last.px - first.px) / range
267
-
268
- return {
269
- to_data: (px: number) => first.value + (px - first.px) / px_per_unit,
270
- domain: [first.value, last.value],
271
- }
272
- }
273
-
274
- // === Boundary Extraction ===
275
-
276
- // Extract phase boundary lines from SVG and convert to data coordinates
277
- function extract_boundaries(
278
- doc: Document,
279
- format: SvgFormat,
280
- x_scale: LinearScale,
281
- y_scale: LinearScale,
282
- ): Boundary[] {
283
- const boundaries: Boundary[] = []
284
- const epsilon = 0.5 // pixel tolerance for classifying horizontal/vertical
285
-
286
- if (format === `mpds`) {
287
- extract_mpds_boundaries(doc, boundaries, x_scale, y_scale, epsilon)
288
- } else if (format === `matplotlib`) {
289
- // Matplotlib: look for line2d_N groups with path elements (skip tick marks at line2d_1..12ish)
290
- for (const group of Array.from(doc.querySelectorAll(`[id^="line2d_"]`))) {
291
- const path_el = group.querySelector(`path`)
292
- if (!path_el) continue
293
-
294
- const d_attr = path_el.getAttribute(`d`)
295
- if (!d_attr) continue
296
-
297
- // Parse "M x1 y1 L x2 y2" path data
298
- const coords = parse_ml_path(d_attr)
299
- if (!coords) continue
300
-
301
- // Skip tick mark lines (short lines, typically < 10px)
302
- const dx = Math.abs(coords.x2 - coords.x1)
303
- const dy = Math.abs(coords.y2 - coords.y1)
304
- if (dx < 15 && dy < 15) continue
305
-
306
- // Check stroke-width to distinguish boundaries from axis lines
307
- const stroke_width = parse_stroke_width(path_el) || 1
308
-
309
- // Axis patches (id="patch_*") are axis borders, not phase boundaries
310
- const parent_id = group.getAttribute(`id`) ?? ``
311
- if (parent_id.startsWith(`patch_`)) continue
312
-
313
- // Only include lines with meaningful stroke
314
- if (stroke_width < 1) continue
315
-
316
- add_boundary(boundaries, coords, x_scale, y_scale, epsilon)
317
- }
318
- } else {
319
- // Simple: <line class="phase-boundary">
320
- for (const line_el of Array.from(
321
- doc.querySelectorAll(`.phase-boundary, line[class*="phase-boundary"]`),
322
- )) {
323
- const x1 = parse_float_attr(line_el, `x1`)
324
- const y1 = parse_float_attr(line_el, `y1`)
325
- const x2 = parse_float_attr(line_el, `x2`)
326
- const y2 = parse_float_attr(line_el, `y2`)
327
- if (x1 === null || y1 === null || x2 === null || y2 === null) continue
328
-
329
- add_boundary(boundaries, { x1, y1, x2, y2 }, x_scale, y_scale, epsilon)
330
- }
331
- }
332
-
333
- return boundaries
334
- }
335
-
336
- // Extract boundaries from MPDS SVGs — simple M x1,y1 L x2,y2 path elements
337
- // inside the plot area with thin stroke (0.216) and dark color
338
- function extract_mpds_boundaries(
339
- doc: Document,
340
- boundaries: Boundary[],
341
- x_scale: LinearScale,
342
- y_scale: LinearScale,
343
- epsilon: number,
344
- ): void {
345
- // Find the plot border to filter boundaries inside it
346
- const plot_rect = find_mpds_plot_rect(doc)
347
- if (!plot_rect) return
348
-
349
- const { left, right, top, bottom } = plot_rect
350
-
351
- for (const path of Array.from(doc.querySelectorAll(`path`))) {
352
- const path_data = path.getAttribute(`d`) ?? ``
353
- const style = path.getAttribute(`style`) ?? ``
354
-
355
- // Skip tick mark paths (stroke-width > 0.3)
356
- const stroke_width = parse_stroke_width(path)
357
- if (stroke_width > 0.3 || stroke_width === 0) continue
358
-
359
- // Skip filled regions (phase region fills)
360
- if (
361
- (style.includes(`fill-rule`) || style.includes(`fill: #`) || style.includes(`fill:#`)) &&
362
- !style.includes(`fill: none`) &&
363
- !style.includes(`fill:none`)
364
- ) {
365
- continue
366
- }
367
-
368
- // Skip red annotation lines
369
- if (style.includes(`#e30016`) || style.includes(`#E30016`)) continue
370
-
371
- // Parse as simple M...L line
372
- const coords = parse_ml_path(path_data)
373
- if (!coords) continue
374
-
375
- // Must be inside the plot area
376
- const inside =
377
- coords.x1 >= left - 1 &&
378
- coords.x1 <= right + 1 &&
379
- coords.x2 >= left - 1 &&
380
- coords.x2 <= right + 1 &&
381
- coords.y1 >= top - 1 &&
382
- coords.y1 <= bottom + 1 &&
383
- coords.y2 >= top - 1 &&
384
- coords.y2 <= bottom + 1
385
- if (!inside) continue
386
-
387
- // Skip very short segments (< 10px)
388
- const dx = Math.abs(coords.x2 - coords.x1)
389
- const dy = Math.abs(coords.y2 - coords.y1)
390
- if (dx < 10 && dy < 10) continue
391
-
392
- // Skip the plot border itself (connects all 4 edges)
393
- const touches_left = Math.abs(coords.x1 - left) < 2 || Math.abs(coords.x2 - left) < 2
394
- const touches_right = Math.abs(coords.x1 - right) < 2 || Math.abs(coords.x2 - right) < 2
395
- if (touches_left && touches_right) continue // spans full width = likely axis
396
-
397
- add_boundary(boundaries, coords, x_scale, y_scale, epsilon)
398
- }
399
- }
400
-
401
- // Find the plot area rectangle in an MPDS SVG
402
- // Handles both M...L...L...L...Z and M...V...H...V...Z formats
403
- function find_mpds_plot_rect(
404
- doc: Document,
405
- ): { left: number; right: number; top: number; bottom: number } | null {
406
- for (const path of Array.from(doc.querySelectorAll(`path`))) {
407
- const path_data = path.getAttribute(`d`) ?? ``
408
- const style = path.getAttribute(`style`) ?? ``
409
- if (!style.includes(`fill: none`) && !style.includes(`fill:none`)) continue
410
- if (!path_data.includes(`Z`) && !path_data.includes(`z`)) continue
411
-
412
- // Parse path into absolute segments and extract corner points
413
- const segments = parse_path_segments(path_data)
414
- if (segments.length < 3) continue // rectangle needs at least 3 segments (4th is Z)
415
-
416
- // Collect all x and y coordinates from segment endpoints
417
- const xs: number[] = []
418
- const ys: number[] = []
419
- for (const [x1, y1, x2, y2] of segments) {
420
- xs.push(Math.round(x1 * 10) / 10, Math.round(x2 * 10) / 10)
421
- ys.push(Math.round(y1 * 10) / 10, Math.round(y2 * 10) / 10)
422
- }
423
-
424
- const unique_x = [...new Set(xs)]
425
- const unique_y = [...new Set(ys)]
426
-
427
- // Rectangle has exactly 2 unique x and 2 unique y values
428
- if (unique_x.length === 2 && unique_y.length === 2) {
429
- return {
430
- left: Math.min(...unique_x),
431
- right: Math.max(...unique_x),
432
- top: Math.min(...unique_y),
433
- bottom: Math.max(...unique_y),
434
- }
435
- }
436
- }
437
- return null
438
- }
439
-
440
- // Add a boundary line, classifying as horizontal or vertical
441
- function add_boundary(
442
- boundaries: Boundary[],
443
- px: { x1: number; y1: number; x2: number; y2: number },
444
- x_scale: LinearScale,
445
- y_scale: LinearScale,
446
- epsilon: number,
447
- ): void {
448
- const is_vertical = Math.abs(px.x1 - px.x2) < epsilon
449
- const is_horizontal = Math.abs(px.y1 - px.y2) < epsilon
450
-
451
- if (!is_vertical && !is_horizontal) return // skip diagonal lines
452
-
453
- const data_x1 = x_scale.to_data(px.x1)
454
- const data_y1 = y_scale.to_data(px.y1)
455
- const data_x2 = x_scale.to_data(px.x2)
456
- const data_y2 = y_scale.to_data(px.y2)
457
-
458
- boundaries.push({
459
- x1: Math.min(data_x1, data_x2),
460
- y1: Math.min(data_y1, data_y2),
461
- x2: Math.max(data_x1, data_x2),
462
- y2: Math.max(data_y1, data_y2),
463
- orientation: is_vertical ? `vertical` : `horizontal`,
464
- })
465
- }
466
-
467
- // === Label Extraction ===
468
-
469
- // Extract phase region labels with their pixel positions
470
- function extract_labels(doc: Document, format: SvgFormat): Label[] {
471
- const labels: Label[] = []
472
-
473
- if (format === `matplotlib`) {
474
- extract_matplotlib_labels(doc, labels)
475
- } else if (format === `mpds`) {
476
- // MPDS labels use garbled encoding; skip label extraction
477
- // Regions will get auto-generated names from infer_regions
478
- } else {
479
- extract_simple_labels(doc, labels)
480
- }
481
-
482
- return labels
483
- }
484
-
485
- // Extract labels from matplotlib SVG using XML comments
486
- function extract_matplotlib_labels(doc: Document, labels: Label[]): void {
487
- // Find text groups with comments containing phase names
488
- // Matplotlib may split multi-line labels (especially rotated ones) into
489
- // separate <g> groups: text_N has "La$_2$NiO$_4$", followed by a sibling
490
- // with "+ La$_3$Ni$_2$O$_7$". We concatenate these continuation groups.
491
- for (const group of Array.from(doc.querySelectorAll(`[id^="text_"]`))) {
492
- let comment = find_comment_text(group)
493
- if (!comment) continue
494
-
495
- // Check following sibling groups for continuation comments starting with "+"
496
- let sibling = group.nextElementSibling
497
- while (sibling) {
498
- // Stop at the next text_N group (that's a separate label)
499
- if (sibling.id?.startsWith(`text_`)) break
500
- const continuation = find_comment_text(sibling)
501
- if (continuation?.trimStart().startsWith(`+`)) {
502
- comment += ` ${continuation.trim()}`
503
- sibling = sibling.nextElementSibling
504
- } else {
505
- break
506
- }
507
- }
508
-
509
- if (!comment.includes(`+`)) continue // skip axis labels (no "+")
510
-
511
- // Clean LaTeX: "La$_2$NiO$_4$ + NiO" -> "La2NiO4 + NiO"
512
- const text = clean_latex(comment.trim())
513
-
514
- // Get position from transform="translate(x, y)"
515
- const pos = parse_translate(group.querySelector(`g[transform]`)) ?? parse_translate(group)
516
- if (!pos) continue
517
-
518
- labels.push({ text, px_x: pos[0], px_y: pos[1] })
519
- }
520
- }
521
-
522
- // Extract labels from simple SVG using class="label-main"
523
- function extract_simple_labels(doc: Document, labels: Label[]): void {
524
- for (const text_el of Array.from(doc.querySelectorAll(`.label-main`))) {
525
- // Get plain text content (strips tspan tags)
526
- const text = (text_el.textContent ?? ``).replaceAll(/\s+/g, ` `).trim()
527
- if (!text.includes(`+`)) continue // skip non-phase labels
528
-
529
- // Get position from transform or x/y attributes
530
- const pos = parse_translate(text_el)
531
- const px_x = pos?.[0] ?? parse_float_attr(text_el, `x`)
532
- const px_y = pos?.[1] ?? parse_float_attr(text_el, `y`)
533
-
534
- if (px_x !== null && px_y !== null) {
535
- labels.push({ text, px_x, px_y })
536
- }
537
- }
538
- }
539
-
540
- // === Component Inference ===
541
-
542
- const split_phase_label = (label: Label) => label.text.split(/\s*\+\s*/)
543
-
544
- // Infer binary components from region labels
545
- function infer_components(labels: Label[]): [string, string] {
546
- // Sort labels by x position, split each into phases
547
- const sorted = labels.toSorted((a, b) => a.px_x - b.px_x)
548
- if (sorted.length < 2) return [`A`, `B`]
549
-
550
- const leftmost = split_phase_label(sorted[0])
551
- const rightmost = split_phase_label(sorted[sorted.length - 1])
552
-
553
- // Component A: the unique phase in the leftmost region that doesn't appear on the right
554
- // For "La2NiO4 + La2O3", La2O3 is the pure A endpoint
555
- let comp_a = `A`
556
- if (leftmost.length === 2) {
557
- const right_phases = new Set(sorted.slice(-3).flatMap(split_phase_label))
558
- comp_a = leftmost.find((phase) => !right_phases.has(phase)) ?? leftmost[1] ?? `A`
559
- }
560
-
561
- // Component B: the unique phase in the rightmost region that doesn't appear on the left
562
- let comp_b = `B`
563
- if (rightmost.length === 2) {
564
- const left_phases = new Set(sorted.slice(0, 3).flatMap(split_phase_label))
565
- comp_b = rightmost.find((phase) => !left_phases.has(phase)) ?? rightmost[1] ?? `B`
566
- }
567
-
568
- return [comp_a, comp_b]
569
- }
570
-
571
- // === Region Inference ===
572
-
573
- // Infer phase regions from orthogonal boundaries using flood-fill on a cell grid
574
- function infer_regions(
575
- boundaries: Boundary[],
576
- labels: Label[],
577
- x_scale: LinearScale,
578
- y_scale: LinearScale,
579
- ): RegionInput[] {
580
- const verticals = boundaries.filter((boundary) => boundary.orientation === `vertical`)
581
- const horizontals = boundaries.filter((boundary) => boundary.orientation === `horizontal`)
582
-
583
- // Collect all unique x and y coordinates (boundaries + domain edges)
584
- const x_coords = collect_unique_sorted([
585
- x_scale.domain[0],
586
- x_scale.domain[1],
587
- ...verticals.map((boundary) => boundary.x1), // x1 === x2 for vertical
588
- ])
589
-
590
- const y_coords = collect_unique_sorted([
591
- y_scale.domain[0],
592
- y_scale.domain[1],
593
- ...horizontals.map((boundary) => boundary.y1), // y1 === y2 for horizontal
594
- ])
595
-
596
- // Build cell grid: cells[col][row]
597
- const n_cols = x_coords.length - 1
598
- const n_rows = y_coords.length - 1
599
- const cell_ids = Array.from({ length: n_cols }, () => Array(n_rows).fill(-1))
600
-
601
- // Check which cell edges have boundaries
602
- const h_walls = Array.from({ length: n_cols }, () => Array(n_rows + 1).fill(false))
603
- const v_walls = Array.from({ length: n_cols + 1 }, () => Array(n_rows).fill(false))
604
-
605
- // Mark horizontal walls (bottom/top of cells)
606
- for (const hb of horizontals) {
607
- const row = find_coord_index(y_coords, hb.y1)
608
- if (row === -1) continue
609
- for (let col = 0; col < n_cols; col++) {
610
- const cell_x_min = x_coords[col]
611
- const cell_x_max = x_coords[col + 1]
612
- // Check if the boundary spans this cell's x range
613
- if (hb.x1 <= cell_x_min + 1e-6 && hb.x2 >= cell_x_max - 1e-6) {
614
- h_walls[col][row] = true
615
- }
616
- }
617
- }
618
-
619
- // Mark vertical walls (left/right of cells)
620
- for (const vb of verticals) {
621
- const col = find_coord_index(x_coords, vb.x1)
622
- if (col === -1) continue
623
- for (let row = 0; row < n_rows; row++) {
624
- const cell_y_min = y_coords[row]
625
- const cell_y_max = y_coords[row + 1]
626
- // Check if the boundary spans this cell's y range
627
- if (vb.y1 <= cell_y_min + 1e-6 && vb.y2 >= cell_y_max - 1e-6) {
628
- v_walls[col][row] = true
629
- }
630
- }
631
- }
632
-
633
- // Plot edges are always walls
634
- for (let col = 0; col < n_cols; col++) {
635
- h_walls[col][0] = true // bottom edge
636
- h_walls[col][n_rows] = true // top edge
637
- }
638
- for (let row = 0; row < n_rows; row++) {
639
- v_walls[0][row] = true // left edge
640
- v_walls[n_cols][row] = true // right edge
641
- }
642
-
643
- // Flood-fill to assign region IDs
644
- let next_region_id = 0
645
- for (let col = 0; col < n_cols; col++) {
646
- for (let row = 0; row < n_rows; row++) {
647
- if (cell_ids[col][row] !== -1) continue
648
- flood_fill(cell_ids, h_walls, v_walls, col, row, n_cols, n_rows, next_region_id)
649
- next_region_id++
650
- }
651
- }
652
-
653
- // Assign labels to regions by checking which region contains each label's position
654
- const region_labels = new Map<number, string>()
655
- for (const label of labels) {
656
- const data_x = x_scale.to_data(label.px_x)
657
- const data_y = y_scale.to_data(label.px_y)
658
-
659
- const col = find_cell_index(x_coords, data_x)
660
- const row = find_cell_index(y_coords, data_y)
661
- if (col >= 0 && col < n_cols && row >= 0 && row < n_rows) {
662
- const region_id = cell_ids[col][row]
663
- if (region_id !== -1) {
664
- region_labels.set(region_id, label.text)
665
- }
666
- }
667
- }
668
-
669
- // Build region polygons from merged cells
670
- const regions: RegionInput[] = []
671
- for (let region_id = 0; region_id < next_region_id; region_id++) {
672
- const name = region_labels.get(region_id) ?? `Region ${region_id + 1}`
673
- // Slug can be empty for non-ASCII labels like "α + β" — fall back to region_N
674
- const slug =
675
- name
676
- .toLowerCase()
677
- .replaceAll(/[^a-z0-9]+/g, `_`)
678
- .replaceAll(/^_|_$/g, ``) || `region_${region_id + 1}`
679
-
680
- // Find bounding box of all cells in this region
681
- let [min_x, max_x] = [Infinity, -Infinity]
682
- let [min_y, max_y] = [Infinity, -Infinity]
683
- for (let col = 0; col < n_cols; col++) {
684
- for (let row = 0; row < n_rows; row++) {
685
- if (cell_ids[col][row] !== region_id) continue
686
- min_x = Math.min(min_x, x_coords[col])
687
- max_x = Math.max(max_x, x_coords[col + 1])
688
- min_y = Math.min(min_y, y_coords[row])
689
- max_y = Math.max(max_y, y_coords[row + 1])
690
- }
691
- }
692
-
693
- if (min_x === Infinity) continue
694
-
695
- const bounds: DiagramPoint[] = [
696
- [round(min_x), round(min_y)],
697
- [round(max_x), round(min_y)],
698
- [round(max_x), round(max_y)],
699
- [round(min_x), round(max_y)],
700
- ]
701
-
702
- regions.push({
703
- id: slug,
704
- name,
705
- color: TWO_PHASE_COLORS[region_id % TWO_PHASE_COLORS.length],
706
- bounds,
707
- })
708
- }
709
-
710
- return regions
711
- }
712
-
713
- // Flood-fill connected cells that share an open edge
714
- function flood_fill(
715
- cell_ids: number[][],
716
- h_walls: boolean[][],
717
- v_walls: boolean[][],
718
- start_col: number,
719
- start_row: number,
720
- n_cols: number,
721
- n_rows: number,
722
- region_id: number,
723
- ): void {
724
- const stack: [number, number][] = [[start_col, start_row]]
725
-
726
- for (let item = stack.pop(); item; item = stack.pop()) {
727
- const [col, row] = item
728
- if (col < 0 || col >= n_cols || row < 0 || row >= n_rows) continue
729
- if (cell_ids[col][row] !== -1) continue
730
-
731
- cell_ids[col][row] = region_id
732
-
733
- // Check neighbors (no wall between them)
734
- // Left neighbor: check v_walls[col][row]
735
- if (col > 0 && !v_walls[col][row]) stack.push([col - 1, row])
736
- // Right neighbor: check v_walls[col+1][row]
737
- if (col < n_cols - 1 && !v_walls[col + 1][row]) stack.push([col + 1, row])
738
- // Bottom neighbor: check h_walls[col][row]
739
- if (row > 0 && !h_walls[col][row]) stack.push([col, row - 1])
740
- // Top neighbor: check h_walls[col][row+1]
741
- if (row < n_rows - 1 && !h_walls[col][row + 1]) stack.push([col, row + 1])
742
- }
743
- }
744
-
745
- // === Curve Generation ===
746
-
747
- // Generate named curves from boundaries for the DiagramInput format
748
- function generate_curves(boundaries: Boundary[]): Record<string, DiagramPoint[]> {
749
- const curves: Record<string, DiagramPoint[]> = {}
750
- const counts = { vertical: 0, horizontal: 0 }
751
-
752
- for (const bnd of boundaries) {
753
- const is_vert = bnd.orientation === `vertical`
754
- const name = `${bnd.orientation}_${counts[bnd.orientation]++}`
755
- curves[name] = [
756
- [round(bnd.x1), round(bnd.y1)],
757
- [round(is_vert ? bnd.x1 : bnd.x2), round(is_vert ? bnd.y2 : bnd.y1)],
758
- ]
759
- }
760
-
761
- return curves
762
- }
763
-
764
- // === Main Entry Point ===
765
-
766
- // Parse a phase diagram SVG string and return a DiagramInput
767
- export function parse_phase_diagram_svg(svg_string: string): DiagramInput {
768
- const parser = new DOMParser()
769
- const doc = parser.parseFromString(svg_string, `image/svg+xml`)
770
-
771
- // Check for parse errors
772
- const parse_error = doc.querySelector(`parsererror`)
773
- if (parse_error) {
774
- throw new Error(`Invalid SVG: ${parse_error.textContent}`)
775
- }
776
-
777
- const format = detect_format(doc)
778
- const { x_scale, y_scale } = extract_axis_scales(doc, format)
779
- const boundaries = extract_boundaries(doc, format, x_scale, y_scale)
780
- const labels = extract_labels(doc, format)
781
- const components = format === `mpds` ? infer_mpds_components(doc) : infer_components(labels)
782
-
783
- if (boundaries.length === 0) {
784
- throw new Error(`No phase boundaries found in SVG`)
785
- }
786
-
787
- const regions = infer_regions(boundaries, labels, x_scale, y_scale)
788
- const curves = generate_curves(boundaries)
789
-
790
- // MPDS SVGs use °C for temperature
791
- const temp_unit = format === `mpds` ? `°C` : `K`
792
-
793
- return {
794
- meta: {
795
- components,
796
- temp_range: y_scale.domain,
797
- temp_unit,
798
- comp_unit: `fraction`,
799
- title: `Imported Phase Diagram`,
800
- },
801
- curves,
802
- regions,
803
- }
804
- }
805
-
806
- // Infer components from MPDS SVGs by finding element-like text content
807
- // MPDS SVGs have readable element names as text (e.g., "Cu", "Si")
808
- function infer_mpds_components(doc: Document): [string, string] {
809
- const elements = new Set<string>()
810
- for (const text_el of Array.from(doc.querySelectorAll(`text`))) {
811
- const content = text_el.textContent?.trim() ?? ``
812
- // Match single/two-letter element symbols, exclude common non-element text
813
- if (/^[A-Z][a-z]?$/.test(content) && content !== `L` && content !== `M`) {
814
- elements.add(content)
815
- }
816
- }
817
- const unique = [...elements]
818
- return [unique[0] ?? `A`, unique[1] ?? `B`]
819
- }
820
-
821
- // === Utility Functions ===
822
-
823
- // Parse stroke-width from style attribute or direct attribute (returns 0 if not found)
824
- function parse_stroke_width(el: Element): number {
825
- const style_match = /stroke-width:\s*([\d.]+)/.exec(el.getAttribute(`style`) ?? ``)
826
- if (style_match) return parseFloat(style_match[1])
827
- const attr = el.getAttribute(`stroke-width`)
828
- return attr ? parseFloat(attr) || 0 : 0
829
- }
830
-
831
- // Parse a float attribute from an SVG element
832
- function parse_float_attr(el: Element, attr: string): number | null {
833
- const val = el.getAttribute(attr)
834
- if (val === null) return null
835
- const parsed = parseFloat(val)
836
- return isNaN(parsed) ? null : parsed
837
- }
838
-
839
- // Extract a number from XML comment nodes inside a group element
840
- function extract_comment_number(group: Element): number | null {
841
- const walker = group.ownerDocument.createTreeWalker(group, NodeFilter.SHOW_COMMENT)
842
- let node: Comment | null
843
- while ((node = walker.nextNode() as Comment | null)) {
844
- const value = parseFloat(node.textContent?.trim() ?? ``)
845
- if (!isNaN(value)) return value
846
- }
847
- return null
848
- }
849
-
850
- // Find the first XML comment text inside or preceding a group
851
- function find_comment_text(group: Element): string | null {
852
- // Check comment nodes inside the group
853
- const walker = group.ownerDocument.createTreeWalker(group, NodeFilter.SHOW_COMMENT)
854
- let node: Comment | null
855
- while ((node = walker.nextNode() as Comment | null)) {
856
- const text = node.textContent?.trim()
857
- if (text && text.length > 1) return text
858
- }
859
-
860
- // Check preceding sibling comments
861
- let sibling = group.previousSibling
862
- while (sibling) {
863
- if (sibling.nodeType === Node.COMMENT_NODE) {
864
- const text = sibling.textContent?.trim()
865
- if (text && text.length > 1) return text
866
- }
867
- if (sibling.nodeType === Node.ELEMENT_NODE) break // stop at previous element
868
- sibling = sibling.previousSibling
869
- }
870
-
871
- return null
872
- }
873
-
874
- // Clean LaTeX subscript notation: "La$_2$NiO$_4$" -> "La2NiO4"
875
- const clean_latex = (text: string): string =>
876
- text
877
- .replaceAll(/\$_\{([^}]*)\}\$/g, `$1`) // $_{10}$ -> 10
878
- .replaceAll(/\$_(\d)\$/g, `$1`) // $_2$ -> 2
879
- .replaceAll(`$`, ``) // remove any remaining $
880
- .replaceAll(/\s+/g, ` `)
881
- .trim()
882
-
883
- // Parse SVG path data into absolute line segments [x1,y1,x2,y2]
884
- // Handles all SVG path commands (M/L/H/V/C/S/Q/T/A/Z, both absolute and relative)
885
- // Curves (C/S/Q/T/A) are approximated as straight lines from start to endpoint
886
- // After M/m, implicit coordinates are treated as L/l per SVG spec
887
- function parse_path_segments(path_str: string): [number, number, number, number][] {
888
- const segments: [number, number, number, number][] = []
889
- let [cursor_x, cursor_y] = [0, 0]
890
- let [start_x, start_y] = [0, 0]
891
- let last_cmd = ``
892
-
893
- // Numbers to skip before the endpoint x,y for each curve command
894
- const curve_skip: Record<string, number> = { C: 4, S: 2, Q: 2, T: 0, A: 5 }
895
- const tokens = path_str.match(/[MmLlHhVvCcSsQqTtAaZz]|[-+]?[\d]*\.?[\d]+(?:[eE][-+]?\d+)?/g)
896
- if (!tokens) return segments
897
-
898
- let idx = 0
899
- const peek = () => tokens[idx]
900
- const next_num = () => parseFloat(tokens[idx++] ?? `0`)
901
-
902
- while (idx < tokens.length) {
903
- let cmd = peek() ?? ``
904
-
905
- if (cmd.length === 1 && /[A-Za-z]/.test(cmd)) {
906
- idx++
907
- last_cmd = cmd
908
- } else {
909
- // Implicit repeat: after M→L, after m→l, others repeat themselves
910
- cmd = last_cmd === `M` ? `L` : last_cmd === `m` ? `l` : last_cmd
911
- }
912
-
913
- if (cmd === `M` || cmd === `m`) {
914
- const next_x = next_num()
915
- const next_y = next_num()
916
- cursor_x = cmd === `M` ? next_x : cursor_x + next_x
917
- cursor_y = cmd === `M` ? next_y : cursor_y + next_y
918
- start_x = cursor_x
919
- start_y = cursor_y
920
- last_cmd = cmd
921
- } else if (cmd === `L` || cmd === `l`) {
922
- const next_x = next_num()
923
- const next_y = next_num()
924
- const x2 = cmd === `L` ? next_x : cursor_x + next_x
925
- const y2 = cmd === `L` ? next_y : cursor_y + next_y
926
- segments.push([cursor_x, cursor_y, x2, y2])
927
- cursor_x = x2
928
- cursor_y = y2
929
- } else if (cmd === `H` || cmd === `h`) {
930
- const next_x = next_num()
931
- const x2 = cmd === `H` ? next_x : cursor_x + next_x
932
- segments.push([cursor_x, cursor_y, x2, cursor_y])
933
- cursor_x = x2
934
- } else if (cmd === `V` || cmd === `v`) {
935
- const next_y = next_num()
936
- const y2 = cmd === `V` ? next_y : cursor_y + next_y
937
- segments.push([cursor_x, cursor_y, cursor_x, y2])
938
- cursor_y = y2
939
- } else if (cmd === `Z` || cmd === `z`) {
940
- if (cursor_x !== start_x || cursor_y !== start_y) {
941
- segments.push([cursor_x, cursor_y, start_x, start_y])
942
- }
943
- cursor_x = start_x
944
- cursor_y = start_y
945
- } else {
946
- // Curve commands: skip control/arc params, use endpoint as straight line
947
- // C=4 skip, S=2, Q=2, T=0, A=5 (numbers before the final x,y endpoint)
948
- const upper = cmd.toUpperCase()
949
- const skip = curve_skip[upper]
950
- if (skip !== undefined) {
951
- for (let skip_idx = 0; skip_idx < skip; skip_idx++) next_num()
952
- const end_x = next_num()
953
- const end_y = next_num()
954
- const x2 = cmd === upper ? end_x : cursor_x + end_x
955
- const y2 = cmd === upper ? end_y : cursor_y + end_y
956
- segments.push([cursor_x, cursor_y, x2, y2])
957
- cursor_x = x2
958
- cursor_y = y2
959
- } else {
960
- idx++ // skip unknown tokens
961
- }
962
- }
963
- }
964
- return segments
965
- }
966
-
967
- // Parse a simple 2-point line path (M...L only). Returns null for multi-segment paths
968
- // to enforce the single-line contract expected by boundary extraction.
969
- function parse_ml_path(
970
- path_str: string,
971
- ): { x1: number; y1: number; x2: number; y2: number } | null {
972
- const segments = parse_path_segments(path_str)
973
- if (segments.length !== 1) return null
974
- const [x1, y1, x2, y2] = segments[0]
975
- return { x1, y1, x2, y2 }
976
- }
977
-
978
- // Parse translate(x, y) or translate(x) from a transform attribute
979
- // Single-arg translate uses implicit y=0 per SVG spec
980
- function parse_translate(el: Element | null): [number, number] | null {
981
- const match = /translate\(\s*([\d.eE+-]+)(?:\s*[,\s]\s*([\d.eE+-]+))?\s*\)/.exec(
982
- el?.getAttribute(`transform`) ?? ``,
983
- )
984
- if (!match) return null
985
- return [parseFloat(match[1]), match[2] ? parseFloat(match[2]) : 0]
986
- }
987
-
988
- // Get translate X or Y from a group's transform attribute
989
- function get_group_translate(el: Element | null, axis: `x` | `y`): number {
990
- const coords = parse_translate(el)
991
- return coords ? coords[axis === `x` ? 0 : 1] : 0
992
- }
993
-
994
- // Collect unique sorted values from an array (with epsilon deduplication)
995
- function collect_unique_sorted(values: number[]): number[] {
996
- if (values.length === 0) return []
997
- const sorted = values.toSorted((a, b) => a - b)
998
- const unique: number[] = [sorted[0]]
999
- for (let idx = 1; idx < sorted.length; idx++) {
1000
- if (Math.abs(sorted[idx] - unique[unique.length - 1]) > 1e-4) {
1001
- unique.push(sorted[idx])
1002
- }
1003
- }
1004
- return unique
1005
- }
1006
-
1007
- // Find the index of a coordinate in a sorted array (with epsilon tolerance)
1008
- function find_coord_index(coords: number[], value: number): number {
1009
- for (let idx = 0; idx < coords.length; idx++) {
1010
- if (Math.abs(coords[idx] - value) < 1e-4) return idx
1011
- }
1012
- return -1
1013
- }
1014
-
1015
- // Find which cell interval a value falls into
1016
- function find_cell_index(coords: number[], value: number): number {
1017
- for (let idx = 0; idx < coords.length - 1; idx++) {
1018
- if (value >= coords[idx] - 1e-4 && value <= coords[idx + 1] + 1e-4) {
1019
- return idx
1020
- }
1021
- }
1022
- return -1
1023
- }