@perspective-dev/viewer 4.0.0

This diff represents the content of publicly available package versions that have been released to one of the supported registries. The information contained in this diff is provided for informational purposes only and reflects changes between package versions as they appear in their respective public registries.
Files changed (394) hide show
  1. package/LICENSE.md +193 -0
  2. package/README.md +4 -0
  3. package/dist/cdn/perspective-viewer.js +3 -0
  4. package/dist/cdn/perspective-viewer.js.map +7 -0
  5. package/dist/css/dracula.css +1 -0
  6. package/dist/css/gruvbox-dark.css +1 -0
  7. package/dist/css/gruvbox.css +1 -0
  8. package/dist/css/icons.css +1 -0
  9. package/dist/css/intl/de.css +1 -0
  10. package/dist/css/intl/es.css +1 -0
  11. package/dist/css/intl/fr.css +1 -0
  12. package/dist/css/intl/ja.css +1 -0
  13. package/dist/css/intl/pt.css +1 -0
  14. package/dist/css/intl/zh.css +1 -0
  15. package/dist/css/intl.css +1 -0
  16. package/dist/css/monokai.css +1 -0
  17. package/dist/css/pro-dark.css +1 -0
  18. package/dist/css/pro.css +1 -0
  19. package/dist/css/solarized-dark.css +1 -0
  20. package/dist/css/solarized.css +1 -0
  21. package/dist/css/themes.css +1 -0
  22. package/dist/css/vaporwave.css +1 -0
  23. package/dist/css/variables.css +0 -0
  24. package/dist/esm/bootstrap.d.ts +1 -0
  25. package/dist/esm/extensions.d.ts +70 -0
  26. package/dist/esm/perspective-viewer.d.ts +39 -0
  27. package/dist/esm/perspective-viewer.inline.js +3 -0
  28. package/dist/esm/perspective-viewer.inline.js.map +7 -0
  29. package/dist/esm/perspective-viewer.js +3 -0
  30. package/dist/esm/perspective-viewer.js.map +7 -0
  31. package/dist/esm/plugin.d.ts +195 -0
  32. package/dist/esm/ts-rs/Aggregate.d.ts +1 -0
  33. package/dist/esm/ts-rs/ColumnConfigValues.d.ts +31 -0
  34. package/dist/esm/ts-rs/CustomDatetimeFormat.d.ts +1 -0
  35. package/dist/esm/ts-rs/CustomDatetimeStyleConfig.d.ts +15 -0
  36. package/dist/esm/ts-rs/CustomNumberFormatConfig.d.ts +18 -0
  37. package/dist/esm/ts-rs/DatetimeColorMode.d.ts +1 -0
  38. package/dist/esm/ts-rs/DatetimeFormatType.d.ts +6 -0
  39. package/dist/esm/ts-rs/Expressions.d.ts +3 -0
  40. package/dist/esm/ts-rs/Filter.d.ts +2 -0
  41. package/dist/esm/ts-rs/FilterReducer.d.ts +1 -0
  42. package/dist/esm/ts-rs/FilterTerm.d.ts +2 -0
  43. package/dist/esm/ts-rs/FormatMode.d.ts +1 -0
  44. package/dist/esm/ts-rs/FormatUnit.d.ts +1 -0
  45. package/dist/esm/ts-rs/NumberBackgroundMode.d.ts +1 -0
  46. package/dist/esm/ts-rs/NumberForegroundMode.d.ts +1 -0
  47. package/dist/esm/ts-rs/OptionalUpdate.d.ts +1 -0
  48. package/dist/esm/ts-rs/PluginConfig.d.ts +2 -0
  49. package/dist/esm/ts-rs/RoundingMode.d.ts +1 -0
  50. package/dist/esm/ts-rs/RoundingPriority.d.ts +1 -0
  51. package/dist/esm/ts-rs/Scalar.d.ts +5 -0
  52. package/dist/esm/ts-rs/SignDisplay.d.ts +1 -0
  53. package/dist/esm/ts-rs/SimpleDatetimeFormat.d.ts +1 -0
  54. package/dist/esm/ts-rs/SimpleDatetimeStyleConfig.d.ts +6 -0
  55. package/dist/esm/ts-rs/Sort.d.ts +2 -0
  56. package/dist/esm/ts-rs/SortDir.d.ts +1 -0
  57. package/dist/esm/ts-rs/StringColorMode.d.ts +1 -0
  58. package/dist/esm/ts-rs/TrailingZeroDisplay.d.ts +1 -0
  59. package/dist/esm/ts-rs/UseGrouping.d.ts +1 -0
  60. package/dist/esm/ts-rs/ViewerConfigUpdate.d.ts +102 -0
  61. package/dist/esm/ts-rs/serde_json/JsonValue.d.ts +3 -0
  62. package/dist/wasm/perspective-viewer.d.ts +1291 -0
  63. package/dist/wasm/perspective-viewer.js +3883 -0
  64. package/dist/wasm/perspective-viewer.wasm +0 -0
  65. package/dist/wasm/perspective-viewer.wasm.d.ts +142 -0
  66. package/dist/wasm/snippets/perspective-viewer-c69283f6f62a5f14/inline0.js +1 -0
  67. package/dist/wasm/snippets/perspective-viewer-c69283f6f62a5f14/inline1.js +1 -0
  68. package/dist/wasm/snippets/perspective-viewer-c69283f6f62a5f14/inline2.js +1 -0
  69. package/dist/wasm/snippets/perspective-viewer-c69283f6f62a5f14/inline3.js +44 -0
  70. package/dist/wasm/snippets/perspective-viewer-c69283f6f62a5f14/inline4.js +1 -0
  71. package/package.json +55 -0
  72. package/src/less/aggregate-selector.less +72 -0
  73. package/src/less/column-dropdown.less +95 -0
  74. package/src/less/column-selector.less +551 -0
  75. package/src/less/column-settings-panel.less +255 -0
  76. package/src/less/column-style.less +319 -0
  77. package/src/less/column-symbol-attributes.less +73 -0
  78. package/src/less/config-selector.less +363 -0
  79. package/src/less/containers/dropdown-menu.less +94 -0
  80. package/src/less/containers/pairs-list.less +46 -0
  81. package/src/less/containers/radio-list.less +29 -0
  82. package/src/less/containers/scroll-panel.less +27 -0
  83. package/src/less/containers/split-panel.less +80 -0
  84. package/src/less/containers/tabs.less +90 -0
  85. package/src/less/dom/checkbox.less +100 -0
  86. package/src/less/dom/scrollbar.less +35 -0
  87. package/src/less/dom/select.less +63 -0
  88. package/src/less/empty-column.less +72 -0
  89. package/src/less/expression-editor.less +152 -0
  90. package/src/less/filter-dropdown.less +53 -0
  91. package/src/less/filter-item.less +126 -0
  92. package/src/less/form/code-editor.less +137 -0
  93. package/src/less/form/debug.less +88 -0
  94. package/src/less/function-dropdown.less +45 -0
  95. package/src/less/plugin-selector.less +173 -0
  96. package/src/less/render-warning.less +81 -0
  97. package/src/less/status-bar.less +551 -0
  98. package/src/less/type-icon.less +68 -0
  99. package/src/less/viewer.less +368 -0
  100. package/src/rust/components/column_dropdown.rs +162 -0
  101. package/src/rust/components/column_selector/active_column.rs +462 -0
  102. package/src/rust/components/column_selector/add_expression_button.rs +59 -0
  103. package/src/rust/components/column_selector/aggregate_selector.rs +186 -0
  104. package/src/rust/components/column_selector/config_selector.rs +674 -0
  105. package/src/rust/components/column_selector/empty_column.rs +134 -0
  106. package/src/rust/components/column_selector/expression_toolbar.rs +45 -0
  107. package/src/rust/components/column_selector/filter_column.rs +530 -0
  108. package/src/rust/components/column_selector/inactive_column.rs +221 -0
  109. package/src/rust/components/column_selector/invalid_column.rs +37 -0
  110. package/src/rust/components/column_selector/pivot_column.rs +93 -0
  111. package/src/rust/components/column_selector/sort_column.rs +135 -0
  112. package/src/rust/components/column_selector.rs +426 -0
  113. package/src/rust/components/column_settings_sidebar/attributes_tab.rs +34 -0
  114. package/src/rust/components/column_settings_sidebar/mod.rs +17 -0
  115. package/src/rust/components/column_settings_sidebar/save_settings.rs +68 -0
  116. package/src/rust/components/column_settings_sidebar/sidebar.rs +396 -0
  117. package/src/rust/components/column_settings_sidebar/style_tab/column_style.rs +177 -0
  118. package/src/rust/components/column_settings_sidebar/style_tab/stub.rs +34 -0
  119. package/src/rust/components/column_settings_sidebar/style_tab/symbol/row_selector.rs +101 -0
  120. package/src/rust/components/column_settings_sidebar/style_tab/symbol/symbol_config.rs +24 -0
  121. package/src/rust/components/column_settings_sidebar/style_tab/symbol/symbol_pairs.rs +185 -0
  122. package/src/rust/components/column_settings_sidebar/style_tab/symbol/symbol_selector.rs +55 -0
  123. package/src/rust/components/column_settings_sidebar/style_tab/symbol.rs +131 -0
  124. package/src/rust/components/column_settings_sidebar/style_tab.rs +231 -0
  125. package/src/rust/components/containers/dragdrop_list.rs +325 -0
  126. package/src/rust/components/containers/dropdown_menu.rs +104 -0
  127. package/src/rust/components/containers/kvpair.rs +47 -0
  128. package/src/rust/components/containers/mod.rs +28 -0
  129. package/src/rust/components/containers/scroll_panel.rs +311 -0
  130. package/src/rust/components/containers/scroll_panel_item.rs +111 -0
  131. package/src/rust/components/containers/select.rs +269 -0
  132. package/src/rust/components/containers/sidebar.rs +89 -0
  133. package/src/rust/components/containers/split_panel.rs +483 -0
  134. package/src/rust/components/containers/tab_list.rs +104 -0
  135. package/src/rust/components/containers/tests/mod.rs +11 -0
  136. package/src/rust/components/containers/tests/split_panel.rs +91 -0
  137. package/src/rust/components/containers/trap_door_panel.rs +50 -0
  138. package/src/rust/components/copy_dropdown.rs +136 -0
  139. package/src/rust/components/datetime_column_style/custom.rs +222 -0
  140. package/src/rust/components/datetime_column_style/simple.rs +122 -0
  141. package/src/rust/components/datetime_column_style.rs +284 -0
  142. package/src/rust/components/editable_header.rs +183 -0
  143. package/src/rust/components/empty_row.rs +169 -0
  144. package/src/rust/components/error_message.rs +56 -0
  145. package/src/rust/components/export_dropdown.rs +153 -0
  146. package/src/rust/components/expression_editor.rs +153 -0
  147. package/src/rust/components/filter_dropdown.rs +144 -0
  148. package/src/rust/components/font_loader.rs +254 -0
  149. package/src/rust/components/form/code_editor.rs +286 -0
  150. package/src/rust/components/form/color_range_selector.rs +115 -0
  151. package/src/rust/components/form/color_selector.rs +54 -0
  152. package/src/rust/components/form/debug.rs +253 -0
  153. package/src/rust/components/form/highlight.rs +50 -0
  154. package/src/rust/components/form/mod.rs +29 -0
  155. package/src/rust/components/form/number_field.rs +72 -0
  156. package/src/rust/components/form/number_input.rs +45 -0
  157. package/src/rust/components/form/number_range_field.rs +77 -0
  158. package/src/rust/components/form/optional_field.rs +53 -0
  159. package/src/rust/components/form/select_field.rs +104 -0
  160. package/src/rust/components/function_dropdown.rs +151 -0
  161. package/src/rust/components/mod.rs +45 -0
  162. package/src/rust/components/modal.rs +171 -0
  163. package/src/rust/components/number_column_style.rs +522 -0
  164. package/src/rust/components/plugin_selector.rs +196 -0
  165. package/src/rust/components/render_warning.rs +178 -0
  166. package/src/rust/components/status_bar.rs +315 -0
  167. package/src/rust/components/status_bar_counter.rs +229 -0
  168. package/src/rust/components/status_indicator.rs +164 -0
  169. package/src/rust/components/string_column_style.rs +186 -0
  170. package/src/rust/components/style/local_style.rs +35 -0
  171. package/src/rust/components/style/mod.rs +52 -0
  172. package/src/rust/components/style/style_cache.rs +132 -0
  173. package/src/rust/components/style/style_provider.rs +65 -0
  174. package/src/rust/components/style_controls/number_string_format/digits_section.rs +140 -0
  175. package/src/rust/components/style_controls/number_string_format/misc_section.rs +56 -0
  176. package/src/rust/components/style_controls/number_string_format/style_section.rs +69 -0
  177. package/src/rust/components/style_controls/number_string_format/types.rs +31 -0
  178. package/src/rust/components/style_controls/number_string_format.rs +259 -0
  179. package/src/rust/components/style_controls.rs +15 -0
  180. package/src/rust/components/type_icon.rs +53 -0
  181. package/src/rust/components/viewer.rs +668 -0
  182. package/src/rust/config/columns_config.rs +132 -0
  183. package/src/rust/config/datetime_column_style/color_mode.rs +62 -0
  184. package/src/rust/config/datetime_column_style/custom.rs +121 -0
  185. package/src/rust/config/datetime_column_style/custom_format.rs +90 -0
  186. package/src/rust/config/datetime_column_style/simple.rs +55 -0
  187. package/src/rust/config/datetime_column_style/simple_format.rs +86 -0
  188. package/src/rust/config/datetime_column_style.rs +104 -0
  189. package/src/rust/config/mod.rs +30 -0
  190. package/src/rust/config/number_column_style.rs +138 -0
  191. package/src/rust/config/number_string_format/enums.rs +298 -0
  192. package/src/rust/config/number_string_format.rs +269 -0
  193. package/src/rust/config/string_column_style.rs +140 -0
  194. package/src/rust/config/view_config.rs +24 -0
  195. package/src/rust/config/viewer_config.rs +337 -0
  196. package/src/rust/custom_elements/column_dropdown.rs +123 -0
  197. package/src/rust/custom_elements/copy_dropdown.rs +112 -0
  198. package/src/rust/custom_elements/debug_plugin.rs +114 -0
  199. package/src/rust/custom_elements/export_dropdown.rs +112 -0
  200. package/src/rust/custom_elements/filter_dropdown.rs +179 -0
  201. package/src/rust/custom_elements/function_dropdown.rs +115 -0
  202. package/src/rust/custom_elements/mod.rs +24 -0
  203. package/src/rust/custom_elements/modal.rs +395 -0
  204. package/src/rust/custom_elements/viewer.rs +880 -0
  205. package/src/rust/custom_events.rs +214 -0
  206. package/src/rust/dragdrop.rs +362 -0
  207. package/src/rust/exprtk/cursor.rs +81 -0
  208. package/src/rust/exprtk/mod.rs +17 -0
  209. package/src/rust/exprtk/tokenize/comment.rs +21 -0
  210. package/src/rust/exprtk/tokenize/number.rs +44 -0
  211. package/src/rust/exprtk/tokenize/string.rs +98 -0
  212. package/src/rust/exprtk/tokenize/symbol.rs +26 -0
  213. package/src/rust/exprtk/tokenize.rs +190 -0
  214. package/src/rust/js/clipboard.rs +77 -0
  215. package/src/rust/js/clipboard_item.rs +21 -0
  216. package/src/rust/js/intersection_observer.rs +32 -0
  217. package/src/rust/js/mimetype.rs +42 -0
  218. package/src/rust/js/mod.rs +29 -0
  219. package/src/rust/js/plugin.rs +167 -0
  220. package/src/rust/js/resize_observer.rs +37 -0
  221. package/src/rust/js/testing.rs +149 -0
  222. package/src/rust/lib.rs +116 -0
  223. package/src/rust/model/columns_iter_set.rs +340 -0
  224. package/src/rust/model/copy_export.rs +157 -0
  225. package/src/rust/model/create_col.rs +59 -0
  226. package/src/rust/model/edit_expression.rs +95 -0
  227. package/src/rust/model/export_app.rs +58 -0
  228. package/src/rust/model/export_method.rs +151 -0
  229. package/src/rust/model/get_viewer_config.rs +85 -0
  230. package/src/rust/model/intersection_observer.rs +81 -0
  231. package/src/rust/model/is_invalid_drop.rs +36 -0
  232. package/src/rust/model/mod.rs +93 -0
  233. package/src/rust/model/plugin_column_styles.rs +106 -0
  234. package/src/rust/model/resize_observer.rs +98 -0
  235. package/src/rust/model/restore_and_render.rs +113 -0
  236. package/src/rust/model/structural.rs +73 -0
  237. package/src/rust/model/update_and_render.rs +74 -0
  238. package/src/rust/presentation.rs +370 -0
  239. package/src/rust/renderer/activate.rs +49 -0
  240. package/src/rust/renderer/limits.rs +200 -0
  241. package/src/rust/renderer/plugin_store.rs +45 -0
  242. package/src/rust/renderer/registry.rs +135 -0
  243. package/src/rust/renderer/render_timer.rs +185 -0
  244. package/src/rust/renderer.rs +463 -0
  245. package/src/rust/session/column_defaults_update.rs +141 -0
  246. package/src/rust/session/drag_drop_update.rs +179 -0
  247. package/src/rust/session/metadata.rs +308 -0
  248. package/src/rust/session/replace_expression_update.rs +131 -0
  249. package/src/rust/session/view_subscription.rs +189 -0
  250. package/src/rust/session.rs +794 -0
  251. package/src/rust/utils/browser/blob.rs +49 -0
  252. package/src/rust/utils/browser/download.rs +26 -0
  253. package/src/rust/utils/browser/mod.rs +24 -0
  254. package/src/rust/utils/browser/request_animation_frame.rs +76 -0
  255. package/src/rust/utils/browser/selection.rs +79 -0
  256. package/src/rust/utils/browser/tests/debounce.rs +114 -0
  257. package/src/rust/utils/browser/tests/mod.rs +13 -0
  258. package/src/rust/utils/custom_element.rs +85 -0
  259. package/src/rust/utils/datetime.rs +49 -0
  260. package/src/rust/utils/debounce.rs +54 -0
  261. package/src/rust/utils/hooks/mod.rs +15 -0
  262. package/src/rust/utils/hooks/use_async_callback.rs +53 -0
  263. package/src/rust/utils/mod.rs +114 -0
  264. package/src/rust/utils/number_format.rs +48 -0
  265. package/src/rust/utils/pubsub.rs +222 -0
  266. package/src/rust/utils/scope.rs +54 -0
  267. package/src/rust/utils/tee.rs +88 -0
  268. package/src/rust/utils/tests/mod.rs +14 -0
  269. package/src/rust/utils/tests/pubsub.rs +95 -0
  270. package/src/rust/utils/tests/request_animation_frame.rs +42 -0
  271. package/src/rust/utils/wasm_abi.rs +61 -0
  272. package/src/rust/utils/weak_scope.rs +39 -0
  273. package/src/svg/align-scroll-icon.svg +7 -0
  274. package/src/svg/bg-pattern.png +0 -0
  275. package/src/svg/boolean-type.svg +3 -0
  276. package/src/svg/checkbox-checked-icon.svg +7 -0
  277. package/src/svg/checkbox-hover.svg +4 -0
  278. package/src/svg/checkbox-off.svg +4 -0
  279. package/src/svg/checkbox-on.svg +4 -0
  280. package/src/svg/checkbox-unchecked-icon.svg +3 -0
  281. package/src/svg/close-icon.svg +6 -0
  282. package/src/svg/column-settings-icon.svg +6 -0
  283. package/src/svg/datagrid-select-column.svg +15 -0
  284. package/src/svg/datagrid-select-region.svg +15 -0
  285. package/src/svg/datagrid-select-row.svg +13 -0
  286. package/src/svg/datagrid.svg +13 -0
  287. package/src/svg/date-type.svg +23 -0
  288. package/src/svg/downloading.gif +0 -0
  289. package/src/svg/drag-handle.svg +10 -0
  290. package/src/svg/drawer-bg-pattern-hidpi.png +0 -0
  291. package/src/svg/drawer-tab-hover.svg +8 -0
  292. package/src/svg/drawer-tab-invert-hover.svg +9 -0
  293. package/src/svg/drawer-tab-invert.svg +8 -0
  294. package/src/svg/drawer-tab.svg +7 -0
  295. package/src/svg/dropdown-selector-light.svg +3 -0
  296. package/src/svg/dropdown-selector.svg +3 -0
  297. package/src/svg/duplicate-icon.svg +4 -0
  298. package/src/svg/editable-icon.svg +5 -0
  299. package/src/svg/export-icon.svg +4 -0
  300. package/src/svg/expression.svg +10 -0
  301. package/src/svg/free-scroll-icon.svg +5 -0
  302. package/src/svg/inactive-mouseover.svg +4 -0
  303. package/src/svg/mega-menu-icons-candlestick.svg +38 -0
  304. package/src/svg/mega-menu-icons-datagrid.svg +38 -0
  305. package/src/svg/mega-menu-icons-heatmap.svg +60 -0
  306. package/src/svg/mega-menu-icons-map-scatter.svg +43 -0
  307. package/src/svg/mega-menu-icons-ohlc.svg +40 -0
  308. package/src/svg/mega-menu-icons-sunburst.svg +39 -0
  309. package/src/svg/mega-menu-icons-treemap.svg +44 -0
  310. package/src/svg/mega-menu-icons-x-bar.svg +36 -0
  311. package/src/svg/mega-menu-icons-x-y-line.svg +38 -0
  312. package/src/svg/mega-menu-icons-x-y-scatter.svg +72 -0
  313. package/src/svg/mega-menu-icons-y-area.svg +73 -0
  314. package/src/svg/mega-menu-icons-y-bar.svg +36 -0
  315. package/src/svg/mega-menu-icons-y-line.svg +36 -0
  316. package/src/svg/mega-menu-icons-y-scatter.svg +72 -0
  317. package/src/svg/number-type.svg +4 -0
  318. package/src/svg/radio-hover.svg +4 -0
  319. package/src/svg/radio-off.svg +3 -0
  320. package/src/svg/radio-on.svg +4 -0
  321. package/src/svg/read-only-icon.svg +6 -0
  322. package/src/svg/revert-icon.svg +4 -0
  323. package/src/svg/sort-abs-asc-icon.svg +3 -0
  324. package/src/svg/sort-abs-col-asc-icon.svg +3 -0
  325. package/src/svg/sort-abs-col-desc-icon.svg +3 -0
  326. package/src/svg/sort-abs-desc-icon.svg +3 -0
  327. package/src/svg/sort-asc-icon.svg +4 -0
  328. package/src/svg/sort-col-asc-icon.svg +4 -0
  329. package/src/svg/sort-col-desc-icon.svg +4 -0
  330. package/src/svg/sort-desc-icon.svg +4 -0
  331. package/src/svg/sort-none-icon.svg +3 -0
  332. package/src/svg/status_error.svg +70 -0
  333. package/src/svg/status_ok.svg +26 -0
  334. package/src/svg/string-type.svg +8 -0
  335. package/src/svg/theme-icon.svg +5 -0
  336. package/src/svg/updating.gif +0 -0
  337. package/src/themes/dracula.less +101 -0
  338. package/src/themes/gruvbox-dark.less +116 -0
  339. package/src/themes/gruvbox.less +152 -0
  340. package/src/themes/icons.less +130 -0
  341. package/src/themes/intl/de.less +102 -0
  342. package/src/themes/intl/es.less +102 -0
  343. package/src/themes/intl/fr.less +102 -0
  344. package/src/themes/intl/ja.less +102 -0
  345. package/src/themes/intl/pt.less +102 -0
  346. package/src/themes/intl/zh.less +102 -0
  347. package/src/themes/intl.less +102 -0
  348. package/src/themes/monokai.less +107 -0
  349. package/src/themes/pro-dark.less +147 -0
  350. package/src/themes/pro.less +186 -0
  351. package/src/themes/solarized-dark.less +78 -0
  352. package/src/themes/solarized.less +102 -0
  353. package/src/themes/themes.less +21 -0
  354. package/src/themes/vaporwave.less +145 -0
  355. package/src/themes/variables.less +24 -0
  356. package/src/ts/bootstrap.ts +26 -0
  357. package/src/ts/extensions.ts +124 -0
  358. package/src/ts/perspective-viewer.cdn.ts +19 -0
  359. package/src/ts/perspective-viewer.inline.ts +23 -0
  360. package/src/ts/perspective-viewer.ts +59 -0
  361. package/src/ts/plugin.ts +279 -0
  362. package/src/ts/ts-rs/Aggregate.ts +3 -0
  363. package/src/ts/ts-rs/ColumnConfigValues.ts +14 -0
  364. package/src/ts/ts-rs/CustomDatetimeFormat.ts +3 -0
  365. package/src/ts/ts-rs/CustomDatetimeStyleConfig.ts +5 -0
  366. package/src/ts/ts-rs/CustomNumberFormatConfig.ts +8 -0
  367. package/src/ts/ts-rs/DatetimeColorMode.ts +3 -0
  368. package/src/ts/ts-rs/DatetimeFormatType.ts +8 -0
  369. package/src/ts/ts-rs/Expressions.ts +3 -0
  370. package/src/ts/ts-rs/Filter.ts +4 -0
  371. package/src/ts/ts-rs/FilterReducer.ts +3 -0
  372. package/src/ts/ts-rs/FilterTerm.ts +4 -0
  373. package/src/ts/ts-rs/FormatMode.ts +3 -0
  374. package/src/ts/ts-rs/FormatUnit.ts +3 -0
  375. package/src/ts/ts-rs/NumberBackgroundMode.ts +3 -0
  376. package/src/ts/ts-rs/NumberForegroundMode.ts +3 -0
  377. package/src/ts/ts-rs/OnUpdateData.ts +8 -0
  378. package/src/ts/ts-rs/OptionalUpdate.ts +3 -0
  379. package/src/ts/ts-rs/PluginConfig.ts +4 -0
  380. package/src/ts/ts-rs/RoundingMode.ts +3 -0
  381. package/src/ts/ts-rs/RoundingPriority.ts +3 -0
  382. package/src/ts/ts-rs/Scalar.ts +7 -0
  383. package/src/ts/ts-rs/SignDisplay.ts +3 -0
  384. package/src/ts/ts-rs/SimpleDatetimeFormat.ts +3 -0
  385. package/src/ts/ts-rs/SimpleDatetimeStyleConfig.ts +4 -0
  386. package/src/ts/ts-rs/Sort.ts +4 -0
  387. package/src/ts/ts-rs/SortDir.ts +3 -0
  388. package/src/ts/ts-rs/StringColorMode.ts +3 -0
  389. package/src/ts/ts-rs/TrailingZeroDisplay.ts +3 -0
  390. package/src/ts/ts-rs/UseGrouping.ts +3 -0
  391. package/src/ts/ts-rs/ViewOnUpdateResp.ts +3 -0
  392. package/src/ts/ts-rs/ViewerConfigUpdate.ts +90 -0
  393. package/src/ts/ts-rs/serde_json/JsonValue.ts +3 -0
  394. package/tsconfig.json +15 -0
