scitex 2.7.3__py3-none-any.whl → 2.8.1__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 (160) hide show
  1. scitex/__version__.py +1 -1
  2. scitex/dev/plt/__init__.py +0 -0
  3. scitex/dev/plt/plot_mpl_axhline.py +0 -0
  4. scitex/dev/plt/plot_mpl_axhspan.py +0 -0
  5. scitex/dev/plt/plot_mpl_axvline.py +0 -0
  6. scitex/dev/plt/plot_mpl_axvspan.py +0 -0
  7. scitex/dev/plt/plot_mpl_bar.py +0 -0
  8. scitex/dev/plt/plot_mpl_barh.py +0 -0
  9. scitex/dev/plt/plot_mpl_boxplot.py +0 -0
  10. scitex/dev/plt/plot_mpl_contour.py +0 -0
  11. scitex/dev/plt/plot_mpl_contourf.py +0 -0
  12. scitex/dev/plt/plot_mpl_errorbar.py +0 -0
  13. scitex/dev/plt/plot_mpl_eventplot.py +0 -0
  14. scitex/dev/plt/plot_mpl_fill.py +0 -0
  15. scitex/dev/plt/plot_mpl_fill_between.py +0 -0
  16. scitex/dev/plt/plot_mpl_hexbin.py +0 -0
  17. scitex/dev/plt/plot_mpl_hist.py +0 -0
  18. scitex/dev/plt/plot_mpl_hist2d.py +0 -0
  19. scitex/dev/plt/plot_mpl_imshow.py +0 -0
  20. scitex/dev/plt/plot_mpl_pcolormesh.py +0 -0
  21. scitex/dev/plt/plot_mpl_pie.py +0 -0
  22. scitex/dev/plt/plot_mpl_plot.py +0 -0
  23. scitex/dev/plt/plot_mpl_quiver.py +0 -0
  24. scitex/dev/plt/plot_mpl_scatter.py +0 -0
  25. scitex/dev/plt/plot_mpl_stackplot.py +0 -0
  26. scitex/dev/plt/plot_mpl_stem.py +0 -0
  27. scitex/dev/plt/plot_mpl_step.py +0 -0
  28. scitex/dev/plt/plot_mpl_violinplot.py +0 -0
  29. scitex/dev/plt/plot_sns_barplot.py +0 -0
  30. scitex/dev/plt/plot_sns_boxplot.py +0 -0
  31. scitex/dev/plt/plot_sns_heatmap.py +0 -0
  32. scitex/dev/plt/plot_sns_histplot.py +0 -0
  33. scitex/dev/plt/plot_sns_kdeplot.py +0 -0
  34. scitex/dev/plt/plot_sns_lineplot.py +0 -0
  35. scitex/dev/plt/plot_sns_scatterplot.py +0 -0
  36. scitex/dev/plt/plot_sns_stripplot.py +0 -0
  37. scitex/dev/plt/plot_sns_swarmplot.py +0 -0
  38. scitex/dev/plt/plot_sns_violinplot.py +0 -0
  39. scitex/dev/plt/plot_stx_bar.py +0 -0
  40. scitex/dev/plt/plot_stx_barh.py +0 -0
  41. scitex/dev/plt/plot_stx_box.py +0 -0
  42. scitex/dev/plt/plot_stx_boxplot.py +0 -0
  43. scitex/dev/plt/plot_stx_conf_mat.py +0 -0
  44. scitex/dev/plt/plot_stx_contour.py +0 -0
  45. scitex/dev/plt/plot_stx_ecdf.py +0 -0
  46. scitex/dev/plt/plot_stx_errorbar.py +0 -0
  47. scitex/dev/plt/plot_stx_fill_between.py +0 -0
  48. scitex/dev/plt/plot_stx_fillv.py +0 -0
  49. scitex/dev/plt/plot_stx_heatmap.py +0 -0
  50. scitex/dev/plt/plot_stx_image.py +0 -0
  51. scitex/dev/plt/plot_stx_imshow.py +0 -0
  52. scitex/dev/plt/plot_stx_joyplot.py +0 -0
  53. scitex/dev/plt/plot_stx_kde.py +0 -0
  54. scitex/dev/plt/plot_stx_line.py +0 -0
  55. scitex/dev/plt/plot_stx_mean_ci.py +0 -0
  56. scitex/dev/plt/plot_stx_mean_std.py +0 -0
  57. scitex/dev/plt/plot_stx_median_iqr.py +0 -0
  58. scitex/dev/plt/plot_stx_raster.py +0 -0
  59. scitex/dev/plt/plot_stx_rectangle.py +0 -0
  60. scitex/dev/plt/plot_stx_scatter.py +0 -0
  61. scitex/dev/plt/plot_stx_shaded_line.py +0 -0
  62. scitex/dev/plt/plot_stx_violin.py +0 -0
  63. scitex/dev/plt/plot_stx_violinplot.py +0 -0
  64. scitex/diagram/README.md +197 -0
  65. scitex/diagram/__init__.py +48 -0
  66. scitex/diagram/_compile.py +312 -0
  67. scitex/diagram/_diagram.py +355 -0
  68. scitex/diagram/_presets.py +173 -0
  69. scitex/diagram/_schema.py +182 -0
  70. scitex/diagram/_split.py +278 -0
  71. scitex/fig/editor/__init__.py +5 -2
  72. scitex/fig/editor/_dearpygui_editor.py +1 -1
  73. scitex/fig/editor/_mpl_editor.py +1 -1
  74. scitex/fig/editor/_qt_editor.py +1 -1
  75. scitex/fig/editor/_tkinter_editor.py +1 -1
  76. scitex/fig/editor/edit/__init__.py +50 -0
  77. scitex/fig/editor/edit/backend_detector.py +109 -0
  78. scitex/fig/editor/edit/bundle_resolver.py +240 -0
  79. scitex/fig/editor/edit/editor_launcher.py +239 -0
  80. scitex/fig/editor/edit/manual_handler.py +53 -0
  81. scitex/fig/editor/edit/panel_loader.py +232 -0
  82. scitex/fig/editor/edit/path_resolver.py +67 -0
  83. scitex/fig/editor/flask_editor/_bbox.py +23 -0
  84. scitex/fig/editor/flask_editor/_core.py +908 -103
  85. scitex/fig/editor/flask_editor/_renderer.py +74 -0
  86. scitex/fig/editor/flask_editor/static/css/base/reset.css +41 -0
  87. scitex/fig/editor/flask_editor/static/css/base/typography.css +16 -0
  88. scitex/fig/editor/flask_editor/static/css/base/variables.css +85 -0
  89. scitex/fig/editor/flask_editor/static/css/components/buttons.css +217 -0
  90. scitex/fig/editor/flask_editor/static/css/components/context-menu.css +93 -0
  91. scitex/fig/editor/flask_editor/static/css/components/dropdown.css +57 -0
  92. scitex/fig/editor/flask_editor/static/css/components/forms.css +112 -0
  93. scitex/fig/editor/flask_editor/static/css/components/modal.css +59 -0
  94. scitex/fig/editor/flask_editor/static/css/components/sections.css +212 -0
  95. scitex/fig/editor/flask_editor/static/css/features/canvas.css +176 -0
  96. scitex/fig/editor/flask_editor/static/css/features/element-inspector.css +190 -0
  97. scitex/fig/editor/flask_editor/static/css/features/loading.css +59 -0
  98. scitex/fig/editor/flask_editor/static/css/features/overlay.css +45 -0
  99. scitex/fig/editor/flask_editor/static/css/features/panel-grid.css +95 -0
  100. scitex/fig/editor/flask_editor/static/css/features/selection.css +101 -0
  101. scitex/fig/editor/flask_editor/static/css/features/statistics.css +138 -0
  102. scitex/fig/editor/flask_editor/static/css/index.css +31 -0
  103. scitex/fig/editor/flask_editor/static/css/layout/container.css +7 -0
  104. scitex/fig/editor/flask_editor/static/css/layout/controls.css +56 -0
  105. scitex/fig/editor/flask_editor/static/css/layout/preview.css +78 -0
  106. scitex/fig/editor/flask_editor/static/js/alignment/axis.js +314 -0
  107. scitex/fig/editor/flask_editor/static/js/alignment/basic.js +107 -0
  108. scitex/fig/editor/flask_editor/static/js/alignment/distribute.js +54 -0
  109. scitex/fig/editor/flask_editor/static/js/canvas/canvas.js +172 -0
  110. scitex/fig/editor/flask_editor/static/js/canvas/dragging.js +258 -0
  111. scitex/fig/editor/flask_editor/static/js/canvas/resize.js +48 -0
  112. scitex/fig/editor/flask_editor/static/js/canvas/selection.js +71 -0
  113. scitex/fig/editor/flask_editor/static/js/core/api.js +288 -0
  114. scitex/fig/editor/flask_editor/static/js/core/state.js +143 -0
  115. scitex/fig/editor/flask_editor/static/js/core/utils.js +245 -0
  116. scitex/fig/editor/flask_editor/static/js/dev/element-inspector.js +992 -0
  117. scitex/fig/editor/flask_editor/static/js/editor/bbox.js +339 -0
  118. scitex/fig/editor/flask_editor/static/js/editor/element-drag.js +286 -0
  119. scitex/fig/editor/flask_editor/static/js/editor/overlay.js +371 -0
  120. scitex/fig/editor/flask_editor/static/js/editor/preview.js +293 -0
  121. scitex/fig/editor/flask_editor/static/js/main.js +426 -0
  122. scitex/fig/editor/flask_editor/static/js/shortcuts/context-menu.js +152 -0
  123. scitex/fig/editor/flask_editor/static/js/shortcuts/keyboard.js +265 -0
  124. scitex/fig/editor/flask_editor/static/js/ui/controls.js +184 -0
  125. scitex/fig/editor/flask_editor/static/js/ui/download.js +57 -0
  126. scitex/fig/editor/flask_editor/static/js/ui/help.js +100 -0
  127. scitex/fig/editor/flask_editor/static/js/ui/theme.js +34 -0
  128. scitex/fig/editor/flask_editor/templates/__init__.py +95 -5
  129. scitex/fig/editor/flask_editor/templates/_html.py +27 -9
  130. scitex/fig/editor/flask_editor/templates/_scripts.py +1928 -131
  131. scitex/fig/editor/flask_editor/templates/_styles.py +363 -51
  132. scitex/fig/io/_bundle.py +97 -12
  133. scitex/io/__init__.py +12 -0
  134. scitex/io/_bundle.py +69 -10
  135. scitex/io/_zip_bundle.py +439 -0
  136. scitex/plt/_subplots/_AxisWrapperMixins/_AdjustmentMixin/__init__.py +0 -0
  137. scitex/plt/_subplots/_AxisWrapperMixins/_AdjustmentMixin/_labels.py +0 -0
  138. scitex/plt/_subplots/_AxisWrapperMixins/_AdjustmentMixin/_metadata.py +0 -0
  139. scitex/plt/_subplots/_AxisWrapperMixins/_AdjustmentMixin/_visual.py +0 -0
  140. scitex/plt/_subplots/_AxisWrapperMixins/_MatplotlibPlotMixin/__init__.py +0 -0
  141. scitex/plt/_subplots/_AxisWrapperMixins/_MatplotlibPlotMixin/_base.py +0 -0
  142. scitex/plt/_subplots/_AxisWrapperMixins/_MatplotlibPlotMixin/_scientific.py +0 -0
  143. scitex/plt/_subplots/_AxisWrapperMixins/_MatplotlibPlotMixin/_statistical.py +0 -0
  144. scitex/plt/_subplots/_AxisWrapperMixins/_MatplotlibPlotMixin/_stx_aliases.py +0 -0
  145. scitex/plt/_subplots/_AxisWrapperMixins/_RawMatplotlibMixin.py +0 -0
  146. scitex/plt/_subplots/_AxisWrapperMixins/_SeabornMixin/__init__.py +0 -0
  147. scitex/plt/_subplots/_AxisWrapperMixins/_SeabornMixin/_base.py +0 -0
  148. scitex/plt/_subplots/_AxisWrapperMixins/_SeabornMixin/_wrappers.py +0 -0
  149. scitex/plt/_subplots/_export_as_csv_formatters/_format_stx_bar.py +0 -0
  150. scitex/plt/_subplots/_export_as_csv_formatters/_format_stx_barh.py +0 -0
  151. scitex/plt/_subplots/_export_as_csv_formatters/_format_stx_errorbar.py +0 -0
  152. scitex/plt/_subplots/_export_as_csv_formatters/_format_stx_scatter.py +0 -0
  153. scitex/plt/io/_layered_bundle.py +0 -0
  154. scitex/schema/_plot.py +0 -0
  155. {scitex-2.7.3.dist-info → scitex-2.8.1.dist-info}/METADATA +1 -1
  156. {scitex-2.7.3.dist-info → scitex-2.8.1.dist-info}/RECORD +78 -22
  157. scitex/fig/editor/_edit.py +0 -751
  158. {scitex-2.7.3.dist-info → scitex-2.8.1.dist-info}/WHEEL +0 -0
  159. {scitex-2.7.3.dist-info → scitex-2.8.1.dist-info}/entry_points.txt +0 -0
  160. {scitex-2.7.3.dist-info → scitex-2.8.1.dist-info}/licenses/LICENSE +0 -0
