@getmicdrop/svelte-components 5.12.0 → 5.14.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 (86) hide show
  1. package/dist/calendar/OrderSummary/OrderSummary.svelte +67 -7
  2. package/dist/calendar/OrderSummary/OrderSummary.svelte.d.ts +2 -0
  3. package/dist/calendar/OrderSummary/OrderSummary.svelte.d.ts.map +1 -1
  4. package/dist/index.spec.js +0 -1
  5. package/dist/patterns/navigation/Header.svelte +23 -27
  6. package/dist/patterns/navigation/Header.svelte.d.ts.map +1 -1
  7. package/dist/primitives/AvatarButton/AvatarButton.svelte +57 -0
  8. package/dist/primitives/AvatarButton/AvatarButton.svelte.d.ts +18 -0
  9. package/dist/primitives/AvatarButton/AvatarButton.svelte.d.ts.map +1 -0
  10. package/dist/primitives/BottomSheet/BottomSheet.spec.js +19 -19
  11. package/dist/primitives/BottomSheet/BottomSheet.svelte +5 -5
  12. package/dist/primitives/BottomSheet/BottomSheet.svelte.d.ts +2 -2
  13. package/dist/primitives/BottomSheet/BottomSheet.svelte.d.ts.map +1 -1
  14. package/dist/primitives/BottomSheet/BottomSheetWrapper.test.svelte +3 -3
  15. package/dist/primitives/BottomSheet/BottomSheetWrapper.test.svelte.d.ts +1 -1
  16. package/dist/primitives/Button/Button.spec.js +16 -14
  17. package/dist/primitives/Button/Button.svelte +9 -45
  18. package/dist/primitives/Button/Button.svelte.d.ts.map +1 -1
  19. package/dist/primitives/CardAction/CardAction.svelte +68 -0
  20. package/dist/primitives/CardAction/CardAction.svelte.d.ts +20 -0
  21. package/dist/primitives/CardAction/CardAction.svelte.d.ts.map +1 -0
  22. package/dist/primitives/Drawer/Drawer.spec.js +33 -33
  23. package/dist/primitives/Drawer/Drawer.svelte +5 -9
  24. package/dist/primitives/Drawer/Drawer.svelte.d.ts +2 -3
  25. package/dist/primitives/Drawer/Drawer.svelte.d.ts.map +1 -1
  26. package/dist/primitives/Input/Input.svelte +1 -1
  27. package/dist/primitives/LandingButton/LandingButton.svelte +92 -0
  28. package/dist/primitives/LandingButton/LandingButton.svelte.d.ts +22 -0
  29. package/dist/primitives/LandingButton/LandingButton.svelte.d.ts.map +1 -0
  30. package/dist/primitives/MenuItem/MenuItem.svelte +85 -0
  31. package/dist/primitives/MenuItem/MenuItem.svelte.d.ts +24 -0
  32. package/dist/primitives/MenuItem/MenuItem.svelte.d.ts.map +1 -0
  33. package/dist/primitives/Modal/Modal.spec.js +7 -7
  34. package/dist/primitives/Modal/Modal.stories.svelte +3 -3
  35. package/dist/primitives/Modal/Modal.svelte +25 -18
  36. package/dist/primitives/Modal/Modal.svelte.d.ts +5 -5
  37. package/dist/primitives/Modal/Modal.svelte.d.ts.map +1 -1
  38. package/dist/primitives/Modal/ModalTestWrapper.svelte +3 -3
  39. package/dist/primitives/Modal/ModalTestWrapper.svelte.d.ts +2 -2
  40. package/dist/primitives/NavItem/NavItem.svelte +75 -0
  41. package/dist/primitives/NavItem/NavItem.svelte.d.ts +20 -0
  42. package/dist/primitives/NavItem/NavItem.svelte.d.ts.map +1 -0
  43. package/dist/primitives/SearchResultItem/SearchResultItem.svelte +109 -0
  44. package/dist/primitives/SearchResultItem/SearchResultItem.svelte.d.ts +26 -0
  45. package/dist/primitives/SearchResultItem/SearchResultItem.svelte.d.ts.map +1 -0
  46. package/dist/primitives/SidebarToggle/SidebarToggle.svelte +55 -0
  47. package/dist/primitives/SidebarToggle/SidebarToggle.svelte.d.ts +18 -0
  48. package/dist/primitives/SidebarToggle/SidebarToggle.svelte.d.ts.map +1 -0
  49. package/dist/primitives/index.d.ts +7 -0
  50. package/dist/primitives/index.js +21 -0
  51. package/dist/recipes/SuperLogin/SuperLogin.svelte +3 -3
  52. package/dist/recipes/SuperLogin/SuperLogin.svelte.d.ts.map +1 -1
  53. package/dist/recipes/inputs/index.d.ts +0 -1
  54. package/dist/recipes/inputs/index.js +0 -1
  55. package/dist/recipes/modals/AlertModal.spec.js +2 -2
  56. package/dist/recipes/modals/AlertModal.svelte +6 -6
  57. package/dist/recipes/modals/AlertModal.svelte.d.ts +3 -3
  58. package/dist/recipes/modals/ConfirmationModal.spec.js +2 -2
  59. package/dist/recipes/modals/ConfirmationModal.svelte +5 -5
  60. package/dist/recipes/modals/ConfirmationModal.svelte.d.ts +3 -3
  61. package/dist/recipes/modals/InputModal.spec.js +2 -2
  62. package/dist/recipes/modals/InputModal.svelte +4 -4
  63. package/dist/recipes/modals/InputModal.svelte.d.ts +3 -3
  64. package/dist/recipes/modals/ModalTestWrapper.spec.js +49 -49
  65. package/dist/recipes/modals/ModalTestWrapper.svelte +3 -3
  66. package/dist/recipes/modals/ModalTestWrapper.svelte.d.ts +2 -2
  67. package/dist/recipes/modals/StatusModal.spec.js +2 -2
  68. package/dist/recipes/modals/StatusModal.svelte +4 -4
  69. package/dist/recipes/modals/StatusModal.svelte.d.ts +3 -3
  70. package/dist/stories/ComponentConsolidation.stories.svelte +10 -10
  71. package/dist/stories/PrimitivesGallery.svelte +25 -21
  72. package/dist/stories/PrimitivesGallery.svelte.d.ts.map +1 -1
  73. package/dist/stories/RecipesGallery.spec.js +9 -18
  74. package/dist/stories/RecipesGallery.svelte +5 -22
  75. package/dist/stories/RecipesGallery.svelte.d.ts.map +1 -1
  76. package/dist/tokens/__tests__/sizing.test.js +5 -7
  77. package/dist/tokens/sizing.d.ts +20 -19
  78. package/dist/tokens/sizing.d.ts.map +1 -1
  79. package/dist/tokens/sizing.js +20 -19
  80. package/package.json +1 -1
  81. package/dist/recipes/inputs/SelectDropdown.spec.d.ts +0 -2
  82. package/dist/recipes/inputs/SelectDropdown.spec.d.ts.map +0 -1
  83. package/dist/recipes/inputs/SelectDropdown.spec.js +0 -518
  84. package/dist/recipes/inputs/SelectDropdown.svelte +0 -171
  85. package/dist/recipes/inputs/SelectDropdown.svelte.d.ts +0 -16
  86. package/dist/recipes/inputs/SelectDropdown.svelte.d.ts.map +0 -1
