premium-ds 0.1.0

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 (257) hide show
  1. package/LICENSE +21 -0
  2. package/README.md +113 -0
  3. package/dist/alert.d.ts +31 -0
  4. package/dist/alert.js +6 -0
  5. package/dist/alert.js.map +1 -0
  6. package/dist/avatar-group.d.ts +13 -0
  7. package/dist/avatar-group.js +3 -0
  8. package/dist/avatar-group.js.map +1 -0
  9. package/dist/avatar.d.ts +25 -0
  10. package/dist/avatar.js +3 -0
  11. package/dist/avatar.js.map +1 -0
  12. package/dist/badge.d.ts +23 -0
  13. package/dist/badge.js +3 -0
  14. package/dist/badge.js.map +1 -0
  15. package/dist/button.d.ts +20 -0
  16. package/dist/button.js +3 -0
  17. package/dist/button.js.map +1 -0
  18. package/dist/checkbox.d.ts +25 -0
  19. package/dist/checkbox.js +3 -0
  20. package/dist/checkbox.js.map +1 -0
  21. package/dist/chunk-2OWHZ4JT.js +36 -0
  22. package/dist/chunk-2OWHZ4JT.js.map +1 -0
  23. package/dist/chunk-34SIXSYL.js +64 -0
  24. package/dist/chunk-34SIXSYL.js.map +1 -0
  25. package/dist/chunk-37O2ZXD6.js +55 -0
  26. package/dist/chunk-37O2ZXD6.js.map +1 -0
  27. package/dist/chunk-4AZL76UJ.js +89 -0
  28. package/dist/chunk-4AZL76UJ.js.map +1 -0
  29. package/dist/chunk-4HSCN5TZ.js +86 -0
  30. package/dist/chunk-4HSCN5TZ.js.map +1 -0
  31. package/dist/chunk-5DDOOT33.js +258 -0
  32. package/dist/chunk-5DDOOT33.js.map +1 -0
  33. package/dist/chunk-5FVHWIMY.js +117 -0
  34. package/dist/chunk-5FVHWIMY.js.map +1 -0
  35. package/dist/chunk-5K6KRJGX.js +147 -0
  36. package/dist/chunk-5K6KRJGX.js.map +1 -0
  37. package/dist/chunk-5PQMQBQC.js +74 -0
  38. package/dist/chunk-5PQMQBQC.js.map +1 -0
  39. package/dist/chunk-7OCTVQ7C.js +95 -0
  40. package/dist/chunk-7OCTVQ7C.js.map +1 -0
  41. package/dist/chunk-7OPMOET7.js +39 -0
  42. package/dist/chunk-7OPMOET7.js.map +1 -0
  43. package/dist/chunk-BXXS7YRC.js +270 -0
  44. package/dist/chunk-BXXS7YRC.js.map +1 -0
  45. package/dist/chunk-CV2Q4YXX.js +272 -0
  46. package/dist/chunk-CV2Q4YXX.js.map +1 -0
  47. package/dist/chunk-EIMMDWIW.js +282 -0
  48. package/dist/chunk-EIMMDWIW.js.map +1 -0
  49. package/dist/chunk-EZ2CWTBE.js +230 -0
  50. package/dist/chunk-EZ2CWTBE.js.map +1 -0
  51. package/dist/chunk-FGHDG3Y4.js +89 -0
  52. package/dist/chunk-FGHDG3Y4.js.map +1 -0
  53. package/dist/chunk-FPP2XLKX.js +127 -0
  54. package/dist/chunk-FPP2XLKX.js.map +1 -0
  55. package/dist/chunk-G6OY35DI.js +295 -0
  56. package/dist/chunk-G6OY35DI.js.map +1 -0
  57. package/dist/chunk-H6KWJNOE.js +65 -0
  58. package/dist/chunk-H6KWJNOE.js.map +1 -0
  59. package/dist/chunk-HGILYGY3.js +45 -0
  60. package/dist/chunk-HGILYGY3.js.map +1 -0
  61. package/dist/chunk-I3BCB4Z5.js +88 -0
  62. package/dist/chunk-I3BCB4Z5.js.map +1 -0
  63. package/dist/chunk-KBWNUUWM.js +582 -0
  64. package/dist/chunk-KBWNUUWM.js.map +1 -0
  65. package/dist/chunk-KN7JFAZ6.js +113 -0
  66. package/dist/chunk-KN7JFAZ6.js.map +1 -0
  67. package/dist/chunk-MEF7PI6U.js +16 -0
  68. package/dist/chunk-MEF7PI6U.js.map +1 -0
  69. package/dist/chunk-NKGMQL6I.js +310 -0
  70. package/dist/chunk-NKGMQL6I.js.map +1 -0
  71. package/dist/chunk-NMFQRGLL.js +127 -0
  72. package/dist/chunk-NMFQRGLL.js.map +1 -0
  73. package/dist/chunk-OUBWD6CX.js +433 -0
  74. package/dist/chunk-OUBWD6CX.js.map +1 -0
  75. package/dist/chunk-PFNXVBLU.js +96 -0
  76. package/dist/chunk-PFNXVBLU.js.map +1 -0
  77. package/dist/chunk-PUPZ4HME.js +165 -0
  78. package/dist/chunk-PUPZ4HME.js.map +1 -0
  79. package/dist/chunk-QFS52OK5.js +690 -0
  80. package/dist/chunk-QFS52OK5.js.map +1 -0
  81. package/dist/chunk-QNC6O3PG.js +45 -0
  82. package/dist/chunk-QNC6O3PG.js.map +1 -0
  83. package/dist/chunk-QUHOXWBK.js +82 -0
  84. package/dist/chunk-QUHOXWBK.js.map +1 -0
  85. package/dist/chunk-UIQGSTBJ.js +106 -0
  86. package/dist/chunk-UIQGSTBJ.js.map +1 -0
  87. package/dist/chunk-UJQKVP6V.js +193 -0
  88. package/dist/chunk-UJQKVP6V.js.map +1 -0
  89. package/dist/chunk-VVPGEAC6.js +11 -0
  90. package/dist/chunk-VVPGEAC6.js.map +1 -0
  91. package/dist/chunk-XA3T5KWA.js +58 -0
  92. package/dist/chunk-XA3T5KWA.js.map +1 -0
  93. package/dist/chunk-YSHJHSJM.js +19 -0
  94. package/dist/chunk-YSHJHSJM.js.map +1 -0
  95. package/dist/chunk-YVHOAVSM.js +182 -0
  96. package/dist/chunk-YVHOAVSM.js.map +1 -0
  97. package/dist/collapse.d.ts +16 -0
  98. package/dist/collapse.js +3 -0
  99. package/dist/collapse.js.map +1 -0
  100. package/dist/count-badge.d.ts +11 -0
  101. package/dist/count-badge.js +4 -0
  102. package/dist/count-badge.js.map +1 -0
  103. package/dist/date-field.d.ts +39 -0
  104. package/dist/date-field.js +8 -0
  105. package/dist/date-field.js.map +1 -0
  106. package/dist/date-range-field.d.ts +30 -0
  107. package/dist/date-range-field.js +8 -0
  108. package/dist/date-range-field.js.map +1 -0
  109. package/dist/datetime-field.d.ts +28 -0
  110. package/dist/datetime-field.js +10 -0
  111. package/dist/datetime-field.js.map +1 -0
  112. package/dist/dialog.d.ts +26 -0
  113. package/dist/dialog.js +7 -0
  114. package/dist/dialog.js.map +1 -0
  115. package/dist/index.d.ts +35 -0
  116. package/dist/index.js +40 -0
  117. package/dist/index.js.map +1 -0
  118. package/dist/motion-tokens.d.ts +29 -0
  119. package/dist/motion-tokens.js +3 -0
  120. package/dist/motion-tokens.js.map +1 -0
  121. package/dist/multi-select.d.ts +25 -0
  122. package/dist/multi-select.js +7 -0
  123. package/dist/multi-select.js.map +1 -0
  124. package/dist/number-field.d.ts +24 -0
  125. package/dist/number-field.js +4 -0
  126. package/dist/number-field.js.map +1 -0
  127. package/dist/otp-field.d.ts +20 -0
  128. package/dist/otp-field.js +3 -0
  129. package/dist/otp-field.js.map +1 -0
  130. package/dist/overlay.d.ts +31 -0
  131. package/dist/overlay.js +4 -0
  132. package/dist/overlay.js.map +1 -0
  133. package/dist/pagination.d.ts +24 -0
  134. package/dist/pagination.js +5 -0
  135. package/dist/pagination.js.map +1 -0
  136. package/dist/radio-group.d.ts +46 -0
  137. package/dist/radio-group.js +6 -0
  138. package/dist/radio-group.js.map +1 -0
  139. package/dist/select-core-SAyS-8w0.d.ts +16 -0
  140. package/dist/select.d.ts +27 -0
  141. package/dist/select.js +7 -0
  142. package/dist/select.js.map +1 -0
  143. package/dist/status-badge.d.ts +17 -0
  144. package/dist/status-badge.js +5 -0
  145. package/dist/status-badge.js.map +1 -0
  146. package/dist/table.d.ts +65 -0
  147. package/dist/table.js +5 -0
  148. package/dist/table.js.map +1 -0
  149. package/dist/tabs.d.ts +44 -0
  150. package/dist/tabs.js +5 -0
  151. package/dist/tabs.js.map +1 -0
  152. package/dist/tag.d.ts +28 -0
  153. package/dist/tag.js +5 -0
  154. package/dist/tag.js.map +1 -0
  155. package/dist/text-field.d.ts +30 -0
  156. package/dist/text-field.js +6 -0
  157. package/dist/text-field.js.map +1 -0
  158. package/dist/textarea.d.ts +33 -0
  159. package/dist/textarea.js +5 -0
  160. package/dist/textarea.js.map +1 -0
  161. package/dist/time-field.d.ts +27 -0
  162. package/dist/time-field.js +6 -0
  163. package/dist/time-field.js.map +1 -0
  164. package/dist/toast-store.d.ts +75 -0
  165. package/dist/toast-store.js +3 -0
  166. package/dist/toast-store.js.map +1 -0
  167. package/dist/toast.d.ts +3 -0
  168. package/dist/toast.js +6 -0
  169. package/dist/toast.js.map +1 -0
  170. package/dist/toggle-tag.d.ts +24 -0
  171. package/dist/toggle-tag.js +4 -0
  172. package/dist/toggle-tag.js.map +1 -0
  173. package/dist/toggle.d.ts +21 -0
  174. package/dist/toggle.js +3 -0
  175. package/dist/toggle.js.map +1 -0
  176. package/dist/tooltip.d.ts +27 -0
  177. package/dist/tooltip.js +4 -0
  178. package/dist/tooltip.js.map +1 -0
  179. package/llms.txt +165 -0
  180. package/package.json +205 -0
  181. package/src/components/alert/Alert.tsx +118 -0
  182. package/src/components/alert/alert.css +136 -0
  183. package/src/components/avatar/Avatar.tsx +128 -0
  184. package/src/components/avatar/AvatarGroup.tsx +50 -0
  185. package/src/components/avatar/avatar.css +200 -0
  186. package/src/components/badge/Badge.tsx +66 -0
  187. package/src/components/badge/CountBadge.tsx +46 -0
  188. package/src/components/badge/StatusBadge.tsx +132 -0
  189. package/src/components/badge/badge.css +243 -0
  190. package/src/components/button/Button.tsx +68 -0
  191. package/src/components/button/button.css +222 -0
  192. package/src/components/checkbox/Checkbox.tsx +90 -0
  193. package/src/components/checkbox/checkbox.css +179 -0
  194. package/src/components/date-picker/DateField.tsx +362 -0
  195. package/src/components/date-picker/DateRangeField.tsx +533 -0
  196. package/src/components/date-picker/DateTimeField.tsx +177 -0
  197. package/src/components/date-picker/TimeField.tsx +100 -0
  198. package/src/components/date-picker/date-picker.css +591 -0
  199. package/src/components/date-picker/date-utils.ts +55 -0
  200. package/src/components/date-picker/field-shell.tsx +78 -0
  201. package/src/components/date-picker/glide-pill.tsx +81 -0
  202. package/src/components/date-picker/time-core.tsx +305 -0
  203. package/src/components/dialog/Dialog.tsx +181 -0
  204. package/src/components/dialog/dialog.css +170 -0
  205. package/src/components/glass/glass.css +100 -0
  206. package/src/components/icon/Icon.tsx +76 -0
  207. package/src/components/icon/IconSlot.tsx +11 -0
  208. package/src/components/icon/icon.css +33 -0
  209. package/src/components/input/NumberField.tsx +117 -0
  210. package/src/components/input/OtpField.tsx +118 -0
  211. package/src/components/input/TextField.tsx +123 -0
  212. package/src/components/input/input.css +335 -0
  213. package/src/components/motion/Collapse.tsx +33 -0
  214. package/src/components/motion/collapse.css +41 -0
  215. package/src/components/overlay/Overlay.tsx +239 -0
  216. package/src/components/overlay/overlay-core.tsx +565 -0
  217. package/src/components/overlay/overlay.css +119 -0
  218. package/src/components/overlay/sheet-drag.tsx +146 -0
  219. package/src/components/pagination/Pagination.tsx +140 -0
  220. package/src/components/pagination/pagination.css +48 -0
  221. package/src/components/radio-group/RadioGroup.tsx +182 -0
  222. package/src/components/radio-group/radio-group.css +277 -0
  223. package/src/components/select/MultiSelect.tsx +251 -0
  224. package/src/components/select/Select.tsx +235 -0
  225. package/src/components/select/select-core.tsx +417 -0
  226. package/src/components/select/select.css +386 -0
  227. package/src/components/table/Table.tsx +433 -0
  228. package/src/components/table/table.css +348 -0
  229. package/src/components/tabs/Tabs.tsx +371 -0
  230. package/src/components/tabs/tabs.css +228 -0
  231. package/src/components/tag/Tag.tsx +145 -0
  232. package/src/components/tag/ToggleTag.tsx +125 -0
  233. package/src/components/tag/tag.css +248 -0
  234. package/src/components/textarea/Textarea.tsx +197 -0
  235. package/src/components/textarea/textarea.css +219 -0
  236. package/src/components/toast/Toast.tsx +349 -0
  237. package/src/components/toast/toast-store.ts +266 -0
  238. package/src/components/toast/toast.css +233 -0
  239. package/src/components/toggle/Toggle.tsx +94 -0
  240. package/src/components/toggle/toggle.css +152 -0
  241. package/src/components/tooltip/Tooltip.tsx +365 -0
  242. package/src/components/tooltip/tooltip.css +86 -0
  243. package/src/index.ts +42 -0
  244. package/src/styles.css +39 -0
  245. package/src/tokens/avatar.css +20 -0
  246. package/src/tokens/color.css +56 -0
  247. package/src/tokens/elevation.css +20 -0
  248. package/src/tokens/fonts.css +3 -0
  249. package/src/tokens/glass.css +21 -0
  250. package/src/tokens/icons.css +7 -0
  251. package/src/tokens/layers.css +6 -0
  252. package/src/tokens/motion-tokens.ts +72 -0
  253. package/src/tokens/motion.css +49 -0
  254. package/src/tokens/radius.css +11 -0
  255. package/src/tokens/semantic.css +75 -0
  256. package/src/tokens/spacing.css +26 -0
  257. package/src/tokens/typography.css +54 -0