@@ -695,6 +695,80 @@ def _apply_annotations(ax, o, axis_fontsize):
695
695
  )
696
696
 
697
697
 
698
+ def render_panel_preview(
699
+ panel_dir,
700
+ dark_mode: bool = False,
701
+ ) -> Tuple[Optional[str], Optional[Dict[str, Any]], Optional[Dict[str, int]]]:
702
+ """Render a panel from its pltz bundle directory with dark mode support.
703
+
704
+ Args:
705
+ panel_dir: Path to the .pltz.d panel directory
706
+ dark_mode: Whether to render with dark mode colors
707
+
708
+ Returns:
709
+ tuple: (base64_image_data, bboxes_dict, image_size) or (None, None, None) on error
710
+ """
711
+ from pathlib import Path
712
+ import json
713
+ import pandas as pd
714
+
715
+ panel_dir = Path(panel_dir)
716
+
717
+ try:
718
+ # Load spec.json
719
+ spec_path = panel_dir / "spec.json"
720
+ if not spec_path.exists():
721
+ # Try legacy format
722
+ for f in panel_dir.glob("*.json"):
723
+ if f.name != "style.json":
724
+ spec_path = f
725
+ break
726
+
727
+ if not spec_path.exists():
728
+ return None, None, None
729
+
730
+ with open(spec_path, "r") as f:
731
+ metadata = json.load(f)
732
+
733
+ # Load CSV data
734
+ csv_data = None
735
+ csv_path = panel_dir / "data.csv"
736
+ if not csv_path.exists():
737
+ for f in panel_dir.glob("*.csv"):
738
+ csv_path = f
739
+ break
740
+
741
+ if csv_path.exists():
742
+ csv_data = pd.read_csv(csv_path)
743
+
744
+ # Load style.json for overrides
745
+ style_path = panel_dir / "style.json"
746
+ overrides = {}
747
+ if style_path.exists():
748
+ with open(style_path, "r") as f:
749
+ style = json.load(f)
750
+ # Convert style to overrides format
751
+ size = style.get("size", {})
752
+ if size:
753
+ width_mm = size.get("width_mm", 80)
754
+ height_mm = size.get("height_mm", 68)
755
+ overrides["fig_size"] = [width_mm / 25.4, height_mm / 25.4]
756
+ overrides["transparent"] = True
757
+
758
+ # Render with dark mode
759
+ return render_preview_with_bboxes(
760
+ csv_data, overrides,
761
+ metadata=metadata,
762
+ dark_mode=dark_mode,
763
+ )
764
+
765
+ except Exception as e:
766
+ import traceback
767
+ print(f"Error rendering panel {panel_dir}: {e}")
768
+ traceback.print_exc()
769
+ return None, None, None
770
+
771
+
698
772
  # Dark mode theme colors
