scitex 2.8.1__py3-none-any.whl → 2.10.2__py3-none-any.whl

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 (415) hide show
  1. scitex/__init__.py +15 -7
  2. scitex/__version__.py +1 -2
  3. scitex/_install_guide.py +250 -0
  4. scitex/_optional_deps.py +206 -39
  5. scitex/ai/_gen_ai/_Groq.py +2 -4
  6. scitex/ai/_gen_ai/_OpenAI.py +5 -2
  7. scitex/ai/_gen_ai/_Perplexity.py +20 -6
  8. scitex/audio/__init__.py +24 -15
  9. scitex/audio/_cross_process_lock.py +139 -0
  10. scitex/audio/_mcp_handlers.py +256 -0
  11. scitex/audio/_mcp_tool_schemas.py +203 -0
  12. scitex/audio/engines/elevenlabs_engine.py +5 -2
  13. scitex/audio/mcp_server.py +98 -457
  14. scitex/bridge/__init__.py +30 -19
  15. scitex/bridge/_figrecipe.py +245 -0
  16. scitex/bridge/_helpers.py +2 -1
  17. scitex/bridge/_plt_vis.py +23 -10
  18. scitex/bridge/_stats_plt.py +18 -5
  19. scitex/bridge/_stats_vis.py +16 -2
  20. scitex/browser/__init__.py +84 -44
  21. scitex/browser/automation/__init__.py +5 -1
  22. scitex/browser/core/BrowserMixin.py +17 -4
  23. scitex/browser/core/__init__.py +11 -2
  24. scitex/browser/remote/CaptchaHandler.py +1 -1
  25. scitex/browser/remote/ZenRowsAPIClient.py +1 -1
  26. scitex/capture/grid.py +487 -0
  27. scitex/capture/mcp_handlers.py +401 -0
  28. scitex/capture/mcp_tool_defs.py +192 -0
  29. scitex/capture/mcp_tools.py +241 -0
  30. scitex/capture/mcp_utils.py +30 -0
  31. scitex/cli/convert.py +421 -0
  32. scitex/cli/main.py +6 -4
  33. scitex/datetime/__init__.py +46 -0
  34. scitex/datetime/_linspace.py +100 -0
  35. scitex/datetime/_normalize_timestamp.py +306 -0
  36. scitex/db/_delete_duplicates.py +4 -4
  37. scitex/db/_sqlite3/_delete_duplicates.py +11 -2
  38. scitex/dev/plt/__init__.py +61 -62
  39. scitex/dev/plt/demo_plotters/__init__.py +0 -0
  40. scitex/dev/plt/demo_plotters/plot_mpl_axhline.py +28 -0
  41. scitex/dev/plt/demo_plotters/plot_mpl_axhspan.py +28 -0
  42. scitex/dev/plt/demo_plotters/plot_mpl_axvline.py +28 -0
  43. scitex/dev/plt/demo_plotters/plot_mpl_axvspan.py +28 -0
  44. scitex/dev/plt/demo_plotters/plot_mpl_bar.py +29 -0
  45. scitex/dev/plt/demo_plotters/plot_mpl_barh.py +29 -0
  46. scitex/dev/plt/demo_plotters/plot_mpl_boxplot.py +28 -0
  47. scitex/dev/plt/demo_plotters/plot_mpl_contour.py +31 -0
  48. scitex/dev/plt/demo_plotters/plot_mpl_contourf.py +31 -0
  49. scitex/dev/plt/demo_plotters/plot_mpl_errorbar.py +30 -0
  50. scitex/dev/plt/demo_plotters/plot_mpl_eventplot.py +28 -0
  51. scitex/dev/plt/demo_plotters/plot_mpl_fill.py +30 -0
  52. scitex/dev/plt/demo_plotters/plot_mpl_fill_between.py +31 -0
  53. scitex/dev/plt/demo_plotters/plot_mpl_hexbin.py +28 -0
  54. scitex/dev/plt/demo_plotters/plot_mpl_hist.py +28 -0
  55. scitex/dev/plt/demo_plotters/plot_mpl_hist2d.py +28 -0
  56. scitex/dev/plt/demo_plotters/plot_mpl_imshow.py +29 -0
  57. scitex/dev/plt/demo_plotters/plot_mpl_pcolormesh.py +31 -0
  58. scitex/dev/plt/demo_plotters/plot_mpl_pie.py +29 -0
  59. scitex/dev/plt/demo_plotters/plot_mpl_plot.py +29 -0
  60. scitex/dev/plt/demo_plotters/plot_mpl_quiver.py +31 -0
  61. scitex/dev/plt/demo_plotters/plot_mpl_scatter.py +28 -0
  62. scitex/dev/plt/demo_plotters/plot_mpl_stackplot.py +31 -0
  63. scitex/dev/plt/demo_plotters/plot_mpl_stem.py +29 -0
  64. scitex/dev/plt/demo_plotters/plot_mpl_step.py +29 -0
  65. scitex/dev/plt/demo_plotters/plot_mpl_violinplot.py +28 -0
  66. scitex/dev/plt/demo_plotters/plot_sns_barplot.py +29 -0
  67. scitex/dev/plt/demo_plotters/plot_sns_boxplot.py +29 -0
  68. scitex/dev/plt/demo_plotters/plot_sns_heatmap.py +28 -0
  69. scitex/dev/plt/demo_plotters/plot_sns_histplot.py +29 -0
  70. scitex/dev/plt/demo_plotters/plot_sns_kdeplot.py +29 -0
  71. scitex/dev/plt/demo_plotters/plot_sns_lineplot.py +31 -0
  72. scitex/dev/plt/demo_plotters/plot_sns_scatterplot.py +29 -0
  73. scitex/dev/plt/demo_plotters/plot_sns_stripplot.py +29 -0
  74. scitex/dev/plt/demo_plotters/plot_sns_swarmplot.py +29 -0
  75. scitex/dev/plt/demo_plotters/plot_sns_violinplot.py +29 -0
  76. scitex/dev/plt/demo_plotters/plot_stx_bar.py +29 -0
  77. scitex/dev/plt/demo_plotters/plot_stx_barh.py +29 -0
  78. scitex/dev/plt/demo_plotters/plot_stx_box.py +28 -0
  79. scitex/dev/plt/demo_plotters/plot_stx_boxplot.py +28 -0
  80. scitex/dev/plt/demo_plotters/plot_stx_conf_mat.py +28 -0
  81. scitex/dev/plt/demo_plotters/plot_stx_contour.py +31 -0
  82. scitex/dev/plt/demo_plotters/plot_stx_ecdf.py +28 -0
  83. scitex/dev/plt/demo_plotters/plot_stx_errorbar.py +30 -0
  84. scitex/dev/plt/demo_plotters/plot_stx_fill_between.py +31 -0
  85. scitex/dev/plt/demo_plotters/plot_stx_fillv.py +28 -0
  86. scitex/dev/plt/demo_plotters/plot_stx_heatmap.py +28 -0
  87. scitex/dev/plt/demo_plotters/plot_stx_image.py +28 -0
  88. scitex/dev/plt/demo_plotters/plot_stx_imshow.py +28 -0
  89. scitex/dev/plt/demo_plotters/plot_stx_joyplot.py +28 -0
  90. scitex/dev/plt/demo_plotters/plot_stx_kde.py +28 -0
  91. scitex/dev/plt/demo_plotters/plot_stx_line.py +28 -0
  92. scitex/dev/plt/demo_plotters/plot_stx_mean_ci.py +28 -0
  93. scitex/dev/plt/demo_plotters/plot_stx_mean_std.py +28 -0
  94. scitex/dev/plt/demo_plotters/plot_stx_median_iqr.py +28 -0
  95. scitex/dev/plt/demo_plotters/plot_stx_raster.py +28 -0
  96. scitex/dev/plt/demo_plotters/plot_stx_rectangle.py +28 -0
  97. scitex/dev/plt/demo_plotters/plot_stx_scatter.py +29 -0
  98. scitex/dev/plt/demo_plotters/plot_stx_shaded_line.py +29 -0
  99. scitex/dev/plt/demo_plotters/plot_stx_violin.py +28 -0
  100. scitex/dev/plt/demo_plotters/plot_stx_violinplot.py +28 -0
  101. scitex/dev/plt/mpl/get_dir_ax.py +46 -0
  102. scitex/dev/plt/mpl/get_signatures.py +176 -0
  103. scitex/dev/plt/mpl/get_signatures_details.py +522 -0
  104. scitex/dict/_pop_keys.py +1 -7
  105. scitex/dsp/__init__.py +15 -10
  106. scitex/dsp/add_noise.py +5 -2
  107. scitex/dsp/example.py +35 -22
  108. scitex/dsp/filt.py +8 -3
  109. scitex/dsp/reference.py +3 -2
  110. scitex/dsp/utils/__init__.py +2 -1
  111. scitex/dsp/utils/_differential_bandpass_filters.py +14 -4
  112. scitex/dt/__init__.py +39 -2
  113. scitex/errors.py +82 -521
  114. scitex/fig/__init__.py +4 -4
  115. scitex/fig/editor/edit/panel_loader.py +1 -1
  116. scitex/fig/io/_bundle.py +7 -7
  117. scitex/fts/README.md +262 -0
  118. scitex/fts/TODO.md +66 -0
  119. scitex/fts/__init__.py +90 -0
  120. scitex/fts/_bundle/README_IN_BUNDLE.md +102 -0
  121. scitex/fts/_bundle/_FTS.py +657 -0
  122. scitex/fts/_bundle/__init__.py +38 -0
  123. scitex/fts/_bundle/_children.py +216 -0
  124. scitex/fts/_bundle/_conversion/__init__.py +15 -0
  125. scitex/fts/_bundle/_conversion/_bundle2dict.py +44 -0
  126. scitex/fts/_bundle/_conversion/_dict2bundle.py +50 -0
  127. scitex/fts/_bundle/_dataclasses/_Axes.py +57 -0
  128. scitex/fts/_bundle/_dataclasses/_BBox.py +54 -0
  129. scitex/fts/_bundle/_dataclasses/_ColumnDef.py +72 -0
  130. scitex/fts/_bundle/_dataclasses/_DataFormat.py +40 -0
  131. scitex/fts/_bundle/_dataclasses/_DataInfo.py +135 -0
  132. scitex/fts/_bundle/_dataclasses/_DataSource.py +44 -0
  133. scitex/fts/_bundle/_dataclasses/_Node.py +319 -0
  134. scitex/fts/_bundle/_dataclasses/_NodeRefs.py +45 -0
  135. scitex/fts/_bundle/_dataclasses/_SizeMM.py +38 -0
  136. scitex/fts/_bundle/_dataclasses/__init__.py +35 -0
  137. scitex/fts/_bundle/_extractors/__init__.py +32 -0
  138. scitex/fts/_bundle/_extractors/_extract_bar.py +131 -0
  139. scitex/fts/_bundle/_extractors/_extract_line.py +71 -0
  140. scitex/fts/_bundle/_extractors/_extract_scatter.py +79 -0
  141. scitex/fts/_bundle/_loader.py +134 -0
  142. scitex/fts/_bundle/_mpl_helpers.py +389 -0
  143. scitex/fts/_bundle/_saver.py +269 -0
  144. scitex/fts/_bundle/_storage.py +200 -0
  145. scitex/fts/_bundle/_utils/__init__.py +55 -0
  146. scitex/fts/_bundle/_utils/_const.py +26 -0
  147. scitex/fts/_bundle/_utils/_errors.py +73 -0
  148. scitex/fts/_bundle/_utils/_generate.py +21 -0
  149. scitex/fts/_bundle/_utils/_types.py +76 -0
  150. scitex/fts/_bundle/_validation.py +434 -0
  151. scitex/fts/_bundle/_zipbundle.py +165 -0
  152. scitex/fts/_fig/__init__.py +22 -0
  153. scitex/fts/_fig/_backend/__init__.py +53 -0
  154. scitex/fts/_fig/_backend/_export.py +165 -0
  155. scitex/fts/_fig/_backend/_parser.py +188 -0
  156. scitex/fts/_fig/_backend/_render.py +538 -0
  157. scitex/fts/_fig/_composite.py +345 -0
  158. scitex/fts/_fig/_dataclasses/_ChannelEncoding.py +46 -0
  159. scitex/fts/_fig/_dataclasses/_Encoding.py +82 -0
  160. scitex/fts/_fig/_dataclasses/_Theme.py +441 -0
  161. scitex/fts/_fig/_dataclasses/_TraceEncoding.py +52 -0
  162. scitex/fts/_fig/_dataclasses/__init__.py +47 -0
  163. scitex/fts/_fig/_editor/__init__.py +14 -0
  164. scitex/fts/_fig/_editor/_cui/__init__.py +33 -0
  165. scitex/fts/_fig/_editor/_cui/_backend_detector.py +39 -0
  166. scitex/fts/_fig/_editor/_cui/_bundle_resolver.py +366 -0
  167. scitex/fts/_fig/_editor/_cui/_editor_launcher.py +175 -0
  168. scitex/fts/_fig/_editor/_cui/_manual_handler.py +52 -0
  169. scitex/fts/_fig/_editor/_cui/_panel_loader.py +246 -0
  170. scitex/fts/_fig/_editor/_cui/_path_resolver.py +66 -0
  171. scitex/fts/_fig/_editor/_defaults.py +300 -0
  172. scitex/fts/_fig/_editor/_gui/__init__.py +11 -0
  173. scitex/fts/_fig/_editor/_gui/_flask_editor/__init__.py +20 -0
  174. scitex/fts/_fig/_editor/_gui/_flask_editor/_bbox.py +1339 -0
  175. scitex/fts/_fig/_editor/_gui/_flask_editor/_core.py +1688 -0
  176. scitex/fts/_fig/_editor/_gui/_flask_editor/_plotter.py +664 -0
  177. scitex/fts/_fig/_editor/_gui/_flask_editor/_renderer.py +853 -0
  178. scitex/fts/_fig/_editor/_gui/_flask_editor/_utils.py +79 -0
  179. scitex/fts/_fig/_editor/_gui/_flask_editor/static/css/base/reset.css +41 -0
  180. scitex/fts/_fig/_editor/_gui/_flask_editor/static/css/base/typography.css +16 -0
  181. scitex/fts/_fig/_editor/_gui/_flask_editor/static/css/base/variables.css +85 -0
  182. scitex/fts/_fig/_editor/_gui/_flask_editor/static/css/components/buttons.css +217 -0
  183. scitex/fts/_fig/_editor/_gui/_flask_editor/static/css/components/context-menu.css +93 -0
  184. scitex/fts/_fig/_editor/_gui/_flask_editor/static/css/components/dropdown.css +57 -0
  185. scitex/fts/_fig/_editor/_gui/_flask_editor/static/css/components/forms.css +112 -0
  186. scitex/fts/_fig/_editor/_gui/_flask_editor/static/css/components/modal.css +59 -0
  187. scitex/fts/_fig/_editor/_gui/_flask_editor/static/css/components/sections.css +212 -0
  188. scitex/fts/_fig/_editor/_gui/_flask_editor/static/css/features/canvas.css +176 -0
  189. scitex/fts/_fig/_editor/_gui/_flask_editor/static/css/features/element-inspector.css +190 -0
  190. scitex/fts/_fig/_editor/_gui/_flask_editor/static/css/features/loading.css +59 -0
  191. scitex/fts/_fig/_editor/_gui/_flask_editor/static/css/features/overlay.css +45 -0
  192. scitex/fts/_fig/_editor/_gui/_flask_editor/static/css/features/panel-grid.css +95 -0
  193. scitex/fts/_fig/_editor/_gui/_flask_editor/static/css/features/selection.css +101 -0
  194. scitex/fts/_fig/_editor/_gui/_flask_editor/static/css/features/statistics.css +138 -0
  195. scitex/fts/_fig/_editor/_gui/_flask_editor/static/css/index.css +31 -0
  196. scitex/fts/_fig/_editor/_gui/_flask_editor/static/css/layout/container.css +7 -0
  197. scitex/fts/_fig/_editor/_gui/_flask_editor/static/css/layout/controls.css +56 -0
  198. scitex/fts/_fig/_editor/_gui/_flask_editor/static/css/layout/preview.css +78 -0
  199. scitex/fts/_fig/_editor/_gui/_flask_editor/static/js/alignment/axis.js +314 -0
  200. scitex/fts/_fig/_editor/_gui/_flask_editor/static/js/alignment/basic.js +107 -0
  201. scitex/fts/_fig/_editor/_gui/_flask_editor/static/js/alignment/distribute.js +54 -0
  202. scitex/fts/_fig/_editor/_gui/_flask_editor/static/js/canvas/canvas.js +172 -0
  203. scitex/fts/_fig/_editor/_gui/_flask_editor/static/js/canvas/dragging.js +258 -0
  204. scitex/fts/_fig/_editor/_gui/_flask_editor/static/js/canvas/resize.js +48 -0
  205. scitex/fts/_fig/_editor/_gui/_flask_editor/static/js/canvas/selection.js +71 -0
  206. scitex/fts/_fig/_editor/_gui/_flask_editor/static/js/core/api.js +288 -0
  207. scitex/fts/_fig/_editor/_gui/_flask_editor/static/js/core/state.js +143 -0
  208. scitex/fts/_fig/_editor/_gui/_flask_editor/static/js/core/utils.js +245 -0
  209. scitex/fts/_fig/_editor/_gui/_flask_editor/static/js/dev/element-inspector.js +992 -0
  210. scitex/fts/_fig/_editor/_gui/_flask_editor/static/js/editor/bbox.js +339 -0
  211. scitex/fts/_fig/_editor/_gui/_flask_editor/static/js/editor/element-drag.js +286 -0
  212. scitex/fts/_fig/_editor/_gui/_flask_editor/static/js/editor/overlay.js +371 -0
  213. scitex/fts/_fig/_editor/_gui/_flask_editor/static/js/editor/preview.js +293 -0
  214. scitex/fts/_fig/_editor/_gui/_flask_editor/static/js/main.js +426 -0
  215. scitex/fts/_fig/_editor/_gui/_flask_editor/static/js/shortcuts/context-menu.js +152 -0
  216. scitex/fts/_fig/_editor/_gui/_flask_editor/static/js/shortcuts/keyboard.js +265 -0
  217. scitex/fts/_fig/_editor/_gui/_flask_editor/static/js/ui/controls.js +184 -0
  218. scitex/fts/_fig/_editor/_gui/_flask_editor/static/js/ui/download.js +57 -0
  219. scitex/fts/_fig/_editor/_gui/_flask_editor/static/js/ui/help.js +100 -0
  220. scitex/fts/_fig/_editor/_gui/_flask_editor/static/js/ui/theme.js +34 -0
  221. scitex/fts/_fig/_editor/_gui/_flask_editor/templates/__init__.py +124 -0
  222. scitex/fts/_fig/_editor/_gui/_flask_editor/templates/_html.py +851 -0
  223. scitex/fts/_fig/_editor/_gui/_flask_editor/templates/_scripts.py +4932 -0
  224. scitex/fts/_fig/_editor/_gui/_flask_editor/templates/_styles.py +1657 -0
  225. scitex/fts/_fig/_editor/_gui/_flask_editor.py +36 -0
  226. scitex/fts/_fig/_models/_Annotations.py +115 -0
  227. scitex/fts/_fig/_models/_Axes.py +152 -0
  228. scitex/fts/_fig/_models/_Figure.py +138 -0
  229. scitex/fts/_fig/_models/_Guides.py +104 -0
  230. scitex/fts/_fig/_models/_Plot.py +123 -0
  231. scitex/fts/_fig/_models/_Styles.py +245 -0
  232. scitex/fts/_fig/_models/__init__.py +80 -0
  233. scitex/fts/_fig/_models/_plot_types/__init__.py +156 -0
  234. scitex/fts/_fig/_models/_plot_types/_bar.py +43 -0
  235. scitex/fts/_fig/_models/_plot_types/_box.py +38 -0
  236. scitex/fts/_fig/_models/_plot_types/_distribution.py +36 -0
  237. scitex/fts/_fig/_models/_plot_types/_errorbar.py +60 -0
  238. scitex/fts/_fig/_models/_plot_types/_histogram.py +30 -0
  239. scitex/fts/_fig/_models/_plot_types/_image.py +61 -0
  240. scitex/fts/_fig/_models/_plot_types/_line.py +57 -0
  241. scitex/fts/_fig/_models/_plot_types/_scatter.py +30 -0
  242. scitex/fts/_fig/_models/_plot_types/_seaborn.py +121 -0
  243. scitex/fts/_fig/_models/_plot_types/_violin.py +36 -0
  244. scitex/fts/_fig/_utils/__init__.py +129 -0
  245. scitex/fts/_fig/_utils/_auto_layout.py +127 -0
  246. scitex/fts/_fig/_utils/_calc_bounds.py +111 -0
  247. scitex/fts/_fig/_utils/_const_sizes.py +48 -0
  248. scitex/fts/_fig/_utils/_convert_coords.py +77 -0
  249. scitex/fts/_fig/_utils/_get_template.py +178 -0
  250. scitex/fts/_fig/_utils/_normalize.py +73 -0
  251. scitex/fts/_fig/_utils/_plot_layout.py +397 -0
  252. scitex/fts/_fig/_utils/_validate.py +197 -0
  253. scitex/fts/_kinds/__init__.py +45 -0
  254. scitex/fts/_kinds/_figure/__init__.py +19 -0
  255. scitex/fts/_kinds/_figure/_composite.py +345 -0
  256. scitex/fts/_kinds/_plot/__init__.py +25 -0
  257. scitex/fts/_kinds/_plot/_backend/__init__.py +53 -0
  258. scitex/fts/_kinds/_plot/_backend/_export.py +165 -0
  259. scitex/fts/_kinds/_plot/_backend/_parser.py +188 -0
  260. scitex/fts/_kinds/_plot/_backend/_render.py +538 -0
  261. scitex/fts/_kinds/_plot/_dataclasses/_ChannelEncoding.py +46 -0
  262. scitex/fts/_kinds/_plot/_dataclasses/_Encoding.py +82 -0
  263. scitex/fts/_kinds/_plot/_dataclasses/_Theme.py +441 -0
  264. scitex/fts/_kinds/_plot/_dataclasses/_TraceEncoding.py +52 -0
  265. scitex/fts/_kinds/_plot/_dataclasses/__init__.py +47 -0
  266. scitex/fts/_kinds/_plot/_models/_Annotations.py +115 -0
  267. scitex/fts/_kinds/_plot/_models/_Axes.py +152 -0
  268. scitex/fts/_kinds/_plot/_models/_Figure.py +138 -0
  269. scitex/fts/_kinds/_plot/_models/_Guides.py +104 -0
  270. scitex/fts/_kinds/_plot/_models/_Plot.py +123 -0
  271. scitex/fts/_kinds/_plot/_models/_Styles.py +245 -0
  272. scitex/fts/_kinds/_plot/_models/__init__.py +80 -0
  273. scitex/fts/_kinds/_plot/_models/_plot_types/__init__.py +156 -0
  274. scitex/fts/_kinds/_plot/_models/_plot_types/_bar.py +43 -0
  275. scitex/fts/_kinds/_plot/_models/_plot_types/_box.py +38 -0
  276. scitex/fts/_kinds/_plot/_models/_plot_types/_distribution.py +36 -0
  277. scitex/fts/_kinds/_plot/_models/_plot_types/_errorbar.py +60 -0
  278. scitex/fts/_kinds/_plot/_models/_plot_types/_histogram.py +30 -0
  279. scitex/fts/_kinds/_plot/_models/_plot_types/_image.py +61 -0
  280. scitex/fts/_kinds/_plot/_models/_plot_types/_line.py +57 -0
  281. scitex/fts/_kinds/_plot/_models/_plot_types/_scatter.py +30 -0
  282. scitex/fts/_kinds/_plot/_models/_plot_types/_seaborn.py +121 -0
  283. scitex/fts/_kinds/_plot/_models/_plot_types/_violin.py +36 -0
  284. scitex/fts/_kinds/_plot/_utils/__init__.py +129 -0
  285. scitex/fts/_kinds/_plot/_utils/_auto_layout.py +127 -0
  286. scitex/fts/_kinds/_plot/_utils/_calc_bounds.py +111 -0
  287. scitex/fts/_kinds/_plot/_utils/_const_sizes.py +48 -0
  288. scitex/fts/_kinds/_plot/_utils/_convert_coords.py +77 -0
  289. scitex/fts/_kinds/_plot/_utils/_get_template.py +178 -0
  290. scitex/fts/_kinds/_plot/_utils/_normalize.py +73 -0
  291. scitex/fts/_kinds/_plot/_utils/_plot_layout.py +397 -0
  292. scitex/fts/_kinds/_plot/_utils/_validate.py +197 -0
  293. scitex/fts/_kinds/_shape/__init__.py +141 -0
  294. scitex/fts/_kinds/_stats/__init__.py +56 -0
  295. scitex/fts/_kinds/_stats/_dataclasses/_Stats.py +423 -0
  296. scitex/fts/_kinds/_stats/_dataclasses/__init__.py +48 -0
  297. scitex/fts/_kinds/_table/__init__.py +72 -0
  298. scitex/fts/_kinds/_table/_latex/__init__.py +93 -0
  299. scitex/fts/_kinds/_table/_latex/_editor/__init__.py +11 -0
  300. scitex/fts/_kinds/_table/_latex/_editor/_app.py +725 -0
  301. scitex/fts/_kinds/_table/_latex/_export.py +279 -0
  302. scitex/fts/_kinds/_table/_latex/_figure_exporter.py +153 -0
  303. scitex/fts/_kinds/_table/_latex/_stats_formatter.py +274 -0
  304. scitex/fts/_kinds/_table/_latex/_table_exporter.py +362 -0
  305. scitex/fts/_kinds/_table/_latex/_utils.py +369 -0
  306. scitex/fts/_kinds/_table/_latex/_validator.py +445 -0
  307. scitex/fts/_kinds/_text/__init__.py +77 -0
  308. scitex/fts/_schemas/data_info.schema.json +75 -0
  309. scitex/fts/_schemas/encoding.schema.json +90 -0
  310. scitex/fts/_schemas/node.schema.json +145 -0
  311. scitex/fts/_schemas/render_manifest.schema.json +62 -0
  312. scitex/fts/_schemas/stats.schema.json +132 -0
  313. scitex/fts/_schemas/theme.schema.json +141 -0
  314. scitex/fts/_stats/__init__.py +48 -0
  315. scitex/fts/_stats/_dataclasses/_Stats.py +423 -0
  316. scitex/fts/_stats/_dataclasses/__init__.py +48 -0
  317. scitex/fts/_tables/__init__.py +65 -0
  318. scitex/fts/_tables/_latex/__init__.py +93 -0
  319. scitex/fts/_tables/_latex/_editor/__init__.py +11 -0
  320. scitex/fts/_tables/_latex/_editor/_app.py +725 -0
  321. scitex/fts/_tables/_latex/_export.py +279 -0
  322. scitex/fts/_tables/_latex/_figure_exporter.py +153 -0
  323. scitex/fts/_tables/_latex/_stats_formatter.py +274 -0
  324. scitex/fts/_tables/_latex/_table_exporter.py +362 -0
  325. scitex/fts/_tables/_latex/_utils.py +369 -0
  326. scitex/fts/_tables/_latex/_validator.py +445 -0
  327. scitex/gen/__init__.py +66 -25
  328. scitex/gen/misc.py +28 -0
  329. scitex/io/__init__.py +47 -32
  330. scitex/io/_load.py +87 -36
  331. scitex/io/_load_modules/__init__.py +10 -7
  332. scitex/io/_load_modules/_pandas.py +6 -1
  333. scitex/io/_save.py +299 -1556
  334. scitex/io/_save_modules/__init__.py +76 -19
  335. scitex/io/_save_modules/_figure_utils.py +90 -0
  336. scitex/io/_save_modules/_image_csv.py +497 -0
  337. scitex/io/_save_modules/_legends.py +91 -0
  338. scitex/io/_save_modules/_pltz_bundle.py +356 -0
  339. scitex/io/_save_modules/_pltz_stx.py +536 -0
  340. scitex/io/_save_modules/_stx_bundle.py +104 -0
  341. scitex/io/_save_modules/_symlink.py +96 -0
  342. scitex/io/_save_modules/_yaml.py +1 -1
  343. scitex/io/_save_modules/_zarr.py +64 -18
  344. scitex/io/bundle/README.md +212 -0
  345. scitex/io/bundle/__init__.py +110 -0
  346. scitex/io/{_bundle.py → bundle/_core.py} +168 -97
  347. scitex/io/bundle/_nested.py +713 -0
  348. scitex/io/bundle/_types.py +74 -0
  349. scitex/io/{_zip_bundle.py → bundle/_zip.py} +93 -45
  350. scitex/io/utils/h5_to_zarr.py +1 -1
  351. scitex/logging/__init__.py +108 -13
  352. scitex/logging/_errors.py +508 -0
  353. scitex/logging/_formatters.py +30 -6
  354. scitex/logging/_warnings.py +261 -0
  355. scitex/plt/__init__.py +4 -1
  356. scitex/plt/_figrecipe.py +236 -0
  357. scitex/plt/_subplots/_AxisWrapper.py +6 -0
  358. scitex/plt/_subplots/_AxisWrapperMixins/_UnitAwareMixin.py +112 -1
  359. scitex/plt/_subplots/_FigWrapper.py +15 -0
  360. scitex/plt/_subplots/_SubplotsWrapper.py +125 -489
  361. scitex/plt/_subplots/_export_as_csv.py +11 -0
  362. scitex/plt/_subplots/_export_as_csv_formatters/__init__.py +2 -0
  363. scitex/plt/_subplots/_export_as_csv_formatters/_format_pcolormesh.py +66 -0
  364. scitex/plt/_subplots/_export_as_csv_formatters/_format_stackplot.py +62 -0
  365. scitex/plt/_subplots/_export_as_csv_formatters/test_formatters.py +208 -0
  366. scitex/plt/_subplots/_fonts.py +71 -0
  367. scitex/plt/_subplots/_mm_layout.py +282 -0
  368. scitex/plt/gallery/__init__.py +99 -2
  369. scitex/plt/styles/_plot_postprocess.py +3 -1
  370. scitex/plt/utils/_configure_mpl.py +16 -19
  371. scitex/repro/_RandomStateManager.py +13 -8
  372. scitex/resource/__init__.py +19 -1
  373. scitex/resource/_utils/_get_env_info.py +13 -25
  374. scitex/schema/__init__.py +149 -160
  375. scitex/schema/_encoding.py +273 -0
  376. scitex/schema/_figure_elements.py +406 -0
  377. scitex/schema/_theme.py +360 -0
  378. scitex/schema/_validation.py +0 -98
  379. scitex/scholar/__init__.py +56 -14
  380. scitex/scholar/auth/ScholarAuthManager.py +1 -1
  381. scitex/scholar/auth/__init__.py +11 -2
  382. scitex/scholar/auth/providers/BaseAuthenticator.py +1 -1
  383. scitex/scholar/auth/providers/EZProxyAuthenticator.py +1 -1
  384. scitex/scholar/auth/providers/OpenAthensAuthenticator.py +1 -1
  385. scitex/scholar/auth/providers/ShibbolethAuthenticator.py +1 -1
  386. scitex/scholar/config/ScholarConfig.py +1 -1
  387. scitex/scholar/core/Scholar.py +1 -1
  388. scitex/session/_decorator.py +18 -16
  389. scitex/session/_lifecycle.py +9 -11
  390. scitex/session/template.py +9 -8
  391. scitex/sh/test_sh.py +72 -0
  392. scitex/sh/test_sh_simple.py +61 -0
  393. scitex/stats/__init__.py +221 -97
  394. scitex/stats/_schema.py +21 -22
  395. scitex/stats/descriptive/_circular.py +212 -351
  396. scitex/stats/descriptive/_describe.py +81 -132
  397. scitex/stats/descriptive/_nan.py +205 -433
  398. scitex/stats/descriptive/_real.py +127 -141
  399. scitex/str/_format_plot_text.py +5 -5
  400. scitex/str/_latex.py +26 -84
  401. scitex/str/_latex_fallback.py +53 -47
  402. scitex/web/_search_pubmed.py +5 -4
  403. scitex/writer/tests/test_diff_between.py +451 -0
  404. scitex/writer/tests/test_document_section.py +311 -0
  405. scitex/writer/tests/test_document_workflow.py +393 -0
  406. scitex/writer/tests/test_writer.py +361 -0
  407. scitex/writer/tests/test_writer_integration.py +303 -0
  408. {scitex-2.8.1.dist-info → scitex-2.10.2.dist-info}/METADATA +368 -183
  409. {scitex-2.8.1.dist-info → scitex-2.10.2.dist-info}/RECORD +412 -97
  410. scitex/scholar/docs/to_claude/guidelines/examples/mgmt/ARCHITECTURE_EXAMPLE.md +0 -905
  411. scitex/scholar/docs/to_claude/guidelines/examples/mgmt/BULLETIN_BOARD_EXAMPLE.md +0 -99
  412. scitex/scholar/docs/to_claude/guidelines/examples/mgmt/PROJECT_DESCRIPTION_EXAMPLE.md +0 -96
  413. {scitex-2.8.1.dist-info → scitex-2.10.2.dist-info}/WHEEL +0 -0
  414. {scitex-2.8.1.dist-info → scitex-2.10.2.dist-info}/entry_points.txt +0 -0
  415. {scitex-2.8.1.dist-info → scitex-2.10.2.dist-info}/licenses/LICENSE +0 -0
