@getmicdrop/svelte-components 5.4.2 → 5.5.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 (217) hide show
  1. package/dist/calendar/AboutShow/AboutShow.svelte +172 -172
  2. package/dist/calendar/Calendar/MiniMonthCalendar.svelte +782 -782
  3. package/dist/calendar/FAQs/FAQs.svelte +75 -75
  4. package/dist/calendar/MonthSwitcher/MonthSwitcher.svelte +126 -126
  5. package/dist/calendar/OrderSummary/OrderSummary.svelte +367 -367
  6. package/dist/calendar/PublicCard/PublicCard.svelte +134 -134
  7. package/dist/calendar/ShowCard/ShowCard.svelte +157 -157
  8. package/dist/calendar/ShowTimeCard/ShowTimeCard.svelte +61 -61
  9. package/dist/components/Layout/Grid.svelte +4 -4
  10. package/dist/components/Layout/Section.svelte +80 -80
  11. package/dist/components/Layout/Sidebar.svelte +108 -108
  12. package/dist/components/Layout/Stack.svelte +6 -6
  13. package/dist/constants/validation.js +91 -91
  14. package/dist/constants/validation.spec.js +64 -64
  15. package/dist/index.d.ts +113 -4
  16. package/dist/index.js +226 -47
  17. package/dist/patterns/data/DataGrid.svelte +45 -45
  18. package/dist/patterns/data/DataList.svelte +24 -24
  19. package/dist/patterns/data/DataTable.svelte +36 -36
  20. package/dist/patterns/forms/FormActions.spec.js +88 -88
  21. package/dist/patterns/forms/FormActions.stories.svelte +97 -97
  22. package/dist/patterns/forms/FormActions.svelte +46 -46
  23. package/dist/patterns/forms/FormGrid.svelte +33 -33
  24. package/dist/patterns/forms/FormSection.svelte +32 -32
  25. package/dist/patterns/forms/FormValidationSummary.stories.svelte +83 -83
  26. package/dist/patterns/forms/FormValidationSummary.svelte +32 -32
  27. package/dist/patterns/layout/Sidebar.svelte +39 -39
  28. package/dist/patterns/navigation/BottomNav.stories.svelte +117 -117
  29. package/dist/patterns/navigation/BottomNav.svelte +20 -20
  30. package/dist/patterns/navigation/Header.stories.svelte +77 -77
  31. package/dist/patterns/navigation/Header.svelte +193 -193
  32. package/dist/patterns/page/PageHeader.svelte +18 -18
  33. package/dist/patterns/page/PageLayout.svelte +40 -40
  34. package/dist/patterns/page/PageLoader.spec.js +54 -54
  35. package/dist/patterns/page/PageLoader.stories.svelte +137 -137
  36. package/dist/patterns/page/PageLoader.svelte +24 -24
  37. package/dist/patterns/page/SectionHeader.svelte +29 -29
  38. package/dist/presets/badges.js +112 -112
  39. package/dist/presets/buttons.js +76 -76
  40. package/dist/presets/index.js +9 -9
  41. package/dist/primitives/Accordion/Accordion.stories.svelte +75 -75
  42. package/dist/primitives/Accordion/Accordion.svelte +42 -42
  43. package/dist/primitives/Accordion/AccordionItem.svelte +95 -95
  44. package/dist/primitives/Alert/Alert.spec.js +170 -170
  45. package/dist/primitives/Alert/Alert.stories.svelte +88 -88
  46. package/dist/primitives/Alert/Alert.svelte +27 -27
  47. package/dist/primitives/Avatar/Avatar.stories.svelte +94 -94
  48. package/dist/primitives/Avatar/Avatar.svelte +66 -66
  49. package/dist/primitives/Badges/Badge.spec.js +103 -103
  50. package/dist/primitives/Badges/Badge.stories.svelte +86 -86
  51. package/dist/primitives/Badges/Badge.svelte +79 -79
  52. package/dist/primitives/BottomSheet/BottomSheet.spec.js +127 -127
  53. package/dist/primitives/BottomSheet/BottomSheet.stories.svelte +83 -83
  54. package/dist/primitives/BottomSheet/BottomSheet.svelte +100 -100
  55. package/dist/primitives/Breadcrumb/Breadcrumb.spec.js +120 -120
  56. package/dist/primitives/Breadcrumb/Breadcrumb.stories.svelte +23 -23
  57. package/dist/primitives/Breadcrumb/Breadcrumb.svelte +89 -89
  58. package/dist/primitives/Button/Button.spec.js +211 -211
  59. package/dist/primitives/Button/Button.stories.svelte +76 -76
  60. package/dist/primitives/Button/Button.svelte +270 -269
  61. package/dist/primitives/Button/Button.svelte.d.ts.map +1 -1
  62. package/dist/primitives/Button/ButtonSaveDemo.spec.js +48 -48
  63. package/dist/primitives/Button/ButtonSaveDemo.svelte +25 -25
  64. package/dist/primitives/Button/ButtonVariantShowcase.svelte +129 -129
  65. package/dist/primitives/Card.spec.js +49 -49
  66. package/dist/primitives/Card.stories.svelte +22 -22
  67. package/dist/primitives/Card.svelte +28 -28
  68. package/dist/primitives/Checkbox/Checkbox.stories.svelte +84 -84
  69. package/dist/primitives/Checkbox/Checkbox.svelte +88 -88
  70. package/dist/primitives/DarkModeToggle.spec.js +357 -357
  71. package/dist/primitives/DarkModeToggle.stories.svelte +57 -57
  72. package/dist/primitives/DarkModeToggle.svelte +136 -136
  73. package/dist/primitives/Drawer/Drawer.stories.svelte +80 -80
  74. package/dist/primitives/Drawer/Drawer.svelte +120 -120
  75. package/dist/primitives/Dropdown/Dropdown.stories.svelte +137 -137
  76. package/dist/primitives/Dropdown/Dropdown.svelte +14 -14
  77. package/dist/primitives/Dropdown/DropdownItem.svelte +80 -80
  78. package/dist/primitives/Icons/ArrowLeft.svelte +8 -8
  79. package/dist/primitives/Icons/ArrowRight.svelte +8 -8
  80. package/dist/primitives/Icons/Availability.svelte +14 -14
  81. package/dist/primitives/Icons/Back.svelte +14 -14
  82. package/dist/primitives/Icons/CheckCircle.svelte +6 -6
  83. package/dist/primitives/Icons/CheckCircleOutline.svelte +15 -15
  84. package/dist/primitives/Icons/ChevronLeft.svelte +4 -4
  85. package/dist/primitives/Icons/ChevronRight.svelte +4 -4
  86. package/dist/primitives/Icons/Copy.svelte +15 -15
  87. package/dist/primitives/Icons/Cross.svelte +5 -5
  88. package/dist/primitives/Icons/DownArrow.svelte +8 -8
  89. package/dist/primitives/Icons/ErrorCircle.svelte +6 -6
  90. package/dist/primitives/Icons/FacebookIcon.svelte +2 -2
  91. package/dist/primitives/Icons/Home.svelte +15 -15
  92. package/dist/primitives/Icons/Icon.spec.js +169 -169
  93. package/dist/primitives/Icons/Icon.stories.svelte +100 -100
  94. package/dist/primitives/Icons/Icon.svelte +52 -52
  95. package/dist/primitives/Icons/IconGallery.stories.svelte +235 -235
  96. package/dist/primitives/Icons/Info.svelte +7 -7
  97. package/dist/primitives/Icons/InstagramIcon.svelte +4 -4
  98. package/dist/primitives/Icons/LogoInstagram.svelte +2 -2
  99. package/dist/primitives/Icons/Message.svelte +15 -15
  100. package/dist/primitives/Icons/MoonIcon.svelte +5 -5
  101. package/dist/primitives/Icons/More.svelte +21 -21
  102. package/dist/primitives/Icons/MoreHori.spec.js +61 -61
  103. package/dist/primitives/Icons/MoreHori.svelte +22 -22
  104. package/dist/primitives/Icons/Notification.svelte +14 -14
  105. package/dist/primitives/Icons/Payment.svelte +14 -14
  106. package/dist/primitives/Icons/Profile.svelte +21 -21
  107. package/dist/primitives/Icons/Reload.svelte +29 -29
  108. package/dist/primitives/Icons/Shows.svelte +21 -21
  109. package/dist/primitives/Icons/Signout.svelte +21 -21
  110. package/dist/primitives/Icons/SunIcon.svelte +8 -8
  111. package/dist/primitives/Icons/TiktokIcon.svelte +2 -2
  112. package/dist/primitives/Icons/TwitterIcon.svelte +2 -2
  113. package/dist/primitives/Icons/WarningIcon.spec.js +18 -18
  114. package/dist/primitives/Icons/WarningIcon.svelte +5 -5
  115. package/dist/primitives/Input/Input.spec.js +573 -573
  116. package/dist/primitives/Input/Input.stories.svelte +139 -139
  117. package/dist/primitives/Input/Input.svelte +391 -391
  118. package/dist/primitives/Input/Select.spec.js +218 -218
  119. package/dist/primitives/Input/Select.stories.svelte +112 -112
  120. package/dist/primitives/Input/Select.svelte +128 -128
  121. package/dist/primitives/Input/Textarea.stories.svelte +137 -137
  122. package/dist/primitives/Input/Textarea.svelte +35 -35
  123. package/dist/primitives/Label/Label.svelte +37 -37
  124. package/dist/primitives/Modal/Modal.spec.js +95 -95
  125. package/dist/primitives/Modal/Modal.stories.svelte +86 -86
  126. package/dist/primitives/Modal/Modal.svelte +158 -158
  127. package/dist/primitives/NumberInput/NumberInput.svelte +106 -0
  128. package/dist/primitives/NumberInput/NumberInput.svelte.d.ts +23 -0
  129. package/dist/primitives/NumberInput/NumberInput.svelte.d.ts.map +1 -0
  130. package/dist/primitives/NumberInput/index.d.ts +2 -0
  131. package/dist/primitives/NumberInput/index.d.ts.map +1 -0
  132. package/dist/primitives/NumberInput/index.js +1 -0
  133. package/dist/primitives/Pagination/Pagination.stories.svelte +76 -76
  134. package/dist/primitives/Pagination/Pagination.svelte +261 -261
  135. package/dist/primitives/Radio/Radio.stories.svelte +80 -80
  136. package/dist/primitives/Radio/Radio.svelte +67 -67
  137. package/dist/primitives/Skeleton/CardPlaceholder.svelte +87 -87
  138. package/dist/primitives/Skeleton/ImagePlaceholder.svelte +59 -59
  139. package/dist/primitives/Skeleton/ListPlaceholder.svelte +76 -76
  140. package/dist/primitives/Skeleton/Skeleton.stories.svelte +151 -151
  141. package/dist/primitives/Skeleton/Skeleton.svelte +26 -26
  142. package/dist/primitives/Spinner/Spinner.spec.js +75 -75
  143. package/dist/primitives/Spinner/Spinner.stories.svelte +29 -29
  144. package/dist/primitives/Spinner/Spinner.svelte +20 -20
  145. package/dist/primitives/Tabs/TabItem.svelte +49 -49
  146. package/dist/primitives/Tabs/Tabs.stories.svelte +112 -112
  147. package/dist/primitives/Tabs/Tabs.svelte +123 -123
  148. package/dist/primitives/Toggle.spec.js +127 -127
  149. package/dist/primitives/Toggle.stories.svelte +92 -92
  150. package/dist/primitives/Toggle.svelte +71 -71
  151. package/dist/primitives/Typography/Typography.svelte +53 -53
  152. package/dist/primitives/ValidationError.spec.js +103 -103
  153. package/dist/primitives/ValidationError.stories.svelte +69 -69
  154. package/dist/primitives/ValidationError.svelte +29 -29
  155. package/dist/recipes/CropImage/CropImage.spec.js +216 -216
  156. package/dist/recipes/CropImage/CropImage.stories.svelte +104 -104
  157. package/dist/recipes/CropImage/CropImage.svelte +238 -238
  158. package/dist/recipes/ImageUploader/ImageUploader.stories.svelte +125 -125
  159. package/dist/recipes/ImageUploader/ImageUploader.svelte +804 -804
  160. package/dist/recipes/Toaster/Toaster.stories.svelte +62 -62
  161. package/dist/recipes/feedback/EmptyState/EmptyState.svelte +1 -1
  162. package/dist/recipes/feedback/ErrorDisplay.spec.js +69 -69
  163. package/dist/recipes/feedback/ErrorDisplay.stories.svelte +101 -101
  164. package/dist/recipes/feedback/ErrorDisplay.svelte +1 -1
  165. package/dist/recipes/feedback/StatusIndicator/StatusIndicator.spec.js +129 -129
  166. package/dist/recipes/feedback/StatusIndicator/StatusIndicator.svelte +157 -157
  167. package/dist/recipes/fields/CheckboxField.svelte +85 -85
  168. package/dist/recipes/fields/FormField.svelte +58 -58
  169. package/dist/recipes/fields/RadioGroup.svelte +95 -95
  170. package/dist/recipes/fields/SelectField.svelte +80 -80
  171. package/dist/recipes/fields/TextareaField.svelte +97 -97
  172. package/dist/recipes/fields/ToggleField.svelte +60 -60
  173. package/dist/recipes/fields/index.js +7 -7
  174. package/dist/recipes/inputs/MultiSelect.spec.js +257 -257
  175. package/dist/recipes/inputs/MultiSelect.stories.svelte +133 -133
  176. package/dist/recipes/inputs/MultiSelect.svelte +249 -276
  177. package/dist/recipes/inputs/MultiSelect.svelte.d.ts +28 -17
  178. package/dist/recipes/inputs/MultiSelect.svelte.d.ts.map +1 -1
  179. package/dist/recipes/inputs/OTPInput.spec.js +238 -238
  180. package/dist/recipes/inputs/OTPInput.stories.svelte +162 -162
  181. package/dist/recipes/inputs/OTPInput.svelte +29 -29
  182. package/dist/recipes/inputs/PasswordInput.svelte +22 -22
  183. package/dist/recipes/inputs/PasswordStrengthIndicator/PasswordStrengthIndicator.spec.js +173 -173
  184. package/dist/recipes/inputs/PasswordStrengthIndicator/PasswordStrengthIndicator.svelte +42 -42
  185. package/dist/recipes/inputs/PlaceAutocomplete/PlaceAutocomplete.spec.js +300 -300
  186. package/dist/recipes/inputs/PlaceAutocomplete/PlaceAutocomplete.stories.svelte +123 -123
  187. package/dist/recipes/inputs/PlaceAutocomplete/PlaceAutocomplete.svelte +326 -326
  188. package/dist/recipes/inputs/Search.svelte +37 -37
  189. package/dist/recipes/inputs/SelectDropdown.svelte +57 -57
  190. package/dist/recipes/modals/AlertModal.svelte +130 -130
  191. package/dist/recipes/modals/ConfirmationModal.spec.js +191 -191
  192. package/dist/recipes/modals/ConfirmationModal.stories.svelte +119 -119
  193. package/dist/recipes/modals/ConfirmationModal.svelte +152 -152
  194. package/dist/recipes/modals/InputModal.svelte +182 -182
  195. package/dist/recipes/modals/ModalStateManager.spec.js +100 -100
  196. package/dist/recipes/modals/ModalStateManager.svelte +77 -77
  197. package/dist/recipes/modals/ModalTestWrapper.svelte +65 -65
  198. package/dist/recipes/modals/StatusModal.svelte +206 -206
  199. package/dist/services/EventService.js +75 -75
  200. package/dist/services/EventService.spec.js +217 -217
  201. package/dist/services/ShowService.spec.js +342 -342
  202. package/dist/stores/auth.js +36 -36
  203. package/dist/stores/auth.spec.js +139 -139
  204. package/dist/stores/toaster.js +13 -13
  205. package/dist/stories/ButtonAuditReview.stories.svelte +14 -14
  206. package/dist/stories/ButtonAuditReview.svelte +427 -427
  207. package/dist/stories/PatternsGallery.stories.svelte +19 -19
  208. package/dist/stories/PatternsGallery.svelte +206 -206
  209. package/dist/stories/PrimitivesGallery.stories.svelte +19 -19
  210. package/dist/stories/PrimitivesGallery.svelte +725 -725
  211. package/dist/stories/RecipesGallery.stories.svelte +19 -19
  212. package/dist/stories/RecipesGallery.svelte +271 -271
  213. package/dist/stories/button-audit-manifest.json +11186 -11186
  214. package/dist/tailwind/preset.cjs +82 -82
  215. package/dist/tokens/tokens.css +87 -87
  216. package/dist/utils/utils.js +354 -354
  217. package/package.json +283 -283