699
773
  DARK_THEME_TEXT_COLOR = "#e8e8e8" # Light gray for visibility on dark background
700
774
  DARK_THEME_SPINE_COLOR = "#e8e8e8"
@@ -0,0 +1,41 @@
1
+ /* =============================================================================
2
+ * Base Styles & Reset
3
+ * ============================================================================= */
4
+ * {
5
+ box-sizing: border-box;
6
+ margin: 0;
7
+ padding: 0;
8
+ }
9
+
10
+ body {
11
+ font-family: -apple-system, BlinkMacSystemFont, 'Segoe UI', Roboto, sans-serif;
12
+ background: var(--workspace-bg-primary);
13
+ color: var(--text-primary);
14
+ transition: background 0.3s, color 0.3s;
15
+ }
16
+
17
+ /* Scrollbar Styling */
18
+ .controls::-webkit-scrollbar,
19
+ .traces-list::-webkit-scrollbar,
20
+ .annotations-list::-webkit-scrollbar {
21
+ width: 6px;
22
+ }
23
+
24
+ .controls::-webkit-scrollbar-track,
25
+ .traces-list::-webkit-scrollbar-track,
26
+ .annotations-list::-webkit-scrollbar-track {
27
+ background: var(--bg-muted);
28
+ }
29
+
30
+ .controls::-webkit-scrollbar-thumb,
31
+ .traces-list::-webkit-scrollbar-thumb,
32
+ .annotations-list::-webkit-scrollbar-thumb {
33
+ background: var(--border-default);
34
+ border-radius: 3px;
35
+ }
36
+
37
+ .controls::-webkit-scrollbar-thumb:hover,
38
+ .traces-list::-webkit-scrollbar-thumb:hover,
39
+ .annotations-list::-webkit-scrollbar-thumb:hover {
40
+ background: var(--text-muted);
41
+ }
@@ -0,0 +1,16 @@
1
+ /* =============================================================================
2
+ * Typography
3
+ * ============================================================================= */
4
+ .filename {
5
+ font-size: 0.8em;
6
+ color: var(--text-muted);
7
+ margin-top: 4px;
8
+ word-break: break-all;
9
+ }
10
+
11
+ .panel-path {
12
+ font-size: 0.75em;
13
+ color: var(--text-muted);
14
+ margin-top: 2px;
15
+ opacity: 0.8;
16
+ }
@@ -0,0 +1,85 @@
1
+ /* =============================================================================
2
+ * SciTeX Color System - Based on scitex-cloud/static/shared/css
3
+ * ============================================================================= */
4
+ :root, [data-theme="light"] {
5
+ /* Brand colors (light mode) */
6
+ --scitex-01: #1a2a40;
7
+ --scitex-02: #34495e;
8
+ --scitex-03: #506b7a;
9
+ --scitex-04: #6c8ba0;
10
+ --scitex-05: #8fa4b0;
11
+ --scitex-06: #b5c7d1;
12
+ --scitex-07: #d4e1e8;
13
+ --white: #fafbfc;
14
+ --gray-subtle: #f6f8fa;
15
+
16
+ /* Semantic tokens */
17
+ --text-primary: var(--scitex-01);
18
+ --text-secondary: var(--scitex-02);
19
+ --text-muted: var(--scitex-04);
20
+ --text-inverse: var(--white);
21
+
22
+ --bg-page: #fefefe;
23
+ --bg-surface: var(--white);
24
+ --bg-muted: var(--gray-subtle);
25
+
26
+ --border-default: var(--scitex-05);
27
+ --border-muted: var(--scitex-06);
28
+
29
+ /* Workspace colors */
30
+ --workspace-bg-primary: #f8f9fa;
31
+ --workspace-bg-secondary: #f3f4f6;
32
+ --workspace-bg-tertiary: #ebedef;
33
+ --workspace-bg-elevated: #ffffff;
34
+ --workspace-border-subtle: #e0e4e8;
35
+ --workspace-border-default: #b5c7d1;
36
+
37
+ /* Status */
38
+ --status-success: #4a9b7e;
39
+ --status-warning: #b8956a;
40
+ --status-error: #a67373;
41
+
42
+ /* CTA */
43
+ --color-cta: #3b82f6;
44
+ --color-cta-hover: #2563eb;
45
+
46
+ /* Preview background (checkered for transparency) */
47
+ --preview-bg: linear-gradient(45deg, #e0e0e0 25%, transparent 25%),
48
+ linear-gradient(-45deg, #e0e0e0 25%, transparent 25%),
49
+ linear-gradient(45deg, transparent 75%, #e0e0e0 75%),
50
+ linear-gradient(-45deg, transparent 75%, #e0e0e0 75%);
51
+ }
52
+
53
+ [data-theme="dark"] {
54
+ /* Semantic tokens (dark mode) */
55
+ --text-primary: var(--scitex-07);
56
+ --text-secondary: var(--scitex-05);
57
+ --text-muted: var(--scitex-04);
58
+ --text-inverse: var(--scitex-01);
59
+
60
+ --bg-page: #0f1419;
61
+ --bg-surface: var(--scitex-01);
62
+ --bg-muted: var(--scitex-02);
63
+
64
+ --border-default: var(--scitex-03);
65
+ --border-muted: var(--scitex-02);
66
+
67
+ /* Workspace colors */
68
+ --workspace-bg-primary: #0d0d0d;
69
+ --workspace-bg-secondary: #151515;
70
+ --workspace-bg-tertiary: #1a1a1a;
71
+ --workspace-bg-elevated: #1f1f1f;
72
+ --workspace-border-subtle: #1a1a1a;
73
+ --workspace-border-default: #3a3a3a;
74
+
75
+ /* Status */
76
+ --status-success: #6ba89a;
77
+ --status-warning: #d4a87a;
78
+ --status-error: #c08888;
79
+
80
+ /* Preview background (darker checkered) */
81
+ --preview-bg: linear-gradient(45deg, #2a2a2a 25%, transparent 25%),
82
+ linear-gradient(-45deg, #2a2a2a 25%, transparent 25%),
83
+ linear-gradient(45deg, transparent 75%, #2a2a2a 75%),
84
+ linear-gradient(-45deg, transparent 75%, #2a2a2a 75%);
85
+ }
@@ -0,0 +1,217 @@
1
+ /* =============================================================================
2
+ * Buttons
3
+ * ============================================================================= */
4
+ .btn {
5
+ width: 100%;
6
+ padding: 7px 12px;
7
+ margin-top: 6px;
8
+ border: none;
9
+ border-radius: 4px;
10
+ cursor: pointer;
11
+ font-size: 0.9em;
12
+ font-weight: 500;
13
+ transition: all 0.2s;
14
+ }
15
+
16
+ .btn-primary {
17
+ background: var(--status-success);
18
+ color: white;
19
+ }
20
+
21
+ .btn-primary:hover {
22
+ filter: brightness(1.1);
23
+ }
24
+
25
+ .btn-secondary {
26
+ background: var(--bg-muted);
27
+ color: var(--text-primary);
28
+ border: 1px solid var(--border-muted);
29
+ }
30
+
31
+ .btn-secondary:hover {
32
+ background: var(--workspace-bg-tertiary);
33
+ }
34
+
35
+ .btn-warning {
36
+ background: #f59e0b;
37
+ color: #000;
38
+ border: 1px solid #d97706;
39
+ }
40
+
41
+ .btn-warning:hover {
42
+ background: #fbbf24;
43
+ }
44
+
45
+ .btn-cta {
46
+ background: var(--color-cta);
47
+ color: white;
48
+ }
49
+
50
+ .btn-cta:hover {
51
+ background: var(--color-cta-hover);
52
+ }
53
+
54
+ .btn-sm {
55
+ padding: 4px 10px;
56
+ font-size: 0.8em;
57
+ width: auto;
58
+ margin: 0;
59
+ }
60
+
61
+ /* Unit Toggle */
62
+ .unit-toggle {
63
+ display: flex;
64
+ gap: 0;
65
+ border-radius: 4px;
66
+ overflow: hidden;
67
+ border: 1px solid var(--border-default);
68
+ }
69
+
70
+ .unit-btn {
71
+ padding: 4px 12px;
72
+ font-size: 0.8em;
73
+ font-weight: 500;
74
+ border: none;
75
+ background: var(--bg-muted);
76
+ color: var(--text-muted);
77
+ cursor: pointer;
78
+ transition: all 0.15s;
79
+ }
80
+
81
+ .unit-btn:first-child {
82
+ border-right: 1px solid var(--border-default);
83
+ }
84
+
85
+ .unit-btn:hover {
86
+ background: var(--bg-surface);
87
+ color: var(--text-secondary);
88
+ }
89
+
90
+ .unit-btn.active {
91
+ background: var(--color-cta);
92
+ color: white;
93
+ }
94
+
95
+ /* Background Type Toggle */
96
+ .bg-toggle {
97
+ display: flex;
98
+ gap: 6px;
99
+ }
100
+
101
+ .bg-btn {
102
+ display: flex;
103
+ flex-direction: column;
104
+ align-items: center;
105
+ gap: 4px;
106
+ padding: 6px 10px;
107
+ font-size: 0.75em;
108
+ border: 1px solid var(--border-default);
109
+ border-radius: 4px;
110
+ background: var(--bg-muted);
111
+ color: var(--text-muted);
112
+ cursor: pointer;
113
+ transition: all 0.15s;
114
+ flex: 1;
115
+ }
116
+
117
+ .bg-btn:hover {
118
+ background: var(--bg-surface);
119
+ color: var(--text-secondary);
120
+ }
121
+
122
+ .bg-btn.active {
123
+ border-color: var(--color-cta);
124
+ background: var(--bg-surface);
125
+ color: var(--text-primary);
126
+ box-shadow: 0 0 0 1px var(--color-cta);
127
+ }
128
+
129
+ .bg-preview {
130
+ width: 20px;
131
+ height: 14px;
132
+ border-radius: 2px;
133
+ border: 1px solid var(--border-default);
134
+ }
135
+
136
+ .bg-preview.white {
137
+ background: #ffffff;
138
+ }
139
+
140
+ .bg-preview.transparent {
141
+ background: linear-gradient(45deg, #ccc 25%, transparent 25%),
142
+ linear-gradient(-45deg, #ccc 25%, transparent 25%),
143
+ linear-gradient(45deg, transparent 75%, #ccc 75%),
144
+ linear-gradient(-45deg, transparent 75%, #ccc 75%);
145
+ background-size: 8px 8px;
146
+ background-position: 0 0, 0 4px, 4px -4px, -4px 0px;
147
+ background-color: #fff;
148
+ }
149
+
150
+ .bg-preview.black {
151
+ background: #000000;
152
+ }
153
+
154
+ /* Axis Tabs (for X/Y/Z axis switching) */
155
+ .axis-tabs {
156
+ display: flex;
157
+ gap: 3px;
158
+ margin-bottom: 8px;
159
+ background: var(--bg-muted);
160
+ padding: 3px;
161
+ border-radius: 6px;
162
+ }
163
+
164
+ .axis-tab {
165
+ flex: 1;
166
+ padding: 4px 8px;
167
+ border: none;
168
+ background: transparent;
169
+ color: var(--text-secondary);
170
+ font-size: 0.8em;
171
+ font-weight: 600;
172
+ cursor: pointer;
173
+ border-radius: 4px;
174
+ transition: all 0.15s ease;
175
+ }
176
+
177
+ .axis-tab:hover {
178
+ background: var(--bg-surface);
179
+ color: var(--text-primary);
180
+ }
181
+
182
+ .axis-tab.active {
183
+ background: var(--scitex-01);
184
+ color: var(--bg-primary);
185
+ }
186
+
187
+ .axis-panel {
188
+ animation: fadeIn 0.15s ease;
189
+ }
190
+
191
+ @keyframes fadeIn {
192
+ from { opacity: 0; }
193
+ to { opacity: 1; }
194
+ }
195
+
196
+ /* Debug toggle button */
197
+ .debug-toggle {
198
+ position: fixed;
199
+ bottom: 10px;
200
+ right: 10px;
201
+ z-index: 1000;
202
+ padding: 8px 12px;
203
+ background: #333;
204
+ color: #fff;
205
+ border: none;
206
+ border-radius: 4px;
207
+ cursor: pointer;
208
+ font-size: 12px;
209
+ }
210
+
211
+ .debug-toggle:hover {
212
+ background: #555;
213
+ }
214
+
215
+ .debug-toggle.active {
216
+ background: #c00;
217
+ }
@@ -0,0 +1,93 @@
1
+ /* =============================================================================
2
+ * Context Menu (Right-Click)
3
+ * ============================================================================= */
4
+ .context-menu {
5
+ position: fixed;
6
+ background: var(--bg-secondary, #2a2a2a);
7
+ border: 1px solid var(--border-color, #444);
8
+ border-radius: 6px;
9
+ box-shadow: 0 4px 16px rgba(0, 0, 0, 0.4);
10
+ min-width: 200px;
11
+ z-index: 10000;
12
+ padding: 4px 0;
13
+ font-size: 13px;
14
+ }
15
+
16
+ .context-menu-item {
17
+ display: flex;
18
+ align-items: center;
19
+ padding: 8px 12px;
20
+ cursor: pointer;
21
+ color: var(--text-primary, #ddd);
22
+ transition: background 0.1s;
23
+ position: relative;
24
+ }
25
+
26
+ .context-menu-item:hover:not(.disabled) {
27
+ background: var(--accent-primary, #4a90d9);
28
+ color: #fff;
29
+ }
30
+
31
+ .context-menu-item.disabled {
32
+ color: var(--text-muted, #666);
33
+ cursor: not-allowed;
34
+ }
35
+
36
+ .context-menu-icon {
37
+ width: 20px;
38
+ margin-right: 8px;
39
+ text-align: center;
40
+ font-size: 14px;
41
+ }
42
+
43
+ .context-menu-shortcut {
44
+ margin-left: auto;
45
+ font-size: 11px;
46
+ color: var(--text-muted, #888);
47
+ padding-left: 16px;
48
+ }
49
+
50
+ .context-menu-item:hover:not(.disabled) .context-menu-shortcut {
51
+ color: rgba(255, 255, 255, 0.7);
52
+ }
53
+
54
+ .context-menu-arrow {
55
+ margin-left: auto;
56
+ font-size: 10px;
57
+ color: var(--text-muted, #888);
58
+ }
59
+
60
+ .context-menu-divider {
61
+ height: 1px;
62
+ background: var(--border-color, #444);
63
+ margin: 4px 0;
64
+ }
65
+
66
+ /* Submenu */
67
+ .context-menu-submenu {
68
+ position: relative;
69
+ }
70
+
71
+ .context-submenu {
72
+ display: none;
73
+ position: absolute;
74
+ left: 100%;
75
+ top: 0;
76
+ background: var(--bg-secondary, #2a2a2a);
77
+ border: 1px solid var(--border-color, #444);
78
+ border-radius: 6px;
79
+ box-shadow: 0 4px 16px rgba(0, 0, 0, 0.4);
80
+ min-width: 150px;
81
+ padding: 4px 0;
82
+ z-index: 10001;
83
+ pointer-events: auto;
84
+ }
85
+
86
+ .context-menu-submenu:hover > .context-submenu,
87
+ .context-submenu:hover {
88
+ display: block;
89
+ }
90
+
91
+ .context-submenu .context-menu-item {
92
+ padding: 6px 12px;
93
+ }
@@ -0,0 +1,57 @@
1
+ /* =============================================================================
2
+ * Download Dropdown
3
+ * ============================================================================= */
4
+ .download-dropdown {
5
+ position: relative;
6
+ display: inline-block;
7
+ }
8
+
9
+ .download-dropdown #download-btn {
10
+ background: #4a90d9;
11
+ border: none;
12
+ color: white;
13
+ padding: 4px 12px;
14
+ font-size: 0.8em;
15
+ border-radius: 3px;
16
+ cursor: pointer;
17
+ }
18
+
19
+ .download-dropdown #download-btn:hover {
20
+ background: #5a9fe9;
21
+ }
22
+
23
+ #download-menu {
24
+ display: none;
25
+ position: absolute;
26
+ top: 100%;
27
+ left: 0;
28
+ background: #2a2a2a;
29
+ border: 1px solid #444;
30
+ border-radius: 4px;
31
+ min-width: 160px;
32
+ box-shadow: 0 4px 12px rgba(0, 0, 0, 0.4);
33
+ z-index: 1000;
34
+ margin-top: 4px;
35
+ padding: 4px 0;
36
+ }
37
+
38
+ .download-item {
39
+ display: block;
40
+ padding: 8px 12px;
41
+ color: #ddd;
42
+ text-decoration: none;
43
+ font-size: 0.85em;
44
+ transition: background 0.15s;
45
+ white-space: nowrap;
46
+ }
47
+
48
+ .download-item:hover {
49
+ background: #3a3a3a;
50
+ color: #fff;
51
+ }
52
+
53
+ .download-divider {
54
+ height: 1px;
55
+ background: #444;
56
+ margin: 4px 0;
57
+ }
@@ -0,0 +1,112 @@
1
+ /* =============================================================================
2
+ * Form Fields
3
+ * ============================================================================= */
4
+ .field {
5
+ margin-bottom: 8px;
6
+ }
7
+
8
+ .field label {
9
+ display: block;
10
+ font-size: 0.78em;
11
+ font-weight: 500;
12
+ margin-bottom: 3px;
13
+ color: var(--text-secondary);
14
+ }
15
+
16
+ .field input[type="text"],
17
+ .field input[type="number"],
18
+ .field select {
19
+ width: 100%;
20
+ padding: 5px 8px;
21
+ border: 1px solid var(--border-muted);
22
+ border-radius: 4px;
23
+ background: var(--bg-surface);
24
+ color: var(--text-primary);
25
+ font-size: 0.82em;
26
+ transition: border-color 0.2s;
27
+ }
28
+
29
+ .field input:focus,
30
+ .field select:focus {
31
+ outline: none;
32
+ border-color: var(--status-success);
33
+ }
34
+
35
+ .field input[type="color"] {
36
+ width: 40px;
37
+ height: 32px;
38
+ padding: 2px;
39
+ border: 1px solid var(--border-muted);
40
+ border-radius: 4px;
41
+ cursor: pointer;
42
+ background: var(--bg-surface);
43
+ }
44
+
45
+ .field-row {
46
+ display: flex;
47
+ gap: 8px;
48
+ }
49
+
50
+ .field-row .field {
51
+ flex: 1;
52
+ margin-bottom: 0;
53
+ }
54
+
55
+ /* Checkbox styling */
56
+ .checkbox-field {
57
+ display: flex;
58
+ align-items: center;
59
+ gap: 6px;
60
+ cursor: pointer;
61
+ padding: 4px 0;
62
+ }
63
+
64
+ .checkbox-field input[type="checkbox"] {
65
+ width: 16px;
66
+ height: 16px;
67
+ accent-color: var(--status-success);
68
+ }
69
+
70
+ .checkbox-field span {
71
+ font-size: 0.85em;
72
+ color: var(--text-primary);
73
+ }
74
+
75
+ /* Color field with input */
76
+ .color-field {
77
+ display: flex;
78
+ align-items: center;
79
+ gap: 8px;
80
+ }
81
+
82
+ .color-field input[type="text"] {
83
+ flex: 1;
84
+ }
85
+
86
+ /* Range slider styling */
87
+ input[type="range"] {
88
+ width: 100%;
89
+ height: 6px;
90
+ border-radius: 3px;
91
+ background: var(--bg-tertiary);
92
+ outline: none;
93
+ -webkit-appearance: none;
94
+ }
95
+
96
+ input[type="range"]::-webkit-slider-thumb {
97
+ -webkit-appearance: none;
98
+ width: 16px;
99
+ height: 16px;
100
+ border-radius: 50%;
101
+ background: var(--accent);
102
+ cursor: pointer;
103
+ }
104
+
105
+ input[type="range"]::-moz-range-thumb {
106
+ width: 16px;
107
+ height: 16px;
108
+ border-radius: 50%;
109
+ background: var(--accent);
110
+ cursor: pointer;
111
+ border: none;
112
+ }