flock-core 0.5.0b50__py3-none-any.whl → 0.5.0b52__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.

Potentially problematic release.


This version of flock-core might be problematic. Click here for more details.

Files changed (117) hide show
  1. flock/dashboard/launcher.py +1 -1
  2. flock/frontend/README.md +678 -0
  3. flock/frontend/docs/DESIGN_SYSTEM.md +1980 -0
  4. flock/frontend/index.html +12 -0
  5. flock/frontend/package-lock.json +4347 -0
  6. flock/frontend/package.json +48 -0
  7. flock/frontend/src/App.tsx +79 -0
  8. flock/frontend/src/__tests__/e2e/critical-scenarios.test.tsx +587 -0
  9. flock/frontend/src/__tests__/integration/filtering-e2e.test.tsx +387 -0
  10. flock/frontend/src/__tests__/integration/graph-rendering.test.tsx +640 -0
  11. flock/frontend/src/__tests__/integration/indexeddb-persistence.test.tsx +699 -0
  12. flock/frontend/src/components/common/BuildInfo.tsx +39 -0
  13. flock/frontend/src/components/common/EmptyState.module.css +115 -0
  14. flock/frontend/src/components/common/EmptyState.tsx +128 -0
  15. flock/frontend/src/components/common/ErrorBoundary.module.css +169 -0
  16. flock/frontend/src/components/common/ErrorBoundary.tsx +118 -0
  17. flock/frontend/src/components/common/KeyboardShortcutsDialog.css +251 -0
  18. flock/frontend/src/components/common/KeyboardShortcutsDialog.tsx +151 -0
  19. flock/frontend/src/components/common/LoadingSpinner.module.css +97 -0
  20. flock/frontend/src/components/common/LoadingSpinner.tsx +29 -0
  21. flock/frontend/src/components/controls/PublishControl.css +547 -0
  22. flock/frontend/src/components/controls/PublishControl.test.tsx +543 -0
  23. flock/frontend/src/components/controls/PublishControl.tsx +432 -0
  24. flock/frontend/src/components/details/DetailWindowContainer.tsx +62 -0
  25. flock/frontend/src/components/details/LiveOutputTab.test.tsx +792 -0
  26. flock/frontend/src/components/details/LiveOutputTab.tsx +220 -0
  27. flock/frontend/src/components/details/MessageHistoryTab.tsx +299 -0
  28. flock/frontend/src/components/details/NodeDetailWindow.test.tsx +501 -0
  29. flock/frontend/src/components/details/NodeDetailWindow.tsx +218 -0
  30. flock/frontend/src/components/details/RunStatusTab.tsx +307 -0
  31. flock/frontend/src/components/details/tabs.test.tsx +1015 -0
  32. flock/frontend/src/components/filters/CorrelationIDFilter.module.css +102 -0
  33. flock/frontend/src/components/filters/CorrelationIDFilter.test.tsx +197 -0
  34. flock/frontend/src/components/filters/CorrelationIDFilter.tsx +121 -0
  35. flock/frontend/src/components/filters/FilterBar.module.css +29 -0
  36. flock/frontend/src/components/filters/FilterBar.test.tsx +133 -0
  37. flock/frontend/src/components/filters/FilterBar.tsx +33 -0
  38. flock/frontend/src/components/filters/FilterPills.module.css +79 -0
  39. flock/frontend/src/components/filters/FilterPills.test.tsx +173 -0
  40. flock/frontend/src/components/filters/FilterPills.tsx +67 -0
  41. flock/frontend/src/components/filters/TimeRangeFilter.module.css +91 -0
  42. flock/frontend/src/components/filters/TimeRangeFilter.test.tsx +154 -0
  43. flock/frontend/src/components/filters/TimeRangeFilter.tsx +105 -0
  44. flock/frontend/src/components/graph/AgentNode.test.tsx +75 -0
  45. flock/frontend/src/components/graph/AgentNode.tsx +322 -0
  46. flock/frontend/src/components/graph/GraphCanvas.tsx +406 -0
  47. flock/frontend/src/components/graph/MessageFlowEdge.tsx +128 -0
  48. flock/frontend/src/components/graph/MessageNode.test.tsx +62 -0
  49. flock/frontend/src/components/graph/MessageNode.tsx +116 -0
  50. flock/frontend/src/components/graph/MiniMap.tsx +47 -0
  51. flock/frontend/src/components/graph/TransformEdge.tsx +123 -0
  52. flock/frontend/src/components/layout/DashboardLayout.css +407 -0
  53. flock/frontend/src/components/layout/DashboardLayout.tsx +300 -0
  54. flock/frontend/src/components/layout/Header.module.css +88 -0
  55. flock/frontend/src/components/layout/Header.tsx +52 -0
  56. flock/frontend/src/components/modules/EventLogModule.test.tsx +401 -0
  57. flock/frontend/src/components/modules/EventLogModule.tsx +396 -0
  58. flock/frontend/src/components/modules/EventLogModuleWrapper.tsx +17 -0
  59. flock/frontend/src/components/modules/ModuleRegistry.test.ts +333 -0
  60. flock/frontend/src/components/modules/ModuleRegistry.ts +85 -0
  61. flock/frontend/src/components/modules/ModuleWindow.tsx +155 -0
  62. flock/frontend/src/components/modules/registerModules.ts +20 -0
  63. flock/frontend/src/components/settings/AdvancedSettings.tsx +175 -0
  64. flock/frontend/src/components/settings/AppearanceSettings.tsx +185 -0
  65. flock/frontend/src/components/settings/GraphSettings.tsx +110 -0
  66. flock/frontend/src/components/settings/SettingsPanel.css +327 -0
  67. flock/frontend/src/components/settings/SettingsPanel.tsx +131 -0
  68. flock/frontend/src/components/settings/ThemeSelector.tsx +298 -0
  69. flock/frontend/src/hooks/useKeyboardShortcuts.ts +148 -0
  70. flock/frontend/src/hooks/useModulePersistence.test.ts +442 -0
  71. flock/frontend/src/hooks/useModulePersistence.ts +154 -0
  72. flock/frontend/src/hooks/useModules.ts +139 -0
  73. flock/frontend/src/hooks/usePersistence.ts +139 -0
  74. flock/frontend/src/main.tsx +13 -0
  75. flock/frontend/src/services/api.ts +213 -0
  76. flock/frontend/src/services/indexeddb.test.ts +793 -0
  77. flock/frontend/src/services/indexeddb.ts +794 -0
  78. flock/frontend/src/services/layout.test.ts +437 -0
  79. flock/frontend/src/services/layout.ts +146 -0
  80. flock/frontend/src/services/themeApplicator.ts +140 -0
  81. flock/frontend/src/services/themeService.ts +77 -0
  82. flock/frontend/src/services/websocket.test.ts +595 -0
  83. flock/frontend/src/services/websocket.ts +685 -0
  84. flock/frontend/src/store/filterStore.test.ts +242 -0
  85. flock/frontend/src/store/filterStore.ts +103 -0
  86. flock/frontend/src/store/graphStore.test.ts +186 -0
  87. flock/frontend/src/store/graphStore.ts +414 -0
  88. flock/frontend/src/store/moduleStore.test.ts +253 -0
  89. flock/frontend/src/store/moduleStore.ts +57 -0
  90. flock/frontend/src/store/settingsStore.ts +188 -0
  91. flock/frontend/src/store/streamStore.ts +68 -0
  92. flock/frontend/src/store/uiStore.test.ts +54 -0
  93. flock/frontend/src/store/uiStore.ts +110 -0
  94. flock/frontend/src/store/wsStore.ts +34 -0
  95. flock/frontend/src/styles/index.css +15 -0
  96. flock/frontend/src/styles/scrollbar.css +47 -0
  97. flock/frontend/src/styles/variables.css +488 -0
  98. flock/frontend/src/test/setup.ts +1 -0
  99. flock/frontend/src/types/filters.ts +14 -0
  100. flock/frontend/src/types/graph.ts +55 -0
  101. flock/frontend/src/types/modules.ts +7 -0
  102. flock/frontend/src/types/theme.ts +55 -0
  103. flock/frontend/src/utils/mockData.ts +85 -0
  104. flock/frontend/src/utils/performance.ts +16 -0
  105. flock/frontend/src/utils/transforms.test.ts +860 -0
  106. flock/frontend/src/utils/transforms.ts +323 -0
  107. flock/frontend/src/vite-env.d.ts +17 -0
  108. flock/frontend/tsconfig.json +27 -0
  109. flock/frontend/tsconfig.node.json +11 -0
  110. flock/frontend/vite.config.ts +25 -0
  111. flock/frontend/vitest.config.ts +11 -0
  112. flock/helper/cli_helper.py +1 -1
  113. {flock_core-0.5.0b50.dist-info → flock_core-0.5.0b52.dist-info}/METADATA +1 -1
  114. {flock_core-0.5.0b50.dist-info → flock_core-0.5.0b52.dist-info}/RECORD +117 -7
  115. {flock_core-0.5.0b50.dist-info → flock_core-0.5.0b52.dist-info}/WHEEL +0 -0
  116. {flock_core-0.5.0b50.dist-info → flock_core-0.5.0b52.dist-info}/entry_points.txt +0 -0
  117. {flock_core-0.5.0b50.dist-info → flock_core-0.5.0b52.dist-info}/licenses/LICENSE +0 -0