@@ -0,0 +1,277 @@
1
+ /* RadioGroup - single-select in two skins: rows (dot + label) and cards (selectable tiles). */
2
+
3
+ /* dot dims (18 / sm 16, inner 8 / sm 7) are bespoke, intentionally off the token ramps - don't "fix" to a token */
4
+ .rg {
5
+ --rg-dot: 18px;
6
+ --rg-dot-inner: 8px;
7
+
8
+ display: grid;
9
+ border: 0;
10
+ margin: 0;
11
+ padding: 0;
12
+ min-width: 0;
13
+ }
14
+ .rg--sm {
15
+ --rg-dot: 16px;
16
+ --rg-dot-inner: 7px;
17
+ }
18
+
19
+ .rg__head {
20
+ margin-bottom: var(--space-4);
21
+ }
22
+ .rg__label {
23
+ font: var(--type-caption);
24
+ font-weight: var(--weight-medium);
25
+ color: var(--text-body);
26
+ display: inline-flex;
27
+ align-items: center;
28
+ gap: var(--space-1);
29
+ line-height: 1;
30
+ padding: 0;
31
+ }
32
+ .rg__req {
33
+ color: var(--danger);
34
+ }
35
+ .rg__optional {
36
+ color: var(--text-subtle);
37
+ font-weight: var(--weight-regular);
38
+ }
39
+ .rg__helper {
40
+ font: var(--type-caption);
41
+ color: var(--text-muted);
42
+ margin-top: var(--space-1);
43
+ max-width: var(--measure-prose);
44
+ }
45
+
46
+ .rg__options {
47
+ display: grid;
48
+ gap: var(--space-1);
49
+ }
50
+ .rg--horizontal .rg__options {
51
+ display: flex;
52
+ flex-wrap: wrap;
53
+ gap: var(--space-2) var(--space-5);
54
+ }
55
+ .rg--cards .rg__options {
56
+ gap: var(--space-3);
57
+ }
58
+ .rg--cards.rg--horizontal .rg__options {
59
+ display: flex;
60
+ flex-wrap: wrap;
61
+ gap: var(--space-3);
62
+ }
63
+ .rg--cards.rg--horizontal .rg-opt {
64
+ flex: 1 1 0;
65
+ min-width: 12rem;
66
+ }
67
+
68
+ /* align-items:flex-start pins the dot to the first text line when a description wraps below */
69
+ .rg-opt {
70
+ position: relative;
71
+ display: flex;
72
+ align-items: flex-start;
73
+ gap: var(--space-3);
74
+ padding: var(--space-2) var(--space-3);
75
+ border-radius: var(--radius-md);
76
+ cursor: pointer;
77
+ transition: var(--transition-colors);
78
+ -webkit-tap-highlight-color: transparent;
79
+ }
80
+ .rg-opt:hover:not(.is-disabled) {
81
+ background: var(--bg-subtle);
82
+ }
83
+
84
+ .rg-opt__input {
85
+ position: absolute;
86
+ width: 1px;
87
+ height: 1px;
88
+ margin: 0;
89
+ padding: 0;
90
+ opacity: 0;
91
+ pointer-events: none;
92
+ }
93
+
94
+ /* height = the label's line-box, so the dot optically centers on the first line */
95
+ .rg-opt__control {
96
+ flex: none;
97
+ height: var(--leading-body);
98
+ display: grid;
99
+ place-items: center;
100
+ }
101
+ .rg--sm .rg-opt__control {
102
+ height: var(--leading-caption);
103
+ }
104
+
105
+ .rg-opt__dot {
106
+ position: relative;
107
+ width: var(--rg-dot);
108
+ height: var(--rg-dot);
109
+ border-radius: var(--radius-full);
110
+ /* transparent so the ring-centre gap shows the surface behind (card tint), never a white halo */
111
+ background: transparent;
112
+ border: var(--border-hairline) solid var(--border-strong);
113
+ /* transform stays in the bundle so the press scale eases instead of snapping */
114
+ transition:
115
+ var(--transition-control),
116
+ transform var(--duration-fast) var(--ease-standard);
117
+ }
118
+ /* single marker glides between dots via Motion layoutId; centred by inset:0 + margin:auto, so no CSS transition here - Motion owns the move */
119
+ .rg__marker {
120
+ position: absolute;
121
+ inset: 0;
122
+ margin: auto;
123
+ width: var(--rg-dot-inner);
124
+ height: var(--rg-dot-inner);
125
+ border-radius: var(--radius-full);
126
+ background: var(--accent);
127
+ pointer-events: none;
128
+ will-change: transform;
129
+ }
130
+
131
+ .rg-opt__body {
132
+ display: grid;
133
+ gap: var(--space-px);
134
+ padding-top: 0;
135
+ }
136
+ .rg-opt__label {
137
+ font: var(--type-body);
138
+ color: var(--text-strong);
139
+ }
140
+ .rg--sm .rg-opt__label {
141
+ font: var(--type-caption);
142
+ font-weight: var(--weight-medium);
143
+ }
144
+ .rg-opt__desc {
145
+ font: var(--type-caption);
146
+ color: var(--text-muted);
147
+ text-wrap: pretty;
148
+ }
149
+
150
+ .rg-opt:hover:not(.is-disabled):not(.is-selected) .rg-opt__dot {
151
+ border-color: var(--text-muted);
152
+ }
153
+ .rg-opt:active:not(.is-disabled) .rg-opt__dot {
154
+ transform: scale(0.92);
155
+ }
156
+
157
+ /* .is-selected (React) drives selection, not :checked/:has - Chromium doesn't reliably invalidate those on a radio's implicit sibling-uncheck */
158
+ .rg-opt.is-selected .rg-opt__dot {
159
+ border-color: var(--accent);
160
+ }
161
+
162
+ .rg-opt__input:focus-visible ~ .rg-opt__control .rg-opt__dot {
163
+ border-color: var(--accent);
164
+ box-shadow: var(--ring-accent);
165
+ }
166
+
167
+ .rg-opt.is-disabled {
168
+ cursor: not-allowed;
169
+ }
170
+ .rg-opt.is-disabled .rg-opt__label {
171
+ color: var(--text-disabled);
172
+ }
173
+ .rg-opt.is-disabled .rg-opt__desc {
174
+ color: var(--text-disabled);
175
+ }
176
+ .rg-opt.is-disabled .rg-opt__dot {
177
+ background: var(--bg-muted);
178
+ border-color: var(--border-default);
179
+ }
180
+ .rg-opt.is-disabled.is-selected .rg-opt__dot {
181
+ border-color: var(--border-strong);
182
+ }
183
+ .rg-opt.is-disabled.is-selected .rg__marker {
184
+ background: var(--text-disabled);
185
+ }
186
+
187
+ .rg--cards .rg-opt {
188
+ position: relative;
189
+ align-items: center;
190
+ padding: var(--space-4);
191
+ background: var(--bg-surface);
192
+ border: var(--border-hairline) solid var(--border-default);
193
+ border-radius: var(--radius-lg);
194
+ box-shadow: var(--shadow-xs);
195
+ transition:
196
+ var(--transition-control),
197
+ transform var(--duration-fast) var(--ease-standard);
198
+ }
199
+ .rg--cards .rg-opt:hover:not(.is-disabled):not(.is-selected) {
200
+ background: var(--bg-surface);
201
+ border-color: var(--border-strong);
202
+ box-shadow: var(--shadow-sm);
203
+ }
204
+ .rg--cards .rg-opt:active:not(.is-disabled) {
205
+ transform: translateY(var(--space-px));
206
+ box-shadow: var(--shadow-xs);
207
+ }
208
+
209
+ .rg--cards .rg-opt__control {
210
+ order: 3;
211
+ margin-left: auto;
212
+ }
213
+ .rg-opt__icon {
214
+ flex: none;
215
+ display: inline-flex;
216
+ color: var(--text-muted);
217
+ transition: color var(--duration-fast) var(--ease-standard);
218
+ }
219
+ .rg-opt__icon svg {
220
+ display: block;
221
+ }
222
+
223
+ /* selected fill is a shared layoutId layer that glides between cards - the card paints no selected bg/border of its own */
224
+ .rg__card-fill {
225
+ position: absolute;
226
+
227
+ /* negative inset covers the card's hairline so the accent edge replaces it */
228
+ inset: calc(-1 * var(--border-hairline));
229
+ z-index: 0;
230
+ border: var(--border-hairline) solid var(--accent);
231
+ border-radius: var(--radius-lg);
232
+ background: var(--accent-subtle);
233
+ pointer-events: none;
234
+ }
235
+ .rg--cards .rg-opt > .rg-opt__icon,
236
+ .rg--cards .rg-opt > .rg-opt__control,
237
+ .rg--cards .rg-opt > .rg-opt__body {
238
+ position: relative;
239
+ z-index: 1;
240
+ }
241
+ .rg--cards .rg-opt.is-selected .rg-opt__icon {
242
+ color: var(--text-accent);
243
+ }
244
+ .rg--cards .rg-opt:has(.rg-opt__input:focus-visible) {
245
+ box-shadow: var(--ring-accent);
246
+ }
247
+
248
+ .rg--cards .rg-opt.is-disabled {
249
+ background: var(--bg-subtle);
250
+ border-color: var(--border-subtle);
251
+ box-shadow: none;
252
+ }
253
+ .rg--cards .rg-opt.is-disabled .rg-opt__icon {
254
+ color: var(--text-disabled);
255
+ }
256
+
257
+ /* in a <Collapse>: absent = zero reserved space; the padding-top is the options-message gap, inside the collapsible area */
258
+ .rg__msg {
259
+ font: var(--type-caption);
260
+ color: var(--danger-text);
261
+ display: flex;
262
+ align-items: center;
263
+ gap: var(--space-1);
264
+ min-height: 1em;
265
+ padding-top: var(--space-3);
266
+ }
267
+ .rg__msg i,
268
+ .rg__msg svg {
269
+ flex: none;
270
+ }
271
+
272
+ .rg.is-error .rg-opt:not(.is-selected) .rg-opt__dot {
273
+ border-color: var(--danger);
274
+ }
275
+ .rg.is-error.rg--cards .rg-opt:not(.is-selected) {
276
+ border-color: var(--danger);
277
+ }
@@ -0,0 +1,251 @@
1
+ 'use client';
2
+
3
+ /* MultiSelect - many-of custom listbox; committing toggles a row, the menu stays open. */
4
+ import * as React from 'react';
5
+ import {
6
+ useControllable as useMsControllable,
7
+ normalize as msNormalize,
8
+ matches as msMatches,
9
+ useSelectMenu as useMsMenu,
10
+ SelectTrigger as MsTrigger,
11
+ SelectMenu as MsMenu,
12
+ SearchField as MsSearchField,
13
+ FilterRow as MsFilterRow,
14
+ EmptyRow as MsEmptyRow,
15
+ LoadingRows as MsLoadingRows,
16
+ type SelectOption,
17
+ type SelectGroup,
18
+ } from './select-core';
19
+ import { IconSlot } from '../icon/IconSlot';
20
+
21
+ export interface MultiSelectProps {
22
+ options: SelectOption[] | SelectGroup[];
23
+ /** Controlled value. */
24
+ value?: string[];
25
+ defaultValue?: string[];
26
+ /** Fires with the NEXT array and the option that was toggled. */
27
+ onChange?: (value: string[], toggled: SelectOption) => void;
28
+ placeholder?: string;
29
+ size?: 'sm' | 'default' | 'lg';
30
+ disabled?: boolean;
31
+ invalid?: boolean;
32
+ loading?: boolean;
33
+ searchable?: boolean;
34
+ searchPlaceholder?: string;
35
+ /** Your own icon node pinned before the trigger label. */
36
+ leadingIcon?: React.ReactNode;
37
+ id?: string;
38
+ ariaLabel?: string;
39
+ }
40
+
41
+ function CheckboxTick({ checked }: { checked: boolean }) {
42
+ return (
43
+ <span className="cbx" aria-hidden="true">
44
+ <input type="checkbox" className="cbx__input" tabIndex={-1} checked={checked} readOnly />
45
+ <span className="cbx__box">
46
+ <svg className="cbx__mark" viewBox="0 0 16 16" fill="none">
47
+ <path className="cbx__tick" d="M3.5 8.5 L6.75 11.5 L12.5 4.75" />
48
+ </svg>
49
+ <span className="cbx__dash"></span>
50
+ </span>
51
+ </span>
52
+ );
53
+ }
54
+
55
+ export function MultiSelect({
56
+ options = [],
57
+ value: controlledValue,
58
+ defaultValue = [],
59
+ onChange,
60
+ placeholder = 'Select options',
61
+ size = 'default',
62
+ disabled = false,
63
+ invalid = false,
64
+ loading = false,
65
+ searchable = false,
66
+ searchPlaceholder = 'Filter options',
67
+ leadingIcon = null,
68
+ id,
69
+ ariaLabel,
70
+ }: MultiSelectProps) {
71
+ const { useState, useRef, useEffect, useId } = React;
72
+ const [value, setValue] = useMsControllable<string[]>(
73
+ controlledValue,
74
+ defaultValue,
75
+ onChange as ((next: string[], opt?: SelectOption) => void) | undefined,
76
+ );
77
+ const [open, setOpen] = useState(false);
78
+ const [query, setQuery] = useState('');
79
+
80
+ const triggerRef = useRef<HTMLButtonElement>(null),
81
+ listRef = useRef<HTMLDivElement>(null),
82
+ searchRef = useRef<HTMLInputElement>(null);
83
+ const baseId = id || 'mselect-' + useId();
84
+ const menuId = baseId + '-menu';
85
+ const listId = baseId + '-list';
86
+
87
+ const { groups, flat } = msNormalize(options);
88
+ const values = Array.isArray(value) ? value : [];
89
+ const isSelected = (v: string) => values.indexOf(v) !== -1;
90
+ const selectedOptions = flat.filter((o) => isSelected(o.value));
91
+ const navItems: SelectOption[] = [];
92
+ groups.forEach((g) =>
93
+ g.options.forEach((o) => {
94
+ if (msMatches(o, query)) navItems.push(o);
95
+ }),
96
+ );
97
+
98
+ const show = () => {
99
+ if (!disabled && !loading) setOpen(true);
100
+ };
101
+ const hide = () => setOpen(false);
102
+ const returnFocus = () => triggerRef.current && triggerRef.current.focus();
103
+
104
+ function commit(opt: SelectOption) {
105
+ if (!opt || opt.disabled) return;
106
+ const next = isSelected(opt.value)
107
+ ? values.filter((v) => v !== opt.value)
108
+ : [...values, opt.value];
109
+ setValue(next, opt);
110
+ }
111
+
112
+ useEffect(() => {
113
+ if (!open) setQuery('');
114
+ }, [open]);
115
+
116
+ const { activeIdx, setActiveIdx, onMenuKeyDown } = useMsMenu({
117
+ open,
118
+ close: hide,
119
+ returnFocus,
120
+ triggerRef,
121
+ listRef,
122
+ searchRef,
123
+ menuId,
124
+ navItems,
125
+ isSelected,
126
+ commit,
127
+ searchable,
128
+ });
129
+
130
+ const isPlaceholder = !loading && selectedOptions.length === 0;
131
+ const adId = open && activeIdx >= 0 ? baseId + '-opt-' + activeIdx : undefined;
132
+ let vIdx = -1;
133
+
134
+ return (
135
+ <div
136
+ className="select"
137
+ data-multiple="true"
138
+ data-size={size === 'default' ? undefined : size}
139
+ data-open={open ? 'true' : undefined}
140
+ data-disabled={disabled ? 'true' : undefined}
141
+ data-invalid={invalid ? 'true' : undefined}
142
+ data-loading={loading ? 'true' : undefined}
143
+ >
144
+ <MsTrigger
145
+ triggerRef={triggerRef}
146
+ baseId={baseId}
147
+ open={open}
148
+ disabled={disabled}
149
+ invalid={invalid}
150
+ ariaLabel={ariaLabel}
151
+ adId={adId}
152
+ show={show}
153
+ hide={hide}
154
+ leading={leadingIcon || (selectedOptions.length === 1 && selectedOptions[0].icon) || null}
155
+ text={loading ? 'Loading...' : isPlaceholder ? placeholder : selectedOptions[0].label}
156
+ isPlaceholder={isPlaceholder}
157
+ count={selectedOptions.length - 1}
158
+ />
159
+
160
+ <MsMenu open={open} menuId={menuId}>
161
+ {searchable && !loading && (
162
+ <MsSearchField
163
+ searchRef={searchRef}
164
+ query={query}
165
+ onQuery={setQuery}
166
+ onKeyDown={onMenuKeyDown}
167
+ placeholder={searchPlaceholder}
168
+ listId={listId}
169
+ adId={adId}
170
+ />
171
+ )}
172
+
173
+ <div
174
+ className="select__list"
175
+ ref={listRef}
176
+ id={listId}
177
+ role="listbox"
178
+ aria-multiselectable="true"
179
+ tabIndex={-1}
180
+ aria-label={ariaLabel}
181
+ onKeyDown={searchable ? undefined : onMenuKeyDown}
182
+ >
183
+ {loading ? (
184
+ <MsLoadingRows />
185
+ ) : (
186
+ <React.Fragment>
187
+ {groups.map((g, gi) => (
188
+ <div
189
+ className="select__group"
190
+ role="group"
191
+ aria-label={g.label || undefined}
192
+ key={gi}
193
+ >
194
+ {g.label && g.options.some((o) => msMatches(o, query)) && (
195
+ <div className="select__group-label">{g.label}</div>
196
+ )}
197
+ {g.options.map((opt) => {
198
+ const visible = msMatches(opt, query);
199
+ const i = visible ? ((vIdx += 1), vIdx) : -1;
200
+ const isSel = isSelected(opt.value);
201
+ const row = (
202
+ <div
203
+ key={opt.value}
204
+ id={visible ? baseId + '-opt-' + i : undefined}
205
+ data-idx={visible ? i : undefined}
206
+ className="select__option"
207
+ role="option"
208
+ aria-selected={isSel}
209
+ aria-hidden={!visible || undefined}
210
+ aria-disabled={opt.disabled || undefined}
211
+ data-selected={isSel ? 'true' : undefined}
212
+ data-active={visible && i === activeIdx ? 'true' : undefined}
213
+ data-disabled={opt.disabled ? 'true' : undefined}
214
+ onMouseEnter={() => visible && !opt.disabled && setActiveIdx(i)}
215
+ onMouseDown={(e) => e.preventDefault() /* keep focus on list */}
216
+ onClick={() => visible && commit(opt)}
217
+ >
218
+ {opt.icon && (
219
+ <span className="select__option-icon">
220
+ <IconSlot size="sm">{opt.icon}</IconSlot>
221
+ </span>
222
+ )}
223
+ <span className="select__option-text">
224
+ <span className="select__option-label">{opt.label}</span>
225
+ {opt.description && (
226
+ <span className="select__option-desc">{opt.description}</span>
227
+ )}
228
+ </span>
229
+ <span className="select__option-check">
230
+ <CheckboxTick checked={isSel} />
231
+ </span>
232
+ </div>
233
+ );
234
+ return searchable ? (
235
+ <MsFilterRow key={opt.value} visible={visible}>
236
+ {row}
237
+ </MsFilterRow>
238
+ ) : (
239
+ row
240
+ );
241
+ })}
242
+ </div>
243
+ ))}
244
+ {navItems.length === 0 && <MsEmptyRow query={query} />}
245
+ </React.Fragment>
246
+ )}
247
+ </div>
248
+ </MsMenu>
249
+ </div>
250
+ );
251
+ }