@@ -0,0 +1,1657 @@
1
+ #!/usr/bin/env python3
2
+ # File: ./src/scitex/vis/editor/flask_editor/templates/styles.py
3
+ """CSS styles for the Flask editor UI.
4
+
5
+ DEPRECATED: This inline CSS module is kept for fallback compatibility only.
6
+ The CSS has been modularized into static/css/ directory:
7
+ - static/css/index.css (main entry point with @imports)
8
+ - static/css/base/ (variables, reset, typography)
9
+ - static/css/layout/ (container, preview, controls)
10
+ - static/css/components/ (buttons, forms, sections, dropdown, modal, context-menu)
11
+ - static/css/features/ (canvas, panel-grid, selection, overlay, loading, statistics)
12
+
13
+ To use static files (recommended):
14
+ Set USE_STATIC_FILES = True in templates/__init__.py
15
+
16
+ To use this inline version (fallback):
17
+ Set USE_STATIC_FILES = False in templates/__init__.py
18
+ """
19
+
20
+ CSS_STYLES = """
21
+ /* =============================================================================
22
+ * SciTeX Color System - Based on scitex-cloud/static/shared/css
23
+ * ============================================================================= */
24
+ :root, [data-theme="light"] {
25
+ /* Brand colors (light mode) */
26
+ --scitex-01: #1a2a40;
27
+ --scitex-02: #34495e;
28
+ --scitex-03: #506b7a;
29
+ --scitex-04: #6c8ba0;
30
+ --scitex-05: #8fa4b0;
31
+ --scitex-06: #b5c7d1;
32
+ --scitex-07: #d4e1e8;
33
+ --white: #fafbfc;
34
+ --gray-subtle: #f6f8fa;
35
+
36
+ /* Semantic tokens */
37
+ --text-primary: var(--scitex-01);
38
+ --text-secondary: var(--scitex-02);
39
+ --text-muted: var(--scitex-04);
40
+ --text-inverse: var(--white);
41
+
42
+ --bg-page: #fefefe;
43
+ --bg-surface: var(--white);
44
+ --bg-muted: var(--gray-subtle);
45
+
46
+ --border-default: var(--scitex-05);
47
+ --border-muted: var(--scitex-06);
48
+
49
+ /* Workspace colors */
50
+ --workspace-bg-primary: #f8f9fa;
51
+ --workspace-bg-secondary: #f3f4f6;
52
+ --workspace-bg-tertiary: #ebedef;
53
+ --workspace-bg-elevated: #ffffff;
54
+ --workspace-border-subtle: #e0e4e8;
55
+ --workspace-border-default: #b5c7d1;
56
+
57
+ /* Status */
58
+ --status-success: #4a9b7e;
59
+ --status-warning: #b8956a;
60
+ --status-error: #a67373;
61
+
62
+ /* CTA */
63
+ --color-cta: #3b82f6;
64
+ --color-cta-hover: #2563eb;
65
+
66
+ /* Preview background (checkered for transparency) */
67
+ --preview-bg: linear-gradient(45deg, #e0e0e0 25%, transparent 25%),
68
+ linear-gradient(-45deg, #e0e0e0 25%, transparent 25%),
69
+ linear-gradient(45deg, transparent 75%, #e0e0e0 75%),
70
+ linear-gradient(-45deg, transparent 75%, #e0e0e0 75%);
71
+ }
72
+
73
+ [data-theme="dark"] {
74
+ /* Semantic tokens (dark mode) */
75
+ --text-primary: var(--scitex-07);
76
+ --text-secondary: var(--scitex-05);
77
+ --text-muted: var(--scitex-04);
78
+ --text-inverse: var(--scitex-01);
79
+
80
+ --bg-page: #0f1419;
81
+ --bg-surface: var(--scitex-01);
82
+ --bg-muted: var(--scitex-02);
83
+
84
+ --border-default: var(--scitex-03);
85
+ --border-muted: var(--scitex-02);
86
+
87
+ /* Workspace colors */
88
+ --workspace-bg-primary: #0d0d0d;
89
+ --workspace-bg-secondary: #151515;
90
+ --workspace-bg-tertiary: #1a1a1a;
91
+ --workspace-bg-elevated: #1f1f1f;
92
+ --workspace-border-subtle: #1a1a1a;
93
+ --workspace-border-default: #3a3a3a;
94
+
95
+ /* Status */
96
+ --status-success: #6ba89a;
97
+ --status-warning: #d4a87a;
98
+ --status-error: #c08888;
99
+
100
+ /* Preview background (darker checkered) */
101
+ --preview-bg: linear-gradient(45deg, #2a2a2a 25%, transparent 25%),
102
+ linear-gradient(-45deg, #2a2a2a 25%, transparent 25%),
103
+ linear-gradient(45deg, transparent 75%, #2a2a2a 75%),
104
+ linear-gradient(-45deg, transparent 75%, #2a2a2a 75%);
105
+ }
106
+
107
+ /* =============================================================================
108
+ * Base Styles
109
+ * ============================================================================= */
110
+ * { box-sizing: border-box; margin: 0; padding: 0; }
111
+
112
+ body {
113
+ font-family: -apple-system, BlinkMacSystemFont, 'Segoe UI', Roboto, sans-serif;
114
+ background: var(--workspace-bg-primary);
115
+ color: var(--text-primary);
116
+ transition: background 0.3s, color 0.3s;
117
+ }
118
+
119
+ .container { display: flex; height: 100vh; }
120
+
121
+ /* =============================================================================
122
+ * Preview Panel
123
+ * ============================================================================= */
124
+ .preview {
125
+ flex: 2;
126
+ padding: 20px;
127
+ display: flex;
128
+ align-items: center;
129
+ justify-content: center;
130
+ background: var(--workspace-bg-secondary);
131
+ }
132
+
133
+ .preview-wrapper {
134
+ background: var(--preview-bg);
135
+ background-size: 20px 20px;
136
+ background-position: 0 0, 0 10px, 10px -10px, -10px 0px;
137
+ border-radius: 8px;
138
+ padding: 10px;
139
+ box-shadow: 0 4px 20px rgba(0,0,0,0.15);
140
+ }
141
+
142
+ .preview img {
143
+ max-width: 100%;
144
+ max-height: calc(100vh - 80px);
145
+ display: block;
146
+ }
147
+
148
+ /* Hover overlay for interactive selection */
149
+ .preview-container {
150
+ position: relative;
151
+ display: inline-block;
152
+ }
153
+
154
+ .hover-overlay {
155
+ position: absolute;
156
+ top: 0;
157
+ left: 0;
158
+ pointer-events: none;
159
+ }
160
+
161
+ /* Color-neutral hover/selection: white outline with glow works with any element color */
162
+ .hover-rect {
163
+ fill: none;
164
+ stroke: rgba(255, 255, 255, 0.9);
165
+ stroke-width: 2;
166
+ pointer-events: none;
167
+ filter: drop-shadow(0 0 3px rgba(0, 0, 0, 0.5)) drop-shadow(0 0 1px rgba(255, 255, 255, 0.8));
168
+ }
169
+
170
+ .selected-rect {
171
+ fill: none;
172
+ stroke: rgba(255, 255, 255, 1);
173
+ stroke-width: 2;
174
+ stroke-dasharray: 4 2;
175
+ pointer-events: none;
176
+ filter: drop-shadow(0 0 4px rgba(0, 0, 0, 0.7)) drop-shadow(0 0 2px rgba(255, 255, 255, 1));
177
+ }
178
+
179
+ .hover-label {
180
+ font-size: 10px;
181
+ fill: rgba(255, 255, 255, 0.9);
182
+ pointer-events: none;
183
+ filter: drop-shadow(0 0 2px rgba(0, 0, 0, 0.8));
184
+ }
185
+
186
+ .selected-label {
187
+ font-size: 10px;
188
+ fill: rgba(255, 255, 255, 1);
189
+ pointer-events: none;
190
+ filter: drop-shadow(0 0 3px rgba(0, 0, 0, 0.9));
191
+ }
192
+
193
+ .hover-path {
194
+ fill: none;
195
+ stroke: rgba(255, 255, 255, 0.9);
196
+ stroke-width: 6;
197
+ stroke-linecap: round;
198
+ stroke-linejoin: round;
199
+ pointer-events: none;
200
+ filter: drop-shadow(0 0 3px rgba(0, 0, 0, 0.6));
201
+ }
202
+
203
+ .selected-path {
204
+ fill: none;
205
+ stroke: rgba(255, 255, 255, 1);
206
+ stroke-width: 6;
207
+ stroke-linecap: round;
208
+ stroke-linejoin: round;
209
+ stroke-dasharray: 8 4;
210
+ pointer-events: none;
211
+ filter: drop-shadow(0 0 4px rgba(0, 0, 0, 0.7));
212
+ }
213
+
214
+ /* Debug mode - show all hit areas */
215
+ .debug-rect {
216
+ fill: rgba(255, 0, 0, 0.15);
217
+ stroke: rgba(255, 0, 0, 0.6);
218
+ stroke-width: 1;
219
+ stroke-dasharray: 4 2;
220
+ pointer-events: none;
221
+ }
222
+
223
+ .debug-rect-trace {
224
+ fill: rgba(0, 255, 0, 0.15);
225
+ stroke: rgba(0, 200, 0, 0.8);
226
+ stroke-width: 1;
227
+ stroke-dasharray: 3 2;
228
+ pointer-events: none;
229
+ }
230
+
231
+ .debug-rect-legend {
232
+ fill: rgba(255, 165, 0, 0.2);
233
+ stroke: rgba(255, 140, 0, 0.8);
234
+ stroke-width: 1;
235
+ stroke-dasharray: 3 2;
236
+ pointer-events: none;
237
+ }
238
+
239
+ .debug-label {
240
+ font-size: 8px;
241
+ fill: rgba(255, 0, 0, 0.9);
242
+ pointer-events: none;
243
+ font-family: monospace;
244
+ }
245
+
246
+ .debug-path {
247
+ fill: none;
248
+ stroke: rgba(0, 200, 0, 0.7);
249
+ stroke-width: 2;
250
+ stroke-linecap: round;
251
+ stroke-linejoin: round;
252
+ stroke-dasharray: 5 3;
253
+ pointer-events: none;
254
+ }
255
+
256
+ .debug-toggle {
257
+ position: fixed;
258
+ bottom: 10px;
259
+ right: 10px;
260
+ z-index: 1000;
261
+ padding: 8px 12px;
262
+ background: #333;
263
+ color: #fff;
264
+ border: none;
265
+ border-radius: 4px;
266
+ cursor: pointer;
267
+ font-size: 12px;
268
+ }
269
+
270
+ .debug-toggle:hover {
271
+ background: #555;
272
+ }
273
+
274
+ .debug-toggle.active {
275
+ background: #c00;
276
+ }
277
+
278
+ /* Hover scatter: White ring around points */
279
+ .hover-scatter {
280
+ fill: none;
281
+ stroke: rgba(255, 255, 255, 0.9);
282
+ stroke-width: 3;
283
+ pointer-events: none;
284
+ filter: drop-shadow(0 0 2px rgba(0, 0, 0, 0.6));
285
+ }
286
+
287
+ /* Selected scatter: Dashed white ring */
288
+ .selected-scatter {
289
+ fill: none;
290
+ stroke: rgba(255, 255, 255, 1);
291
+ stroke-width: 3;
292
+ stroke-dasharray: 3 2;
293
+ pointer-events: none;
294
+ filter: drop-shadow(0 0 3px rgba(0, 0, 0, 0.7));
295
+ }
296
+
297
+ /* =============================================================================
298
+ * Selected Element Panel
299
+ * ============================================================================= */
300
+ #section-selected {
301
+ border: 2px solid var(--accent-muted);
302
+ background: var(--bg-secondary);
303
+ }
304
+
305
+ #section-selected .section-header {
306
+ background: var(--accent-muted);
307
+ color: var(--text-primary);
308
+ font-weight: 600;
309
+ }
310
+
311
+ .selected-info {
312
+ display: flex;
313
+ align-items: center;
314
+ gap: 10px;
315
+ margin-bottom: 12px;
316
+ padding: 8px;
317
+ background: var(--bg-primary);
318
+ border-radius: 4px;
319
+ }
320
+
321
+ .element-type-badge {
322
+ padding: 4px 10px;
323
+ border-radius: 12px;
324
+ font-size: 11px;
325
+ font-weight: 600;
326
+ text-transform: uppercase;
327
+ }
328
+
329
+ .element-type-badge.trace { background: #3498db; color: white; }
330
+ .element-type-badge.scatter { background: #e74c3c; color: white; }
331
+ .element-type-badge.fill { background: #9b59b6; color: white; }
332
+ .element-type-badge.bar { background: #f39c12; color: white; }
333
+ .element-type-badge.label { background: #2ecc71; color: white; }
334
+ .element-type-badge.panel { background: #34495e; color: white; }
335
+ .element-type-badge.legend { background: #1abc9c; color: white; }
336
+
337
+ .element-axis-info {
338
+ font-size: 12px;
339
+ color: var(--text-secondary);
340
+ }
341
+
342
+ .element-props {
343
+ border-top: 1px solid var(--border-color);
344
+ padding-top: 12px;
345
+ margin-top: 8px;
346
+ }
347
+
348
+ /* Range slider styling */
349
+ input[type="range"] {
350
+ width: 100%;
351
+ height: 6px;
352
+ border-radius: 3px;
353
+ background: var(--bg-tertiary);
354
+ outline: none;
355
+ -webkit-appearance: none;
356
+ }
357
+
358
+ input[type="range"]::-webkit-slider-thumb {
359
+ -webkit-appearance: none;
360
+ width: 16px;
361
+ height: 16px;
362
+ border-radius: 50%;
363
+ background: var(--accent);
364
+ cursor: pointer;
365
+ }
366
+
367
+ input[type="range"]::-moz-range-thumb {
368
+ width: 16px;
369
+ height: 16px;
370
+ border-radius: 50%;
371
+ background: var(--accent);
372
+ cursor: pointer;
373
+ border: none;
374
+ }
375
+
376
+ /* =============================================================================
377
+ * Controls Panel
378
+ * ============================================================================= */
379
+ .controls {
380
+ flex: 1;
381
+ min-width: 320px;
382
+ max-width: 420px;
383
+ background: var(--workspace-bg-elevated);
384
+ border-left: 1px solid var(--workspace-border-default);
385
+ overflow-y: auto;
386
+ display: flex;
387
+ flex-direction: column;
388
+ }
389
+
390
+ .controls-header {
391
+ padding: 12px 16px;
392
+ border-bottom: 1px solid var(--workspace-border-subtle);
393
+ display: flex;
394
+ justify-content: space-between;
395
+ align-items: center;
396
+ background: var(--bg-surface);
397
+ position: sticky;
398
+ top: 0;
399
+ z-index: 10;
400
+ }
401
+
402
+ .controls-header h1 {
403
+ font-size: 1.0em;
404
+ font-weight: 600;
405
+ color: var(--status-success);
406
+ }
407
+
408
+ .controls-body {
409
+ padding: 0 14px 14px;
410
+ flex: 1;
411
+ }
412
+
413
+ .filename {
414
+ font-size: 0.8em;
415
+ color: var(--text-muted);
416
+ margin-top: 4px;
417
+ word-break: break-all;
418
+ }
419
+
420
+ .panel-path {
421
+ font-size: 0.75em;
422
+ color: var(--text-muted);
423
+ margin-top: 2px;
424
+ opacity: 0.8;
425
+ }
426
+
427
+ /* Theme toggle */
428
+ .theme-toggle {
429
+ background: transparent;
430
+ border: 1px solid var(--border-muted);
431
+ color: var(--text-secondary);
432
+ cursor: pointer;
433
+ font-size: 16px;
434
+ padding: 6px 10px;
435
+ border-radius: 6px;
436
+ transition: all 0.2s;
437
+ display: flex;
438
+ align-items: center;
439
+ gap: 6px;
440
+ }
441
+
442
+ .theme-toggle:hover {
443
+ background: var(--bg-muted);
444
+ border-color: var(--border-default);
445
+ }
446
+
447
+ /* =============================================================================
448
+ * Section Headers
449
+ * ============================================================================= */
450
+ .section {
451
+ margin-top: 10px;
452
+ }
453
+
454
+ .section-header {
455
+ font-size: 0.72em;
456
+ font-weight: 600;
457
+ text-transform: uppercase;
458
+ letter-spacing: 0.5px;
459
+ color: var(--text-inverse);
460
+ background: var(--status-success);
461
+ padding: 6px 10px;
462
+ border-radius: 4px;
463
+ margin-bottom: 8px;
464
+ }
465
+
466
+ /* =============================================================================
467
+ * Form Fields
468
+ * ============================================================================= */
469
+ .field { margin-bottom: 8px; }
470
+
471
+ .field label {
472
+ display: block;
473
+ font-size: 0.78em;
474
+ font-weight: 500;
475
+ margin-bottom: 3px;
476
+ color: var(--text-secondary);
477
+ }
478
+
479
+ .field input[type="text"],
480
+ .field input[type="number"],
481
+ .field select {
482
+ width: 100%;
483
+ padding: 5px 8px;
484
+ border: 1px solid var(--border-muted);
485
+ border-radius: 4px;
486
+ background: var(--bg-surface);
487
+ color: var(--text-primary);
488
+ font-size: 0.82em;
489
+ transition: border-color 0.2s;
490
+ }
491
+
492
+ .field input:focus,
493
+ .field select:focus {
494
+ outline: none;
495
+ border-color: var(--status-success);
496
+ }
497
+
498
+ .field input[type="color"] {
499
+ width: 40px;
500
+ height: 32px;
501
+ padding: 2px;
502
+ border: 1px solid var(--border-muted);
503
+ border-radius: 4px;
504
+ cursor: pointer;
505
+ background: var(--bg-surface);
506
+ }
507
+
508
+ .field-row {
509
+ display: flex;
510
+ gap: 8px;
511
+ }
512
+
513
+ .field-row .field { flex: 1; margin-bottom: 0; }
514
+
515
+ /* Checkbox styling */
516
+ .checkbox-field {
517
+ display: flex;
518
+ align-items: center;
519
+ gap: 6px;
520
+ cursor: pointer;
521
+ padding: 4px 0;
522
+ }
523
+
524
+ .checkbox-field input[type="checkbox"] {
525
+ width: 16px;
526
+ height: 16px;
527
+ accent-color: var(--status-success);
528
+ }
529
+
530
+ .checkbox-field span {
531
+ font-size: 0.85em;
532
+ color: var(--text-primary);
533
+ }
534
+
535
+ /* Color field with input */
536
+ .color-field {
537
+ display: flex;
538
+ align-items: center;
539
+ gap: 8px;
540
+ }
541
+
542
+ .color-field input[type="text"] {
543
+ flex: 1;
544
+ }
545
+
546
+ /* =============================================================================
547
+ * Traces Section
548
+ * ============================================================================= */
549
+ .traces-list {
550
+ max-height: 200px;
551
+ overflow-y: auto;
552
+ border: 1px solid var(--border-muted);
553
+ border-radius: 4px;
554
+ background: var(--bg-muted);
555
+ }
556
+
557
+ .trace-item {
558
+ display: flex;
559
+ align-items: center;
560
+ gap: 8px;
561
+ padding: 8px 10px;
562
+ border-bottom: 1px solid var(--border-muted);
563
+ font-size: 0.85em;
564
+ }
565
+
566
+ .trace-item:last-child { border-bottom: none; }
567
+
568
+ .trace-item-highlight {
569
+ background: var(--status-success);
570
+ color: var(--text-inverse);
571
+ animation: traceHighlightFade 1.5s ease-out forwards;
572
+ }
573
+
574
+ @keyframes traceHighlightFade {
575
+ 0% { background: var(--status-success); }
576
+ 100% { background: var(--bg-muted); }
577
+ }
578
+
579
+ .trace-color {
580
+ width: 24px;
581
+ height: 24px;
582
+ border-radius: 4px;
583
+ border: 1px solid var(--border-default);
584
+ cursor: pointer;
585
+ }
586
+
587
+ .trace-label {
588
+ flex: 1;
589
+ color: var(--text-primary);
590
+ }
591
+
592
+ .trace-style select {
593
+ padding: 4px 6px;
594
+ font-size: 0.8em;
595
+ border: 1px solid var(--border-muted);
596
+ border-radius: 3px;
597
+ background: var(--bg-surface);
598
+ color: var(--text-primary);
599
+ }
600
+
601
+ /* =============================================================================
602
+ * Annotations
603
+ * ============================================================================= */
604
+ .annotations-list {
605
+ margin-top: 10px;
606
+ max-height: 120px;
607
+ overflow-y: auto;
608
+ }
609
+
610
+ .annotation-item {
611
+ display: flex;
612
+ justify-content: space-between;
613
+ align-items: center;
614
+ padding: 6px 10px;
615
+ background: var(--bg-muted);
616
+ border-radius: 4px;
617
+ margin-bottom: 5px;
618
+ font-size: 0.85em;
619
+ }
620
+
621
+ .annotation-item span { color: var(--text-primary); }
622
+
623
+ .annotation-item button {
624
+ padding: 3px 8px;
625
+ font-size: 0.75em;
626
+ background: var(--status-error);
627
+ border: none;
628
+ border-radius: 3px;
629
+ color: white;
630
+ cursor: pointer;
631
+ }
632
+
633
+ /* =============================================================================
634
+ * Buttons
635
+ * ============================================================================= */
636
+ .btn {
637
+ width: 100%;
638
+ padding: 7px 12px;
639
+ margin-top: 6px;
640
+ border: none;
641
+ border-radius: 4px;
642
+ cursor: pointer;
643
+ font-size: 0.9em;
644
+ font-weight: 500;
645
+ transition: all 0.2s;
646
+ }
647
+
648
+ .btn-primary {
649
+ background: var(--status-success);
650
+ color: white;
651
+ }
652
+
653
+ .btn-primary:hover {
654
+ filter: brightness(1.1);
655
+ }
656
+
657
+ .btn-secondary {
658
+ background: var(--bg-muted);
659
+ color: var(--text-primary);
660
+ border: 1px solid var(--border-muted);
661
+ }
662
+
663
+ .btn-secondary:hover {
664
+ background: var(--workspace-bg-tertiary);
665
+ }
666
+
667
+ .btn-cta {
668
+ background: var(--color-cta);
669
+ color: white;
670
+ }
671
+
672
+ .btn-cta:hover {
673
+ background: var(--color-cta-hover);
674
+ }
675
+
676
+ /* =============================================================================
677
+ * Unit Toggle
678
+ * ============================================================================= */
679
+ .unit-toggle {
680
+ display: flex;
681
+ gap: 0;
682
+ border-radius: 4px;
683
+ overflow: hidden;
684
+ border: 1px solid var(--border-default);
685
+ }
686
+
687
+ .unit-btn {
688
+ padding: 4px 12px;
689
+ font-size: 0.8em;
690
+ font-weight: 500;
691
+ border: none;
692
+ background: var(--bg-muted);
693
+ color: var(--text-muted);
694
+ cursor: pointer;
695
+ transition: all 0.15s;
696
+ }
697
+
698
+ .unit-btn:first-child {
699
+ border-right: 1px solid var(--border-default);
700
+ }
701
+
702
+ .unit-btn:hover {
703
+ background: var(--bg-surface);
704
+ color: var(--text-secondary);
705
+ }
706
+
707
+ .unit-btn.active {
708
+ background: var(--color-cta);
709
+ color: white;
710
+ }
711
+
712
+ /* =============================================================================
713
+ * Background Type Toggle
714
+ * ============================================================================= */
715
+ .bg-toggle {
716
+ display: flex;
717
+ gap: 6px;
718
+ }
719
+
720
+ .bg-btn {
721
+ display: flex;
722
+ flex-direction: column;
723
+ align-items: center;
724
+ gap: 4px;
725
+ padding: 6px 10px;
726
+ font-size: 0.75em;
727
+ border: 1px solid var(--border-default);
728
+ border-radius: 4px;
729
+ background: var(--bg-muted);
730
+ color: var(--text-muted);
731
+ cursor: pointer;
732
+ transition: all 0.15s;
733
+ flex: 1;
734
+ }
735
+
736
+ .bg-btn:hover {
737
+ background: var(--bg-surface);
738
+ color: var(--text-secondary);
739
+ }
740
+
741
+ .bg-btn.active {
742
+ border-color: var(--color-cta);
743
+ background: var(--bg-surface);
744
+ color: var(--text-primary);
745
+ box-shadow: 0 0 0 1px var(--color-cta);
746
+ }
747
+
748
+ .bg-preview {
749
+ width: 20px;
750
+ height: 14px;
751
+ border-radius: 2px;
752
+ border: 1px solid var(--border-default);
753
+ }
754
+
755
+ .bg-preview.white {
756
+ background: #ffffff;
757
+ }
758
+
759
+ .bg-preview.transparent {
760
+ background: linear-gradient(45deg, #ccc 25%, transparent 25%),
761
+ linear-gradient(-45deg, #ccc 25%, transparent 25%),
762
+ linear-gradient(45deg, transparent 75%, #ccc 75%),
763
+ linear-gradient(-45deg, transparent 75%, #ccc 75%);
764
+ background-size: 8px 8px;
765
+ background-position: 0 0, 0 4px, 4px -4px, -4px 0px;
766
+ background-color: #fff;
767
+ }
768
+
769
+ .bg-preview.black {
770
+ background: #000000;
771
+ }
772
+
773
+ /* =============================================================================
774
+ * Section Hints
775
+ * ============================================================================= */
776
+ .section-hint {
777
+ font-size: 0.75em;
778
+ color: var(--text-muted);
779
+ font-style: italic;
780
+ margin-bottom: 10px;
781
+ padding: 6px 8px;
782
+ background: var(--bg-muted);
783
+ border-radius: 4px;
784
+ border-left: 2px solid var(--border-default);
785
+ }
786
+
787
+ /* =============================================================================
788
+ * Element Statistics
789
+ * ============================================================================= */
790
+ .element-stats {
791
+ margin-top: 12px;
792
+ padding: 10px;
793
+ background: var(--bg-muted);
794
+ border-radius: 4px;
795
+ border: 1px solid var(--border-muted);
796
+ }
797
+
798
+ .stats-header {
799
+ font-size: 0.8em;
800
+ font-weight: 600;
801
+ color: var(--text-secondary);
802
+ margin-bottom: 8px;
803
+ text-transform: uppercase;
804
+ letter-spacing: 0.5px;
805
+ }
806
+
807
+ .stats-grid {
808
+ display: grid;
809
+ grid-template-columns: repeat(3, 1fr);
810
+ gap: 8px;
811
+ }
812
+
813
+ .stat-item {
814
+ display: flex;
815
+ flex-direction: column;
816
+ gap: 2px;
817
+ }
818
+
819
+ .stat-label {
820
+ font-size: 0.7em;
821
+ color: var(--text-muted);
822
+ text-transform: uppercase;
823
+ }
824
+
825
+ .stat-value {
826
+ font-size: 0.85em;
827
+ font-weight: 500;
828
+ color: var(--text-primary);
829
+ font-family: 'SF Mono', Monaco, 'Cascadia Code', monospace;
830
+ }
831
+
832
+ /* =============================================================================
833
+ * Status Bar
834
+ * ============================================================================= */
835
+ .status-bar {
836
+ margin-top: 16px;
837
+ padding: 10px 12px;
838
+ border-radius: 4px;
839
+ background: var(--bg-muted);
840
+ font-size: 0.8em;
841
+ color: var(--text-secondary);
842
+ border-left: 3px solid var(--status-success);
843
+ }
844
+
845
+ .status-bar.error {
846
+ border-left-color: var(--status-error);
847
+ }
848
+
849
+ /* =============================================================================
850
+ * Collapsible Sections
851
+ * ============================================================================= */
852
+ .section-toggle {
853
+ cursor: pointer;
854
+ display: flex;
855
+ align-items: center;
856
+ gap: 8px;
857
+ }
858
+
859
+ .section-toggle::before {
860
+ content: "\\25BC";
861
+ font-size: 0.7em;
862
+ transition: transform 0.2s;
863
+ }
864
+
865
+ .section-toggle.collapsed::before {
866
+ transform: rotate(-90deg);
867
+ }
868
+
869
+ .section-content {
870
+ overflow: hidden;
871
+ transition: max-height 0.3s ease;
872
+ }
873
+
874
+ .section-content.collapsed {
875
+ max-height: 0 !important;
876
+ }
877
+
878
+ /* =============================================================================
879
+ * Scrollbar Styling
880
+ * ============================================================================= */
881
+ .controls::-webkit-scrollbar,
882
+ .traces-list::-webkit-scrollbar,
883
+ .annotations-list::-webkit-scrollbar {
884
+ width: 6px;
885
+ }
886
+
887
+ .controls::-webkit-scrollbar-track,
888
+ .traces-list::-webkit-scrollbar-track,
889
+ .annotations-list::-webkit-scrollbar-track {
890
+ background: var(--bg-muted);
891
+ }
892
+
893
+ .controls::-webkit-scrollbar-thumb,
894
+ .traces-list::-webkit-scrollbar-thumb,
895
+ .annotations-list::-webkit-scrollbar-thumb {
896
+ background: var(--border-default);
897
+ border-radius: 3px;
898
+ }
899
+
900
+ .controls::-webkit-scrollbar-thumb:hover,
901
+ .traces-list::-webkit-scrollbar-thumb:hover,
902
+ .annotations-list::-webkit-scrollbar-thumb:hover {
903
+ background: var(--text-muted);
904
+ }
905
+
906
+ /* =============================================================================
907
+ * Statistics Section
908
+ * ============================================================================= */
909
+ #stats-container {
910
+ font-family: 'JetBrains Mono', 'Fira Code', 'Monaco', monospace;
911
+ font-size: 11px;
912
+ line-height: 1.5;
913
+ }
914
+
915
+ .stats-loading {
916
+ color: var(--text-muted);
917
+ font-style: italic;
918
+ padding: 10px;
919
+ text-align: center;
920
+ }
921
+
922
+ .stats-empty {
923
+ color: var(--text-muted);
924
+ padding: 10px;
925
+ text-align: center;
926
+ background: var(--bg-muted);
927
+ border-radius: 4px;
928
+ }
929
+
930
+ .stats-card {
931
+ background: var(--bg-surface);
932
+ border: 1px solid var(--workspace-border-subtle);
933
+ border-radius: 6px;
934
+ padding: 12px;
935
+ margin-bottom: 10px;
936
+ }
937
+
938
+ .stats-card-header {
939
+ display: flex;
940
+ justify-content: space-between;
941
+ align-items: center;
942
+ margin-bottom: 8px;
943
+ padding-bottom: 8px;
944
+ border-bottom: 1px solid var(--workspace-border-subtle);
945
+ }
946
+
947
+ .stats-card-title {
948
+ font-weight: 600;
949
+ font-size: 12px;
950
+ color: var(--text-primary);
951
+ }
952
+
953
+ .stats-significance {
954
+ display: inline-block;
955
+ padding: 2px 8px;
956
+ border-radius: 4px;
957
+ font-weight: bold;
958
+ font-size: 11px;
959
+ }
960
+
961
+ .stats-significance.sig-high {
962
+ background: #dcfce7;
963
+ color: #166534;
964
+ }
965
+
966
+ .stats-significance.sig-medium {
967
+ background: #fef9c3;
968
+ color: #854d0e;
969
+ }
970
+
971
+ .stats-significance.sig-low {
972
+ background: #fee2e2;
973
+ color: #991b1b;
974
+ }
975
+
976
+ .stats-significance.sig-ns {
977
+ background: var(--bg-muted);
978
+ color: var(--text-muted);
979
+ }
980
+
981
+ .stats-row {
982
+ display: flex;
983
+ justify-content: space-between;
984
+ padding: 3px 0;
985
+ }
986
+
987
+ .stats-label {
988
+ color: var(--text-secondary);
989
+ }
990
+
991
+ .stats-value {
992
+ font-weight: 500;
993
+ color: var(--text-primary);
994
+ }
995
+
996
+ .stats-groups {
997
+ display: grid;
998
+ grid-template-columns: 1fr 1fr;
999
+ gap: 8px;
1000
+ margin-top: 8px;
1001
+ padding-top: 8px;
1002
+ border-top: 1px dashed var(--workspace-border-subtle);
1003
+ }
1004
+
1005
+ .stats-group {
1006
+ background: var(--bg-muted);
1007
+ padding: 8px;
1008
+ border-radius: 4px;
1009
+ font-size: 10px;
1010
+ }
1011
+
1012
+ .stats-group-name {
1013
+ font-weight: 600;
1014
+ margin-bottom: 4px;
1015
+ color: var(--text-primary);
1016
+ }
1017
+
1018
+ .stats-correction-badge {
1019
+ display: inline-block;
1020
+ background: var(--color-cta);
1021
+ color: white;
1022
+ padding: 2px 6px;
1023
+ border-radius: 3px;
1024
+ font-size: 9px;
1025
+ margin-left: 6px;
1026
+ }
1027
+
1028
+ .stats-summary-header {
1029
+ background: var(--scitex-01);
1030
+ color: var(--text-inverse);
1031
+ padding: 10px 12px;
1032
+ border-radius: 6px 6px 0 0;
1033
+ font-weight: 600;
1034
+ margin-bottom: 0;
1035
+ }
1036
+
1037
+ .stats-summary-body {
1038
+ background: var(--bg-surface);
1039
+ border: 1px solid var(--scitex-01);
1040
+ border-top: none;
1041
+ border-radius: 0 0 6px 6px;
1042
+ padding: 12px;
1043
+ }
1044
+
1045
+ /* =============================================================================
1046
+ * Subsection Headers (for grouped controls like X/Y axis settings)
1047
+ * ============================================================================= */
1048
+ .subsection-header {
1049
+ font-size: 0.68em;
1050
+ font-weight: 600;
1051
+ text-transform: uppercase;
1052
+ letter-spacing: 0.3px;
1053
+ color: var(--text-secondary);
1054
+ padding: 4px 6px;
1055
+ margin-bottom: 6px;
1056
+ margin-top: 8px;
1057
+ background: var(--bg-muted);
1058
+ border-left: 2px solid var(--status-success);
1059
+ border-radius: 0 4px 4px 0;
1060
+ }
1061
+
1062
+ .subsection-header:first-child {
1063
+ margin-top: 0;
1064
+ }
1065
+
1066
+ /* =============================================================================
1067
+ * Axis Tabs (for X/Y/Z axis switching)
1068
+ * ============================================================================= */
1069
+ .axis-tabs {
1070
+ display: flex;
1071
+ gap: 3px;
1072
+ margin-bottom: 8px;
1073
+ background: var(--bg-muted);
1074
+ padding: 3px;
1075
+ border-radius: 6px;
1076
+ }
1077
+
1078
+ .axis-tab {
1079
+ flex: 1;
1080
+ padding: 4px 8px;
1081
+ border: none;
1082
+ background: transparent;
1083
+ color: var(--text-secondary);
1084
+ font-size: 0.8em;
1085
+ font-weight: 600;
1086
+ cursor: pointer;
1087
+ border-radius: 4px;
1088
+ transition: all 0.15s ease;
1089
+ }
1090
+
1091
+ .axis-tab:hover {
1092
+ background: var(--bg-surface);
1093
+ color: var(--text-primary);
1094
+ }
1095
+
1096
+ .axis-tab.active {
1097
+ background: var(--scitex-01);
1098
+ color: var(--bg-primary);
1099
+ }
1100
+
1101
+ .axis-panel {
1102
+ animation: fadeIn 0.15s ease;
1103
+ }
1104
+
1105
+ @keyframes fadeIn {
1106
+ from { opacity: 0; }
1107
+ to { opacity: 1; }
1108
+ }
1109
+
1110
+ /* =============================================================================
1111
+ * Loading Spinner Overlay
1112
+ * ============================================================================= */
1113
+ .loading-overlay {
1114
+ position: absolute;
1115
+ top: 0;
1116
+ left: 0;
1117
+ right: 0;
1118
+ bottom: 0;
1119
+ background: rgba(0, 0, 0, 0.3);
1120
+ display: flex;
1121
+ align-items: center;
1122
+ justify-content: center;
1123
+ z-index: 100;
1124
+ border-radius: 8px;
1125
+ }
1126
+
1127
+ /* Global loading overlay - non-intrusive top bar style */
1128
+ .global-loading-overlay {
1129
+ position: fixed;
1130
+ top: 0;
1131
+ left: 0;
1132
+ width: 100%;
1133
+ height: 36px;
1134
+ background: var(--workspace-bg-elevated);
1135
+ border-bottom: 1px solid var(--workspace-border-default);
1136
+ display: flex;
1137
+ align-items: center;
1138
+ justify-content: center;
1139
+ z-index: 9999;
1140
+ gap: 8px;
1141
+ box-shadow: 0 2px 8px rgba(0,0,0,0.1);
1142
+ }
1143
+
1144
+ .global-loading-overlay .loading-text {
1145
+ color: var(--text-secondary);
1146
+ font-size: 12px;
1147
+ font-weight: 500;
1148
+ }
1149
+
1150
+ .spinner {
1151
+ width: 16px;
1152
+ height: 16px;
1153
+ border: 2px solid var(--scitex-06);
1154
+ border-top: 2px solid var(--status-success);
1155
+ border-radius: 50%;
1156
+ animation: spin 0.6s linear infinite;
1157
+ }
1158
+
1159
+ .spinner-small {
1160
+ width: 14px;
1161
+ height: 14px;
1162
+ border-width: 2px;
1163
+ }
1164
+
1165
+ @keyframes spin {
1166
+ 0% { transform: rotate(0deg); }
1167
+ 100% { transform: rotate(360deg); }
1168
+ }
1169
+
1170
+ /* =============================================================================
1171
+ * Panel Grid View (for multi-panel figz bundles)
1172
+ * ============================================================================= */
1173
+ .panel-grid-section {
1174
+ width: 100%;
1175
+ margin-bottom: 20px;
1176
+ background: var(--workspace-bg-elevated);
1177
+ border-radius: 8px;
1178
+ padding: 16px;
1179
+ box-shadow: 0 2px 8px rgba(0,0,0,0.1);
1180
+ }
1181
+
1182
+ .panel-grid-header {
1183
+ display: flex;
1184
+ justify-content: space-between;
1185
+ align-items: center;
1186
+ margin-bottom: 12px;
1187
+ }
1188
+
1189
+ .panel-grid-header h3 {
1190
+ font-size: 1em;
1191
+ font-weight: 600;
1192
+ color: var(--text-primary);
1193
+ margin: 0;
1194
+ }
1195
+
1196
+ .panel-grid {
1197
+ display: grid;
1198
+ grid-template-columns: repeat(auto-fit, minmax(180px, 1fr));
1199
+ gap: 12px;
1200
+ }
1201
+
1202
+ .panel-card {
1203
+ background: var(--preview-bg);
1204
+ background-size: 16px 16px;
1205
+ background-position: 0 0, 0 8px, 8px -8px, -8px 0px;
1206
+ border-radius: 6px;
1207
+ overflow: hidden;
1208
+ cursor: pointer;
1209
+ transition: all 0.2s ease;
1210
+ border: 2px solid transparent;
1211
+ position: relative;
1212
+ }
1213
+
1214
+ .panel-card:hover {
1215
+ border-color: var(--color-cta);
1216
+ transform: translateY(-2px);
1217
+ box-shadow: 0 4px 12px rgba(0,0,0,0.15);
1218
+ }
1219
+
1220
+ .panel-card.active {
1221
+ border-color: var(--status-success);
1222
+ box-shadow: 0 0 0 2px var(--status-success);
1223
+ }
1224
+
1225
+ .panel-card img {
1226
+ width: 100%;
1227
+ height: auto;
1228
+ display: block;
1229
+ }
1230
+
1231
+ /* Dark mode: invert plot colors for visibility */
1232
+ [data-theme="dark"] .panel-card img,
1233
+ [data-theme="dark"] .panel-canvas-item img,
1234
+ [data-theme="dark"] .preview-wrapper img {
1235
+ filter: invert(0.88) hue-rotate(180deg);
1236
+ }
1237
+
1238
+ .panel-card-container {
1239
+ position: relative;
1240
+ display: inline-block;
1241
+ width: 100%;
1242
+ }
1243
+
1244
+ .panel-card-overlay {
1245
+ position: absolute;
1246
+ top: 0;
1247
+ left: 0;
1248
+ pointer-events: none;
1249
+ z-index: 5;
1250
+ }
1251
+
1252
+ .panel-card-label {
1253
+ position: absolute;
1254
+ top: 6px;
1255
+ left: 6px;
1256
+ background: rgba(0,0,0,0.7);
1257
+ color: white;
1258
+ padding: 3px 8px;
1259
+ border-radius: 4px;
1260
+ font-size: 0.75em;
1261
+ font-weight: 600;
1262
+ }
1263
+
1264
+ .panel-card-loading {
1265
+ display: flex;
1266
+ align-items: center;
1267
+ justify-content: center;
1268
+ min-height: 120px;
1269
+ color: var(--text-muted);
1270
+ font-size: 0.85em;
1271
+ }
1272
+
1273
+ /* Panel Navigation Header */
1274
+ .preview-header {
1275
+ display: flex;
1276
+ justify-content: space-between;
1277
+ align-items: center;
1278
+ padding: 8px 12px;
1279
+ background: var(--workspace-bg-elevated);
1280
+ border-radius: 8px 8px 0 0;
1281
+ border-bottom: 1px solid var(--workspace-border-subtle);
1282
+ }
1283
+
1284
+ #current-panel-name {
1285
+ font-weight: 600;
1286
+ color: var(--text-primary);
1287
+ font-size: 0.9em;
1288
+ }
1289
+
1290
+ .panel-nav {
1291
+ display: flex;
1292
+ align-items: center;
1293
+ gap: 8px;
1294
+ }
1295
+
1296
+ #panel-indicator {
1297
+ font-size: 0.8em;
1298
+ color: var(--text-muted);
1299
+ min-width: 50px;
1300
+ text-align: center;
1301
+ }
1302
+
1303
+ .btn-sm {
1304
+ padding: 4px 10px;
1305
+ font-size: 0.8em;
1306
+ width: auto;
1307
+ margin: 0;
1308
+ }
1309
+
1310
+ /* Preview area adjustments when panel header is visible */
1311
+ .preview-wrapper:has(.preview-header:not([style*="display: none"])) {
1312
+ border-radius: 0 0 8px 8px;
1313
+ }
1314
+
1315
+ .preview {
1316
+ flex-direction: column;
1317
+ align-items: stretch;
1318
+ padding: 20px;
1319
+ overflow-y: auto;
1320
+ }
1321
+
1322
+ .preview > .preview-wrapper {
1323
+ flex-shrink: 0;
1324
+ }
1325
+
1326
+ /* =============================================================================
1327
+ * Unified Panel Canvas (matches figz export layout)
1328
+ * Exactly matches SciTeX Cloud's .plot-preview-area styling
1329
+ * ============================================================================= */
1330
+ .panel-canvas {
1331
+ position: relative;
1332
+ min-height: 400px;
1333
+ /* White background with grid pattern - exact match to SciTeX Cloud */
1334
+ background: #ffffff;
1335
+ background-image:
1336
+ linear-gradient(0deg, transparent 23px, #e5e5e5 23px),
1337
+ linear-gradient(90deg, transparent 23px, #e5e5e5 23px);
1338
+ background-size: 24px 24px;
1339
+ border-radius: 8px;
1340
+ overflow: visible; /* Allow panels to extend slightly if needed */
1341
+ }
1342
+
1343
+ /* Dark mode: dark canvas background for eye comfort */
1344
+ [data-theme="dark"] .panel-canvas,
1345
+ .dark-mode .panel-canvas {
1346
+ background: #1a1a1a;
1347
+ background-image:
1348
+ linear-gradient(0deg, transparent 23px, #2a2a2a 23px),
1349
+ linear-gradient(90deg, transparent 23px, #2a2a2a 23px);
1350
+ background-size: 24px 24px;
1351
+ }
1352
+
1353
+ /* Hide grid when toggled off (G key) */
1354
+ .panel-canvas.hide-grid {
1355
+ background-image: none !important;
1356
+ }
1357
+
1358
+ /* Unified canvas - panels appear as single figure matching figz export */
1359
+ .panel-canvas-item {
1360
+ position: absolute;
1361
+ user-select: none;
1362
+ background: transparent;
1363
+ overflow: visible;
1364
+ border: none;
1365
+ cursor: grab; /* Indicate draggability */
1366
+ }
1367
+
1368
+ .panel-canvas-item:hover {
1369
+ z-index: 10;
1370
+ }
1371
+
1372
+ .panel-canvas-item:active {
1373
+ cursor: grabbing;
1374
+ }
1375
+
1376
+ .panel-canvas-item.active {
1377
+ outline: 2px solid var(--status-success);
1378
+ outline-offset: 2px;
1379
+ }
1380
+
1381
+ .panel-canvas-item .panel-card-container {
1382
+ width: 100%;
1383
+ height: 100%;
1384
+ position: relative;
1385
+ }
1386
+
1387
+ .panel-canvas-item img {
1388
+ width: 100%;
1389
+ height: 100%;
1390
+ object-fit: contain; /* Preserve aspect ratio - never distort figures */
1391
+ pointer-events: none;
1392
+ display: block;
1393
+ }
1394
+
1395
+ .panel-canvas-label {
1396
+ position: absolute;
1397
+ top: 4px;
1398
+ left: 4px;
1399
+ background: rgba(0,0,0,0.6);
1400
+ color: white;
1401
+ padding: 2px 6px;
1402
+ border-radius: 3px;
1403
+ font-size: 0.65em;
1404
+ font-weight: 600;
1405
+ cursor: move;
1406
+ opacity: 1; /* Show by default */
1407
+ transition: opacity 0.2s, background 0.2s;
1408
+ z-index: 5;
1409
+ }
1410
+
1411
+ .panel-canvas-item:hover .panel-canvas-label {
1412
+ opacity: 1;
1413
+ background: rgba(0,0,0,0.8); /* Slightly darker on hover */
1414
+ }
1415
+
1416
+ /* Drag handle for panel repositioning */
1417
+ .panel-drag-handle {
1418
+ position: absolute;
1419
+ top: 4px;
1420
+ right: 4px;
1421
+ background: rgba(0,0,0,0.5);
1422
+ color: white;
1423
+ padding: 2px 4px;
1424
+ border-radius: 3px;
1425
+ font-size: 0.7em;
1426
+ cursor: move;
1427
+ opacity: 0;
1428
+ transition: opacity 0.2s, background 0.2s;
1429
+ z-index: 5;
1430
+ user-select: none;
1431
+ }
1432
+
1433
+ .panel-canvas-item:hover .panel-drag-handle {
1434
+ opacity: 1;
1435
+ }
1436
+
1437
+ .panel-drag-handle:hover {
1438
+ background: rgba(0,0,0,0.8);
1439
+ }
1440
+
1441
+ .panel-drag-handle:active {
1442
+ background: var(--accent-primary);
1443
+ }
1444
+
1445
+ /* Position indicator while dragging */
1446
+ .panel-position-indicator {
1447
+ position: absolute;
1448
+ bottom: 4px;
1449
+ left: 4px;
1450
+ background: rgba(0,0,0,0.7);
1451
+ color: #4fc3f7;
1452
+ padding: 2px 6px;
1453
+ border-radius: 3px;
1454
+ font-size: 0.6em;
1455
+ font-family: monospace;
1456
+ pointer-events: none;
1457
+ opacity: 0;
1458
+ transition: opacity 0.3s;
1459
+ z-index: 5;
1460
+ }
1461
+
1462
+ /* Dragging state */
1463
+ .panel-canvas-item.dragging {
1464
+ opacity: 0.85;
1465
+ z-index: 100;
1466
+ box-shadow: 0 8px 24px rgba(0,0,0,0.3);
1467
+ outline: 2px dashed var(--accent-primary);
1468
+ outline-offset: 2px;
1469
+ cursor: grabbing;
1470
+ }
1471
+
1472
+ .panel-canvas-item.dragging .panel-position-indicator {
1473
+ opacity: 1;
1474
+ }
1475
+
1476
+ .canvas-controls {
1477
+ display: flex;
1478
+ gap: 8px;
1479
+ margin-bottom: 8px;
1480
+ align-items: center;
1481
+ }
1482
+
1483
+ .canvas-controls button {
1484
+ padding: 4px 12px;
1485
+ font-size: 0.8em;
1486
+ }
1487
+
1488
+ /* Toolbar separator */
1489
+ .toolbar-separator {
1490
+ width: 1px;
1491
+ height: 24px;
1492
+ background: #555;
1493
+ margin: 0 4px;
1494
+ }
1495
+
1496
+ /* Download dropdown in toolbar */
1497
+ .download-dropdown {
1498
+ position: relative;
1499
+ display: inline-block;
1500
+ }
1501
+
1502
+ .download-dropdown #download-btn {
1503
+ background: #4a90d9;
1504
+ border: none;
1505
+ color: white;
1506
+ padding: 4px 12px;
1507
+ font-size: 0.8em;
1508
+ border-radius: 3px;
1509
+ cursor: pointer;
1510
+ }
1511
+
1512
+ .download-dropdown #download-btn:hover {
1513
+ background: #5a9fe9;
1514
+ }
1515
+
1516
+ #download-menu {
1517
+ display: none;
1518
+ position: absolute;
1519
+ top: 100%;
1520
+ left: 0;
1521
+ background: #2a2a2a;
1522
+ border: 1px solid #444;
1523
+ border-radius: 4px;
1524
+ min-width: 160px;
1525
+ box-shadow: 0 4px 12px rgba(0, 0, 0, 0.4);
1526
+ z-index: 1000;
1527
+ margin-top: 4px;
1528
+ padding: 4px 0;
1529
+ }
1530
+
1531
+ .download-item {
1532
+ display: block;
1533
+ padding: 8px 12px;
1534
+ color: #ddd;
1535
+ text-decoration: none;
1536
+ font-size: 0.85em;
1537
+ transition: background 0.15s;
1538
+ white-space: nowrap;
1539
+ }
1540
+
1541
+ .download-item:hover {
1542
+ background: #3a3a3a;
1543
+ color: #fff;
1544
+ }
1545
+
1546
+ .download-divider {
1547
+ height: 1px;
1548
+ background: #444;
1549
+ margin: 4px 0;
1550
+ }
1551
+
1552
+ /* =============================================================================
1553
+ Context Menu (Right-Click)
1554
+ ============================================================================= */
1555
+ .context-menu {
1556
+ position: fixed;
1557
+ background: var(--bg-secondary, #2a2a2a);
1558
+ border: 1px solid var(--border-color, #444);
1559
+ border-radius: 6px;
1560
+ box-shadow: 0 4px 16px rgba(0, 0, 0, 0.4);
1561
+ min-width: 200px;
1562
+ z-index: 10000;
1563
+ padding: 4px 0;
1564
+ font-size: 13px;
1565
+ }
1566
+
1567
+ .context-menu-item {
1568
+ display: flex;
1569
+ align-items: center;
1570
+ padding: 8px 12px;
1571
+ cursor: pointer;
1572
+ color: var(--text-primary, #ddd);
1573
+ transition: background 0.1s;
1574
+ position: relative;
1575
+ }
1576
+
1577
+ .context-menu-item:hover:not(.disabled) {
1578
+ background: var(--accent-primary, #4a90d9);
1579
+ color: #fff;
1580
+ }
1581
+
1582
+ .context-menu-item.disabled {
1583
+ color: var(--text-muted, #666);
1584
+ cursor: not-allowed;
1585
+ }
1586
+
1587
+ .context-menu-icon {
1588
+ width: 20px;
1589
+ margin-right: 8px;
1590
+ text-align: center;
1591
+ font-size: 14px;
1592
+ }
1593
+
1594
+ .context-menu-shortcut {
1595
+ margin-left: auto;
1596
+ font-size: 11px;
1597
+ color: var(--text-muted, #888);
1598
+ padding-left: 16px;
1599
+ }
1600
+
1601
+ .context-menu-item:hover:not(.disabled) .context-menu-shortcut {
1602
+ color: rgba(255, 255, 255, 0.7);
1603
+ }
1604
+
1605
+ .context-menu-arrow {
1606
+ margin-left: auto;
1607
+ font-size: 10px;
1608
+ color: var(--text-muted, #888);
1609
+ }
1610
+
1611
+ .context-menu-divider {
1612
+ height: 1px;
1613
+ background: var(--border-color, #444);
1614
+ margin: 4px 0;
1615
+ }
1616
+
1617
+ /* Submenu */
1618
+ .context-menu-submenu {
1619
+ position: relative;
1620
+ }
1621
+
1622
+ .context-submenu {
1623
+ display: none;
1624
+ position: absolute;
1625
+ left: 100%;
1626
+ top: 0;
1627
+ background: var(--bg-secondary, #2a2a2a);
1628
+ border: 1px solid var(--border-color, #444);
1629
+ border-radius: 6px;
1630
+ box-shadow: 0 4px 16px rgba(0, 0, 0, 0.4);
1631
+ min-width: 150px;
1632
+ padding: 4px 0;
1633
+ }
1634
+
1635
+ .context-menu-submenu:hover > .context-submenu {
1636
+ display: block;
1637
+ }
1638
+
1639
+ .context-submenu .context-menu-item {
1640
+ padding: 6px 12px;
1641
+ }
1642
+
1643
+ /* Panel selection styling */
1644
+ .panel-canvas-item.active {
1645
+ outline: 3px solid var(--accent-primary, #4a90d9);
1646
+ outline-offset: 2px;
1647
+ z-index: 100;
1648
+ }
1649
+
1650
+ .panel-canvas-item:hover:not(.active) {
1651
+ outline: 2px dashed var(--accent-secondary, #6ab04c);
1652
+ outline-offset: 2px;
1653
+ }
1654
+ """
1655
+
1656
+
1657
+ # EOF