@@ -0,0 +1,116 @@
1
+ import { memo } from 'react';
2
+ import { NodeProps, Handle, Position } from '@xyflow/react';
3
+ import JsonView from '@uiw/react-json-view';
4
+ import { MessageNodeData } from '../../types/graph';
5
+
6
+ const MessageNode = memo(({ data, selected }: NodeProps) => {
7
+ const nodeData = data as MessageNodeData;
8
+ const artifactType = nodeData.artifactType;
9
+ const payload = nodeData.payload;
10
+ const producedBy = nodeData.producedBy;
11
+ const timestamp = nodeData.timestamp;
12
+ const isStreaming = nodeData.isStreaming || false;
13
+ const streamingText = nodeData.streamingText || '';
14
+
15
+ return (
16
+ <div
17
+ className={`message-node ${selected ? 'selected' : ''}`}
18
+ style={{
19
+ padding: '12px',
20
+ border: `2px solid ${selected ? '#8b5cf6' : '#e2e8f0'}`,
21
+ borderRadius: '8px',
22
+ backgroundColor: '#fefce8',
23
+ minWidth: '300px',
24
+ maxWidth: '600px',
25
+ width: 'auto',
26
+ boxShadow: selected ? '0 4px 12px rgba(139,92,246,0.3)' : '0 2px 8px rgba(0,0,0,0.1)',
27
+ transition: 'all 0.2s ease',
28
+ }}
29
+ >
30
+ <Handle type="target" position={Position.Left} />
31
+ <Handle type="source" position={Position.Right} />
32
+
33
+ {/* Header */}
34
+ <div style={{
35
+ marginBottom: '10px',
36
+ paddingBottom: '8px',
37
+ borderBottom: '1px solid #e2e8f0'
38
+ }}>
39
+ <div style={{
40
+ fontWeight: 700,
41
+ fontSize: '14px',
42
+ marginBottom: '4px',
43
+ color: '#713f12',
44
+ fontFamily: 'monospace'
45
+ }}>
46
+ {artifactType}
47
+ </div>
48
+ <div style={{
49
+ fontSize: '11px',
50
+ color: '#a8a29e',
51
+ display: 'flex',
52
+ justifyContent: 'space-between',
53
+ alignItems: 'center'
54
+ }}>
55
+ <div>by: <span style={{ color: '#78716c', fontWeight: 600 }}>{producedBy}</span></div>
56
+ <div>{new Date(timestamp).toLocaleTimeString()}</div>
57
+ </div>
58
+ </div>
59
+
60
+ {/* Content: Streaming text or JSON Payload */}
61
+ <div style={{
62
+ fontSize: '12px',
63
+ maxHeight: '600px',
64
+ overflowY: 'auto',
65
+ overflowX: 'auto',
66
+ whiteSpace: 'pre-wrap',
67
+ wordBreak: 'break-word'
68
+ }}>
69
+ {isStreaming ? (
70
+ /* Streaming: Show raw text with cursor */
71
+ <div style={{
72
+ fontFamily: 'ui-monospace, SFMono-Regular, Menlo, Monaco, Consolas, monospace',
73
+ color: '#78716c',
74
+ lineHeight: '1.5'
75
+ }}>
76
+ {streamingText}
77
+ <span style={{
78
+ display: 'inline-block',
79
+ width: '8px',
80
+ height: '14px',
81
+ backgroundColor: '#8b5cf6',
82
+ marginLeft: '2px',
83
+ animation: 'blink 1s step-end infinite'
84
+ }} />
85
+ <style>{`
86
+ @keyframes blink {
87
+ 0%, 50% { opacity: 1; }
88
+ 51%, 100% { opacity: 0; }
89
+ }
90
+ `}</style>
91
+ </div>
92
+ ) : (
93
+ /* Complete: Show beautiful JSON */
94
+ <JsonView
95
+ value={payload}
96
+ collapsed={false}
97
+ displayDataTypes={false}
98
+ shortenTextAfterLength={0}
99
+ style={{
100
+ backgroundColor: 'transparent',
101
+ fontSize: '12px',
102
+ fontFamily: 'ui-monospace, SFMono-Regular, Menlo, Monaco, Consolas, monospace',
103
+ '--w-rjv-line-color': '#78716c',
104
+ '--w-rjv-key-string': '#713f12',
105
+ '--w-rjv-info-color': '#a8a29e',
106
+ } as React.CSSProperties}
107
+ />
108
+ )}
109
+ </div>
110
+ </div>
111
+ );
112
+ });
113
+
114
+ MessageNode.displayName = 'MessageNode';
115
+
116
+ export default MessageNode;
@@ -0,0 +1,47 @@
1
+ import React from 'react';
2
+ import { MiniMap as ReactFlowMiniMap } from '@xyflow/react';
3
+
4
+ /**
5
+ * Phase 4: Graph Visualization & Dual Views - MiniMap Component
6
+ *
7
+ * Wrapper around React Flow's MiniMap component for graph navigation.
8
+ * Features:
9
+ * - Positioned in bottom-right corner
10
+ * - Size: 150x100px
11
+ * - Color-coded nodes: agents (blue), messages (green)
12
+ *
13
+ * SPECIFICATION: docs/specs/003-real-time-dashboard/PLAN.md Phase 4
14
+ */
15
+
16
+ const MiniMap: React.FC = () => {
17
+ // Color function for nodes in minimap - using design system colors
18
+ const nodeColor = (node: any) => {
19
+ switch (node.type) {
20
+ case 'agent':
21
+ return '#3b82f6'; // Blue for agents (var(--color-node-agent-border))
22
+ case 'message':
23
+ return '#f59e0b'; // Amber for messages (var(--color-node-message-border))
24
+ default:
25
+ return '#64748b'; // Gray for other nodes (var(--color-idle))
26
+ }
27
+ };
28
+
29
+ return (
30
+ <ReactFlowMiniMap
31
+ nodeColor={nodeColor}
32
+ style={{
33
+ width: 150,
34
+ height: 100,
35
+ backgroundColor: 'var(--color-bg-surface)',
36
+ border: 'var(--border-default)',
37
+ borderRadius: 'var(--radius-lg)',
38
+ }}
39
+ maskColor="rgba(10, 10, 11, 0.7)"
40
+ position="bottom-right"
41
+ pannable
42
+ zoomable
43
+ />
44
+ );
45
+ };
46
+
47
+ export default MiniMap;
@@ -0,0 +1,123 @@
1
+ import React from 'react';
2
+ import {
3
+ BaseEdge,
4
+ EdgeProps,
5
+ getBezierPath,
6
+ getStraightPath,
7
+ getSmoothStepPath,
8
+ getSimpleBezierPath,
9
+ EdgeLabelRenderer
10
+ } from '@xyflow/react';
11
+ import { useSettingsStore } from '../../store/settingsStore';
12
+
13
+ /**
14
+ * Phase 4: Graph Visualization & Dual Views - TransformEdge Component
15
+ *
16
+ * Custom edge for Blackboard View showing transformation between artifacts.
17
+ * Features:
18
+ * - Dashed line style to indicate transformation
19
+ * - Label with agent name
20
+ * - Arrow marker at target
21
+ *
22
+ * SPECIFICATION: docs/specs/003-real-time-dashboard/PLAN.md Phase 4
23
+ */
24
+
25
+ export interface TransformEdgeData {
26
+ transformerAgent: string;
27
+ runId: string;
28
+ durationMs?: number;
29
+ labelOffset?: number; // Phase 11: Vertical offset to prevent label overlap
30
+ }
31
+
32
+ const TransformEdge: React.FC<EdgeProps> = ({
33
+ id,
34
+ sourceX,
35
+ sourceY,
36
+ targetX,
37
+ targetY,
38
+ sourcePosition,
39
+ targetPosition,
40
+ label,
41
+ style = {},
42
+ markerEnd,
43
+ data,
44
+ selected,
45
+ }) => {
46
+ // Get edge settings from settings store
47
+ const edgeType = useSettingsStore((state) => state.graph.edgeType);
48
+ const edgeStrokeWidth = useSettingsStore((state) => state.graph.edgeStrokeWidth);
49
+ const showEdgeLabels = useSettingsStore((state) => state.graph.showEdgeLabels);
50
+
51
+ // Use appropriate path function based on settings
52
+ const getPath = () => {
53
+ const params = { sourceX, sourceY, sourcePosition, targetX, targetY, targetPosition };
54
+ switch (edgeType) {
55
+ case 'straight':
56
+ return getStraightPath(params);
57
+ case 'smoothstep':
58
+ return getSmoothStepPath(params);
59
+ case 'simplebezier':
60
+ return getSimpleBezierPath(params);
61
+ case 'bezier':
62
+ default:
63
+ return getBezierPath(params);
64
+ }
65
+ };
66
+
67
+ const [edgePath, labelX, labelY] = getPath();
68
+
69
+ // Phase 11 Bug Fix: Apply label offset to prevent overlap when multiple edges exist
70
+ const edgeData = data as TransformEdgeData | undefined;
71
+ const labelOffset = edgeData?.labelOffset || 0;
72
+
73
+ const [isHovered, setIsHovered] = React.useState(false);
74
+
75
+ // Determine edge color based on state
76
+ const edgeColor = selected || isHovered ? 'var(--color-edge-active)' : 'var(--color-edge-default)';
77
+
78
+ return (
79
+ <>
80
+ <BaseEdge
81
+ id={id}
82
+ path={edgePath}
83
+ style={{
84
+ ...style,
85
+ stroke: edgeColor,
86
+ strokeWidth: isHovered || selected ? edgeStrokeWidth + 1 : edgeStrokeWidth,
87
+ strokeDasharray: '5,5',
88
+ transition: 'var(--transition-all)',
89
+ filter: isHovered || selected ? `drop-shadow(0 0 4px ${edgeColor})` : 'none',
90
+ }}
91
+ markerEnd={markerEnd}
92
+ />
93
+ {label && showEdgeLabels && (
94
+ <EdgeLabelRenderer>
95
+ <div
96
+ style={{
97
+ position: 'absolute',
98
+ transform: `translate(-50%, -50%) translate(${labelX}px,${labelY + labelOffset}px)`,
99
+ fontSize: 'var(--font-size-caption)',
100
+ fontWeight: 'var(--font-weight-medium)',
101
+ background: 'var(--color-edge-label-bg)',
102
+ color: 'var(--color-edge-label-text)',
103
+ padding: 'var(--spacing-1) var(--spacing-2)',
104
+ borderRadius: 'var(--radius-sm)',
105
+ border: `var(--border-width-1) solid ${edgeColor}`,
106
+ pointerEvents: 'all',
107
+ backdropFilter: 'blur(var(--blur-sm))',
108
+ boxShadow: 'var(--shadow-sm)',
109
+ transition: 'var(--transition-all)',
110
+ }}
111
+ className="nodrag nopan"
112
+ onMouseEnter={() => setIsHovered(true)}
113
+ onMouseLeave={() => setIsHovered(false)}
114
+ >
115
+ {label}
116
+ </div>
117
+ </EdgeLabelRenderer>
118
+ )}
119
+ </>
120
+ );
121
+ };
122
+
123
+ export default TransformEdge;
@@ -0,0 +1,407 @@
1
+ /* ========================================
2
+ Dashboard Layout - Premium Dark Theme
3
+ Uses design system from frontend/src/styles/variables.css
4
+ ======================================== */
5
+
6
+ /* Layout Container */
7
+ .dashboard-layout {
8
+ width: 100vw;
9
+ height: 100vh;
10
+ display: flex;
11
+ flex-direction: column;
12
+ background: var(--color-bg-base);
13
+ overflow: hidden;
14
+ }
15
+
16
+ /* ========================================
17
+ Header
18
+ ======================================== */
19
+
20
+ .dashboard-header {
21
+ display: flex;
22
+ justify-content: space-between;
23
+ align-items: center;
24
+ gap: var(--space-layout-md);
25
+ padding: var(--space-component-md) var(--space-layout-sm);
26
+ background: var(--color-bg-surface);
27
+ border-bottom: var(--border-subtle);
28
+ box-shadow: var(--shadow-sm);
29
+ z-index: 100;
30
+ }
31
+
32
+ .dashboard-title {
33
+ margin: 0;
34
+ font-size: var(--font-size-h3);
35
+ font-weight: var(--font-weight-semibold);
36
+ color: var(--color-text-primary);
37
+ letter-spacing: var(--letter-spacing-tight);
38
+ white-space: nowrap;
39
+ }
40
+
41
+ /* ========================================
42
+ View Toggle - Premium Pill Design
43
+ ======================================== */
44
+
45
+ .view-toggle-container {
46
+ display: flex;
47
+ align-items: center;
48
+ gap: var(--gap-md);
49
+ flex: 1;
50
+ justify-content: center;
51
+ }
52
+
53
+ .view-toggle-label {
54
+ font-size: var(--font-size-body-sm);
55
+ font-weight: var(--font-weight-medium);
56
+ color: var(--color-text-tertiary);
57
+ letter-spacing: var(--letter-spacing-wide);
58
+ text-transform: uppercase;
59
+ font-size: var(--font-size-caption);
60
+ }
61
+
62
+ .view-toggle-group {
63
+ display: inline-flex;
64
+ background: var(--color-bg-elevated);
65
+ padding: var(--spacing-1);
66
+ border-radius: var(--radius-full);
67
+ border: var(--border-subtle);
68
+ box-shadow: var(--shadow-inner);
69
+ gap: var(--gap-xs);
70
+ }
71
+
72
+ .view-toggle-button {
73
+ position: relative;
74
+ padding: var(--space-component-sm) var(--space-component-lg);
75
+ background: transparent;
76
+ color: var(--color-text-tertiary);
77
+ border: none;
78
+ border-radius: var(--radius-full);
79
+ font-family: var(--font-family-sans);
80
+ font-size: var(--font-size-body-sm);
81
+ font-weight: var(--font-weight-medium);
82
+ cursor: pointer;
83
+ transition: var(--transition-colors), var(--transition-transform);
84
+ white-space: nowrap;
85
+ }
86
+
87
+ .view-toggle-button:hover:not(.active) {
88
+ color: var(--color-text-secondary);
89
+ background: var(--color-bg-overlay);
90
+ }
91
+
92
+ .view-toggle-button.active {
93
+ background: var(--color-primary-500);
94
+ color: var(--color-text-on-primary);
95
+ box-shadow: var(--shadow-sm);
96
+ }
97
+
98
+ .view-toggle-button:active {
99
+ transform: scale(0.98);
100
+ }
101
+
102
+ .view-toggle-button:focus-visible {
103
+ outline: none;
104
+ box-shadow: var(--shadow-glow-primary);
105
+ }
106
+
107
+ /* ========================================
108
+ Dashboard Actions
109
+ ======================================== */
110
+
111
+ .dashboard-actions {
112
+ display: flex;
113
+ align-items: center;
114
+ gap: var(--gap-md);
115
+ }
116
+
117
+ /* Icon Button - Subtle, Icon-based Clear Button */
118
+ .icon-button {
119
+ display: inline-flex;
120
+ align-items: center;
121
+ justify-content: center;
122
+ width: 32px;
123
+ height: 32px;
124
+ padding: var(--spacing-2);
125
+ background: transparent;
126
+ color: var(--color-text-tertiary);
127
+ border: none;
128
+ border-radius: var(--radius-md);
129
+ cursor: pointer;
130
+ transition: var(--transition-colors), var(--transition-transform);
131
+ }
132
+
133
+ .icon-button:hover {
134
+ background: var(--color-bg-overlay);
135
+ color: var(--color-text-secondary);
136
+ }
137
+
138
+ .icon-button:active {
139
+ transform: scale(0.95);
140
+ }
141
+
142
+ .icon-button:focus-visible {
143
+ outline: none;
144
+ box-shadow: var(--shadow-glow-primary);
145
+ }
146
+
147
+ .help-button:hover {
148
+ color: var(--color-primary-500);
149
+ background: var(--color-primary-50);
150
+ }
151
+
152
+ .help-button:focus-visible {
153
+ box-shadow: var(--shadow-glow-primary);
154
+ }
155
+
156
+ .clear-button:hover {
157
+ color: var(--color-error);
158
+ background: var(--color-error-bg);
159
+ }
160
+
161
+ .clear-button:focus-visible {
162
+ box-shadow: var(--shadow-glow-error);
163
+ }
164
+
165
+ /* Controls Toggle - Elegant Premium Button */
166
+ .controls-toggle {
167
+ display: inline-flex;
168
+ align-items: center;
169
+ gap: var(--gap-sm);
170
+ padding: var(--space-component-sm) var(--space-component-md);
171
+ background: var(--color-bg-elevated);
172
+ color: var(--color-text-secondary);
173
+ border: var(--border-default);
174
+ border-radius: var(--radius-md);
175
+ font-family: var(--font-family-sans);
176
+ font-size: var(--font-size-body-sm);
177
+ font-weight: var(--font-weight-medium);
178
+ cursor: pointer;
179
+ transition: var(--transition-colors), var(--transition-shadow), var(--transition-transform);
180
+ white-space: nowrap;
181
+ }
182
+
183
+ /* Primary variant - for important actions like Publish */
184
+ .controls-toggle.primary {
185
+ background: var(--color-primary-500);
186
+ color: var(--color-text-on-primary);
187
+ border-color: var(--color-primary-500);
188
+ box-shadow: var(--shadow-md);
189
+ font-weight: var(--font-weight-semibold);
190
+ }
191
+
192
+ .controls-toggle.primary:hover {
193
+ background: var(--color-primary-600);
194
+ border-color: var(--color-primary-600);
195
+ box-shadow: var(--shadow-lg);
196
+ }
197
+
198
+ .controls-toggle.primary.active {
199
+ background: var(--color-primary-700);
200
+ border-color: var(--color-primary-700);
201
+ }
202
+
203
+ .controls-toggle:hover {
204
+ background: var(--color-bg-surface);
205
+ color: var(--color-text-primary);
206
+ border-color: var(--color-border-strong);
207
+ box-shadow: var(--shadow-sm);
208
+ }
209
+
210
+ .controls-toggle.active {
211
+ background: var(--color-success);
212
+ color: var(--color-text-on-primary);
213
+ border-color: var(--color-success);
214
+ box-shadow: var(--shadow-sm);
215
+ }
216
+
217
+ .controls-toggle.active:hover {
218
+ background: var(--color-success-dark);
219
+ border-color: var(--color-success-dark);
220
+ }
221
+
222
+ .controls-toggle:active {
223
+ transform: scale(0.98);
224
+ }
225
+
226
+ .controls-toggle:focus-visible {
227
+ outline: none;
228
+ box-shadow: var(--shadow-glow-primary);
229
+ }
230
+
231
+ .controls-toggle.active:focus-visible {
232
+ box-shadow: var(--shadow-glow-success);
233
+ }
234
+
235
+ .controls-toggle svg {
236
+ width: 16px;
237
+ height: 16px;
238
+ flex-shrink: 0;
239
+ transition: var(--transition-transform);
240
+ }
241
+
242
+ .controls-toggle.active svg {
243
+ transform: rotate(45deg);
244
+ }
245
+
246
+ /* ========================================
247
+ Main Content Area
248
+ ======================================== */
249
+
250
+ .dashboard-main {
251
+ flex: 1;
252
+ display: flex;
253
+ overflow: hidden;
254
+ position: relative;
255
+ }
256
+
257
+ .graph-container {
258
+ flex: 1;
259
+ overflow: hidden;
260
+ position: relative;
261
+ }
262
+
263
+ /* ========================================
264
+ Controls Sidebar - Premium Glassmorphism
265
+ ======================================== */
266
+
267
+ .controls-sidebar {
268
+ width: 400px;
269
+ background: var(--color-glass-bg);
270
+ backdrop-filter: blur(var(--blur-lg));
271
+ -webkit-backdrop-filter: blur(var(--blur-lg));
272
+ border-left: var(--border-subtle);
273
+ box-shadow: var(--shadow-lg);
274
+ overflow-y: auto;
275
+ animation: slideInRight var(--duration-slow) var(--ease-smooth);
276
+ }
277
+
278
+ @keyframes slideInRight {
279
+ from {
280
+ opacity: 0;
281
+ transform: translateX(20px);
282
+ }
283
+ to {
284
+ opacity: 1;
285
+ transform: translateX(0);
286
+ }
287
+ }
288
+
289
+ .controls-sidebar-inner {
290
+ padding: var(--space-layout-sm);
291
+ }
292
+
293
+ .controls-title {
294
+ margin: 0 0 var(--space-layout-sm) 0;
295
+ font-size: var(--font-size-h4);
296
+ font-weight: var(--font-weight-semibold);
297
+ color: var(--color-text-primary);
298
+ letter-spacing: var(--letter-spacing-tight);
299
+ }
300
+
301
+ .control-section {
302
+ margin-bottom: var(--space-layout-md);
303
+ }
304
+
305
+ .control-section-title {
306
+ margin: 0 0 var(--space-component-md) 0;
307
+ font-size: var(--font-size-h5);
308
+ font-weight: var(--font-weight-medium);
309
+ color: var(--color-text-secondary);
310
+ }
311
+
312
+ /* ========================================
313
+ Scrollbar Styling
314
+ ======================================== */
315
+
316
+ .controls-sidebar::-webkit-scrollbar {
317
+ width: 8px;
318
+ }
319
+
320
+ .controls-sidebar::-webkit-scrollbar-track {
321
+ background: var(--color-bg-elevated);
322
+ }
323
+
324
+ .controls-sidebar::-webkit-scrollbar-thumb {
325
+ background: var(--color-border-default);
326
+ border-radius: var(--radius-full);
327
+ transition: var(--transition-colors);
328
+ }
329
+
330
+ .controls-sidebar::-webkit-scrollbar-thumb:hover {
331
+ background: var(--color-border-strong);
332
+ }
333
+
334
+ /* ========================================
335
+ Responsive Design
336
+ ======================================== */
337
+
338
+ @media (max-width: 1024px) {
339
+ .dashboard-header {
340
+ flex-wrap: wrap;
341
+ gap: var(--gap-md);
342
+ }
343
+
344
+ .view-toggle-container {
345
+ order: 3;
346
+ width: 100%;
347
+ justify-content: center;
348
+ }
349
+
350
+ .controls-sidebar {
351
+ width: 320px;
352
+ }
353
+ }
354
+
355
+ @media (max-width: 768px) {
356
+ .dashboard-title {
357
+ font-size: var(--font-size-h4);
358
+ }
359
+
360
+ .view-toggle-button {
361
+ padding: var(--spacing-2) var(--spacing-4);
362
+ font-size: var(--font-size-caption);
363
+ }
364
+
365
+ .controls-toggle span {
366
+ display: none;
367
+ }
368
+
369
+ .controls-sidebar {
370
+ width: 280px;
371
+ }
372
+ }
373
+
374
+ /* ========================================
375
+ Accessibility - Reduced Motion
376
+ ======================================== */
377
+
378
+ @media (prefers-reduced-motion: reduce) {
379
+ .controls-sidebar {
380
+ animation: none;
381
+ }
382
+
383
+ .view-toggle-button,
384
+ .icon-button,
385
+ .controls-toggle {
386
+ transition: none;
387
+ }
388
+
389
+ .controls-toggle svg {
390
+ transition: none;
391
+ }
392
+ }
393
+
394
+ /* ========================================
395
+ Print Styles
396
+ ======================================== */
397
+
398
+ @media print {
399
+ .dashboard-header,
400
+ .controls-sidebar {
401
+ display: none;
402
+ }
403
+
404
+ .dashboard-main {
405
+ flex: 1;
406
+ }
407
+ }