@countermeasure-platform/web-components 1.3.3-dev.33.1 → 1.3.4

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 (64) hide show
  1. package/README.md +31 -0
  2. package/dist/component-D5sRm1fq.js +389 -0
  3. package/dist/component-D5sRm1fq.js.map +1 -0
  4. package/dist/components/index.js +27 -27
  5. package/dist/icons/index.d.ts +12 -0
  6. package/dist/icons/index.d.ts.map +1 -1
  7. package/dist/icons/index.js +12 -0
  8. package/dist/icons/index.js.map +1 -1
  9. package/dist/index.d.ts +1 -1
  10. package/dist/index.d.ts.map +1 -1
  11. package/dist/index.js +127 -126
  12. package/dist/layout/app-shell.d.ts +50 -3
  13. package/dist/layout/app-shell.d.ts.map +1 -1
  14. package/dist/layout/app-shell.js +142 -13
  15. package/dist/layout/app-shell.js.map +1 -1
  16. package/dist/layout/core-app-chrome.d.ts +81 -0
  17. package/dist/layout/core-app-chrome.d.ts.map +1 -0
  18. package/dist/layout/core-app-chrome.js +349 -0
  19. package/dist/layout/core-app-chrome.js.map +1 -0
  20. package/dist/layout/index.d.ts +4 -2
  21. package/dist/layout/index.d.ts.map +1 -1
  22. package/dist/layout/index.js +88 -87
  23. package/dist/layout/index.js.map +1 -1
  24. package/dist/react/primitives/badge.d.ts +1 -1
  25. package/dist/react/primitives/button.d.ts +2 -2
  26. package/dist/react/primitives/copy-button.d.ts +1 -1
  27. package/dist/react/primitives/stat-card.d.ts +1 -1
  28. package/dist/react/primitives/toggle.d.ts +1 -1
  29. package/dist/react/sidebar.d.ts +4 -4
  30. package/dist/react/sidebar.d.ts.map +1 -1
  31. package/dist/react/sidebar.js +18 -16
  32. package/dist/react/sidebar.js.map +1 -1
  33. package/dist/sidebar/component.d.ts +24 -2
  34. package/dist/sidebar/component.d.ts.map +1 -1
  35. package/dist/sidebar/enhance.d.ts +8 -0
  36. package/dist/sidebar/enhance.d.ts.map +1 -1
  37. package/dist/sidebar/index.d.ts +3 -0
  38. package/dist/sidebar/index.d.ts.map +1 -1
  39. package/dist/sidebar/index.js +81 -28
  40. package/dist/sidebar/index.js.map +1 -1
  41. package/dist/sidebar/types.d.ts +126 -4
  42. package/dist/sidebar/types.d.ts.map +1 -1
  43. package/dist/styles/layout.css +252 -0
  44. package/dist/styles/sidebar.css +313 -5
  45. package/package.json +6 -1
  46. package/src/icons/icons.test.ts +9 -0
  47. package/src/icons/index.ts +12 -0
  48. package/src/index.ts +11 -0
  49. package/src/layout/app-shell.test.ts +204 -0
  50. package/src/layout/app-shell.ts +362 -3
  51. package/src/layout/core-app-chrome.test.ts +507 -0
  52. package/src/layout/core-app-chrome.ts +662 -0
  53. package/src/layout/index.ts +36 -2
  54. package/src/react/sidebar.test.tsx +104 -3
  55. package/src/react/sidebar.tsx +26 -4
  56. package/src/sidebar/component.test.ts +395 -1
  57. package/src/sidebar/component.ts +661 -86
  58. package/src/sidebar/enhance.ts +118 -0
  59. package/src/sidebar/index.ts +144 -0
  60. package/src/sidebar/types.ts +143 -4
  61. package/src/styles/layout.css +252 -0
  62. package/src/styles/sidebar.css +313 -5
  63. package/dist/component-Bxhxf21c.js +0 -167
  64. package/dist/component-Bxhxf21c.js.map +0 -1
@@ -32,6 +32,10 @@
32
32
  color: var(--color-text);
33
33
  }
34
34
 
