figrecipe 0.6.0__py3-none-any.whl → 0.9.0__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 (269) hide show
  1. figrecipe/__init__.py +161 -1030
  2. figrecipe/__main__.py +12 -0
  3. figrecipe/_api/__init__.py +48 -0
  4. figrecipe/_api/_extract.py +108 -0
  5. figrecipe/_api/_notebook.py +61 -0
  6. figrecipe/_api/_panel.py +113 -0
  7. figrecipe/_api/_save.py +287 -0
  8. figrecipe/_api/_seaborn_proxy.py +34 -0
  9. figrecipe/_api/_style_manager.py +153 -0
  10. figrecipe/_api/_subplots.py +333 -0
  11. figrecipe/_api/_validate.py +82 -0
  12. figrecipe/_cli/__init__.py +7 -0
  13. figrecipe/_cli/_compose.py +87 -0
  14. figrecipe/_cli/_convert.py +117 -0
  15. figrecipe/_cli/_crop.py +82 -0
  16. figrecipe/_cli/_edit.py +70 -0
  17. figrecipe/_cli/_extract.py +128 -0
  18. figrecipe/_cli/_fonts.py +47 -0
  19. figrecipe/_cli/_info.py +67 -0
  20. figrecipe/_cli/_main.py +58 -0
  21. figrecipe/_cli/_reproduce.py +79 -0
  22. figrecipe/_cli/_style.py +77 -0
  23. figrecipe/_cli/_validate.py +66 -0
  24. figrecipe/_cli/_version.py +50 -0
  25. figrecipe/_composition/__init__.py +32 -0
  26. figrecipe/_composition/_alignment.py +452 -0
  27. figrecipe/_composition/_compose.py +179 -0
  28. figrecipe/_composition/_import_axes.py +127 -0
  29. figrecipe/_composition/_visibility.py +125 -0
  30. figrecipe/_dev/__init__.py +4 -93
  31. figrecipe/_dev/_plotters.py +76 -0
  32. figrecipe/_dev/_run_demos.py +56 -0
  33. figrecipe/_dev/browser/__init__.py +69 -0
  34. figrecipe/_dev/browser/_audio.py +240 -0
  35. figrecipe/_dev/browser/_caption.py +356 -0
  36. figrecipe/_dev/browser/_click_effect.py +146 -0
  37. figrecipe/_dev/browser/_cursor.py +196 -0
  38. figrecipe/_dev/browser/_highlight.py +105 -0
  39. figrecipe/_dev/browser/_narration.py +237 -0
  40. figrecipe/_dev/browser/_recorder.py +446 -0
  41. figrecipe/_dev/browser/_utils.py +178 -0
  42. figrecipe/_dev/browser/_video_trim/__init__.py +152 -0
  43. figrecipe/_dev/browser/_video_trim/_detection.py +223 -0
  44. figrecipe/_dev/browser/_video_trim/_markers.py +140 -0
  45. figrecipe/_dev/demo_plotters/__init__.py +35 -166
  46. figrecipe/_dev/demo_plotters/_categories.py +81 -0
  47. figrecipe/_dev/demo_plotters/_figure_creators.py +119 -0
  48. figrecipe/_dev/demo_plotters/_helpers.py +31 -0
  49. figrecipe/_dev/demo_plotters/_registry.py +50 -0
  50. figrecipe/_dev/demo_plotters/bar_categorical/__init__.py +4 -0
  51. figrecipe/_dev/demo_plotters/contour_surface/__init__.py +4 -0
  52. figrecipe/_dev/demo_plotters/distribution/__init__.py +4 -0
  53. figrecipe/_dev/demo_plotters/image_matrix/__init__.py +4 -0
  54. figrecipe/_dev/demo_plotters/line_curve/__init__.py +4 -0
  55. figrecipe/_dev/demo_plotters/{plot_plot.py → line_curve/plot_plot.py} +3 -2
  56. figrecipe/_dev/demo_plotters/scatter_points/__init__.py +4 -0
  57. figrecipe/_dev/demo_plotters/special/__init__.py +4 -0
  58. figrecipe/_dev/demo_plotters/{plot_pie.py → special/plot_pie.py} +5 -1
  59. figrecipe/_dev/demo_plotters/spectral_signal/__init__.py +4 -0
  60. figrecipe/_dev/demo_plotters/vector_flow/__init__.py +4 -0
  61. figrecipe/_editor/__init__.py +61 -13
  62. figrecipe/_editor/_bbox/__init__.py +43 -0
  63. figrecipe/_editor/_bbox/_collections.py +177 -0
  64. figrecipe/_editor/_bbox/_elements.py +159 -0
  65. figrecipe/_editor/_bbox/_extract.py +402 -0
  66. figrecipe/_editor/_bbox/_extract_axes.py +370 -0
  67. figrecipe/_editor/_bbox/_extract_text.py +466 -0
  68. figrecipe/_editor/_bbox/_lines.py +173 -0
  69. figrecipe/_editor/_bbox/_transforms.py +146 -0
  70. figrecipe/_editor/_call_overrides.py +183 -0
  71. figrecipe/_editor/_datatable_plot_handlers.py +249 -0
  72. figrecipe/_editor/_figure_layout.py +211 -0
  73. figrecipe/_editor/_flask_app.py +200 -1030
  74. figrecipe/_editor/_helpers.py +251 -0
  75. figrecipe/_editor/_hitmap/__init__.py +76 -0
  76. figrecipe/_editor/_hitmap/_artists/__init__.py +21 -0
  77. figrecipe/_editor/_hitmap/_artists/_collections.py +345 -0
  78. figrecipe/_editor/_hitmap/_artists/_images.py +68 -0
  79. figrecipe/_editor/_hitmap/_artists/_lines.py +107 -0
  80. figrecipe/_editor/_hitmap/_artists/_patches.py +163 -0
  81. figrecipe/_editor/_hitmap/_artists/_text.py +190 -0
  82. figrecipe/_editor/_hitmap/_colors.py +181 -0
  83. figrecipe/_editor/_hitmap/_detect.py +194 -0
  84. figrecipe/_editor/_hitmap/_restore.py +154 -0
  85. figrecipe/_editor/_hitmap_main.py +182 -0
  86. figrecipe/_editor/_overrides.py +4 -1
  87. figrecipe/_editor/_plot_types_registry.py +190 -0
  88. figrecipe/_editor/_preferences.py +135 -0
  89. figrecipe/_editor/_render_overrides.py +507 -0
  90. figrecipe/_editor/_renderer.py +81 -186
  91. figrecipe/_editor/_routes_annotation.py +114 -0
  92. figrecipe/_editor/_routes_axis.py +482 -0
  93. figrecipe/_editor/_routes_captions.py +130 -0
  94. figrecipe/_editor/_routes_composition.py +270 -0
  95. figrecipe/_editor/_routes_core.py +126 -0
  96. figrecipe/_editor/_routes_datatable.py +364 -0
  97. figrecipe/_editor/_routes_element.py +335 -0
  98. figrecipe/_editor/_routes_files.py +443 -0
  99. figrecipe/_editor/_routes_image.py +200 -0
  100. figrecipe/_editor/_routes_snapshot.py +94 -0
  101. figrecipe/_editor/_routes_style.py +243 -0
  102. figrecipe/_editor/_templates/__init__.py +116 -1
  103. figrecipe/_editor/_templates/_html.py +154 -64
  104. figrecipe/_editor/_templates/_html_components/__init__.py +13 -0
  105. figrecipe/_editor/_templates/_html_components/_composition_toolbar.py +79 -0
  106. figrecipe/_editor/_templates/_html_components/_file_browser.py +41 -0
  107. figrecipe/_editor/_templates/_html_datatable.py +92 -0
  108. figrecipe/_editor/_templates/_scripts/__init__.py +178 -0
  109. figrecipe/_editor/_templates/_scripts/_accordion.py +328 -0
  110. figrecipe/_editor/_templates/_scripts/_annotation_drag.py +504 -0
  111. figrecipe/_editor/_templates/_scripts/_api.py +228 -0
  112. figrecipe/_editor/_templates/_scripts/_canvas_context_menu.py +182 -0
  113. figrecipe/_editor/_templates/_scripts/_captions.py +231 -0
  114. figrecipe/_editor/_templates/_scripts/_colors.py +485 -0
  115. figrecipe/_editor/_templates/_scripts/_composition.py +283 -0
  116. figrecipe/_editor/_templates/_scripts/_core.py +493 -0
  117. figrecipe/_editor/_templates/_scripts/_datatable/__init__.py +59 -0
  118. figrecipe/_editor/_templates/_scripts/_datatable/_cell_edit.py +97 -0
  119. figrecipe/_editor/_templates/_scripts/_datatable/_clipboard.py +164 -0
  120. figrecipe/_editor/_templates/_scripts/_datatable/_context_menu.py +221 -0
  121. figrecipe/_editor/_templates/_scripts/_datatable/_core.py +150 -0
  122. figrecipe/_editor/_templates/_scripts/_datatable/_editable.py +511 -0
  123. figrecipe/_editor/_templates/_scripts/_datatable/_import.py +161 -0
  124. figrecipe/_editor/_templates/_scripts/_datatable/_plot.py +261 -0
  125. figrecipe/_editor/_templates/_scripts/_datatable/_selection.py +438 -0
  126. figrecipe/_editor/_templates/_scripts/_datatable/_table.py +256 -0
  127. figrecipe/_editor/_templates/_scripts/_datatable/_tabs.py +354 -0
  128. figrecipe/_editor/_templates/_scripts/_debug_snapshot.py +186 -0
  129. figrecipe/_editor/_templates/_scripts/_element_editor.py +325 -0
  130. figrecipe/_editor/_templates/_scripts/_files.py +429 -0
  131. figrecipe/_editor/_templates/_scripts/_files_context_menu.py +240 -0
  132. figrecipe/_editor/_templates/_scripts/_hitmap.py +512 -0
  133. figrecipe/_editor/_templates/_scripts/_image_drop.py +428 -0
  134. figrecipe/_editor/_templates/_scripts/_inspector.py +315 -0
  135. figrecipe/_editor/_templates/_scripts/_labels.py +464 -0
  136. figrecipe/_editor/_templates/_scripts/_legend_drag.py +270 -0
  137. figrecipe/_editor/_templates/_scripts/_modals.py +226 -0
  138. figrecipe/_editor/_templates/_scripts/_multi_select.py +198 -0
  139. figrecipe/_editor/_templates/_scripts/_overlays.py +292 -0
  140. figrecipe/_editor/_templates/_scripts/_panel_drag.py +505 -0
  141. figrecipe/_editor/_templates/_scripts/_panel_drag_snapshot.py +33 -0
  142. figrecipe/_editor/_templates/_scripts/_panel_position.py +463 -0
  143. figrecipe/_editor/_templates/_scripts/_panel_resize.py +230 -0
  144. figrecipe/_editor/_templates/_scripts/_panel_snap.py +307 -0
  145. figrecipe/_editor/_templates/_scripts/_region_select.py +255 -0
  146. figrecipe/_editor/_templates/_scripts/_selection.py +244 -0
  147. figrecipe/_editor/_templates/_scripts/_sync.py +242 -0
  148. figrecipe/_editor/_templates/_scripts/_tabs.py +89 -0
  149. figrecipe/_editor/_templates/_scripts/_undo_redo.py +348 -0
  150. figrecipe/_editor/_templates/_scripts/_view_mode.py +107 -0
  151. figrecipe/_editor/_templates/_scripts/_zoom.py +212 -0
  152. figrecipe/_editor/_templates/_styles/__init__.py +78 -0
  153. figrecipe/_editor/_templates/_styles/_base.py +111 -0
  154. figrecipe/_editor/_templates/_styles/_buttons.py +327 -0
  155. figrecipe/_editor/_templates/_styles/_color_input.py +123 -0
  156. figrecipe/_editor/_templates/_styles/_composition.py +87 -0
  157. figrecipe/_editor/_templates/_styles/_controls.py +430 -0
  158. figrecipe/_editor/_templates/_styles/_datatable/__init__.py +40 -0
  159. figrecipe/_editor/_templates/_styles/_datatable/_editable.py +203 -0
  160. figrecipe/_editor/_templates/_styles/_datatable/_panel.py +268 -0
  161. figrecipe/_editor/_templates/_styles/_datatable/_table.py +479 -0
  162. figrecipe/_editor/_templates/_styles/_datatable/_toolbar.py +384 -0
  163. figrecipe/_editor/_templates/_styles/_datatable/_vars.py +123 -0
  164. figrecipe/_editor/_templates/_styles/_dynamic_props.py +144 -0
  165. figrecipe/_editor/_templates/_styles/_file_browser.py +466 -0
  166. figrecipe/_editor/_templates/_styles/_forms.py +224 -0
  167. figrecipe/_editor/_templates/_styles/_hitmap.py +191 -0
  168. figrecipe/_editor/_templates/_styles/_inspector.py +90 -0
  169. figrecipe/_editor/_templates/_styles/_labels.py +118 -0
  170. figrecipe/_editor/_templates/_styles/_modals.py +127 -0
  171. figrecipe/_editor/_templates/_styles/_overlays.py +130 -0
  172. figrecipe/_editor/_templates/_styles/_preview.py +430 -0
  173. figrecipe/_editor/_templates/_styles/_selection.py +73 -0
  174. figrecipe/_editor/_templates/_styles/_spinner.py +117 -0
  175. figrecipe/_editor/static/audio/click.mp3 +0 -0
  176. figrecipe/_editor/static/click.mp3 +0 -0
  177. figrecipe/_editor/static/icons/favicon.ico +0 -0
  178. figrecipe/_integrations/__init__.py +17 -0
  179. figrecipe/_integrations/_scitex_stats.py +298 -0
  180. figrecipe/_params/_DECORATION_METHODS.py +8 -0
  181. figrecipe/_recorder.py +63 -109
  182. figrecipe/_recorder_utils.py +124 -0
  183. figrecipe/_reproducer/__init__.py +18 -0
  184. figrecipe/_reproducer/_core.py +509 -0
  185. figrecipe/_reproducer/_custom_plots.py +279 -0
  186. figrecipe/_reproducer/_seaborn.py +100 -0
  187. figrecipe/_reproducer/_violin.py +186 -0
  188. figrecipe/_signatures/_kwargs.py +273 -0
  189. figrecipe/_signatures/_loader.py +21 -423
  190. figrecipe/_signatures/_parsing.py +147 -0
  191. figrecipe/_utils/__init__.py +3 -0
  192. figrecipe/_utils/_bundle.py +205 -0
  193. figrecipe/_wrappers/_axes.py +252 -895
  194. figrecipe/_wrappers/_axes_helpers.py +136 -0
  195. figrecipe/_wrappers/_axes_plots.py +418 -0
  196. figrecipe/_wrappers/_axes_seaborn.py +157 -0
  197. figrecipe/_wrappers/_caption_generator.py +218 -0
  198. figrecipe/_wrappers/_figure.py +188 -1
  199. figrecipe/_wrappers/_panel_labels.py +127 -0
  200. figrecipe/_wrappers/_plot_helpers.py +143 -0
  201. figrecipe/_wrappers/_stat_annotation.py +274 -0
  202. figrecipe/_wrappers/_violin_helpers.py +180 -0
  203. figrecipe/styles/__init__.py +8 -6
  204. figrecipe/styles/_dotdict.py +72 -0
  205. figrecipe/styles/_finalize.py +134 -0
  206. figrecipe/styles/_fonts.py +77 -0
  207. figrecipe/styles/_kwargs_converter.py +178 -0
  208. figrecipe/styles/_plot_styles.py +209 -0
  209. figrecipe/styles/_style_applier.py +42 -480
  210. figrecipe/styles/_style_loader.py +16 -192
  211. figrecipe/styles/_themes.py +151 -0
  212. figrecipe/styles/presets/MATPLOTLIB.yaml +2 -1
  213. figrecipe/styles/presets/SCITEX.yaml +40 -28
  214. figrecipe-0.9.0.dist-info/METADATA +427 -0
  215. figrecipe-0.9.0.dist-info/RECORD +277 -0
  216. figrecipe-0.9.0.dist-info/entry_points.txt +2 -0
  217. figrecipe/_editor/_bbox.py +0 -978
  218. figrecipe/_editor/_hitmap.py +0 -937
  219. figrecipe/_editor/_templates/_scripts.py +0 -2778
  220. figrecipe/_editor/_templates/_styles.py +0 -1326
  221. figrecipe/_reproducer.py +0 -975
  222. figrecipe-0.6.0.dist-info/METADATA +0 -394
  223. figrecipe-0.6.0.dist-info/RECORD +0 -90
  224. /figrecipe/_dev/demo_plotters/{plot_bar.py → bar_categorical/plot_bar.py} +0 -0
  225. /figrecipe/_dev/demo_plotters/{plot_barh.py → bar_categorical/plot_barh.py} +0 -0
  226. /figrecipe/_dev/demo_plotters/{plot_contour.py → contour_surface/plot_contour.py} +0 -0
  227. /figrecipe/_dev/demo_plotters/{plot_contourf.py → contour_surface/plot_contourf.py} +0 -0
  228. /figrecipe/_dev/demo_plotters/{plot_tricontour.py → contour_surface/plot_tricontour.py} +0 -0
  229. /figrecipe/_dev/demo_plotters/{plot_tricontourf.py → contour_surface/plot_tricontourf.py} +0 -0
  230. /figrecipe/_dev/demo_plotters/{plot_tripcolor.py → contour_surface/plot_tripcolor.py} +0 -0
  231. /figrecipe/_dev/demo_plotters/{plot_triplot.py → contour_surface/plot_triplot.py} +0 -0
  232. /figrecipe/_dev/demo_plotters/{plot_boxplot.py → distribution/plot_boxplot.py} +0 -0
  233. /figrecipe/_dev/demo_plotters/{plot_ecdf.py → distribution/plot_ecdf.py} +0 -0
  234. /figrecipe/_dev/demo_plotters/{plot_hist.py → distribution/plot_hist.py} +0 -0
  235. /figrecipe/_dev/demo_plotters/{plot_hist2d.py → distribution/plot_hist2d.py} +0 -0
  236. /figrecipe/_dev/demo_plotters/{plot_violinplot.py → distribution/plot_violinplot.py} +0 -0
  237. /figrecipe/_dev/demo_plotters/{plot_hexbin.py → image_matrix/plot_hexbin.py} +0 -0
  238. /figrecipe/_dev/demo_plotters/{plot_imshow.py → image_matrix/plot_imshow.py} +0 -0
  239. /figrecipe/_dev/demo_plotters/{plot_matshow.py → image_matrix/plot_matshow.py} +0 -0
  240. /figrecipe/_dev/demo_plotters/{plot_pcolor.py → image_matrix/plot_pcolor.py} +0 -0
  241. /figrecipe/_dev/demo_plotters/{plot_pcolormesh.py → image_matrix/plot_pcolormesh.py} +0 -0
  242. /figrecipe/_dev/demo_plotters/{plot_spy.py → image_matrix/plot_spy.py} +0 -0
  243. /figrecipe/_dev/demo_plotters/{plot_errorbar.py → line_curve/plot_errorbar.py} +0 -0
  244. /figrecipe/_dev/demo_plotters/{plot_fill.py → line_curve/plot_fill.py} +0 -0
  245. /figrecipe/_dev/demo_plotters/{plot_fill_between.py → line_curve/plot_fill_between.py} +0 -0
  246. /figrecipe/_dev/demo_plotters/{plot_fill_betweenx.py → line_curve/plot_fill_betweenx.py} +0 -0
  247. /figrecipe/_dev/demo_plotters/{plot_stackplot.py → line_curve/plot_stackplot.py} +0 -0
  248. /figrecipe/_dev/demo_plotters/{plot_stairs.py → line_curve/plot_stairs.py} +0 -0
  249. /figrecipe/_dev/demo_plotters/{plot_step.py → line_curve/plot_step.py} +0 -0
  250. /figrecipe/_dev/demo_plotters/{plot_scatter.py → scatter_points/plot_scatter.py} +0 -0
  251. /figrecipe/_dev/demo_plotters/{plot_eventplot.py → special/plot_eventplot.py} +0 -0
  252. /figrecipe/_dev/demo_plotters/{plot_loglog.py → special/plot_loglog.py} +0 -0
  253. /figrecipe/_dev/demo_plotters/{plot_semilogx.py → special/plot_semilogx.py} +0 -0
  254. /figrecipe/_dev/demo_plotters/{plot_semilogy.py → special/plot_semilogy.py} +0 -0
  255. /figrecipe/_dev/demo_plotters/{plot_stem.py → special/plot_stem.py} +0 -0
  256. /figrecipe/_dev/demo_plotters/{plot_acorr.py → spectral_signal/plot_acorr.py} +0 -0
  257. /figrecipe/_dev/demo_plotters/{plot_angle_spectrum.py → spectral_signal/plot_angle_spectrum.py} +0 -0
  258. /figrecipe/_dev/demo_plotters/{plot_cohere.py → spectral_signal/plot_cohere.py} +0 -0
  259. /figrecipe/_dev/demo_plotters/{plot_csd.py → spectral_signal/plot_csd.py} +0 -0
  260. /figrecipe/_dev/demo_plotters/{plot_magnitude_spectrum.py → spectral_signal/plot_magnitude_spectrum.py} +0 -0
  261. /figrecipe/_dev/demo_plotters/{plot_phase_spectrum.py → spectral_signal/plot_phase_spectrum.py} +0 -0
  262. /figrecipe/_dev/demo_plotters/{plot_psd.py → spectral_signal/plot_psd.py} +0 -0
  263. /figrecipe/_dev/demo_plotters/{plot_specgram.py → spectral_signal/plot_specgram.py} +0 -0
  264. /figrecipe/_dev/demo_plotters/{plot_xcorr.py → spectral_signal/plot_xcorr.py} +0 -0
  265. /figrecipe/_dev/demo_plotters/{plot_barbs.py → vector_flow/plot_barbs.py} +0 -0
  266. /figrecipe/_dev/demo_plotters/{plot_quiver.py → vector_flow/plot_quiver.py} +0 -0
  267. /figrecipe/_dev/demo_plotters/{plot_streamplot.py → vector_flow/plot_streamplot.py} +0 -0
  268. {figrecipe-0.6.0.dist-info → figrecipe-0.9.0.dist-info}/WHEEL +0 -0
  269. {figrecipe-0.6.0.dist-info → figrecipe-0.9.0.dist-info}/licenses/LICENSE +0 -0