@@ -0,0 +1,668 @@
1
+ // ┏━━━━━━━━━━━━━━━━━━━━━━━━━━━━━━━━━━━━━━━━━━━━━━━━━━━━━━━━━━━━━━━━━━━━━━━━━━━┓
2
+ // ┃ ██████ ██████ ██████ █ █ █ █ █ █▄ ▀███ █ ┃
3
+ // ┃ ▄▄▄▄▄█ █▄▄▄▄▄ ▄▄▄▄▄█ ▀▀▀▀▀█▀▀▀▀▀ █ ▀▀▀▀▀█ ████████▌▐███ ███▄ ▀█ █ ▀▀▀▀▀ ┃
4
+ // ┃ █▀▀▀▀▀ █▀▀▀▀▀ █▀██▀▀ ▄▄▄▄▄ █ ▄▄▄▄▄█ ▄▄▄▄▄█ ████████▌▐███ █████▄ █ ▄▄▄▄▄ ┃
5
+ // ┃ █ ██████ █ ▀█▄ █ ██████ █ ███▌▐███ ███████▄ █ ┃
6
+ // ┣━━━━━━━━━━━━━━━━━━━━━━━━━━━━━━━━━━━━━━━━━━━━━━━━━━━━━━━━━━━━━━━━━━━━━━━━━━━┫
7
+ // ┃ Copyright (c) 2017, the Perspective Authors. ┃
8
+ // ┃ ╌╌╌╌╌╌╌╌╌╌╌╌╌╌╌╌╌╌╌╌╌╌╌╌╌╌╌╌╌╌╌╌╌╌╌╌╌╌╌╌╌╌╌╌╌╌╌╌╌╌╌╌╌╌╌╌╌╌╌╌╌╌╌╌╌╌╌╌╌╌╌╌╌ ┃
9
+ // ┃ This file is part of the Perspective library, distributed under the terms ┃
10
+ // ┃ of the [Apache License 2.0](https://www.apache.org/licenses/LICENSE-2.0). ┃
11
+ // ┗━━━━━━━━━━━━━━━━━━━━━━━━━━━━━━━━━━━━━━━━━━━━━━━━━━━━━━━━━━━━━━━━━━━━━━━━━━━┛
12
+
13
+ use std::rc::Rc;
14
+
15
+ use futures::channel::oneshot::*;
16
+ use perspective_client::config::ColumnType;
17
+ use wasm_bindgen::prelude::*;
18
+ use yew::prelude::*;
19
+
20
+ use super::column_selector::ColumnSelector;
21
+ use super::containers::split_panel::SplitPanel;
22
+ use super::font_loader::{FontLoader, FontLoaderProps, FontLoaderStatus};
23
+ use super::form::debug::DebugPanel;
24
+ use super::plugin_selector::PluginSelector;
25
+ use super::render_warning::RenderWarning;
26
+ use super::status_bar::StatusBar;
27
+ use super::style::{LocalStyle, StyleProvider};
28
+ use crate::components::column_settings_sidebar::ColumnSettingsSidebar;
29
+ use crate::components::containers::sidebar::SidebarCloseButton;
30
+ use crate::config::*;
31
+ use crate::custom_events::CustomEvents;
32
+ use crate::dragdrop::*;
33
+ use crate::model::*;
34
+ use crate::presentation::Presentation;
35
+ use crate::renderer::*;
36
+ use crate::session::*;
37
+ use crate::utils::*;
38
+ use crate::*;
39
+
40
+ /// Locates a view column.
41
+ /// Table columns are those defined on the table, but their types will reflect
42
+ /// the view type, not the table type.
43
+ #[derive(Clone, Debug, PartialEq)]
44
+ pub enum ColumnLocator {
45
+ Table(String),
46
+ Expression(String),
47
+ NewExpression,
48
+ }
49
+ impl ColumnLocator {
50
+ /// Pulls the column's name from the locator.
51
+ /// If the column is a new expression which has yet to be saved, the
52
+ /// function will return None.
53
+ pub fn name(&self) -> Option<&String> {
54
+ match self {
55
+ Self::Table(s) | Self::Expression(s) => Some(s),
56
+ Self::NewExpression => None,
57
+ }
58
+ }
59
+
60
+ pub fn name_or_default(&self, session: &Session) -> String {
61
+ match self {
62
+ Self::Table(s) | Self::Expression(s) => s.clone(),
63
+ Self::NewExpression => session.metadata().make_new_column_name(None),
64
+ }
65
+ }
66
+
67
+ pub fn is_active(&self, session: &Session) -> bool {
68
+ self.name()
69
+ .map(|name| session.is_column_active(name))
70
+ .unwrap_or_default()
71
+ }
72
+
73
+ #[inline(always)]
74
+ pub fn is_saved_expr(&self) -> bool {
75
+ matches!(self, ColumnLocator::Expression(_))
76
+ }
77
+
78
+ #[inline(always)]
79
+ pub fn is_expr(&self) -> bool {
80
+ matches!(
81
+ self,
82
+ ColumnLocator::Expression(_) | ColumnLocator::NewExpression
83
+ )
84
+ }
85
+
86
+ #[inline(always)]
87
+ pub fn is_new_expr(&self) -> bool {
88
+ matches!(self, ColumnLocator::NewExpression)
89
+ }
90
+
91
+ pub fn view_type(&self, session: &Session) -> Option<ColumnType> {
92
+ let name = self.name().cloned().unwrap_or_default();
93
+ session.metadata().get_column_view_type(name.as_str())
94
+ }
95
+ }
96
+
97
+ #[derive(Properties)]
98
+ pub struct PerspectiveViewerProps {
99
+ pub elem: web_sys::HtmlElement,
100
+ pub session: Session,
101
+ pub renderer: Renderer,
102
+ pub presentation: Presentation,
103
+ pub dragdrop: DragDrop,
104
+ pub custom_events: CustomEvents,
105
+
106
+ #[prop_or_default]
107
+ pub weak_link: WeakScope<PerspectiveViewer>,
108
+ }
109
+
110
+ derive_model!(Renderer, Session, Presentation for PerspectiveViewerProps);
111
+
112
+ impl PartialEq for PerspectiveViewerProps {
113
+ fn eq(&self, _rhs: &Self) -> bool {
114
+ false
115
+ }
116
+ }
117
+
118
+ impl PerspectiveViewerProps {
119
+ fn is_title(&self) -> bool {
120
+ !self.presentation.get_is_workspace() && self.presentation.get_title().is_some()
121
+ }
122
+ }
123
+
124
+ #[derive(Debug)]
125
+ pub enum PerspectiveViewerMsg {
126
+ Resize,
127
+ Reset(bool, Option<Sender<()>>),
128
+ ToggleSettingsInit(Option<SettingsUpdate>, Option<Sender<ApiResult<JsValue>>>),
129
+ ToggleSettingsComplete(SettingsUpdate, Sender<()>),
130
+ ToggleDebug,
131
+ PreloadFontsUpdate,
132
+ RenderLimits(Option<(usize, usize, Option<usize>, Option<usize>)>),
133
+ SettingsPanelSizeUpdate(Option<i32>),
134
+ ColumnSettingsPanelSizeUpdate(Option<i32>),
135
+ Error,
136
+ OpenColumnSettings {
137
+ locator: Option<ColumnLocator>,
138
+ sender: Option<Sender<()>>,
139
+ toggle: bool,
140
+ },
141
+ }
142
+
143
+ pub struct PerspectiveViewer {
144
+ dimensions: Option<(usize, usize, Option<usize>, Option<usize>)>,
145
+ on_rendered: Option<Sender<()>>,
146
+ fonts: FontLoaderProps,
147
+ settings_open: bool,
148
+ debug_open: bool,
149
+ /// The column which will be opened in the ColumnSettingsSidebar
150
+ selected_column: Option<ColumnLocator>,
151
+ selected_column_is_active: bool, // TODO: should we use a struct?
152
+ on_resize: Rc<PubSub<()>>,
153
+ on_dimensions_reset: Rc<PubSub<()>>,
154
+ _subscriptions: [Subscription; 2],
155
+ settings_panel_width_override: Option<i32>,
156
+ column_settings_panel_width_override: Option<i32>,
157
+
158
+ on_close_column_settings: Callback<()>,
159
+ }
160
+
161
+ impl Component for PerspectiveViewer {
162
+ type Message = PerspectiveViewerMsg;
163
+ type Properties = PerspectiveViewerProps;
164
+
165
+ fn create(ctx: &Context<Self>) -> Self {
166
+ *ctx.props().weak_link.borrow_mut() = Some(ctx.link().clone());
167
+ let elem = ctx.props().elem.clone();
168
+ let callback = ctx
169
+ .link()
170
+ .callback(|()| PerspectiveViewerMsg::PreloadFontsUpdate);
171
+
172
+ let session_sub = {
173
+ clone!(
174
+ ctx.props().presentation,
175
+ ctx.props().session,
176
+ plugin_query = ctx.props().get_plugin_column_styles_query()
177
+ );
178
+ let callback = ctx.link().batch_callback(move |(update, render_limits)| {
179
+ if update {
180
+ vec![PerspectiveViewerMsg::RenderLimits(Some(render_limits))]
181
+ } else {
182
+ let locator =
183
+ presentation
184
+ .get_open_column_settings()
185
+ .locator
186
+ .filter(|locator| match &locator {
187
+ ColumnLocator::Table(name) => {
188
+ locator.is_active(&session)
189
+ && plugin_query
190
+ .can_render_column_styles(name)
191
+ .unwrap_or_default()
192
+ },
193
+ _ => true,
194
+ });
195
+
196
+ vec![
197
+ PerspectiveViewerMsg::RenderLimits(Some(render_limits)),
198
+ PerspectiveViewerMsg::OpenColumnSettings {
199
+ locator,
200
+ sender: None,
201
+ toggle: false,
202
+ },
203
+ ]
204
+ }
205
+ });
206
+ ctx.props()
207
+ .renderer
208
+ .render_limits_changed
209
+ .add_listener(callback)
210
+ };
211
+
212
+ let error_sub = ctx
213
+ .props()
214
+ .session
215
+ .table_errored
216
+ .add_listener(ctx.link().callback(|_| PerspectiveViewerMsg::Error));
217
+
218
+ let on_close_column_settings =
219
+ ctx.link()
220
+ .callback(|_| PerspectiveViewerMsg::OpenColumnSettings {
221
+ locator: None,
222
+ sender: None,
223
+ toggle: false,
224
+ });
225
+
226
+ Self {
227
+ dimensions: None,
228
+ on_rendered: None,
229
+ fonts: FontLoaderProps::new(&elem, callback),
230
+ settings_open: false,
231
+ debug_open: false,
232
+ selected_column: None,
233
+ selected_column_is_active: false,
234
+ on_resize: Default::default(),
235
+ on_dimensions_reset: Default::default(),
236
+ _subscriptions: [session_sub, error_sub],
237
+ settings_panel_width_override: None,
238
+ column_settings_panel_width_override: None,
239
+ on_close_column_settings,
240
+ }
241
+ }
242
+
243
+ fn update(&mut self, ctx: &Context<Self>, msg: Self::Message) -> bool {
244
+ let needs_update = self.selected_column.is_some();
245
+ match msg {
246
+ PerspectiveViewerMsg::PreloadFontsUpdate => true,
247
+ PerspectiveViewerMsg::Resize => {
248
+ self.on_resize.emit(());
249
+ false
250
+ },
251
+ PerspectiveViewerMsg::Error => true,
252
+ PerspectiveViewerMsg::Reset(all, sender) => {
253
+ self.selected_column = None;
254
+ clone!(
255
+ ctx.props().renderer,
256
+ ctx.props().session,
257
+ ctx.props().presentation
258
+ );
259
+
260
+ ApiFuture::spawn(async move {
261
+ session.reset(all).await?;
262
+ let columns_config = if all {
263
+ presentation.reset_columns_configs();
264
+ None
265
+ } else {
266
+ Some(presentation.all_columns_configs())
267
+ };
268
+
269
+ renderer.reset(columns_config.as_ref()).await?;
270
+ presentation.reset_available_themes(None).await;
271
+ if all {
272
+ presentation.reset_theme().await?;
273
+ }
274
+
275
+ let result = renderer.draw(session.validate().await?.create_view()).await;
276
+ if let Some(sender) = sender {
277
+ sender.send(()).unwrap();
278
+ }
279
+
280
+ renderer.reset_changed.emit(());
281
+ result
282
+ });
283
+
284
+ needs_update
285
+ },
286
+ PerspectiveViewerMsg::ToggleDebug => {
287
+ self.debug_open = !self.debug_open;
288
+ clone!(ctx.props().renderer, ctx.props().session);
289
+ ApiFuture::spawn(async move {
290
+ renderer.draw(session.validate().await?.create_view()).await
291
+ });
292
+
293
+ true
294
+ },
295
+ PerspectiveViewerMsg::ToggleSettingsInit(Some(SettingsUpdate::Missing), None) => false,
296
+ PerspectiveViewerMsg::ToggleSettingsInit(
297
+ Some(SettingsUpdate::Missing),
298
+ Some(resolve),
299
+ ) => {
300
+ resolve.send(Ok(JsValue::UNDEFINED)).unwrap();
301
+ false
302
+ },
303
+ PerspectiveViewerMsg::ToggleSettingsInit(Some(SettingsUpdate::SetDefault), resolve) => {
304
+ self.init_toggle_settings_task(ctx, Some(false), resolve);
305
+ false
306
+ },
307
+ PerspectiveViewerMsg::ToggleSettingsInit(
308
+ Some(SettingsUpdate::Update(force)),
309
+ resolve,
310
+ ) => {
311
+ self.init_toggle_settings_task(ctx, Some(force), resolve);
312
+ false
313
+ },
314
+ PerspectiveViewerMsg::ToggleSettingsInit(None, resolve) => {
315
+ self.init_toggle_settings_task(ctx, None, resolve);
316
+ false
317
+ },
318
+ PerspectiveViewerMsg::ToggleSettingsComplete(SettingsUpdate::SetDefault, resolve)
319
+ if self.settings_open =>
320
+ {
321
+ self.selected_column = None;
322
+ self.settings_open = false;
323
+ self.on_rendered = Some(resolve);
324
+ true
325
+ },
326
+ PerspectiveViewerMsg::ToggleSettingsComplete(
327
+ SettingsUpdate::Update(force),
328
+ resolve,
329
+ ) if force != self.settings_open => {
330
+ self.selected_column = None;
331
+ self.settings_open = force;
332
+ self.on_rendered = Some(resolve);
333
+ true
334
+ },
335
+ PerspectiveViewerMsg::ToggleSettingsComplete(_, resolve)
336
+ if matches!(self.fonts.get_status(), FontLoaderStatus::Finished) =>
337
+ {
338
+ self.selected_column = None;
339
+ if let Err(e) = resolve.send(()) {
340
+ tracing::error!("toggle settings failed {:?}", e);
341
+ }
342
+
343
+ false
344
+ },
345
+ PerspectiveViewerMsg::ToggleSettingsComplete(_, resolve) => {
346
+ self.selected_column = None;
347
+ self.on_rendered = Some(resolve);
348
+ true
349
+ },
350
+ PerspectiveViewerMsg::RenderLimits(dimensions) => {
351
+ if self.dimensions != dimensions {
352
+ self.dimensions = dimensions;
353
+ true
354
+ } else {
355
+ false
356
+ }
357
+ },
358
+ PerspectiveViewerMsg::OpenColumnSettings {
359
+ locator,
360
+ sender,
361
+ toggle,
362
+ } => {
363
+ let is_active = locator
364
+ .as_ref()
365
+ .map(|l| l.is_active(&ctx.props().session))
366
+ .unwrap_or_default();
367
+
368
+ self.selected_column_is_active = is_active;
369
+ if toggle && self.selected_column == locator {
370
+ self.selected_column = None;
371
+ (false, None)
372
+ } else {
373
+ self.selected_column.clone_from(&locator);
374
+
375
+ locator
376
+ .clone()
377
+ .map(|c| (true, c.name().cloned()))
378
+ .unwrap_or_default()
379
+ };
380
+
381
+ let mut open_column_settings = ctx.props().presentation.get_open_column_settings();
382
+ open_column_settings
383
+ .locator
384
+ .clone_from(&self.selected_column);
385
+
386
+ ctx.props()
387
+ .presentation
388
+ .set_open_column_settings(Some(open_column_settings));
389
+
390
+ if let Some(sender) = sender {
391
+ sender.send(()).unwrap();
392
+ }
393
+
394
+ true
395
+ },
396
+ PerspectiveViewerMsg::SettingsPanelSizeUpdate(Some(x)) => {
397
+ self.settings_panel_width_override = Some(x);
398
+ false
399
+ },
400
+ PerspectiveViewerMsg::SettingsPanelSizeUpdate(None) => {
401
+ self.settings_panel_width_override = None;
402
+ false
403
+ },
404
+ PerspectiveViewerMsg::ColumnSettingsPanelSizeUpdate(Some(x)) => {
405
+ self.column_settings_panel_width_override = Some(x);
406
+ false
407
+ },
408
+ PerspectiveViewerMsg::ColumnSettingsPanelSizeUpdate(None) => {
409
+ self.column_settings_panel_width_override = None;
410
+ false
411
+ },
412
+ }
413
+ }
414
+
415
+ /// This top-level component is mounted to the Custom Element, so it has no
416
+ /// API to provide props - but for sanity if needed, just return true on
417
+ /// change.
418
+ fn changed(&mut self, _ctx: &Context<Self>, _old: &Self::Properties) -> bool {
419
+ true
420
+ }
421
+
422
+ /// On rendered call notify_resize(). This also triggers any registered
423
+ /// async callbacks to the Custom Element API.
424
+ fn rendered(&mut self, ctx: &Context<Self>, _first_render: bool) {
425
+ ctx.props()
426
+ .presentation
427
+ .set_settings_open(Some(self.settings_open))
428
+ .unwrap();
429
+
430
+ if self.on_rendered.is_some()
431
+ && matches!(self.fonts.get_status(), FontLoaderStatus::Finished)
432
+ && self.on_rendered.take().unwrap().send(()).is_err()
433
+ {
434
+ tracing::warn!("Orphan render");
435
+ }
436
+ }
437
+
438
+ /// `PerspectiveViewer` has two basic UI modes - "open" and "closed".
439
+ fn view(&self, ctx: &Context<Self>) -> Html {
440
+ let settings = ctx
441
+ .link()
442
+ .callback(|_| PerspectiveViewerMsg::ToggleSettingsInit(None, None));
443
+
444
+ let on_close_settings = ctx
445
+ .link()
446
+ .callback(|()| PerspectiveViewerMsg::ToggleSettingsInit(None, None));
447
+
448
+ let on_toggle_debug = ctx.link().callback(|_| PerspectiveViewerMsg::ToggleDebug);
449
+ let mut class = classes!("settings-closed");
450
+ if ctx.props().is_title() {
451
+ class.push("titled");
452
+ }
453
+
454
+ let on_open_expr_panel =
455
+ ctx.link()
456
+ .callback(|c| PerspectiveViewerMsg::OpenColumnSettings {
457
+ locator: Some(c),
458
+ sender: None,
459
+ toggle: true,
460
+ });
461
+
462
+ let on_reset = ctx
463
+ .link()
464
+ .callback(|all| PerspectiveViewerMsg::Reset(all, None));
465
+
466
+ let on_split_panel_resize = ctx
467
+ .link()
468
+ .callback(|(x, _)| PerspectiveViewerMsg::SettingsPanelSizeUpdate(Some(x)));
469
+
470
+ let on_column_settings_panel_resize = ctx
471
+ .link()
472
+ .callback(|(x, _)| PerspectiveViewerMsg::ColumnSettingsPanelSizeUpdate(Some(x)));
473
+
474
+ let settings_panel = html! {
475
+ <div id="settings_panel" class="sidebar_column noselect split-panel orient-vertical">
476
+ if self.selected_column.is_none() {
477
+ <SidebarCloseButton
478
+ id="settings_close_button"
479
+ on_close_sidebar={&on_close_settings}
480
+ />
481
+ }
482
+ <SidebarCloseButton
483
+ id={if self.debug_open { "debug_close_button" } else { "debug_open_button" }}
484
+ on_close_sidebar={&on_toggle_debug}
485
+ />
486
+ <PluginSelector
487
+ session={&ctx.props().session}
488
+ renderer={&ctx.props().renderer}
489
+ presentation={&ctx.props().presentation}
490
+ />
491
+ <ColumnSelector
492
+ dragdrop={&ctx.props().dragdrop}
493
+ renderer={&ctx.props().renderer}
494
+ session={&ctx.props().session}
495
+ presentation={&ctx.props().presentation}
496
+ on_resize={&self.on_resize}
497
+ on_open_expr_panel={&on_open_expr_panel}
498
+ on_dimensions_reset={&self.on_dimensions_reset}
499
+ selected_column={self.selected_column.clone()}
500
+ />
501
+ </div>
502
+ };
503
+
504
+ let main_panel = html! {
505
+ <div id="main_column">
506
+ <StatusBar
507
+ id="status_bar"
508
+ session={&ctx.props().session}
509
+ renderer={&ctx.props().renderer}
510
+ presentation={&ctx.props().presentation}
511
+ on_reset={on_reset.clone()}
512
+ />
513
+ <div id="main_panel_container">
514
+ <RenderWarning
515
+ dimensions={self.dimensions}
516
+ session={&ctx.props().session}
517
+ renderer={&ctx.props().renderer}
518
+ />
519
+ <slot />
520
+ </div>
521
+ if let Some(selected_column) = self.selected_column.clone() {
522
+ <SplitPanel
523
+ id="modal_panel"
524
+ reverse=true
525
+ initial_size={self.column_settings_panel_width_override}
526
+ on_reset={ctx.link().callback(|_| PerspectiveViewerMsg::ColumnSettingsPanelSizeUpdate(None))}
527
+ on_resize={on_column_settings_panel_resize}
528
+ >
529
+ <ColumnSettingsSidebar
530
+ session={&ctx.props().session}
531
+ renderer={&ctx.props().renderer}
532
+ custom_events={&ctx.props().custom_events}
533
+ presentation={&ctx.props().presentation}
534
+ {selected_column}
535
+ on_close={self.on_close_column_settings.clone()}
536
+ width_override={self.column_settings_panel_width_override}
537
+ is_active={self.selected_column_is_active}
538
+ />
539
+ <></>
540
+ </SplitPanel>
541
+ }
542
+ </div>
543
+ };
544
+
545
+ html! {
546
+ <>
547
+ <StyleProvider root={ctx.props().elem.clone()}>
548
+ <LocalStyle href={css!("viewer")} />
549
+ if self.settings_open && ctx.props().session.has_table() {
550
+ if self.debug_open {
551
+ <SplitPanel
552
+ id="app_panel"
553
+ reverse=true
554
+ initial_size={self.settings_panel_width_override}
555
+ on_reset={ctx.link().callback(|_| PerspectiveViewerMsg::SettingsPanelSizeUpdate(None))}
556
+ on_resize={on_split_panel_resize}
557
+ on_resize_finished={ctx.props().render_callback()}
558
+ >
559
+ <DebugPanel
560
+ session={ctx.props().session()}
561
+ renderer={ctx.props().renderer()}
562
+ presentation={ctx.props().presentation()}
563
+ />
564
+ { settings_panel }
565
+ { main_panel }
566
+ </SplitPanel>
567
+ } else {
568
+ <SplitPanel
569
+ id="app_panel"
570
+ reverse=true
571
+ initial_size={self.settings_panel_width_override}
572
+ on_reset={ctx.link().callback(|_| PerspectiveViewerMsg::SettingsPanelSizeUpdate(None))}
573
+ on_resize={on_split_panel_resize}
574
+ on_resize_finished={ctx.props().resize_callback()}
575
+ >
576
+ { settings_panel }
577
+ { main_panel }
578
+ </SplitPanel>
579
+ }
580
+ } else {
581
+ <RenderWarning
582
+ dimensions={self.dimensions}
583
+ session={&ctx.props().session}
584
+ renderer={&ctx.props().renderer}
585
+ />
586
+ if ctx.props().is_title() || !ctx.props().session.has_table() || ctx.props().session.is_errored() {
587
+ <StatusBar
588
+ id="status_bar"
589
+ session={&ctx.props().session}
590
+ renderer={&ctx.props().renderer}
591
+ presentation={&ctx.props().presentation}
592
+ {on_reset}
593
+ />
594
+ }
595
+ <div id="main_panel_container" {class}><slot /></div>
596
+ if !ctx.props().presentation.get_is_workspace() {
597
+ <div
598
+ id="settings_button"
599
+ class={if ctx.props().is_title() { "noselect button closed titled" } else { "noselect button closed" }}
600
+ onmousedown={settings}
601
+ />
602
+ }
603
+ }
604
+ </StyleProvider>
605
+ <FontLoader ..self.fonts.clone() />
606
+ </>
607
+ }
608
+ }
609
+
610
+ fn destroy(&mut self, _ctx: &Context<Self>) {}
611
+ }
612
+
613
+ impl PerspectiveViewer {
614
+ /// Toggle the settings, or force the settings panel either open (true) or
615
+ /// closed (false) explicitly. In order to reduce apparent
616
+ /// screen-shear, `toggle_settings()` uses a somewhat complex render
617
+ /// order: it first resize the plugin's `<div>` without moving it,
618
+ /// using `overflow: hidden` to hide the extra draw area; then,
619
+ /// after the _async_ drawing of the plugin is complete, it will send a
620
+ /// message to complete the toggle action and re-render the element with
621
+ /// the settings removed.
622
+ ///
623
+ /// # Arguments
624
+ /// * `force` - Whether to explicitly set the settings panel state to
625
+ /// Open/Close (`Some(true)`/`Some(false)`), or to just toggle the current
626
+ /// state (`None`).
627
+ fn init_toggle_settings_task(
628
+ &mut self,
629
+ ctx: &Context<Self>,
630
+ force: Option<bool>,
631
+ sender: Option<Sender<ApiResult<JsValue>>>,
632
+ ) {
633
+ let is_open = ctx.props().presentation.is_settings_open();
634
+ match force {
635
+ Some(force) if is_open == force => {
636
+ if let Some(sender) = sender {
637
+ sender.send(Ok(JsValue::UNDEFINED)).unwrap();
638
+ }
639
+ },
640
+ Some(_) | None => {
641
+ let force = !is_open;
642
+ let callback = ctx.link().callback(move |resolve| {
643
+ let update = SettingsUpdate::Update(force);
644
+ PerspectiveViewerMsg::ToggleSettingsComplete(update, resolve)
645
+ });
646
+
647
+ clone!(ctx.props().renderer, ctx.props().session);
648
+ ApiFuture::spawn(async move {
649
+ let result = if session.js_get_table().is_some() {
650
+ renderer.presize(force, callback.emit_async_safe()).await
651
+ } else {
652
+ callback.emit_async_safe().await?;
653
+ Ok(JsValue::UNDEFINED)
654
+ };
655
+
656
+ if let Some(sender) = sender {
657
+ let msg = result.ignore_view_delete();
658
+ sender
659
+ .send(msg.map(|x| x.unwrap_or(JsValue::UNDEFINED)))
660
+ .into_apierror()?;
661
+ };
662
+
663
+ Ok(JsValue::undefined())
664
+ });
665
+ },
666
+ };
667
+ }
668
+ }