35
+ .cmm-app-shell--product {
36
+ background: var(--color-bg-base, var(--color-bg));
37
+ }
38
+
35
39
  .cmm-app-shell__sidebar {
36
40
  min-width: 0;
37
41
  }
@@ -49,12 +53,249 @@
49
53
  padding: var(--space-6);
50
54
  }
51
55
 
56
+ .cmm-app-shell--product .cmm-app-shell__content {
57
+ padding: var(--space-6) var(--space-8);
58
+ }
59
+
52
60
  .cmm-app-shell__metric-strip {
53
61
  display: grid;
54
62
  grid-template-columns: repeat(6, minmax(0, 1fr));
55
63
  gap: var(--space-2);
56
64
  }
57
65
 
66
+ .cmm-topbar {
67
+ position: sticky;
68
+ top: 0;
69
+ z-index: 40;
70
+ display: flex;
71
+ min-height: 3.5rem;
72
+ flex-shrink: 0;
73
+ align-items: center;
74
+ justify-content: space-between;
75
+ gap: var(--space-4);
76
+ padding: 0 var(--space-5);
77
+ border-bottom: 1px solid var(--color-border, hsl(var(--border)));
78
+ background: color-mix(in srgb, var(--color-surface, hsl(var(--card))) 70%, transparent);
79
+ -webkit-backdrop-filter: blur(40px);
80
+ backdrop-filter: blur(40px);
81
+ }
82
+
83
+ .cmm-app-shell--product .cmm-topbar {
84
+ border-bottom-color: var(--color-border-glass, var(--color-border));
85
+ background: color-mix(in srgb, var(--color-card, var(--color-surface)) 70%, transparent);
86
+ }
87
+
88
+ .cmm-topbar__breadcrumbs {
89
+ display: flex;
90
+ min-width: 0;
91
+ align-items: center;
92
+ gap: 0.5rem;
93
+ }
94
+
95
+ .cmm-topbar__product {
96
+ flex-shrink: 0;
97
+ color: var(--color-text-muted);
98
+ font-size: 0.625rem;
99
+ font-weight: 700;
100
+ letter-spacing: 0.18em;
101
+ text-transform: uppercase;
102
+ }
103
+
104
+ .cmm-topbar__separator {
105
+ flex-shrink: 0;
106
+ color: var(--color-border);
107
+ font-size: 0.75rem;
108
+ }
109
+
110
+ .cmm-topbar__breadcrumb,
111
+ .cmm-topbar__current {
112
+ min-width: 0;
113
+ overflow: hidden;
114
+ color: var(--color-text-muted);
115
+ font-size: 0.875rem;
116
+ font-weight: 500;
117
+ text-overflow: ellipsis;
118
+ white-space: nowrap;
119
+ }
120
+
121
+ .cmm-topbar__breadcrumb {
122
+ text-decoration: none;
123
+ }
124
+
125
+ .cmm-topbar__breadcrumb:hover {
126
+ color: var(--color-text);
127
+ }
128
+
129
+ .cmm-topbar__current {
130
+ color: var(--color-text);
131
+ }
132
+
133
+ .cmm-topbar__actions {
134
+ display: flex;
135
+ flex-shrink: 0;
136
+ align-items: center;
137
+ gap: 0.625rem;
138
+ }
139
+
140
+ .cmm-topbar__action,
141
+ .cmm-topbar__user-trigger {
142
+ position: relative;
143
+ display: inline-flex;
144
+ align-items: center;
145
+ justify-content: center;
146
+ min-width: 2.25rem;
147
+ height: 2.25rem;
148
+ padding: 0 0.625rem;
149
+ border: 1px solid transparent;
150
+ border-radius: 0.5rem;
151
+ background: transparent;
152
+ color: var(--color-text-muted);
153
+ text-decoration: none;
154
+ transition:
155
+ background 0.15s,
156
+ border-color 0.15s,
157
+ color 0.15s;
158
+ }
159
+
160
+ .cmm-topbar__action:hover,
161
+ .cmm-topbar__user-trigger:hover,
162
+ .cmm-topbar__user-trigger[aria-expanded='true'] {
163
+ border-color: var(--color-border);
164
+ background: var(--color-surface-hover);
165
+ color: var(--color-text);
166
+ }
167
+
168
+ .cmm-topbar__action svg,
169
+ .cmm-topbar__user-trigger svg {
170
+ width: 1rem;
171
+ height: 1rem;
172
+ }
173
+
174
+ .cmm-topbar__action-label {
175
+ overflow: hidden;
176
+ text-overflow: ellipsis;
177
+ white-space: nowrap;
178
+ }
179
+
180
+ .cmm-topbar__action-badge {
181
+ position: absolute;
182
+ top: 0.4rem;
183
+ right: 0.4rem;
184
+ min-width: 0.45rem;
185
+ height: 0.45rem;
186
+ border-radius: 9999px;
187
+ background: var(--color-primary, hsl(var(--primary)));
188
+ color: var(--color-primary-foreground, hsl(var(--primary-foreground)));
189
+ font-size: 0.5625rem;
190
+ line-height: 1;
191
+ }
192
+
193
+ .cmm-topbar__action-badge:not(:empty) {
194
+ top: 0.2rem;
195
+ right: 0.2rem;
196
+ display: inline-flex;
197
+ min-width: 1rem;
198
+ height: 1rem;
199
+ align-items: center;
200
+ justify-content: center;
201
+ padding: 0 0.25rem;
202
+ }
203
+
204
+ .cmm-topbar__user {
205
+ position: relative;
206
+ }
207
+
208
+ .cmm-topbar__user-trigger {
209
+ gap: 0.5rem;
210
+ padding: 0.125rem 0.5rem 0.125rem 0.125rem;
211
+ }
212
+
213
+ .cmm-topbar__avatar {
214
+ display: inline-flex;
215
+ width: 2.125rem;
216
+ height: 2.125rem;
217
+ align-items: center;
218
+ justify-content: center;
219
+ overflow: hidden;
220
+ border-radius: 0.5rem;
221
+ background: var(--color-primary, hsl(var(--primary)));
222
+ color: var(--color-primary-foreground, hsl(var(--primary-foreground)));
223
+ font-size: 0.75rem;
224
+ font-weight: 700;
225
+ }
226
+
227
+ .cmm-topbar__avatar img {
228
+ width: 100%;
229
+ height: 100%;
230
+ object-fit: cover;
231
+ }
232
+
233
+ .cmm-topbar__user-menu {
234
+ position: absolute;
235
+ top: calc(100% + 0.5rem);
236
+ right: 0;
237
+ z-index: 80;
238
+ width: 18rem;
239
+ overflow: hidden;
240
+ border: 1px solid var(--color-border);
241
+ border-radius: 0.75rem;
242
+ background: var(--color-surface);
243
+ box-shadow: var(--shadow-xl, 0 24px 60px rgb(0 0 0 / 30%));
244
+ }
245
+
246
+ .cmm-topbar__user-menu[hidden] {
247
+ display: none;
248
+ }
249
+
250
+ .cmm-topbar__user-profile {
251
+ display: grid;
252
+ gap: 0.125rem;
253
+ padding: 0.75rem;
254
+ border-bottom: 1px solid var(--color-border);
255
+ }
256
+
257
+ .cmm-topbar__user-name {
258
+ overflow: hidden;
259
+ color: var(--color-text);
260
+ font-size: 0.875rem;
261
+ font-weight: 700;
262
+ text-overflow: ellipsis;
263
+ white-space: nowrap;
264
+ }
265
+
266
+ .cmm-topbar__user-detail {
267
+ overflow: hidden;
268
+ color: var(--color-text-muted);
269
+ font-size: 0.75rem;
270
+ text-overflow: ellipsis;
271
+ white-space: nowrap;
272
+ }
273
+
274
+ .cmm-topbar__menu-item {
275
+ display: flex;
276
+ width: 100%;
277
+ align-items: center;
278
+ gap: 0.625rem;
279
+ padding: 0.625rem 0.75rem;
280
+ border: 0;
281
+ background: transparent;
282
+ color: var(--color-text-muted);
283
+ font-size: 0.875rem;
284
+ text-align: left;
285
+ text-decoration: none;
286
+ }
287
+
288
+ .cmm-topbar__menu-item:hover {
289
+ background: var(--color-surface-hover);
290
+ color: var(--color-text);
291
+ }
292
+
293
+ .cmm-topbar__menu-item svg {
294
+ width: 1rem;
295
+ height: 1rem;
296
+ flex-shrink: 0;
297
+ }
298
+
58
299
  @media (max-width: 1023px) {
59
300
  .cmm-app-shell {
60
301
  display: flex;
@@ -65,6 +306,17 @@
65
306
  padding: var(--space-4);
66
307
  }
67
308
 
309
+ .cmm-topbar {
310
+ min-height: 3.25rem;
311
+ padding: 0 var(--space-4);
312
+ }
313
+
314
+ .cmm-topbar__product,
315
+ .cmm-topbar__separator,
316
+ .cmm-topbar__breadcrumb {
317
+ display: none;
318
+ }
319
+
68
320
  .cmm-app-shell__metric-strip {
69
321
  grid-template-columns: repeat(2, minmax(0, 1fr));
70
322
  }
@@ -483,6 +483,16 @@ button.sidebar__brand {
483
483
  color: var(--color-text);
484
484
  }
485
485
 
486
+ .sidebar__item--disabled {
487
+ cursor: not-allowed;
488
+ opacity: 0.45;
489
+ }
490
+
491
+ .sidebar__item--disabled:hover {
492
+ background: transparent;
493
+ color: var(--color-text-muted);
494
+ }
495
+
486
496
  .sidebar__item-label {
487
497
  flex: 1;
488
498
  min-width: 0;
@@ -491,6 +501,24 @@ button.sidebar__brand {
491
501
  white-space: nowrap;
492
502
  }
493
503
 
504
+ .sidebar__tag {
505
+ display: inline-flex;
506
+ align-items: center;
507
+ justify-content: center;
508
+ flex-shrink: 0;
509
+ min-height: 1rem;
510
+ padding: 0 0.4rem;
511
+ border: 1px solid var(--color-border);
512
+ border-radius: 0.375rem;
513
+ background: var(--color-surface-hover);
514
+ color: var(--color-text-muted);
515
+ font-size: 0.5625rem;
516
+ font-weight: 700;
517
+ line-height: 1;
518
+ letter-spacing: 0.08em;
519
+ text-transform: uppercase;
520
+ }
521
+
494
522
  /* Unified Nav Sections */
495
523
  .sidebar__section {
496
524
  display: flex;
@@ -680,6 +708,7 @@ button.sidebar__brand {
680
708
  Scope Selector (Z2)
681
709
  ----------------------------------------------------------------------------- */
682
710
  .sidebar__scope {
711
+ position: relative;
683
712
  padding: var(--density-padding-y) var(--density-padding-x);
684
713
  display: flex;
685
714
  flex-direction: column;
@@ -712,6 +741,145 @@ button.sidebar__brand {
712
741
  box-shadow: 0 0 0 2px hsl(var(--primary) / 0.15);
713
742
  }
714
743
 
744
+ .sidebar__scope-trigger {
745
+ display: flex;
746
+ align-items: center;
747
+ width: 100%;
748
+ min-width: 0;
749
+ gap: 0.5rem;
750
+ padding: 0.45rem 0.625rem;
751
+ border: 1px solid var(--color-border);
752
+ border-radius: 0.5rem;
753
+ background: var(--color-surface);
754
+ color: var(--color-text);
755
+ text-align: left;
756
+ cursor: pointer;
757
+ transition:
758
+ background 0.15s,
759
+ border-color 0.15s,
760
+ color 0.15s;
761
+ }
762
+
763
+ .sidebar__scope-trigger:hover {
764
+ border-color: var(--color-border-strong, var(--color-border));
765
+ background: var(--color-surface-hover);
766
+ }
767
+
768
+ .sidebar__scope-icon,
769
+ .sidebar__scope-chevron {
770
+ flex-shrink: 0;
771
+ width: 0.875rem;
772
+ height: 0.875rem;
773
+ color: var(--color-text-muted);
774
+ }
775
+
776
+ .sidebar__scope-copy {
777
+ display: flex;
778
+ min-width: 0;
779
+ flex: 1;
780
+ flex-direction: column;
781
+ gap: 0.125rem;
782
+ }
783
+
784
+ .sidebar__scope-current {
785
+ overflow: hidden;
786
+ color: var(--color-text);
787
+ font-size: 0.75rem;
788
+ font-weight: 600;
789
+ text-overflow: ellipsis;
790
+ white-space: nowrap;
791
+ }
792
+
793
+ .sidebar__scope-menu {
794
+ position: absolute;
795
+ z-index: 80;
796
+ top: calc(100% + 0.25rem);
797
+ right: var(--density-padding-x);
798
+ left: var(--density-padding-x);
799
+ overflow: hidden;
800
+ padding: 0.25rem;
801
+ border: 1px solid var(--color-border);
802
+ border-radius: 0.625rem;
803
+ background: var(--color-surface);
804
+ box-shadow: var(--shadow-lg);
805
+ }
806
+
807
+ .sidebar__scope-menu[hidden] {
808
+ display: none;
809
+ }
810
+
811
+ .sidebar__scope-option {
812
+ display: flex;
813
+ align-items: center;
814
+ width: 100%;
815
+ gap: 0.5rem;
816
+ padding: 0.45rem 0.5rem;
817
+ border: 0;
818
+ border-radius: 0.5rem;
819
+ background: transparent;
820
+ color: var(--color-text-muted);
821
+ font-size: 0.75rem;
822
+ text-align: left;
823
+ }
824
+
825
+ .sidebar__scope-option:hover {
826
+ background: var(--color-surface-hover);
827
+ color: var(--color-text);
828
+ }
829
+
830
+ .sidebar__scope-option[aria-selected='true'] {
831
+ color: hsl(var(--primary));
832
+ }
833
+
834
+ .sidebar__scope-option[disabled] {
835
+ cursor: not-allowed;
836
+ opacity: 0.5;
837
+ }
838
+
839
+ .sidebar__scope-option-label {
840
+ min-width: 0;
841
+ flex: 1;
842
+ overflow: hidden;
843
+ text-overflow: ellipsis;
844
+ white-space: nowrap;
845
+ }
846
+
847
+ .sidebar__scope-dot {
848
+ width: 0.375rem;
849
+ height: 0.375rem;
850
+ flex-shrink: 0;
851
+ border-radius: 9999px;
852
+ background: var(--color-text-muted);
853
+ }
854
+
855
+ .sidebar__scope-dot--active {
856
+ background: #34d399;
857
+ }
858
+
859
+ .sidebar__scope-dot--suspended,
860
+ .sidebar__scope-dot--warning {
861
+ background: #fbbf24;
862
+ }
863
+
864
+ .sidebar__scope-dot--archived,
865
+ .sidebar__scope-dot--neutral {
866
+ background: #94a3b8;
867
+ }
868
+
869
+ .sidebar__scope-dot--error {
870
+ background: #fb7185;
871
+ }
872
+
873
+ .sidebar__scope-check {
874
+ width: 0.75rem;
875
+ height: 0.75rem;
876
+ opacity: 0;
877
+ }
878
+
879
+ .sidebar__scope-option[aria-selected='true'] .sidebar__scope-check {
880
+ opacity: 1;
881
+ }
882
+
715
883
  /* -----------------------------------------------------------------------------
716
884
  Action Bar (Z3)
717
885
  ----------------------------------------------------------------------------- */
@@ -1039,11 +1207,12 @@ button.sidebar__brand {
1039
1207
  }
1040
1208
 
1041
1209
  /* Unified collapsed rules */
1042
- .sidebar--collapsed .sidebar__item-label,
1043
- .sidebar--collapsed .sidebar__badge,
1044
- .sidebar--collapsed .sidebar__section-label,
1045
- .sidebar--collapsed .sidebar__scope,
1046
- .sidebar--collapsed .sidebar__action-bar,
1210
+ .sidebar--collapsed .sidebar__item-label,
1211
+ .sidebar--collapsed .sidebar__badge,
1212
+ .sidebar--collapsed .sidebar__tag,
1213
+ .sidebar--collapsed .sidebar__section-label,
1214
+ .sidebar--collapsed .sidebar__scope,
1215
+ .sidebar--collapsed .sidebar__action-bar,
1047
1216
  .sidebar--collapsed .sidebar__user-details,
1048
1217
  .sidebar--collapsed .sidebar__user-chevron,
1049
1218
  .sidebar--collapsed .sidebar__footer-action-label {
@@ -1498,6 +1667,105 @@ button.sidebar__brand {
1498
1667
  border-color: var(--cmm-sidebar-primary-border);
1499
1668
  }
1500
1669
 
1670
+ .sidebar--product .sidebar__tag,
1671
+ .sidebar--analyst-console .sidebar__tag,
1672
+ .sidebar--security .sidebar__tag,
1673
+ .sidebar--admin .sidebar__tag {
1674
+ border-color: var(--cmm-sidebar-border);
1675
+ background: var(--cmm-sidebar-hover);
1676
+ color: var(--cmm-sidebar-text-muted);
1677
+ }
1678
+
1679
+ .sidebar--product .sidebar__scope,
1680
+ .sidebar--analyst-console .sidebar__scope,
1681
+ .sidebar--security .sidebar__scope,
1682
+ .sidebar--admin .sidebar__scope {
1683
+ padding: 0 0.75rem 0.625rem;
1684
+ }
1685
+
1686
+ .sidebar--product .sidebar__scope-trigger,
1687
+ .sidebar--analyst-console .sidebar__scope-trigger,
1688
+ .sidebar--security .sidebar__scope-trigger,
1689
+ .sidebar--admin .sidebar__scope-trigger {
1690
+ min-height: 2rem;
1691
+ border-color: var(--cmm-sidebar-border);
1692
+ background: var(--cmm-sidebar-hover);
1693
+ color: var(--cmm-sidebar-text);
1694
+ }
1695
+
1696
+ .sidebar--product .sidebar__scope-trigger:hover,
1697
+ .sidebar--analyst-console .sidebar__scope-trigger:hover,
1698
+ .sidebar--security .sidebar__scope-trigger:hover,
1699
+ .sidebar--admin .sidebar__scope-trigger:hover,
1700
+ .sidebar--product .sidebar__scope-trigger[aria-expanded='true'],
1701
+ .sidebar--analyst-console .sidebar__scope-trigger[aria-expanded='true'],
1702
+ .sidebar--security .sidebar__scope-trigger[aria-expanded='true'],
1703
+ .sidebar--admin .sidebar__scope-trigger[aria-expanded='true'] {
1704
+ border-color: var(--cmm-sidebar-primary-border);
1705
+ background: var(--cmm-sidebar-active);
1706
+ color: var(--cmm-sidebar-text-active);
1707
+ }
1708
+
1709
+ .sidebar--product .sidebar__scope-icon,
1710
+ .sidebar--product .sidebar__scope-chevron,
1711
+ .sidebar--analyst-console .sidebar__scope-icon,
1712
+ .sidebar--analyst-console .sidebar__scope-chevron,
1713
+ .sidebar--security .sidebar__scope-icon,
1714
+ .sidebar--security .sidebar__scope-chevron,
1715
+ .sidebar--admin .sidebar__scope-icon,
1716
+ .sidebar--admin .sidebar__scope-chevron {
1717
+ color: var(--cmm-sidebar-text-muted);
1718
+ }
1719
+
1720
+ .sidebar--product .sidebar__scope-label,
1721
+ .sidebar--analyst-console .sidebar__scope-label,
1722
+ .sidebar--security .sidebar__scope-label,
1723
+ .sidebar--admin .sidebar__scope-label {
1724
+ color: var(--cmm-sidebar-text-muted);
1725
+ font-size: 0.5625rem;
1726
+ letter-spacing: 0.14em;
1727
+ }
1728
+
1729
+ .sidebar--product .sidebar__scope-current,
1730
+ .sidebar--analyst-console .sidebar__scope-current,
1731
+ .sidebar--security .sidebar__scope-current,
1732
+ .sidebar--admin .sidebar__scope-current {
1733
+ color: var(--cmm-sidebar-text);
1734
+ }
1735
+
1736
+ .sidebar--product .sidebar__scope-menu,
1737
+ .sidebar--analyst-console .sidebar__scope-menu,
1738
+ .sidebar--security .sidebar__scope-menu,
1739
+ .sidebar--admin .sidebar__scope-menu {
1740
+ border-color: var(--cmm-sidebar-border);
1741
+ background: var(--cmm-sidebar-bg);
1742
+ box-shadow: 0 16px 42px rgb(0 0 0 / 35%);
1743
+ -webkit-backdrop-filter: blur(40px);
1744
+ backdrop-filter: blur(40px);
1745
+ }
1746
+
1747
+ .sidebar--product .sidebar__scope-option,
1748
+ .sidebar--analyst-console .sidebar__scope-option,
1749
+ .sidebar--security .sidebar__scope-option,
1750
+ .sidebar--admin .sidebar__scope-option {
1751
+ color: var(--cmm-sidebar-text-muted);
1752
+ }
1753
+
1754
+ .sidebar--product .sidebar__scope-option:hover,
1755
+ .sidebar--analyst-console .sidebar__scope-option:hover,
1756
+ .sidebar--security .sidebar__scope-option:hover,
1757
+ .sidebar--admin .sidebar__scope-option:hover {
1758
+ background: var(--cmm-sidebar-hover);
1759
+ color: var(--cmm-sidebar-text-active);
1760
+ }
1761
+
1762
+ .sidebar--product .sidebar__scope-option[aria-selected='true'],
1763
+ .sidebar--analyst-console .sidebar__scope-option[aria-selected='true'],
1764
+ .sidebar--security .sidebar__scope-option[aria-selected='true'],
1765
+ .sidebar--admin .sidebar__scope-option[aria-selected='true'] {
1766
+ color: var(--cmm-sidebar-primary);
1767
+ }
1768
+
1501
1769
  .sidebar--product .sidebar__footer,
1502
1770
  .sidebar--analyst-console .sidebar__footer,
1503
1771
  .sidebar--security .sidebar__footer,
@@ -1638,6 +1906,46 @@ button.sidebar__brand {
1638
1906
  top: var(--sidebar-collapse-edge-top);
1639
1907
  }
1640
1908
 
1909
+ .sidebar--product.sidebar--collapsed .sidebar__scope,
1910
+ .sidebar--analyst-console.sidebar--collapsed .sidebar__scope,
1911
+ .sidebar--security.sidebar--collapsed .sidebar__scope,
1912
+ .sidebar--admin.sidebar--collapsed .sidebar__scope {
1913
+ display: flex;
1914
+ padding: 0 0.5rem 0.5rem;
1915
+ }
1916
+
1917
+ .sidebar--product.sidebar--collapsed .sidebar__scope-trigger,
1918
+ .sidebar--analyst-console.sidebar--collapsed .sidebar__scope-trigger,
1919
+ .sidebar--security.sidebar--collapsed .sidebar__scope-trigger,
1920
+ .sidebar--admin.sidebar--collapsed .sidebar__scope-trigger {
1921
+ justify-content: center;
1922
+ width: 2.5rem;
1923
+ min-height: 2.5rem;
1924
+ padding: 0;
1925
+ border-radius: 9999px;
1926
+ }
1927
+
1928
+ .sidebar--product.sidebar--collapsed .sidebar__scope-copy,
1929
+ .sidebar--product.sidebar--collapsed .sidebar__scope-chevron,
1930
+ .sidebar--analyst-console.sidebar--collapsed .sidebar__scope-copy,
1931
+ .sidebar--analyst-console.sidebar--collapsed .sidebar__scope-chevron,
1932
+ .sidebar--security.sidebar--collapsed .sidebar__scope-copy,
1933
+ .sidebar--security.sidebar--collapsed .sidebar__scope-chevron,
1934
+ .sidebar--admin.sidebar--collapsed .sidebar__scope-copy,
1935
+ .sidebar--admin.sidebar--collapsed .sidebar__scope-chevron {
1936
+ display: none;
1937
+ }
1938
+
1939
+ .sidebar--product.sidebar--collapsed .sidebar__scope-menu,
1940
+ .sidebar--analyst-console.sidebar--collapsed .sidebar__scope-menu,
1941
+ .sidebar--security.sidebar--collapsed .sidebar__scope-menu,
1942
+ .sidebar--admin.sidebar--collapsed .sidebar__scope-menu {
1943
+ top: 0;
1944
+ right: auto;
1945
+ left: calc(100% + 0.5rem);
1946
+ width: 14rem;
1947
+ }
1948
+
1641
1949
  /* -----------------------------------------------------------------------------
1642
1950
  Mobile Responsive Styles
1643
1951
  ----------------------------------------------------------------------------- */
package/package.json CHANGED
@@ -1,6 +1,6 @@
1
1
  {
2
2
  "name": "@countermeasure-platform/web-components",
3
- "version": "1.3.3-dev.33.1",
3
+ "version": "1.3.4",
4
4
  "description": "Shared web components for CounterMeasure applications - consolidates common frontend functionality across projects.",
5
5
  "type": "module",
6
6
  "main": "./dist/index.js",
@@ -57,6 +57,11 @@
57
57
  "import": "./dist/layout/app-shell.js",
58
58
  "types": "./dist/layout/app-shell.d.ts"
59
59
  },
60
+ "./layout/core-app-chrome": {
61
+ "source": "./src/layout/core-app-chrome.ts",
62
+ "import": "./dist/layout/core-app-chrome.js",
63
+ "types": "./dist/layout/core-app-chrome.d.ts"
64
+ },
60
65
  "./layout/page-header": {
61
66
  "source": "./src/layout/page-header.ts",
62
67
  "import": "./dist/layout/page-header.js",
@@ -102,6 +102,15 @@ describe('Icons Module', () => {
102
102
  expect(getIconSvgInner('lucide', 'detections')).toBe(getIconSvgInner('lucide', 'Shield'))
103
103
  })
104
104
 
105
+ it('maps product chrome aliases used by native app navigation', () => {
106
+ expect(resolveLucideIconName('capacity')).toBe('Gauge')
107
+ expect(resolveLucideIconName('api-access')).toBe('KeyRound')
108
+ expect(resolveLucideIconName('intel-reports')).toBe('FileText')
109
+ expect(resolveLucideIconName('notifications')).toBe('Siren')
110
+ expect(resolveLucideIconName('runners')).toBe('Server')
111
+ expect(getIconSvgInner('lucide', 'capacity')).toBe(getIconSvgInner('lucide', 'Gauge'))
112
+ })
113
+
105
114
  it('keeps semantic aliases backed by existing canonical lucide paths', () => {
106
115
  for (const [alias, canonicalName] of Object.entries(SEMANTIC_ICON_ALIASES)) {
107
116
  expect(getLucideIconPath(canonicalName), `${alias} maps to ${canonicalName}`).toBeDefined()
@@ -49,17 +49,29 @@ export const SEMANTIC_ICON_ALIASES = {
49
49
  alerts: 'Siren',
50
50
  analytics: 'BarChart3',
51
51
  api: 'Code2',
52
+ 'api-access': 'KeyRound',
52
53
  authentication: 'Lock',
54
+ actors: 'Users',
55
+ bell: 'Siren',
56
+ capacity: 'Gauge',
57
+ 'circle-help': 'CircleAlert',
53
58
  code: 'Code2',
54
59
  confidence: 'ShieldCheck',
55
60
  coverage: 'ShieldCheck',
56
61
  docs: 'BookOpen',
62
+ 'intel-reports': 'FileText',
57
63
  integrations: 'Plug',
58
64
  library: 'BookOpen',
65
+ mitre: 'Grid3X3',
66
+ notifications: 'Siren',
59
67
  passkeys: 'KeyRound',
68
+ pipeline: 'Workflow',
60
69
  platform: 'Building2',
61
70
  reports: 'FileText',
71
+ runners: 'Server',
62
72
  telemetry: 'Activity',
73
+ tenants: 'Building2',
74
+ theme: 'Sun',
63
75
  topology: 'Network',
64
76
  } as const satisfies Record<string, LucideIconName>
65
77