@@ -0,0 +1,430 @@
1
+ #!/usr/bin/env python3
2
+ # -*- coding: utf-8 -*-
3
+ """Controls panel CSS styles for the figure editor.
4
+
5
+ This module contains CSS for:
6
+ - Controls panel layout
7
+ - Tab navigation
8
+ - Sections and form elements
9
+ - Field highlighting
10
+ """
11
+
12
+ STYLES_CONTROLS = """
13
+ /* Controls Panel */
14
+ .controls-panel {
15
+ width: 350px;
16
+ min-width: 280px;
17
+ max-width: 500px;
18
+ display: flex;
19
+ flex-direction: column;
20
+ background: var(--bg-primary);
21
+ overflow: hidden;
22
+ position: relative;
23
+ border-left: 1px solid var(--border-color);
24
+ transition: width 0.2s ease-out, min-width 0.2s ease-out;
25
+ order: 20; /* Always stay at far right */
26
+ }
27
+
28
+ .controls-panel.collapsed {
29
+ width: 42px;
30
+ min-width: 42px;
31
+ }
32
+
33
+ .controls-panel.collapsed .controls-sections,
34
+ .controls-panel.collapsed .style-info,
35
+ .controls-panel.collapsed .override-status,
36
+ .controls-panel.collapsed #theme-modal,
37
+ .controls-panel.collapsed #shortcuts-modal,
38
+ .controls-panel.collapsed .properties-resize {
39
+ display: none;
40
+ }
41
+
42
+ .controls-panel.collapsed .controls-header {
43
+ flex-direction: column;
44
+ padding: 10px 6px;
45
+ gap: 8px;
46
+ }
47
+
48
+ .controls-panel.collapsed .controls-header .header-title {
49
+ justify-content: center;
50
+ }
51
+
52
+ .controls-panel.collapsed .controls-header h2,
53
+ .controls-panel.collapsed .controls-header span {
54
+ display: none;
55
+ }
56
+
57
+ .controls-panel.collapsed .controls-actions {
58
+ flex-direction: column;
59
+ gap: 4px;
60
+ width: 100%;
61
+ }
62
+
63
+ .controls-panel.collapsed .controls-actions {
64
+ display: none;
65
+ }
66
+
67
+ /* Flip collapse button when collapsed (now points left to expand) */
68
+ .controls-panel.collapsed .header-title .btn-collapse {
69
+ transform: rotate(180deg);
70
+ }
71
+
72
+ /* Properties resize handle */
73
+ .properties-resize {
74
+ width: 4px;
75
+ cursor: col-resize;
76
+ background: transparent;
77
+ position: absolute;
78
+ left: 0;
79
+ top: 0;
80
+ bottom: 0;
81
+ z-index: 10;
82
+ transition: background 0.2s;
83
+ }
84
+
85
+ .properties-resize:hover,
86
+ .properties-resize.resizing {
87
+ background: var(--accent-color);
88
+ }
89
+
90
+ .controls-header {
91
+ display: flex;
92
+ justify-content: space-between;
93
+ align-items: center;
94
+ padding: 0 12px;
95
+ height: var(--panel-header-height);
96
+ min-height: var(--panel-header-height);
97
+ background: var(--panel-header-bg);
98
+ border-bottom: 1px solid var(--border-color);
99
+ }
100
+
101
+ .controls-header .header-title {
102
+ display: flex;
103
+ align-items: center;
104
+ gap: 8px;
105
+ }
106
+
107
+ .controls-header .header-title .btn-collapse {
108
+ width: 28px;
109
+ height: 28px;
110
+ padding: 0;
111
+ border: none;
112
+ background: transparent;
113
+ color: var(--text-secondary);
114
+ cursor: pointer;
115
+ font-size: 14px;
116
+ display: flex;
117
+ align-items: center;
118
+ justify-content: center;
119
+ border-radius: 4px;
120
+ transition: all 0.2s;
121
+ }
122
+
123
+ .controls-header .header-title .btn-collapse:hover {
124
+ background: var(--bg-tertiary);
125
+ color: var(--text-primary);
126
+ }
127
+
128
+ .controls-header h2 {
129
+ font-size: 16px;
130
+ font-weight: 600;
131
+ margin: 0;
132
+ }
133
+
134
+ /* Panel label matching FILES, DATA, CANVAS style */
135
+ .controls-header span {
136
+ font-size: 11px;
137
+ font-weight: 600;
138
+ margin: 0;
139
+ text-transform: uppercase;
140
+ letter-spacing: 0.5px;
141
+ color: var(--text-secondary);
142
+ }
143
+
144
+ .controls-actions {
145
+ display: flex;
146
+ gap: 8px;
147
+ align-items: center;
148
+ }
149
+
150
+ .controls-sections {
151
+ flex: 1;
152
+ overflow-y: auto;
153
+ padding: 8px 12px 8px 8px; /* Extra right padding for scrollbar clearance */
154
+ }
155
+
156
+ /* Tab Navigation */
157
+ .tab-navigation {
158
+ display: flex;
159
+ gap: 0;
160
+ background: var(--bg-secondary);
161
+ border-radius: 8px;
162
+ padding: 4px;
163
+ margin-bottom: 12px;
164
+ }
165
+
166
+ .tab-btn {
167
+ flex: 1;
168
+ padding: 10px 16px;
169
+ font-size: 13px;
170
+ font-weight: 500;
171
+ border: none;
172
+ background: transparent;
173
+ border-radius: 6px;
174
+ cursor: pointer;
175
+ transition: all 0.15s;
176
+ color: var(--text-secondary);
177
+ }
178
+
179
+ .tab-btn:hover {
180
+ background: var(--bg-tertiary);
181
+ color: var(--text-primary);
182
+ }
183
+
184
+ .tab-btn.active {
185
+ background: var(--accent-color);
186
+ color: white;
187
+ box-shadow: 0 2px 4px rgba(0, 0, 0, 0.1);
188
+ }
189
+
190
+ /* Tab Content */
191
+ .tab-content {
192
+ display: none;
193
+ }
194
+
195
+ .tab-content.active {
196
+ display: block;
197
+ }
198
+
199
+ .tab-hint {
200
+ text-align: center;
201
+ padding: 40px 20px;
202
+ color: var(--text-secondary);
203
+ background: var(--bg-secondary);
204
+ border-radius: 8px;
205
+ margin-bottom: 12px;
206
+ }
207
+
208
+ .tab-hint p {
209
+ margin: 0;
210
+ }
211
+
212
+ .tab-hint .hint-sub {
213
+ font-size: 12px;
214
+ margin-top: 8px;
215
+ opacity: 0.7;
216
+ }
217
+
218
+ /* Selected element panel in Element tab */
219
+ .selected-element-header {
220
+ display: flex;
221
+ align-items: center;
222
+ gap: 10px;
223
+ padding: 12px;
224
+ background: var(--bg-secondary);
225
+ border-radius: 8px;
226
+ margin-bottom: 12px;
227
+ }
228
+
229
+ .element-type-badge {
230
+ padding: 4px 10px;
231
+ font-size: 11px;
232
+ font-weight: 600;
233
+ text-transform: uppercase;
234
+ background: var(--accent-color);
235
+ color: white;
236
+ border-radius: 4px;
237
+ }
238
+
239
+ .element-name {
240
+ font-size: 14px;
241
+ font-weight: 500;
242
+ color: var(--text-primary);
243
+ }
244
+
245
+ /* Legacy toggle styles for backward compatibility */
246
+ .view-mode-toggle {
247
+ display: none; /* Hidden - replaced by tabs */
248
+ }
249
+
250
+ .btn-toggle {
251
+ padding: 6px 12px;
252
+ font-size: 12px;
253
+ font-weight: 500;
254
+ border: 1px solid var(--border-color);
255
+ background: var(--bg-primary);
256
+ border-radius: 4px;
257
+ cursor: pointer;
258
+ transition: all 0.15s;
259
+ }
260
+
261
+ .btn-toggle:hover {
262
+ background: var(--bg-tertiary);
263
+ }
264
+
265
+ .btn-toggle.active {
266
+ background: var(--accent-color);
267
+ border-color: var(--accent-color);
268
+ color: white;
269
+ }
270
+
271
+ .selection-hint {
272
+ margin-left: auto;
273
+ font-size: 11px;
274
+ color: var(--text-secondary);
275
+ font-style: italic;
276
+ }
277
+
278
+ /* Filtering mode: hide non-matching sections */
279
+ .controls-sections.filter-mode .section.section-hidden {
280
+ display: none;
281
+ }
282
+
283
+ .controls-sections.filter-mode .form-row.field-hidden {
284
+ display: none;
285
+ }
286
+
287
+ /* Show matching sections with highlight in filter mode */
288
+ .controls-sections.filter-mode .section.section-visible {
289
+ border-color: var(--accent-color);
290
+ }
291
+
292
+ .controls-sections.filter-mode .section.section-visible summary {
293
+ background: rgba(37, 99, 235, 0.08);
294
+ }
295
+
296
+ [data-theme="dark"] .controls-sections.filter-mode .section.section-visible summary {
297
+ background: rgba(59, 130, 246, 0.12);
298
+ }
299
+
300
+ /* Sections */
301
+ .section {
302
+ margin-bottom: 8px;
303
+ border: 1px solid var(--border-color);
304
+ border-radius: 6px;
305
+ overflow: hidden;
306
+ }
307
+
308
+ .section summary {
309
+ padding: 10px 14px;
310
+ background: var(--bg-secondary);
311
+ cursor: pointer;
312
+ font-weight: 500;
313
+ user-select: none;
314
+ list-style: none;
315
+ display: flex;
316
+ align-items: center;
317
+ }
318
+
319
+ .section summary::before {
320
+ content: '\\25B6';
321
+ font-size: 10px;
322
+ margin-right: 8px;
323
+ transition: transform 0.2s;
324
+ }
325
+
326
+ .section[open] summary::before {
327
+ transform: rotate(90deg);
328
+ }
329
+
330
+ .section-highlighted {
331
+ border-color: var(--accent-color);
332
+ box-shadow: 0 0 0 2px rgba(37, 99, 235, 0.2);
333
+ }
334
+
335
+ .section-highlighted summary {
336
+ background: rgba(37, 99, 235, 0.1);
337
+ border-left: 3px solid var(--accent-color);
338
+ }
339
+
340
+ [data-theme="dark"] .section-highlighted {
341
+ box-shadow: 0 0 0 2px rgba(59, 130, 246, 0.3);
342
+ }
343
+
344
+ [data-theme="dark"] .section-highlighted summary {
345
+ background: rgba(59, 130, 246, 0.15);
346
+ }
347
+
348
+ /* Field highlighting for selected element */
349
+ .field-highlighted {
350
+ background: rgba(37, 99, 235, 0.08);
351
+ border-radius: 4px;
352
+ padding: 6px 8px;
353
+ margin: -6px -8px;
354
+ margin-bottom: 8px;
355
+ border-left: 3px solid var(--accent-color);
356
+ }
357
+
358
+ .field-highlighted:last-child {
359
+ margin-bottom: 0;
360
+ }
361
+
362
+ [data-theme="dark"] .field-highlighted {
363
+ background: rgba(59, 130, 246, 0.12);
364
+ }
365
+
366
+ .section-content {
367
+ padding: 12px 14px;
368
+ background: var(--bg-primary);
369
+ }
370
+
371
+ /* Panel Indicator */
372
+ .panel-indicator-row {
373
+ align-items: center;
374
+ }
375
+
376
+ .panel-indicator {
377
+ padding: 6px 12px;
378
+ font-size: 13px;
379
+ font-weight: 500;
380
+ color: var(--text-secondary);
381
+ background: var(--bg-secondary);
382
+ border-radius: 6px;
383
+ border: 1px solid var(--border-color);
384
+ display: inline-block;
385
+ }
386
+
387
+ .panel-indicator.panel-selected {
388
+ color: var(--accent-color);
389
+ background: rgba(37, 99, 235, 0.1);
390
+ border-color: var(--accent-color);
391
+ font-weight: 600;
392
+ }
393
+
394
+ [data-theme="dark"] .panel-indicator.panel-selected {
395
+ background: rgba(59, 130, 246, 0.15);
396
+ }
397
+
398
+ /* Override Status - subtle indicator */
399
+ .override-status {
400
+ display: flex;
401
+ align-items: center;
402
+ gap: 6px;
403
+ padding: 4px 12px;
404
+ font-size: 11px;
405
+ color: var(--text-tertiary);
406
+ border-bottom: 1px solid var(--border-color);
407
+ }
408
+
409
+ .override-indicator {
410
+ display: flex;
411
+ align-items: center;
412
+ gap: 4px;
413
+ }
414
+
415
+ .override-indicator::before {
416
+ content: '';
417
+ width: 6px;
418
+ height: 6px;
419
+ background: var(--accent-color);
420
+ border-radius: 50%;
421
+ }
422
+
423
+ .override-timestamp {
424
+ opacity: 0.6;
425
+ }
426
+ """
427
+
428
+ __all__ = ["STYLES_CONTROLS"]
429
+
430
+ # EOF
@@ -0,0 +1,40 @@
1
+ #!/usr/bin/env python3
2
+ # -*- coding: utf-8 -*-
3
+ """Datatable CSS styles - orchestrator.
4
+
5
+ Combines all datatable CSS modules:
6
+ - _panel.py: Panel layout, header, toggle
7
+ - _toolbar.py: Toolbar, import, hints
8
+ - _table.py: Spreadsheet table
9
+ - _vars.py: Variable assignment with color linking
10
+ - _editable.py: Editable table styles
11
+ """
12
+
13
+ from ._editable import CSS_DATATABLE_EDITABLE
14
+ from ._panel import CSS_DATATABLE_PANEL
15
+ from ._table import CSS_DATATABLE_TABLE
16
+ from ._toolbar import CSS_DATATABLE_TOOLBAR
17
+ from ._vars import CSS_DATATABLE_VARS
18
+
19
+
20
+ def get_styles_datatable() -> str:
21
+ """Generate combined datatable CSS."""
22
+ return (
23
+ CSS_DATATABLE_PANEL
24
+ + "\n"
25
+ + CSS_DATATABLE_TOOLBAR
26
+ + "\n"
27
+ + CSS_DATATABLE_TABLE
28
+ + "\n"
29
+ + CSS_DATATABLE_VARS
30
+ + "\n"
31
+ + CSS_DATATABLE_EDITABLE
32
+ )
33
+
34
+
35
+ # For backward compatibility
36
+ STYLES_DATATABLE = get_styles_datatable()
37
+
38
+ __all__ = ["STYLES_DATATABLE", "get_styles_datatable"]
39
+
40
+ # EOF
@@ -0,0 +1,203 @@
1
+ #!/usr/bin/env python3
2
+ # -*- coding: utf-8 -*-
3
+ """CSS for editable datatable functionality."""
4
+
5
+ CSS_DATATABLE_EDITABLE = """
6
+ /* ============================================================================
7
+ Create New Table Button
8
+ ============================================================================ */
9
+ .datatable-create-new {
10
+ text-align: center;
11
+ padding: 8px;
12
+ border-top: 1px solid var(--dt-border);
13
+ }
14
+
15
+ .btn-create-csv {
16
+ background: var(--dt-accent);
17
+ color: white;
18
+ border: none;
19
+ padding: 8px 16px;
20
+ border-radius: 4px;
21
+ cursor: pointer;
22
+ font-size: 13px;
23
+ transition: background 0.2s;
24
+ }
25
+
26
+ .btn-create-csv:hover {
27
+ background: var(--dt-accent-hover);
28
+ }
29
+
30
+ /* ============================================================================
31
+ Editable Table Styles
32
+ ============================================================================ */
33
+ .editable-table-wrapper {
34
+ display: flex;
35
+ flex-direction: column;
36
+ flex: 1;
37
+ min-height: 0; /* Allow flex shrinking */
38
+ }
39
+
40
+ .editable-table-actions {
41
+ display: flex;
42
+ gap: 4px;
43
+ padding: 4px;
44
+ background: var(--dt-header-bg);
45
+ border-bottom: 1px solid var(--dt-border);
46
+ flex-shrink: 0;
47
+ }
48
+
49
+ .editable-table-actions .btn-small {
50
+ padding: 2px 8px;
51
+ font-size: 11px;
52
+ background: var(--dt-bg);
53
+ border: 1px solid var(--dt-border);
54
+ border-radius: 3px;
55
+ cursor: pointer;
56
+ color: var(--dt-text);
57
+ }
58
+
59
+ .editable-table-actions .btn-small:hover {
60
+ background: var(--dt-row-hover);
61
+ }
62
+
63
+ .editable-table-actions .btn-danger {
64
+ color: #dc3545;
65
+ }
66
+
67
+ .editable-table-actions .btn-danger:hover {
68
+ background: rgba(220, 53, 69, 0.1);
69
+ }
70
+
71
+ /* ============================================================================
72
+ Large Table Performance (inspired by vis_app patterns)
73
+ ============================================================================ */
74
+ /* Table container with scrolling */
75
+ .editable-table-scroll {
76
+ overflow: auto;
77
+ flex: 1;
78
+ min-height: 0; /* Allow flex shrinking */
79
+ }
80
+
81
+ /* Fixed row height for predictable scrolling */
82
+ .datatable-table.editable {
83
+ border-collapse: separate;
84
+ border-spacing: 0;
85
+ }
86
+
87
+ .datatable-table.editable tr {
88
+ height: 28px;
89
+ }
90
+
91
+ /* Sticky header row */
92
+ .datatable-table.editable thead th {
93
+ position: sticky;
94
+ top: 0;
95
+ z-index: 10;
96
+ background: var(--dt-header-bg);
97
+ }
98
+
99
+ /* Sticky row numbers column */
100
+ .datatable-table.editable .row-num {
101
+ position: sticky;
102
+ left: 0;
103
+ z-index: 5;
104
+ background: var(--dt-header-bg);
105
+ }
106
+
107
+ /* Corner cell (row # header) - highest z-index */
108
+ .datatable-table.editable thead th.row-num {
109
+ z-index: 11;
110
+ }
111
+
112
+ /* Editable table specific styles */
113
+ .datatable-table.editable th {
114
+ padding: 2px;
115
+ }
116
+
117
+ .editable-header {
118
+ display: flex;
119
+ align-items: center;
120
+ gap: 2px;
121
+ }
122
+
123
+ .col-name-input {
124
+ flex: 1;
125
+ min-width: 40px;
126
+ max-width: 80px;
127
+ padding: 2px 4px;
128
+ font-size: 11px;
129
+ border: 1px solid transparent;
130
+ background: transparent;
131
+ color: var(--dt-text);
132
+ border-radius: 2px;
133
+ }
134
+
135
+ .col-name-input:hover,
136
+ .col-name-input:focus {
137
+ border-color: var(--dt-accent);
138
+ background: var(--dt-bg);
139
+ outline: none;
140
+ }
141
+
142
+ .col-type-select {
143
+ width: 28px;
144
+ padding: 1px;
145
+ font-size: 10px;
146
+ border: 1px solid var(--dt-border);
147
+ background: var(--dt-bg);
148
+ color: var(--dt-text-muted);
149
+ border-radius: 2px;
150
+ cursor: pointer;
151
+ }
152
+
153
+ .editable-cell {
154
+ padding: 0 !important;
155
+ }
156
+
157
+ .editable-cell input {
158
+ width: 100%;
159
+ padding: 4px 6px;
160
+ font-size: 12px;
161
+ border: none;
162
+ background: transparent;
163
+ color: var(--dt-text);
164
+ box-sizing: border-box;
165
+ }
166
+
167
+ .editable-cell input:hover {
168
+ background: var(--dt-row-hover);
169
+ }
170
+
171
+ .editable-cell input:focus {
172
+ outline: 2px solid var(--dt-accent);
173
+ outline-offset: -2px;
174
+ background: var(--dt-bg);
175
+ }
176
+
177
+ .editable-cell input[type="number"] {
178
+ text-align: right;
179
+ -moz-appearance: textfield; /* Firefox */
180
+ }
181
+
182
+ /* Hide number spinbuttons completely */
183
+ .editable-cell input[type="number"]::-webkit-inner-spin-button,
184
+ .editable-cell input[type="number"]::-webkit-outer-spin-button {
185
+ -webkit-appearance: none;
186
+ margin: 0;
187
+ display: none;
188
+ }
189
+
190
+ /* ============================================================================
191
+ Element Color Indicator on Column Headers
192
+ ============================================================================ */
193
+ .datatable-table.editable th.has-element-color {
194
+ border-left-width: 3px;
195
+ border-left-style: solid;
196
+ /* Color is set via inline style */
197
+ position: relative;
198
+ }
199
+ """
200
+
201
+ __all__ = ["CSS_DATATABLE_EDITABLE"]
202
+
203
+ # EOF