@@ -1,518 +0,0 @@
1
- import { render, screen, fireEvent, waitFor } from "@testing-library/svelte";
2
- import userEvent from "@testing-library/user-event";
3
- import { expect, describe, test, vi, beforeEach } from "vitest";
4
- import SelectDropdown from "./SelectDropdown.svelte";
5
-
6
- // Mock transitions to be instant for testing
7
- vi.mock('../../utils/transitions.js', () => ({
8
- bloom: () => ({ duration: 0, delay: 0, css: () => '' }),
9
- safeSlide: () => ({ duration: 0, delay: 0, css: () => '' }),
10
- }));
11
-
12
- const sampleOptions = [
13
- { label: "Option 1", value: "opt1" },
14
- { label: "Option 2", value: "opt2" },
15
- { label: "Option 3", value: "opt3" },
16
- { label: "Option 4", value: "opt4" },
17
- ];
18
-
19
- function setupTest(args = {}) {
20
- const user = userEvent.setup();
21
- const { component, container } = render(SelectDropdown, {
22
- props: {
23
- options: sampleOptions,
24
- ...args,
25
- },
26
- });
27
- return { user, component, container };
28
- }
29
-
30
- describe("SelectDropdown Component Tests", () => {
31
- beforeEach(() => {
32
- // Clear any existing event listeners
33
- document.body.innerHTML = "";
34
- });
35
-
36
- test("Renders with default placeholder", () => {
37
- setupTest();
38
- expect(screen.getByRole("button", { name: /select/i })).toBeInTheDocument();
39
- expect(screen.getByText("Select")).toBeInTheDocument();
40
- });
41
-
42
- test("Renders with custom placeholder", () => {
43
- setupTest({ placeholder: "Choose an option" });
44
- expect(screen.getByText("Choose an option")).toBeInTheDocument();
45
- });
46
-
47
- test("Shows selected option label", () => {
48
- setupTest({ selected: sampleOptions[0] });
49
- expect(screen.getByText("Option 1")).toBeInTheDocument();
50
- });
51
-
52
- test("Opens dropdown on click", async () => {
53
- const { user } = setupTest();
54
- const trigger = screen.getByRole("button");
55
-
56
- await user.click(trigger);
57
-
58
- expect(screen.getByRole("listbox")).toBeInTheDocument();
59
- expect(screen.getByText("Option 1")).toBeInTheDocument();
60
- expect(screen.getByText("Option 2")).toBeInTheDocument();
61
- expect(screen.getByText("Option 3")).toBeInTheDocument();
62
- expect(screen.getByText("Option 4")).toBeInTheDocument();
63
- });
64
-
65
- test("Displays all options in dropdown", async () => {
66
- const { user } = setupTest();
67
- const trigger = screen.getByRole("button");
68
-
69
- await user.click(trigger);
70
-
71
- const options = screen.getAllByRole("option");
72
- expect(options).toHaveLength(4);
73
- });
74
-
75
- test("Selects option on click", async () => {
76
- const { user } = setupTest();
77
- const trigger = screen.getByRole("button");
78
-
79
- await user.click(trigger);
80
- const option2 = screen.getByRole("option", { name: "Option 2" });
81
- await user.click(option2);
82
-
83
- // Should show selected option in trigger
84
- expect(screen.getByText("Option 2")).toBeInTheDocument();
85
- });
86
-
87
- test("Closes dropdown after selection", async () => {
88
- const { user } = setupTest();
89
- const trigger = screen.getByRole("button");
90
-
91
- await user.click(trigger);
92
- const option1 = screen.getByRole("option", { name: "Option 1" });
93
- await user.click(option1);
94
-
95
- // Dropdown should be closed
96
- expect(screen.queryByRole("listbox")).not.toBeInTheDocument();
97
- });
98
-
99
- test("Calls onselect callback when option is selected", async () => {
100
- const onSelectSpy = vi.fn();
101
- const { user } = setupTest({ onselect: onSelectSpy });
102
- const trigger = screen.getByRole("button");
103
-
104
- await user.click(trigger);
105
- const option2 = screen.getByRole("option", { name: "Option 2" });
106
- await user.click(option2);
107
-
108
- expect(onSelectSpy).toHaveBeenCalledWith(sampleOptions[1]);
109
- });
110
-
111
- test("Does not open dropdown when disabled", async () => {
112
- const { user } = setupTest({ disabled: true });
113
- const trigger = screen.getByRole("button");
114
-
115
- expect(trigger).toBeDisabled();
116
- await user.click(trigger);
117
-
118
- expect(screen.queryByRole("listbox")).not.toBeInTheDocument();
119
- });
120
-
121
- test("Applies disabled styles when disabled", () => {
122
- setupTest({ disabled: true });
123
- const trigger = screen.getByRole("button");
124
-
125
- expect(trigger).toHaveClass("opacity-50");
126
- expect(trigger).toHaveClass("cursor-not-allowed");
127
- });
128
-
129
- test("Applies custom className", () => {
130
- const { container } = setupTest({ class: "custom-class" });
131
- const dropdown = container.querySelector(".relative");
132
-
133
- expect(dropdown).toHaveClass("custom-class");
134
- });
135
-
136
- test("Closes dropdown on Escape key", async () => {
137
- const { user } = setupTest();
138
- const trigger = screen.getByRole("button");
139
-
140
- await user.click(trigger);
141
- expect(screen.getByRole("listbox")).toBeInTheDocument();
142
-
143
- await user.keyboard("{Escape}");
144
- expect(screen.queryByRole("listbox")).not.toBeInTheDocument();
145
- });
146
-
147
- test("Navigates options with ArrowDown key", async () => {
148
- const { user } = setupTest();
149
- const trigger = screen.getByRole("button");
150
-
151
- await user.click(trigger);
152
- await waitFor(() => expect(screen.getByRole("listbox")).toBeInTheDocument());
153
-
154
- const options = screen.getAllByRole("option");
155
- await user.keyboard("{ArrowDown}");
156
-
157
- // First option should receive focus
158
- expect(options[0]).toHaveFocus();
159
-
160
- await user.keyboard("{ArrowDown}");
161
- // Second option should receive focus
162
- expect(options[1]).toHaveFocus();
163
- });
164
-
165
- test("Navigates options with ArrowUp key", async () => {
166
- const { user } = setupTest();
167
- const trigger = screen.getByRole("button");
168
-
169
- await user.click(trigger);
170
- await waitFor(() => expect(screen.getByRole("listbox")).toBeInTheDocument());
171
-
172
- const options = screen.getAllByRole("option");
173
- await user.keyboard("{ArrowUp}");
174
-
175
- // Should wrap to last option
176
- expect(options[3]).toHaveFocus();
177
-
178
- await user.keyboard("{ArrowUp}");
179
- // Should move to third option
180
- expect(options[2]).toHaveFocus();
181
- });
182
-
183
- test("Navigates to first option with Home key", async () => {
184
- const { user } = setupTest();
185
- const trigger = screen.getByRole("button");
186
-
187
- await user.click(trigger);
188
- await waitFor(() => expect(screen.getByRole("listbox")).toBeInTheDocument());
189
-
190
- const options = screen.getAllByRole("option");
191
- await user.keyboard("{ArrowDown}");
192
- await user.keyboard("{ArrowDown}");
193
- await user.keyboard("{Home}");
194
-
195
- expect(options[0]).toHaveFocus();
196
- });
197
-
198
- test("Navigates to last option with End key", async () => {
199
- const { user } = setupTest();
200
- const trigger = screen.getByRole("button");
201
-
202
- await user.click(trigger);
203
- await waitFor(() => expect(screen.getByRole("listbox")).toBeInTheDocument());
204
-
205
- const options = screen.getAllByRole("option");
206
- await user.keyboard("{End}");
207
-
208
- expect(options[3]).toHaveFocus();
209
- });
210
-
211
- test("Closes dropdown on Tab key", async () => {
212
- const { user } = setupTest();
213
- const trigger = screen.getByRole("button");
214
-
215
- await user.click(trigger);
216
- expect(screen.getByRole("listbox")).toBeInTheDocument();
217
-
218
- await user.keyboard("{Tab}");
219
- expect(screen.queryByRole("listbox")).not.toBeInTheDocument();
220
- });
221
-
222
- test("Returns focus to trigger after Escape", async () => {
223
- const { user } = setupTest();
224
- const trigger = screen.getByRole("button");
225
-
226
- await user.click(trigger);
227
- await user.keyboard("{Escape}");
228
-
229
- expect(trigger).toHaveFocus();
230
- });
231
-
232
- test("Returns focus to trigger after selection", async () => {
233
- const { user } = setupTest();
234
- const trigger = screen.getByRole("button");
235
-
236
- await user.click(trigger);
237
- const option1 = screen.getByRole("option", { name: "Option 1" });
238
- await user.click(option1);
239
-
240
- await waitFor(() => expect(trigger).toHaveFocus());
241
- });
242
-
243
- test("Closes dropdown when clicking outside", async () => {
244
- const { user } = setupTest();
245
- const trigger = screen.getByRole("button");
246
-
247
- await user.click(trigger);
248
- expect(screen.getByRole("listbox")).toBeInTheDocument();
249
-
250
- // Click outside the dropdown
251
- await user.click(document.body);
252
-
253
- await waitFor(() => {
254
- expect(screen.queryByRole("listbox")).not.toBeInTheDocument();
255
- });
256
- });
257
-
258
- test("Does not close dropdown when clicking inside container", async () => {
259
- const { user } = setupTest();
260
- const trigger = screen.getByRole("button");
261
-
262
- await user.click(trigger);
263
- expect(screen.getByRole("listbox")).toBeInTheDocument();
264
-
265
- // Click on the trigger again
266
- await user.click(trigger);
267
-
268
- // Should toggle closed
269
- expect(screen.queryByRole("listbox")).not.toBeInTheDocument();
270
- });
271
-
272
- test("Highlights selected option in dropdown", async () => {
273
- const { user } = setupTest({ selected: sampleOptions[1] });
274
- const trigger = screen.getByRole("button");
275
-
276
- await user.click(trigger);
277
-
278
- const selectedOption = screen.getByRole("option", { name: "Option 2" });
279
- expect(selectedOption).toHaveAttribute("aria-selected", "true");
280
- expect(selectedOption).toHaveClass("bg-blue-50");
281
- });
282
-
283
- test("Toggles dropdown on successive clicks", async () => {
284
- const { user } = setupTest();
285
- const trigger = screen.getByRole("button");
286
-
287
- // Open
288
- await user.click(trigger);
289
- expect(screen.getByRole("listbox")).toBeInTheDocument();
290
-
291
- // Close
292
- await user.click(trigger);
293
- expect(screen.queryByRole("listbox")).not.toBeInTheDocument();
294
-
295
- // Open again
296
- await user.click(trigger);
297
- expect(screen.getByRole("listbox")).toBeInTheDocument();
298
- });
299
-
300
- test("Sets aria-haspopup attribute on trigger", () => {
301
- setupTest();
302
- const trigger = screen.getByRole("button");
303
-
304
- expect(trigger).toHaveAttribute("aria-haspopup", "listbox");
305
- });
306
-
307
- test("Sets aria-expanded based on dropdown state", async () => {
308
- const { user } = setupTest();
309
- const trigger = screen.getByRole("button");
310
-
311
- expect(trigger).toHaveAttribute("aria-expanded", "false");
312
-
313
- await user.click(trigger);
314
- expect(trigger).toHaveAttribute("aria-expanded", "true");
315
-
316
- await user.click(trigger);
317
- expect(trigger).toHaveAttribute("aria-expanded", "false");
318
- });
319
-
320
- test("Rotates chevron icon when dropdown is open", async () => {
321
- const { user, container } = setupTest();
322
- const trigger = screen.getByRole("button");
323
-
324
- const chevron = container.querySelector("svg");
325
- expect(chevron).not.toHaveClass("rotate-180");
326
-
327
- await user.click(trigger);
328
- expect(chevron).toHaveClass("rotate-180");
329
- });
330
-
331
- test("Handles empty options array", () => {
332
- setupTest({ options: [] });
333
- expect(screen.getByRole("button")).toBeInTheDocument();
334
- });
335
-
336
- test("Updates displayed value when selected prop changes", async () => {
337
- const { container } = render(SelectDropdown, {
338
- props: {
339
- options: sampleOptions,
340
- selected: sampleOptions[0],
341
- },
342
- });
343
-
344
- expect(screen.getByText("Option 1")).toBeInTheDocument();
345
-
346
- // Re-render with new selected prop (Svelte 5 compatible approach)
347
- container.innerHTML = "";
348
- render(SelectDropdown, {
349
- props: {
350
- options: sampleOptions,
351
- selected: sampleOptions[2],
352
- },
353
- target: container,
354
- });
355
-
356
- await waitFor(() => {
357
- expect(screen.getByText("Option 3")).toBeInTheDocument();
358
- });
359
- });
360
-
361
- test("Shows all options even when one is selected", async () => {
362
- const { user } = setupTest({ selected: sampleOptions[0] });
363
- const trigger = screen.getByRole("button");
364
-
365
- await user.click(trigger);
366
-
367
- const options = screen.getAllByRole("option");
368
- expect(options).toHaveLength(4);
369
- });
370
-
371
- test("Does not interfere with keyboard events outside dropdown", async () => {
372
- const { user } = setupTest();
373
- const trigger = screen.getByRole("button");
374
-
375
- // Don't open dropdown
376
- await user.keyboard("{ArrowDown}");
377
-
378
- // Dropdown should not open
379
- expect(screen.queryByRole("listbox")).not.toBeInTheDocument();
380
- });
381
-
382
- test("Wraps focus correctly with ArrowDown", async () => {
383
- const { user } = setupTest();
384
- const trigger = screen.getByRole("button");
385
-
386
- await user.click(trigger);
387
- await waitFor(() => expect(screen.getByRole("listbox")).toBeInTheDocument());
388
-
389
- const options = screen.getAllByRole("option");
390
-
391
- // Navigate to last item
392
- await user.keyboard("{End}");
393
- expect(options[3]).toHaveFocus();
394
-
395
- // Press ArrowDown - should wrap to first
396
- await user.keyboard("{ArrowDown}");
397
- expect(options[0]).toHaveFocus();
398
- });
399
-
400
- test("Wraps focus correctly with ArrowUp", async () => {
401
- const { user } = setupTest();
402
- const trigger = screen.getByRole("button");
403
-
404
- await user.click(trigger);
405
- await waitFor(() => expect(screen.getByRole("listbox")).toBeInTheDocument());
406
-
407
- const options = screen.getAllByRole("option");
408
-
409
- // First option
410
- await user.keyboard("{Home}");
411
- expect(options[0]).toHaveFocus();
412
-
413
- // Press ArrowUp - should wrap to last
414
- await user.keyboard("{ArrowUp}");
415
- expect(options[3]).toHaveFocus();
416
- });
417
-
418
- test("Applies correct styling classes to dropdown menu", async () => {
419
- const { user } = setupTest();
420
- const trigger = screen.getByRole("button");
421
-
422
- await user.click(trigger);
423
-
424
- const menu = screen.getByRole("listbox");
425
- expect(menu).toHaveClass("absolute");
426
- expect(menu).toHaveClass("z-10");
427
- expect(menu).toHaveClass("rounded-lg");
428
- expect(menu).toHaveClass("shadow-lg");
429
- });
430
-
431
- test("Truncates long selected option labels", () => {
432
- setupTest({
433
- selected: { label: "Very long option name that should be truncated", value: "long" },
434
- });
435
- const span = screen.getByText("Very long option name that should be truncated");
436
- expect(span).toHaveClass("truncate");
437
- });
438
-
439
- test("Allows null selected value", () => {
440
- setupTest({ selected: null });
441
- expect(screen.getByText("Select")).toBeInTheDocument();
442
- });
443
-
444
- test("Handles option selection with Enter key", async () => {
445
- const onSelectSpy = vi.fn();
446
- const { user } = setupTest({ onselect: onSelectSpy });
447
- const trigger = screen.getByRole("button");
448
-
449
- await user.click(trigger);
450
- await waitFor(() => expect(screen.getByRole("listbox")).toBeInTheDocument());
451
-
452
- await user.keyboard("{ArrowDown}");
453
- const options = screen.getAllByRole("option");
454
-
455
- // Simulate Enter key on focused option
456
- await fireEvent.click(options[0]);
457
-
458
- expect(onSelectSpy).toHaveBeenCalledWith(sampleOptions[0]);
459
- });
460
-
461
- test("Maintains dropdown position with max-height scroll", async () => {
462
- const manyOptions = Array.from({ length: 20 }, (_, i) => ({
463
- label: `Option ${i + 1}`,
464
- value: `opt${i + 1}`,
465
- }));
466
-
467
- const { user } = setupTest({ options: manyOptions });
468
- const trigger = screen.getByRole("button");
469
-
470
- await user.click(trigger);
471
-
472
- const menu = screen.getByRole("listbox");
473
- expect(menu).toHaveClass("max-h-60");
474
- expect(menu).toHaveClass("overflow-y-auto");
475
- });
476
-
477
- test("Initializes focusedIndex to -1 when opening dropdown", async () => {
478
- const { user } = setupTest();
479
- const trigger = screen.getByRole("button");
480
-
481
- await user.click(trigger);
482
-
483
- // No option should be focused initially
484
- const options = screen.getAllByRole("option");
485
- options.forEach(option => {
486
- expect(option).not.toHaveFocus();
487
- });
488
- });
489
-
490
- test("Opens and closes dropdown instances independently", async () => {
491
- const { user: user1 } = setupTest({ placeholder: "Dropdown 1" });
492
- const { user: user2 } = setupTest({ placeholder: "Dropdown 2" });
493
-
494
- const trigger1 = screen.getByRole("button", { name: "Dropdown 1" });
495
- const trigger2 = screen.getByRole("button", { name: "Dropdown 2" });
496
-
497
- // Open first dropdown
498
- await user1.click(trigger1);
499
- expect(screen.getAllByRole("listbox")).toHaveLength(1);
500
-
501
- // Opening second dropdown closes first (click-outside behavior)
502
- await user2.click(trigger2);
503
-
504
- // Should have exactly one listbox open (the second one)
505
- const listboxes = screen.getAllByRole("listbox");
506
- expect(listboxes).toHaveLength(1);
507
- });
508
-
509
- test("Maintains proper z-index for dropdown overlay", async () => {
510
- const { user } = setupTest();
511
- const trigger = screen.getByRole("button");
512
-
513
- await user.click(trigger);
514
-
515
- const menu = screen.getByRole("listbox");
516
- expect(menu).toHaveClass("z-10");
517
- });
518
- });
@@ -1,171 +0,0 @@
1
- <script lang="ts">
2
- import { tick } from "svelte";
3
- import { ChevronDownOutline } from "../../primitives/Icons";
4
- import { bloom } from "../../utils/transitions.js";
5
-
6
- interface SelectOption {
7
- label: string;
8
- value: string;
9
- }
10
-
11
- interface Props {
12
- options?: SelectOption[];
13
- selected?: SelectOption | null;
14
- placeholder?: string;
15
- class?: string;
16
- disabled?: boolean;
17
- onselect?: (option: SelectOption) => void;
18
- }
19
-
20
- let {
21
- options = [],
22
- selected = $bindable(null),
23
- placeholder = "Select",
24
- class: className = "",
25
- disabled = false,
26
- onselect,
27
- }: Props = $props();
28
-
29
- let isOpen = $state(false);
30
- let triggerRef = $state<HTMLButtonElement>();
31
- let menuRef = $state<HTMLDivElement>();
32
- let containerRef = $state<HTMLDivElement>();
33
- let focusedIndex = $state(-1);
34
- let optionElements = $state<HTMLElement[]>([]);
35
-
36
- async function toggleDropdown() {
37
- if (disabled) return;
38
- isOpen = !isOpen;
39
- if (isOpen) {
40
- focusedIndex = -1;
41
- await tick();
42
- updateOptionElements();
43
- }
44
- }
45
-
46
- function updateOptionElements() {
47
- if (menuRef) {
48
- optionElements = Array.from(menuRef.querySelectorAll('[role="option"]'));
49
- }
50
- }
51
-
52
- function focusOption(index: number) {
53
- if (optionElements.length === 0) return;
54
- if (index < 0) index = optionElements.length - 1;
55
- if (index >= optionElements.length) index = 0;
56
- focusedIndex = index;
57
- optionElements[focusedIndex]?.focus();
58
- }
59
-
60
- function selectOption(option: SelectOption) {
61
- selected = option;
62
- onselect?.(option);
63
- isOpen = false;
64
- triggerRef?.focus();
65
- }
66
-
67
- // CRITICAL FIX: Click-outside detection using mousedown for reliable detection
68
- function handleClickOutside(event: MouseEvent) {
69
- if (!isOpen) return;
70
-
71
- // Check if click is inside the container (trigger or menu)
72
- if (containerRef && containerRef.contains(event.target as Node)) {
73
- return;
74
- }
75
-
76
- // Click was outside - close the dropdown
77
- isOpen = false;
78
- }
79
-
80
- function handleKeyDown(event: KeyboardEvent) {
81
- if (!isOpen) return;
82
-
83
- // Only handle if event originated from within this dropdown
84
- if (!menuRef?.contains(event.target as Node) && !triggerRef?.contains(event.target as Node)) {
85
- return;
86
- }
87
-
88
- switch (event.key) {
89
- case "Escape":
90
- isOpen = false;
91
- triggerRef?.focus();
92
- event.preventDefault();
93
- break;
94
- case "ArrowDown":
95
- focusOption(focusedIndex + 1);
96
- event.preventDefault();
97
- break;
98
- case "ArrowUp":
99
- focusOption(focusedIndex - 1);
100
- event.preventDefault();
101
- break;
102
- case "Home":
103
- focusOption(0);
104
- event.preventDefault();
105
- break;
106
- case "End":
107
- focusOption(optionElements.length - 1);
108
- event.preventDefault();
109
- break;
110
- case "Tab":
111
- isOpen = false;
112
- break;
113
- }
114
- }
115
-
116
- $effect(() => {
117
- if (typeof window !== "undefined") {
118
- // Use mousedown for click-outside to capture before click completes
119
- document.addEventListener("mousedown", handleClickOutside, true);
120
- document.addEventListener("keydown", handleKeyDown);
121
- return () => {
122
- document.removeEventListener("mousedown", handleClickOutside, true);
123
- document.removeEventListener("keydown", handleKeyDown);
124
- };
125
- }
126
- });
127
- </script>
128
-
129
- <div bind:this={containerRef} class="relative inline-block text-left {className}">
130
- <button
131
- bind:this={triggerRef}
132
- type="button"
133
- class="inline-flex items-center justify-between w-full px-4 py-2.5 text-sm font-medium text-gray-900 bg-white border border-gray-300 rounded-lg hover:bg-gray-100 focus:ring-4 focus:outline-hidden focus:ring-blue-300 dark:bg-gray-700 dark:text-white dark:border-gray-600 dark:hover:bg-gray-600 dark:focus:ring-blue-800 {disabled ? 'opacity-50 cursor-not-allowed' : ''}"
134
- onclick={toggleDropdown}
135
- aria-haspopup="listbox"
136
- aria-expanded={isOpen}
137
- aria-label={placeholder}
138
- {disabled}
139
- >
140
- <span class="truncate">
141
- {selected?.label || placeholder}
142
- </span>
143
- <ChevronDownOutline class="w-4 h-4 ml-2 transition-transform duration-200 {isOpen ? 'rotate-180' : ''}" />
144
- </button>
145
-
146
- {#if isOpen}
147
- <div
148
- bind:this={menuRef}
149
- class="absolute z-10 mt-1 w-full bg-white divide-y divide-gray-100 rounded-lg shadow-lg dark:bg-gray-700 dark:divide-gray-600 max-h-60 overflow-y-auto"
150
- role="listbox"
151
- aria-label={placeholder}
152
- transition:bloom={{ origin: "top left" }}
153
- >
154
- <ul class="py-2 text-sm text-gray-700 dark:text-gray-200">
155
- {#each options as option}
156
- <li>
157
- <button
158
- type="button"
159
- class="block w-full text-left px-4 py-2 hover:bg-gray-100 dark:hover:bg-gray-600 dark:hover:text-white {selected?.value === option.value ? 'bg-blue-50 text-blue-700 dark:bg-gray-600 dark:text-white' : ''}"
160
- onclick={() => selectOption(option)}
161
- role="option"
162
- aria-selected={selected?.value === option.value}
163
- >
164
- {option.label}
165
- </button>
166
- </li>
167
- {/each}
168
- </ul>
169
- </div>
170
- {/if}
171
- </div>
@@ -1,16 +0,0 @@
1
- interface SelectOption {
2
- label: string;
3
- value: string;
4
- }
5
- interface Props {
6
- options?: SelectOption[];
7
- selected?: SelectOption | null;
8
- placeholder?: string;
9
- class?: string;
10
- disabled?: boolean;
11
- onselect?: (option: SelectOption) => void;
12
- }
13
- declare const SelectDropdown: import("svelte").Component<Props, {}, "selected">;
14
- type SelectDropdown = ReturnType<typeof SelectDropdown>;
15
- export default SelectDropdown;
16
- //# sourceMappingURL=SelectDropdown.svelte.d.ts.map
@@ -1 +0,0 @@
1
- {"version":3,"file":"SelectDropdown.svelte.d.ts","sourceRoot":"","sources":["../../../src/lib/recipes/inputs/SelectDropdown.svelte.ts"],"names":[],"mappings":"AAQE,UAAU,YAAY;IACpB,KAAK,EAAE,MAAM,CAAC;IACd,KAAK,EAAE,MAAM,CAAC;CACf;AAED,UAAU,KAAK;IACb,OAAO,CAAC,EAAE,YAAY,EAAE,CAAC;IACzB,QAAQ,CAAC,EAAE,YAAY,GAAG,IAAI,CAAC;IAC/B,WAAW,CAAC,EAAE,MAAM,CAAC;IACrB,KAAK,CAAC,EAAE,MAAM,CAAC;IACf,QAAQ,CAAC,EAAE,OAAO,CAAC;IACnB,QAAQ,CAAC,EAAE,CAAC,MAAM,EAAE,YAAY,KAAK,IAAI,CAAC;CAC3C;AA8IH,QAAA,MAAM,cAAc,mDAAwC,CAAC;AAC7D,KAAK,cAAc,GAAG,UAAU,CAAC,OAAO,cAAc,CAAC,CAAC;AACxD,eAAe,cAAc,CAAC"}