@@ -1,257 +1,257 @@
1
- import { render, screen, fireEvent } from "@testing-library/svelte";
2
- import userEvent from "@testing-library/user-event";
3
- import { expect, describe, test, vi } from "vitest";
4
- import MultiSelect from "./MultiSelect.svelte";
5
-
6
- const sampleItems = [
7
- { name: "Option 1", value: "opt1" },
8
- { name: "Option 2", value: "opt2" },
9
- { name: "Option 3", value: "opt3" },
10
- { name: "Option 4", value: "opt4" },
11
- ];
12
-
13
- function setupTest(args = {}) {
14
- const user = userEvent.setup();
15
- const { component } = render(MultiSelect, {
16
- props: {
17
- items: sampleItems,
18
- ...args,
19
- },
20
- });
21
- return { user, component };
22
- }
23
-
24
- describe("MultiSelect Component Tests", () => {
25
- test("Renders multiselect with label", () => {
26
- setupTest({
27
- label: "Test Label",
28
- id: "test-multiselect",
29
- });
30
- expect(screen.getByText("Test Label")).toBeInTheDocument();
31
- });
32
-
33
- test("Shows placeholder when no value selected", () => {
34
- setupTest({
35
- placeholder: "Select options",
36
- });
37
- expect(screen.getByText("Select options")).toBeInTheDocument();
38
- });
39
-
40
- test("Opens dropdown on click", async () => {
41
- const { user } = setupTest();
42
- const trigger = screen.getByRole("button", { name: /select options/i });
43
-
44
- await user.click(trigger);
45
-
46
- expect(screen.getByRole("listbox")).toBeInTheDocument();
47
- expect(screen.getByText("Option 1")).toBeInTheDocument();
48
- expect(screen.getByText("Option 2")).toBeInTheDocument();
49
- });
50
-
51
- test("Selects multiple items", async () => {
52
- const { user } = setupTest();
53
- const trigger = screen.getByRole("button", { name: /select options/i });
54
-
55
- await user.click(trigger);
56
- // Click on options in dropdown
57
- const options = screen.getAllByRole("option");
58
- await user.click(options[0]); // Option 1
59
- await user.click(options[2]); // Option 3
60
-
61
- // Both tags should be visible (may have duplicates due to dropdown still open)
62
- expect(screen.getAllByText("Option 1").length).toBeGreaterThanOrEqual(1);
63
- expect(screen.getAllByText("Option 3").length).toBeGreaterThanOrEqual(1);
64
- });
65
-
66
- test("Dropdown stays open after selection", async () => {
67
- const { user } = setupTest();
68
- const trigger = screen.getByRole("button", { name: /select options/i });
69
-
70
- await user.click(trigger);
71
- await user.click(screen.getByText("Option 1"));
72
-
73
- // Dropdown should still be open for multi-select
74
- expect(screen.getByRole("listbox")).toBeInTheDocument();
75
- });
76
-
77
- test("Shows selected values as tags", () => {
78
- setupTest({
79
- value: ["opt1", "opt2"],
80
- });
81
- expect(screen.getByText("Option 1")).toBeInTheDocument();
82
- expect(screen.getByText("Option 2")).toBeInTheDocument();
83
- });
84
-
85
- test("Removes item when clicking tag remove button", async () => {
86
- const { user } = setupTest({
87
- value: ["opt1", "opt2"],
88
- });
89
-
90
- // Find the remove button for Option 1
91
- const removeButtons = screen.getAllByRole("button", { name: /remove/i });
92
- await user.click(removeButtons[0]);
93
-
94
- // Option 1 should be removed, Option 2 should remain
95
- expect(screen.queryByText("Option 1")).not.toBeInTheDocument();
96
- expect(screen.getByText("Option 2")).toBeInTheDocument();
97
- });
98
-
99
- test("Clears all selections when clicking clear button", async () => {
100
- const { user } = setupTest({
101
- value: ["opt1", "opt2", "opt3"],
102
- });
103
-
104
- const clearButton = screen.getByRole("button", { name: /clear all/i });
105
- await user.click(clearButton);
106
-
107
- expect(screen.getByText("Select options")).toBeInTheDocument();
108
- expect(screen.queryByText("Option 1")).not.toBeInTheDocument();
109
- });
110
-
111
- test("Hides clear button when hideClear is true", () => {
112
- setupTest({
113
- value: ["opt1"],
114
- hideClear: true,
115
- });
116
- expect(screen.queryByRole("button", { name: /clear all/i })).not.toBeInTheDocument();
117
- });
118
-
119
- test("Dispatches change event on selection", async () => {
120
- const { user, component } = setupTest();
121
- const changeSpy = vi.fn();
122
- component.$on("change", changeSpy);
123
-
124
- const trigger = screen.getByRole("button", { name: /select options/i });
125
- await user.click(trigger);
126
- await user.click(screen.getByText("Option 2"));
127
-
128
- expect(changeSpy).toHaveBeenCalled();
129
- expect(changeSpy.mock.calls[0][0].detail.value).toContain("opt2");
130
- });
131
-
132
- test("Shows required indicator when required", () => {
133
- setupTest({
134
- label: "Required Field",
135
- required: true,
136
- });
137
- expect(screen.getByText("*")).toBeInTheDocument();
138
- });
139
-
140
- test("Displays error state", () => {
141
- setupTest({
142
- error: "Please select at least one option",
143
- });
144
- expect(screen.getByText("Please select at least one option")).toBeInTheDocument();
145
- });
146
-
147
- test("Disables multiselect when disabled prop is true", async () => {
148
- const { user } = setupTest({
149
- disabled: true,
150
- });
151
- const trigger = screen.getByRole("button");
152
-
153
- expect(trigger).toBeDisabled();
154
- await user.click(trigger);
155
- expect(screen.queryByRole("listbox")).not.toBeInTheDocument();
156
- });
157
-
158
- test("Navigates options with keyboard", async () => {
159
- const { user } = setupTest();
160
- const trigger = screen.getByRole("button", { name: /select options/i });
161
-
162
- await user.click(trigger);
163
- await user.keyboard("{ArrowDown}");
164
- await user.keyboard("{Enter}");
165
-
166
- // First option should be selected
167
- expect(screen.getByText("Option 1")).toBeInTheDocument();
168
- });
169
-
170
- test("Closes dropdown on Escape key", async () => {
171
- const { user } = setupTest();
172
- const trigger = screen.getByRole("button", { name: /select options/i });
173
-
174
- await user.click(trigger);
175
- expect(screen.getByRole("listbox")).toBeInTheDocument();
176
-
177
- await user.keyboard("{Escape}");
178
- expect(screen.queryByRole("listbox")).not.toBeInTheDocument();
179
- });
180
-
181
- test("Toggles item off when clicking selected item", async () => {
182
- const { user } = setupTest({
183
- value: ["opt1"],
184
- });
185
- // Get the main trigger button (not remove buttons)
186
- const triggers = screen.getAllByRole("button");
187
- const trigger = triggers.find(b => b.getAttribute("aria-haspopup") === "listbox");
188
-
189
- await user.click(trigger);
190
-
191
- // Click Option 1 to deselect it
192
- const option1 = screen.getByRole("option", { name: /option 1/i });
193
- await user.click(option1);
194
-
195
- // Should show placeholder now
196
- expect(screen.getByText("Select options")).toBeInTheDocument();
197
- });
198
-
199
- test("Applies animate focus class by default", () => {
200
- setupTest();
201
- const trigger = screen.getByRole("button");
202
- expect(trigger).toHaveClass("multiselect-animate-focus");
203
- });
204
-
205
- test("Does not apply animate focus when disabled", () => {
206
- setupTest({
207
- animateFocus: false,
208
- });
209
- const trigger = screen.getByRole("button");
210
- expect(trigger).not.toHaveClass("multiselect-animate-focus");
211
- });
212
-
213
- test("Shows checkbox for each option", async () => {
214
- const { user } = setupTest();
215
- const trigger = screen.getByRole("button", { name: /select options/i });
216
-
217
- await user.click(trigger);
218
-
219
- // Each option should have a checkbox element
220
- const options = screen.getAllByRole("option");
221
- options.forEach(option => {
222
- expect(option.querySelector(".multiselect-checkbox")).toBeInTheDocument();
223
- });
224
- });
225
-
226
- test("Shows checked checkbox for selected items", async () => {
227
- const { user } = setupTest({
228
- value: ["opt1"],
229
- });
230
- // Get the main trigger button (not remove buttons)
231
- const triggers = screen.getAllByRole("button");
232
- const trigger = triggers.find(b => b.getAttribute("aria-haspopup") === "listbox");
233
-
234
- await user.click(trigger);
235
-
236
- const selectedOption = screen.getByRole("option", { name: /option 1/i });
237
- expect(selectedOption.querySelector(".multiselect-checkbox-checked")).toBeInTheDocument();
238
- });
239
-
240
- test("Sets aria-multiselectable on listbox", async () => {
241
- const { user } = setupTest();
242
- const trigger = screen.getByRole("button", { name: /select options/i });
243
-
244
- await user.click(trigger);
245
-
246
- const listbox = screen.getByRole("listbox");
247
- expect(listbox).toHaveAttribute("aria-multiselectable", "true");
248
- });
249
-
250
- test("Initializes empty value as array", () => {
251
- setupTest({
252
- value: null,
253
- });
254
- // Should not crash and show placeholder
255
- expect(screen.getByText("Select options")).toBeInTheDocument();
256
- });
257
- });
1
+ import { render, screen, fireEvent } from "@testing-library/svelte";
2
+ import userEvent from "@testing-library/user-event";
3
+ import { expect, describe, test, vi } from "vitest";
4
+ import MultiSelect from "./MultiSelect.svelte";
5
+
6
+ const sampleItems = [
7
+ { name: "Option 1", value: "opt1" },
8
+ { name: "Option 2", value: "opt2" },
9
+ { name: "Option 3", value: "opt3" },
10
+ { name: "Option 4", value: "opt4" },
11
+ ];
12
+
13
+ function setupTest(args = {}) {
14
+ const user = userEvent.setup();
15
+ const { component } = render(MultiSelect, {
16
+ props: {
17
+ items: sampleItems,
18
+ ...args,
19
+ },
20
+ });
21
+ return { user, component };
22
+ }
23
+
24
+ describe("MultiSelect Component Tests", () => {
25
+ test("Renders multiselect with label", () => {
26
+ setupTest({
27
+ label: "Test Label",
28
+ id: "test-multiselect",
29
+ });
30
+ expect(screen.getByText("Test Label")).toBeInTheDocument();
31
+ });
32
+
33
+ test("Shows placeholder when no value selected", () => {
34
+ setupTest({
35
+ placeholder: "Select options",
36
+ });
37
+ expect(screen.getByText("Select options")).toBeInTheDocument();
38
+ });
39
+
40
+ test("Opens dropdown on click", async () => {
41
+ const { user } = setupTest();
42
+ const trigger = screen.getByRole("button", { name: /select options/i });
43
+
44
+ await user.click(trigger);
45
+
46
+ expect(screen.getByRole("listbox")).toBeInTheDocument();
47
+ expect(screen.getByText("Option 1")).toBeInTheDocument();
48
+ expect(screen.getByText("Option 2")).toBeInTheDocument();
49
+ });
50
+
51
+ test("Selects multiple items", async () => {
52
+ const { user } = setupTest();
53
+ const trigger = screen.getByRole("button", { name: /select options/i });
54
+
55
+ await user.click(trigger);
56
+ // Click on options in dropdown
57
+ const options = screen.getAllByRole("option");
58
+ await user.click(options[0]); // Option 1
59
+ await user.click(options[2]); // Option 3
60
+
61
+ // Both tags should be visible (may have duplicates due to dropdown still open)
62
+ expect(screen.getAllByText("Option 1").length).toBeGreaterThanOrEqual(1);
63
+ expect(screen.getAllByText("Option 3").length).toBeGreaterThanOrEqual(1);
64
+ });
65
+
66
+ test("Dropdown stays open after selection", async () => {
67
+ const { user } = setupTest();
68
+ const trigger = screen.getByRole("button", { name: /select options/i });
69
+
70
+ await user.click(trigger);
71
+ await user.click(screen.getByText("Option 1"));
72
+
73
+ // Dropdown should still be open for multi-select
74
+ expect(screen.getByRole("listbox")).toBeInTheDocument();
75
+ });
76
+
77
+ test("Shows selected values as tags", () => {
78
+ setupTest({
79
+ value: ["opt1", "opt2"],
80
+ });
81
+ expect(screen.getByText("Option 1")).toBeInTheDocument();
82
+ expect(screen.getByText("Option 2")).toBeInTheDocument();
83
+ });
84
+
85
+ test("Removes item when clicking tag remove button", async () => {
86
+ const { user } = setupTest({
87
+ value: ["opt1", "opt2"],
88
+ });
89
+
90
+ // Find the remove button for Option 1
91
+ const removeButtons = screen.getAllByRole("button", { name: /remove/i });
92
+ await user.click(removeButtons[0]);
93
+
94
+ // Option 1 should be removed, Option 2 should remain
95
+ expect(screen.queryByText("Option 1")).not.toBeInTheDocument();
96
+ expect(screen.getByText("Option 2")).toBeInTheDocument();
97
+ });
98
+
99
+ test("Clears all selections when clicking clear button", async () => {
100
+ const { user } = setupTest({
101
+ value: ["opt1", "opt2", "opt3"],
102
+ });
103
+
104
+ const clearButton = screen.getByRole("button", { name: /clear all/i });
105
+ await user.click(clearButton);
106
+
107
+ expect(screen.getByText("Select options")).toBeInTheDocument();
108
+ expect(screen.queryByText("Option 1")).not.toBeInTheDocument();
109
+ });
110
+
111
+ test("Hides clear button when hideClear is true", () => {
112
+ setupTest({
113
+ value: ["opt1"],
114
+ hideClear: true,
115
+ });
116
+ expect(screen.queryByRole("button", { name: /clear all/i })).not.toBeInTheDocument();
117
+ });
118
+
119
+ test("Dispatches change event on selection", async () => {
120
+ const { user, component } = setupTest();
121
+ const changeSpy = vi.fn();
122
+ component.$on("change", changeSpy);
123
+
124
+ const trigger = screen.getByRole("button", { name: /select options/i });
125
+ await user.click(trigger);
126
+ await user.click(screen.getByText("Option 2"));
127
+
128
+ expect(changeSpy).toHaveBeenCalled();
129
+ expect(changeSpy.mock.calls[0][0].detail.value).toContain("opt2");
130
+ });
131
+
132
+ test("Shows required indicator when required", () => {
133
+ setupTest({
134
+ label: "Required Field",
135
+ required: true,
136
+ });
137
+ expect(screen.getByText("*")).toBeInTheDocument();
138
+ });
139
+
140
+ test("Displays error state", () => {
141
+ setupTest({
142
+ error: "Please select at least one option",
143
+ });
144
+ expect(screen.getByText("Please select at least one option")).toBeInTheDocument();
145
+ });
146
+
147
+ test("Disables multiselect when disabled prop is true", async () => {
148
+ const { user } = setupTest({
149
+ disabled: true,
150
+ });
151
+ const trigger = screen.getByRole("button");
152
+
153
+ expect(trigger).toBeDisabled();
154
+ await user.click(trigger);
155
+ expect(screen.queryByRole("listbox")).not.toBeInTheDocument();
156
+ });
157
+
158
+ test("Navigates options with keyboard", async () => {
159
+ const { user } = setupTest();
160
+ const trigger = screen.getByRole("button", { name: /select options/i });
161
+
162
+ await user.click(trigger);
163
+ await user.keyboard("{ArrowDown}");
164
+ await user.keyboard("{Enter}");
165
+
166
+ // First option should be selected
167
+ expect(screen.getByText("Option 1")).toBeInTheDocument();
168
+ });
169
+
170
+ test("Closes dropdown on Escape key", async () => {
171
+ const { user } = setupTest();
172
+ const trigger = screen.getByRole("button", { name: /select options/i });
173
+
174
+ await user.click(trigger);
175
+ expect(screen.getByRole("listbox")).toBeInTheDocument();
176
+
177
+ await user.keyboard("{Escape}");
178
+ expect(screen.queryByRole("listbox")).not.toBeInTheDocument();
179
+ });
180
+
181
+ test("Toggles item off when clicking selected item", async () => {
182
+ const { user } = setupTest({
183
+ value: ["opt1"],
184
+ });
185
+ // Get the main trigger button (not remove buttons)
186
+ const triggers = screen.getAllByRole("button");
187
+ const trigger = triggers.find(b => b.getAttribute("aria-haspopup") === "listbox");
188
+
189
+ await user.click(trigger);
190
+
191
+ // Click Option 1 to deselect it
192
+ const option1 = screen.getByRole("option", { name: /option 1/i });
193
+ await user.click(option1);
194
+
195
+ // Should show placeholder now
196
+ expect(screen.getByText("Select options")).toBeInTheDocument();
197
+ });
198
+
199
+ test("Applies animate focus class by default", () => {
200
+ setupTest();
201
+ const trigger = screen.getByRole("button");
202
+ expect(trigger).toHaveClass("multiselect-animate-focus");
203
+ });
204
+
205
+ test("Does not apply animate focus when disabled", () => {
206
+ setupTest({
207
+ animateFocus: false,
208
+ });
209
+ const trigger = screen.getByRole("button");
210
+ expect(trigger).not.toHaveClass("multiselect-animate-focus");
211
+ });
212
+
213
+ test("Shows checkbox for each option", async () => {
214
+ const { user } = setupTest();
215
+ const trigger = screen.getByRole("button", { name: /select options/i });
216
+
217
+ await user.click(trigger);
218
+
219
+ // Each option should have a checkbox element
220
+ const options = screen.getAllByRole("option");
221
+ options.forEach(option => {
222
+ expect(option.querySelector(".multiselect-checkbox")).toBeInTheDocument();
223
+ });
224
+ });
225
+
226
+ test("Shows checked checkbox for selected items", async () => {
227
+ const { user } = setupTest({
228
+ value: ["opt1"],
229
+ });
230
+ // Get the main trigger button (not remove buttons)
231
+ const triggers = screen.getAllByRole("button");
232
+ const trigger = triggers.find(b => b.getAttribute("aria-haspopup") === "listbox");
233
+
234
+ await user.click(trigger);
235
+
236
+ const selectedOption = screen.getByRole("option", { name: /option 1/i });
237
+ expect(selectedOption.querySelector(".multiselect-checkbox-checked")).toBeInTheDocument();
238
+ });
239
+
240
+ test("Sets aria-multiselectable on listbox", async () => {
241
+ const { user } = setupTest();
242
+ const trigger = screen.getByRole("button", { name: /select options/i });
243
+
244
+ await user.click(trigger);
245
+
246
+ const listbox = screen.getByRole("listbox");
247
+ expect(listbox).toHaveAttribute("aria-multiselectable", "true");
248
+ });
249
+
250
+ test("Initializes empty value as array", () => {
251
+ setupTest({
252
+ value: null,
253
+ });
254
+ // Should not crash and show placeholder
255
+ expect(screen.getByText("Select options")).toBeInTheDocument();
256
+ });
257
+ });