@getmicdrop/svelte-components 5.17.1 → 5.17.3

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 (382) hide show
  1. package/dist/calendar/AboutShow/AboutShow.svelte +187 -187
  2. package/dist/calendar/Calendar/MiniMonthCalendar.svelte +782 -782
  3. package/dist/calendar/FAQs/FAQs.svelte +77 -77
  4. package/dist/calendar/MonthSwitcher/MonthSwitcher.svelte +126 -126
  5. package/dist/calendar/OrderSummary/OrderSummary.svelte +457 -457
  6. package/dist/calendar/PublicCard/PublicCard.svelte +146 -146
  7. package/dist/calendar/ShowCard/ShowCard.svelte +157 -157
  8. package/dist/calendar/ShowTimeCard/ShowTimeCard.svelte +61 -61
  9. package/dist/components/Heading.spec.d.ts +2 -0
  10. package/dist/components/Heading.spec.d.ts.map +1 -0
  11. package/dist/components/Heading.spec.js +89 -0
  12. package/dist/components/Heading.svelte +60 -60
  13. package/dist/components/Layout/AppShell.svelte +104 -104
  14. package/dist/components/Layout/ContentSection.svelte +80 -80
  15. package/dist/components/Layout/Grid.svelte +4 -4
  16. package/dist/components/Layout/Heading.svelte +81 -81
  17. package/dist/components/Layout/PageContainer.svelte +69 -69
  18. package/dist/components/Layout/Responsive.svelte +75 -75
  19. package/dist/components/Layout/Section.svelte +80 -80
  20. package/dist/components/Layout/ShowOnDesktop.svelte +37 -37
  21. package/dist/components/Layout/ShowOnMobile.svelte +37 -37
  22. package/dist/components/Layout/Sidebar.svelte +108 -108
  23. package/dist/components/Layout/Stack.spec.js +1 -1
  24. package/dist/components/Layout/Stack.svelte +6 -6
  25. package/dist/components/Layout/Text.svelte +87 -87
  26. package/dist/components/Layout/TwoColumn.svelte +108 -108
  27. package/dist/components/Layout/__tests__/AppShell.test.js +140 -0
  28. package/dist/components/Text.spec.d.ts +2 -0
  29. package/dist/components/Text.spec.d.ts.map +1 -0
  30. package/dist/components/Text.spec.js +89 -0
  31. package/dist/components/Text.svelte +53 -53
  32. package/dist/constants/validation.js +91 -91
  33. package/dist/constants/validation.spec.js +64 -64
  34. package/dist/datetime/__tests__/format.test.js +1 -1
  35. package/dist/datetime/__tests__/parse.test.js +1 -1
  36. package/dist/datetime/__tests__/timezone.test.js +1 -1
  37. package/dist/datetime/parse.js +1 -1
  38. package/dist/forms/createFormStore.svelte.js +1 -0
  39. package/dist/forms/createFormStore.svelte.spec.d.ts +2 -0
  40. package/dist/forms/createFormStore.svelte.spec.d.ts.map +1 -0
  41. package/dist/forms/createFormStore.svelte.spec.js +388 -0
  42. package/dist/index.js +57 -57
  43. package/dist/index.spec.js +369 -369
  44. package/dist/patterns/chat/ChatActivityNotice.spec.d.ts +2 -0
  45. package/dist/patterns/chat/ChatActivityNotice.spec.d.ts.map +1 -0
  46. package/dist/patterns/chat/ChatActivityNotice.spec.js +59 -0
  47. package/dist/patterns/chat/ChatActivityNotice.svelte +41 -41
  48. package/dist/patterns/chat/ChatBubble.spec.d.ts +2 -0
  49. package/dist/patterns/chat/ChatBubble.spec.d.ts.map +1 -0
  50. package/dist/patterns/chat/ChatBubble.spec.js +91 -0
  51. package/dist/patterns/chat/ChatBubble.svelte +95 -95
  52. package/dist/patterns/chat/ChatContainer.spec.d.ts +2 -0
  53. package/dist/patterns/chat/ChatContainer.spec.d.ts.map +1 -0
  54. package/dist/patterns/chat/ChatContainer.spec.js +30 -0
  55. package/dist/patterns/chat/ChatContainer.svelte +46 -46
  56. package/dist/patterns/chat/ChatDateDivider.spec.d.ts +2 -0
  57. package/dist/patterns/chat/ChatDateDivider.spec.d.ts.map +1 -0
  58. package/dist/patterns/chat/ChatDateDivider.spec.js +30 -0
  59. package/dist/patterns/chat/ChatDateDivider.svelte +27 -27
  60. package/dist/patterns/chat/ChatInvitationBubble.spec.d.ts +2 -0
  61. package/dist/patterns/chat/ChatInvitationBubble.spec.d.ts.map +1 -0
  62. package/dist/patterns/chat/ChatInvitationBubble.spec.js +46 -0
  63. package/dist/patterns/chat/ChatInvitationBubble.svelte +37 -37
  64. package/dist/patterns/chat/ChatInvitationNotice.spec.d.ts +2 -0
  65. package/dist/patterns/chat/ChatInvitationNotice.spec.d.ts.map +1 -0
  66. package/dist/patterns/chat/ChatInvitationNotice.spec.js +32 -0
  67. package/dist/patterns/chat/ChatInvitationNotice.svelte +27 -27
  68. package/dist/patterns/chat/ChatMessageGroup.spec.d.ts +2 -0
  69. package/dist/patterns/chat/ChatMessageGroup.spec.d.ts.map +1 -0
  70. package/dist/patterns/chat/ChatMessageGroup.spec.js +58 -0
  71. package/dist/patterns/chat/ChatMessageGroup.svelte +57 -57
  72. package/dist/patterns/chat/ChatSlotUpdate.spec.d.ts +2 -0
  73. package/dist/patterns/chat/ChatSlotUpdate.spec.d.ts.map +1 -0
  74. package/dist/patterns/chat/ChatSlotUpdate.spec.js +65 -0
  75. package/dist/patterns/chat/ChatSlotUpdate.svelte +46 -46
  76. package/dist/patterns/chat/ChatStatusBadge.spec.d.ts +2 -0
  77. package/dist/patterns/chat/ChatStatusBadge.spec.d.ts.map +1 -0
  78. package/dist/patterns/chat/ChatStatusBadge.spec.js +79 -0
  79. package/dist/patterns/chat/ChatStatusBadge.svelte +91 -91
  80. package/dist/patterns/chat/ChatStatusTransition.spec.d.ts +2 -0
  81. package/dist/patterns/chat/ChatStatusTransition.spec.d.ts.map +1 -0
  82. package/dist/patterns/chat/ChatStatusTransition.spec.js +81 -0
  83. package/dist/patterns/chat/ChatStatusTransition.svelte +64 -64
  84. package/dist/patterns/chat/ChatTextBubble.spec.d.ts +2 -0
  85. package/dist/patterns/chat/ChatTextBubble.spec.d.ts.map +1 -0
  86. package/dist/patterns/chat/ChatTextBubble.spec.js +35 -0
  87. package/dist/patterns/chat/ChatTextBubble.svelte +41 -41
  88. package/dist/patterns/chat/index.js +22 -22
  89. package/dist/patterns/data/DataGrid.svelte +45 -45
  90. package/dist/patterns/data/DataList.svelte +24 -24
  91. package/dist/patterns/data/DataTable.spec.js +61 -0
  92. package/dist/patterns/data/DataTable.svelte +36 -36
  93. package/dist/patterns/forms/FormActions.spec.js +95 -95
  94. package/dist/patterns/forms/FormActions.stories.svelte +97 -97
  95. package/dist/patterns/forms/FormActions.svelte +46 -46
  96. package/dist/patterns/forms/FormGrid.spec.js +34 -0
  97. package/dist/patterns/forms/FormGrid.svelte +33 -33
  98. package/dist/patterns/forms/FormSection.svelte +32 -32
  99. package/dist/patterns/forms/FormValidationSummary.stories.svelte +83 -83
  100. package/dist/patterns/forms/FormValidationSummary.svelte +74 -74
  101. package/dist/patterns/index.js +21 -21
  102. package/dist/patterns/layout/Sidebar.spec.js +240 -1
  103. package/dist/patterns/layout/Sidebar.svelte +39 -39
  104. package/dist/patterns/layout/SidebarTestWrapper.svelte +34 -0
  105. package/dist/patterns/layout/SidebarTestWrapper.svelte.d.ts +23 -0
  106. package/dist/patterns/layout/SidebarTestWrapper.svelte.d.ts.map +1 -0
  107. package/dist/patterns/layout/index.js +29 -29
  108. package/dist/patterns/navigation/BottomNav.stories.svelte +117 -117
  109. package/dist/patterns/navigation/BottomNav.svelte +74 -74
  110. package/dist/patterns/navigation/Header.spec.js +123 -0
  111. package/dist/patterns/navigation/Header.stories.svelte +77 -77
  112. package/dist/patterns/navigation/Header.svelte +251 -251
  113. package/dist/patterns/page/PageHeader.svelte +18 -18
  114. package/dist/patterns/page/PageLayout.svelte +40 -40
  115. package/dist/patterns/page/PageLoader.spec.js +57 -57
  116. package/dist/patterns/page/PageLoader.stories.svelte +137 -137
  117. package/dist/patterns/page/PageLoader.svelte +24 -24
  118. package/dist/patterns/page/SectionHeader.svelte +29 -29
  119. package/dist/presets/badges.js +112 -112
  120. package/dist/presets/buttons.js +76 -76
  121. package/dist/presets/index.js +9 -9
  122. package/dist/primitives/Accordion/Accordion.spec.js +112 -2
  123. package/dist/primitives/Accordion/Accordion.stories.svelte +75 -75
  124. package/dist/primitives/Accordion/Accordion.svelte +42 -42
  125. package/dist/primitives/Accordion/AccordionItem.svelte +95 -95
  126. package/dist/primitives/Accordion/AccordionToggleWrapper.test.svelte +28 -0
  127. package/dist/primitives/Accordion/AccordionToggleWrapper.test.svelte.d.ts +7 -0
  128. package/dist/primitives/Accordion/AccordionToggleWrapper.test.svelte.d.ts.map +1 -0
  129. package/dist/primitives/Alert/Alert.spec.js +173 -173
  130. package/dist/primitives/Alert/Alert.stories.svelte +88 -88
  131. package/dist/primitives/Alert/Alert.svelte +27 -27
  132. package/dist/primitives/Avatar/Avatar.spec.js +23 -0
  133. package/dist/primitives/Avatar/Avatar.stories.svelte +94 -94
  134. package/dist/primitives/Avatar/Avatar.svelte +66 -66
  135. package/dist/primitives/AvatarButton/AvatarButton.svelte +57 -57
  136. package/dist/primitives/Badges/Badge.spec.js +144 -144
  137. package/dist/primitives/Badges/Badge.stories.svelte +86 -86
  138. package/dist/primitives/Badges/Badge.svelte +99 -99
  139. package/dist/primitives/BottomSheet/BottomSheet.spec.js +238 -136
  140. package/dist/primitives/BottomSheet/BottomSheet.stories.svelte +83 -83
  141. package/dist/primitives/BottomSheet/BottomSheet.svelte +115 -115
  142. package/dist/primitives/BottomSheet/BottomSheetWithActions.test.svelte +20 -0
  143. package/dist/primitives/BottomSheet/BottomSheetWithActions.test.svelte.d.ts +10 -0
  144. package/dist/primitives/BottomSheet/BottomSheetWithActions.test.svelte.d.ts.map +1 -0
  145. package/dist/primitives/Breadcrumb/Breadcrumb.spec.js +123 -123
  146. package/dist/primitives/Breadcrumb/Breadcrumb.stories.svelte +23 -23
  147. package/dist/primitives/Breadcrumb/Breadcrumb.svelte +99 -99
  148. package/dist/primitives/Button/Button.spec.js +225 -225
  149. package/dist/primitives/Button/Button.stories.svelte +76 -76
  150. package/dist/primitives/Button/Button.svelte +278 -278
  151. package/dist/primitives/Button/ButtonGroup.spec.d.ts +2 -0
  152. package/dist/primitives/Button/ButtonGroup.spec.d.ts.map +1 -0
  153. package/dist/primitives/Button/ButtonGroup.spec.js +44 -0
  154. package/dist/primitives/Button/ButtonGroup.svelte +50 -50
  155. package/dist/primitives/Button/ButtonSaveDemo.spec.js +146 -146
  156. package/dist/primitives/Button/ButtonSaveDemo.svelte +25 -25
  157. package/dist/primitives/Button/ButtonVariantShowcase.svelte +129 -129
  158. package/dist/primitives/Card.spec.js +49 -49
  159. package/dist/primitives/Card.stories.svelte +22 -22
  160. package/dist/primitives/Card.svelte +28 -28
  161. package/dist/primitives/CardAction/CardAction.svelte +68 -68
  162. package/dist/primitives/Checkbox/Checkbox.spec.js +32 -0
  163. package/dist/primitives/Checkbox/Checkbox.stories.svelte +84 -84
  164. package/dist/primitives/Checkbox/Checkbox.svelte +88 -88
  165. package/dist/primitives/DarkModeToggle.spec.js +390 -390
  166. package/dist/primitives/DarkModeToggle.stories.svelte +57 -57
  167. package/dist/primitives/DarkModeToggle.svelte +136 -136
  168. package/dist/primitives/Drawer/Drawer.spec.js +437 -0
  169. package/dist/primitives/Drawer/Drawer.stories.svelte +80 -80
  170. package/dist/primitives/Drawer/Drawer.svelte +224 -224
  171. package/dist/primitives/Drawer/DrawerTestWrapper.svelte +86 -0
  172. package/dist/primitives/Drawer/DrawerTestWrapper.svelte.d.ts +26 -0
  173. package/dist/primitives/Drawer/DrawerTestWrapper.svelte.d.ts.map +1 -0
  174. package/dist/primitives/Dropdown/Dropdown.spec.js +116 -0
  175. package/dist/primitives/Dropdown/Dropdown.stories.svelte +137 -137
  176. package/dist/primitives/Dropdown/Dropdown.svelte +170 -170
  177. package/dist/primitives/Dropdown/DropdownDivider.spec.d.ts +2 -0
  178. package/dist/primitives/Dropdown/DropdownDivider.spec.d.ts.map +1 -0
  179. package/dist/primitives/Dropdown/DropdownDivider.spec.js +30 -0
  180. package/dist/primitives/Dropdown/DropdownDivider.svelte +9 -9
  181. package/dist/primitives/Dropdown/DropdownItem.spec.js +155 -1
  182. package/dist/primitives/Dropdown/DropdownItem.svelte +80 -80
  183. package/dist/primitives/Dropdown/DropdownItemTestWrapper.svelte +43 -0
  184. package/dist/primitives/Dropdown/DropdownItemTestWrapper.svelte.d.ts +17 -0
  185. package/dist/primitives/Dropdown/DropdownItemTestWrapper.svelte.d.ts.map +1 -0
  186. package/dist/primitives/Helper/Helper.spec.d.ts +2 -0
  187. package/dist/primitives/Helper/Helper.spec.d.ts.map +1 -0
  188. package/dist/primitives/Helper/Helper.spec.js +57 -0
  189. package/dist/primitives/Helper/Helper.svelte +33 -33
  190. package/dist/primitives/Icons/ArrowLeft.svelte +8 -8
  191. package/dist/primitives/Icons/ArrowRight.svelte +8 -8
  192. package/dist/primitives/Icons/Availability.svelte +14 -14
  193. package/dist/primitives/Icons/Back.svelte +14 -14
  194. package/dist/primitives/Icons/CheckCircle.svelte +6 -6
  195. package/dist/primitives/Icons/CheckCircleOutline.svelte +15 -15
  196. package/dist/primitives/Icons/ChevronLeft.svelte +4 -4
  197. package/dist/primitives/Icons/ChevronRight.svelte +4 -4
  198. package/dist/primitives/Icons/Copy.svelte +15 -15
  199. package/dist/primitives/Icons/Cross.svelte +5 -5
  200. package/dist/primitives/Icons/DownArrow.svelte +8 -8
  201. package/dist/primitives/Icons/ErrorCircle.svelte +6 -6
  202. package/dist/primitives/Icons/FacebookIcon.svelte +2 -2
  203. package/dist/primitives/Icons/Home.svelte +15 -15
  204. package/dist/primitives/Icons/Icon.spec.js +169 -169
  205. package/dist/primitives/Icons/Icon.stories.svelte +100 -100
  206. package/dist/primitives/Icons/Icon.svelte +52 -52
  207. package/dist/primitives/Icons/IconGallery.stories.svelte +235 -235
  208. package/dist/primitives/Icons/Info.svelte +7 -7
  209. package/dist/primitives/Icons/InstagramIcon.svelte +4 -4
  210. package/dist/primitives/Icons/LogoInstagram.svelte +2 -2
  211. package/dist/primitives/Icons/Message.svelte +15 -15
  212. package/dist/primitives/Icons/MoonIcon.svelte +5 -5
  213. package/dist/primitives/Icons/More.svelte +21 -21
  214. package/dist/primitives/Icons/MoreHori.spec.js +61 -61
  215. package/dist/primitives/Icons/MoreHori.svelte +22 -22
  216. package/dist/primitives/Icons/Notification.svelte +14 -14
  217. package/dist/primitives/Icons/Payment.svelte +14 -14
  218. package/dist/primitives/Icons/Profile.svelte +21 -21
  219. package/dist/primitives/Icons/Reload.svelte +29 -29
  220. package/dist/primitives/Icons/Shows.svelte +21 -21
  221. package/dist/primitives/Icons/Signout.svelte +21 -21
  222. package/dist/primitives/Icons/SunIcon.svelte +8 -8
  223. package/dist/primitives/Icons/TiktokIcon.svelte +2 -2
  224. package/dist/primitives/Icons/TwitterIcon.svelte +2 -2
  225. package/dist/primitives/Icons/WarningIcon.spec.js +18 -18
  226. package/dist/primitives/Icons/WarningIcon.svelte +5 -5
  227. package/dist/primitives/Input/Input.spec.js +1235 -573
  228. package/dist/primitives/Input/Input.stories.svelte +139 -139
  229. package/dist/primitives/Input/Input.svelte +423 -423
  230. package/dist/primitives/Input/Select.spec.js +632 -218
  231. package/dist/primitives/Input/Select.stories.svelte +112 -112
  232. package/dist/primitives/Input/Select.svelte +252 -252
  233. package/dist/primitives/Input/Textarea.stories.svelte +137 -137
  234. package/dist/primitives/Input/Textarea.svelte +105 -105
  235. package/dist/primitives/Label/Label.spec.js +9 -0
  236. package/dist/primitives/Label/Label.svelte +37 -37
  237. package/dist/primitives/LandingButton/LandingButton.spec.d.ts +2 -0
  238. package/dist/primitives/LandingButton/LandingButton.spec.d.ts.map +1 -0
  239. package/dist/primitives/LandingButton/LandingButton.spec.js +61 -0
  240. package/dist/primitives/LandingButton/LandingButton.svelte +92 -92
  241. package/dist/primitives/MenuItem/MenuItem.spec.d.ts +2 -0
  242. package/dist/primitives/MenuItem/MenuItem.spec.d.ts.map +1 -0
  243. package/dist/primitives/MenuItem/MenuItem.spec.js +130 -0
  244. package/dist/primitives/MenuItem/MenuItem.svelte +85 -85
  245. package/dist/primitives/Modal/Modal.spec.js +314 -99
  246. package/dist/primitives/Modal/Modal.stories.svelte +86 -86
  247. package/dist/primitives/Modal/Modal.svelte +181 -181
  248. package/dist/primitives/NavItem/NavItem.spec.d.ts +2 -0
  249. package/dist/primitives/NavItem/NavItem.spec.d.ts.map +1 -0
  250. package/dist/primitives/NavItem/NavItem.spec.js +97 -0
  251. package/dist/primitives/NavItem/NavItem.svelte +75 -75
  252. package/dist/primitives/Pagination/Pagination.stories.svelte +76 -76
  253. package/dist/primitives/Pagination/Pagination.svelte +261 -261
  254. package/dist/primitives/Radio/Radio.stories.svelte +80 -80
  255. package/dist/primitives/Radio/Radio.svelte +67 -67
  256. package/dist/primitives/SearchResultItem/SearchResultItem.spec.d.ts +2 -0
  257. package/dist/primitives/SearchResultItem/SearchResultItem.spec.d.ts.map +1 -0
  258. package/dist/primitives/SearchResultItem/SearchResultItem.spec.js +78 -0
  259. package/dist/primitives/SearchResultItem/SearchResultItem.svelte +109 -109
  260. package/dist/primitives/SidebarToggle/SidebarToggle.spec.d.ts +2 -0
  261. package/dist/primitives/SidebarToggle/SidebarToggle.spec.d.ts.map +1 -0
  262. package/dist/primitives/SidebarToggle/SidebarToggle.spec.js +61 -0
  263. package/dist/primitives/SidebarToggle/SidebarToggle.svelte +55 -55
  264. package/dist/primitives/Skeleton/CardPlaceholder.svelte +87 -87
  265. package/dist/primitives/Skeleton/ImagePlaceholder.svelte +59 -59
  266. package/dist/primitives/Skeleton/ListPlaceholder.svelte +76 -76
  267. package/dist/primitives/Skeleton/Skeleton.stories.svelte +151 -151
  268. package/dist/primitives/Skeleton/Skeleton.svelte +26 -26
  269. package/dist/primitives/Spinner/Spinner.spec.js +84 -71
  270. package/dist/primitives/Spinner/Spinner.stories.svelte +29 -29
  271. package/dist/primitives/Spinner/Spinner.svelte +20 -20
  272. package/dist/primitives/Tabs/TabItem.svelte +49 -49
  273. package/dist/primitives/Tabs/Tabs.stories.svelte +112 -112
  274. package/dist/primitives/Tabs/Tabs.svelte +137 -137
  275. package/dist/primitives/Toggle.spec.js +221 -146
  276. package/dist/primitives/Toggle.stories.svelte +92 -92
  277. package/dist/primitives/Toggle.svelte +141 -141
  278. package/dist/primitives/ToggleTestWrapper.svelte +30 -0
  279. package/dist/primitives/ToggleTestWrapper.svelte.d.ts +29 -0
  280. package/dist/primitives/ToggleTestWrapper.svelte.d.ts.map +1 -0
  281. package/dist/primitives/Tooltip/Tooltip.spec.d.ts +2 -0
  282. package/dist/primitives/Tooltip/Tooltip.spec.d.ts.map +1 -0
  283. package/dist/primitives/Tooltip/Tooltip.spec.js +126 -0
  284. package/dist/primitives/Tooltip/Tooltip.svelte +83 -83
  285. package/dist/primitives/Typography/Typography.svelte +53 -53
  286. package/dist/primitives/ValidationError.spec.js +103 -103
  287. package/dist/primitives/ValidationError.stories.svelte +69 -69
  288. package/dist/primitives/ValidationError.svelte +29 -29
  289. package/dist/primitives/index.js +113 -113
  290. package/dist/recipes/CropImage/CropImage.spec.js +208 -208
  291. package/dist/recipes/CropImage/CropImage.stories.svelte +104 -104
  292. package/dist/recipes/CropImage/CropImage.svelte +241 -241
  293. package/dist/recipes/ImageUploader/ImageUploader.stories.svelte +125 -125
  294. package/dist/recipes/ImageUploader/ImageUploader.svelte +992 -992
  295. package/dist/recipes/SuperLogin/SuperLogin.spec.js +2 -2
  296. package/dist/recipes/SuperLogin/SuperLogin.svelte +1 -1
  297. package/dist/recipes/Toaster/Toaster.stories.svelte +62 -62
  298. package/dist/recipes/feedback/EmptyState/EmptyState.svelte +1 -1
  299. package/dist/recipes/feedback/ErrorDisplay.spec.js +69 -69
  300. package/dist/recipes/feedback/ErrorDisplay.stories.svelte +101 -101
  301. package/dist/recipes/feedback/ErrorDisplay.svelte +1 -1
  302. package/dist/recipes/feedback/StatusIndicator/StatusIndicator.spec.js +133 -133
  303. package/dist/recipes/feedback/StatusIndicator/StatusIndicator.svelte +157 -157
  304. package/dist/recipes/fields/CheckboxField.svelte +85 -85
  305. package/dist/recipes/fields/FormField.svelte +58 -58
  306. package/dist/recipes/fields/RadioGroup.svelte +95 -95
  307. package/dist/recipes/fields/SelectField.svelte +82 -82
  308. package/dist/recipes/fields/TextareaField.svelte +97 -97
  309. package/dist/recipes/fields/ToggleField.svelte +60 -60
  310. package/dist/recipes/fields/index.js +7 -7
  311. package/dist/recipes/inputs/MultiSelect.spec.js +263 -263
  312. package/dist/recipes/inputs/MultiSelect.stories.svelte +133 -133
  313. package/dist/recipes/inputs/MultiSelect.svelte +283 -283
  314. package/dist/recipes/inputs/OTPInput.spec.js +251 -251
  315. package/dist/recipes/inputs/OTPInput.stories.svelte +162 -162
  316. package/dist/recipes/inputs/OTPInput.svelte +117 -117
  317. package/dist/recipes/inputs/PasswordInput.svelte +22 -22
  318. package/dist/recipes/inputs/PasswordStrengthIndicator/PasswordStrengthIndicator.svelte +131 -131
  319. package/dist/recipes/inputs/PlaceAutocomplete/PlaceAutocomplete.stories.svelte +123 -123
  320. package/dist/recipes/inputs/PlaceAutocomplete/PlaceAutocomplete.svelte +336 -336
  321. package/dist/recipes/inputs/Search.spec.js +66 -2
  322. package/dist/recipes/inputs/Search.svelte +102 -102
  323. package/dist/recipes/inputs/index.js +7 -7
  324. package/dist/recipes/modals/AlertModal.svelte +130 -130
  325. package/dist/recipes/modals/ConfirmationModal.spec.js +396 -206
  326. package/dist/recipes/modals/ConfirmationModal.stories.svelte +119 -119
  327. package/dist/recipes/modals/ConfirmationModal.svelte +162 -162
  328. package/dist/recipes/modals/InputModal.svelte +182 -182
  329. package/dist/recipes/modals/ModalStateManager.spec.js +100 -100
  330. package/dist/recipes/modals/ModalStateManager.svelte +77 -77
  331. package/dist/recipes/modals/ModalTestWrapper.svelte +65 -65
  332. package/dist/recipes/modals/StatusModal.svelte +206 -206
  333. package/dist/services/EventService.js +79 -79
  334. package/dist/services/EventService.spec.js +217 -217
  335. package/dist/services/ShowService.js +144 -144
  336. package/dist/services/ShowService.spec.js +345 -345
  337. package/dist/stores/auth.svelte.spec.d.ts +2 -0
  338. package/dist/stores/auth.svelte.spec.d.ts.map +1 -0
  339. package/dist/stores/auth.svelte.spec.js +112 -0
  340. package/dist/stores/formDataStore.svelte.spec.d.ts +2 -0
  341. package/dist/stores/formDataStore.svelte.spec.d.ts.map +1 -0
  342. package/dist/stores/formDataStore.svelte.spec.js +150 -0
  343. package/dist/stores/formSave.svelte.spec.d.ts +2 -0
  344. package/dist/stores/formSave.svelte.spec.d.ts.map +1 -0
  345. package/dist/stores/formSave.svelte.spec.js +196 -0
  346. package/dist/stores/navigation.spec.d.ts +2 -0
  347. package/dist/stores/navigation.spec.d.ts.map +1 -0
  348. package/dist/stores/navigation.spec.js +53 -0
  349. package/dist/stores/toaster.js +13 -13
  350. package/dist/stories/ButtonAuditReview.stories.svelte +14 -14
  351. package/dist/stories/ButtonAuditReview.svelte +427 -427
  352. package/dist/stories/PatternsGallery.stories.svelte +19 -19
  353. package/dist/stories/PatternsGallery.svelte +206 -206
  354. package/dist/stories/PrimitivesGallery.stories.svelte +19 -19
  355. package/dist/stories/PrimitivesGallery.svelte +756 -756
  356. package/dist/stories/RecipesGallery.stories.svelte +19 -19
  357. package/dist/stories/RecipesGallery.svelte +454 -454
  358. package/dist/stories/button-audit-manifest.json +11186 -11186
  359. package/dist/tailwind/preset.cjs +82 -82
  360. package/dist/telemetry.js +402 -402
  361. package/dist/telemetry.server.js +212 -212
  362. package/dist/telemetry.server.spec.js +437 -437
  363. package/dist/telemetry.spec.js +1168 -1168
  364. package/dist/tokens/tokens.css +87 -87
  365. package/dist/tokens/typography-base.css +163 -163
  366. package/dist/tokens/utilities.css +353 -353
  367. package/dist/utils/apiConfig.js +117 -117
  368. package/dist/utils/apiConfig.spec.js +219 -219
  369. package/dist/utils/greetings.js +187 -187
  370. package/dist/utils/haptic.spec.d.ts +2 -0
  371. package/dist/utils/haptic.spec.d.ts.map +1 -0
  372. package/dist/utils/haptic.spec.js +153 -0
  373. package/dist/utils/imageOptimizer.spec.d.ts +2 -0
  374. package/dist/utils/imageOptimizer.spec.d.ts.map +1 -0
  375. package/dist/utils/imageOptimizer.spec.js +201 -0
  376. package/dist/utils/imageValidation.js +121 -121
  377. package/dist/utils/logger.spec.d.ts +2 -0
  378. package/dist/utils/logger.spec.d.ts.map +1 -0
  379. package/dist/utils/logger.spec.js +95 -0
  380. package/dist/utils/transitions.js +4 -4
  381. package/dist/utils/utils.js +693 -693
  382. package/package.json +292 -293
@@ -1,390 +1,390 @@
1
- import { render, screen, fireEvent } from "@testing-library/svelte";
2
- import userEvent from "@testing-library/user-event";
3
- import { expect, describe, test, vi, beforeEach, afterEach } from "vitest";
4
- import DarkModeToggle from "./DarkModeToggle.svelte";
5
-
6
- // Mock localStorage
7
- const localStorageMock = {
8
- store: {},
9
- getItem: vi.fn((key) => localStorageMock.store[key] || null),
10
- setItem: vi.fn((key, value) => {
11
- localStorageMock.store[key] = value;
12
- }),
13
- removeItem: vi.fn((key) => {
14
- delete localStorageMock.store[key];
15
- }),
16
- clear: vi.fn(() => {
17
- localStorageMock.store = {};
18
- }),
19
- };
20
-
21
- // Mock matchMedia
22
- const matchMediaMock = vi.fn((query) => ({
23
- matches: false,
24
- media: query,
25
- onchange: null,
26
- addListener: vi.fn(),
27
- removeListener: vi.fn(),
28
- addEventListener: vi.fn(),
29
- removeEventListener: vi.fn(),
30
- dispatchEvent: vi.fn(),
31
- }));
32
-
33
- function setupTest() {
34
- const user = userEvent.setup();
35
- const { component, container } = render(DarkModeToggle);
36
- return { user, component, container };
37
- }
38
-
39
- describe("DarkModeToggle Component Tests", () => {
40
- beforeEach(() => {
41
- localStorageMock.clear();
42
- Object.defineProperty(window, "localStorage", { value: localStorageMock });
43
- Object.defineProperty(window, "matchMedia", { value: matchMediaMock });
44
- // Reset any dark class
45
- document.documentElement.classList.remove("dark");
46
- });
47
-
48
- afterEach(() => {
49
- vi.clearAllMocks();
50
- });
51
-
52
- test("Renders toggle button", () => {
53
- setupTest();
54
- const button = screen.getByRole("button", { name: /toggle theme/i });
55
- expect(button).toBeInTheDocument();
56
- });
57
-
58
- test("Has aria-label for accessibility", () => {
59
- setupTest();
60
- const button = screen.getByRole("button");
61
- expect(button).toHaveAttribute("aria-label", "Toggle theme");
62
- });
63
-
64
- test("Has title attribute for tooltip", () => {
65
- setupTest();
66
- const button = screen.getByRole("button");
67
- expect(button).toHaveAttribute("title");
68
- });
69
-
70
- test("Contains sun icon SVG", () => {
71
- const { container } = setupTest();
72
- const sunIcon = container.querySelector('span.text-amber-500 svg');
73
- expect(sunIcon).toBeInTheDocument();
74
- });
75
-
76
- test("Contains moon icon SVG", () => {
77
- const { container } = setupTest();
78
- const moonIcon = container.querySelector('span:not(.text-amber-500) svg');
79
- expect(moonIcon).toBeInTheDocument();
80
- });
81
-
82
- test("Contains auto mode split icon", () => {
83
- const { container } = setupTest();
84
- // In auto mode, there should be a div with absolute positioning containing the split icon
85
- const autoDiv = container.querySelector('div.absolute.inset-0');
86
- expect(autoDiv).toBeInTheDocument();
87
- });
88
-
89
- test("Auto icon is visible in auto mode (default)", () => {
90
- const { container } = setupTest();
91
- const autoDiv = container.querySelector('div.absolute.inset-0');
92
- expect(autoDiv).toBeInTheDocument();
93
- });
94
-
95
- test("Sun and moon single icons have hidden classes in auto mode", () => {
96
- const { container } = setupTest();
97
- // In auto mode, the main sun/moon icons should have opacity-0 and scale-0 classes
98
- const sunIcon = container.querySelector('span.text-amber-500');
99
- const moonIcon = container.querySelectorAll('span')[1]; // Second span is moon
100
- expect(sunIcon).toHaveClass('opacity-0');
101
- expect(sunIcon).toHaveClass('scale-0');
102
- expect(moonIcon).toHaveClass('opacity-0');
103
- expect(moonIcon).toHaveClass('scale-0');
104
- });
105
-
106
- test("Button has relative class for positioning", () => {
107
- setupTest();
108
- const button = screen.getByRole("button");
109
- expect(button).toHaveClass("relative");
110
- });
111
-
112
- test("Button has focus outline removed", () => {
113
- setupTest();
114
- const button = screen.getByRole("button");
115
- expect(button).toHaveClass("focus:outline-hidden");
116
- });
117
-
118
- test("Button click handler is called on interaction", async () => {
119
- const { user } = setupTest();
120
- const button = screen.getByRole("button");
121
-
122
- // Verify button is interactive
123
- await user.click(button);
124
-
125
- // After click, localStorage should be updated (auto -> light)
126
- expect(localStorageMock.setItem).toHaveBeenCalled();
127
- });
128
-
129
- test("SVG icons have correct dimensions", () => {
130
- const { container } = setupTest();
131
- const svgs = container.querySelectorAll('svg');
132
- // All SVGs should have width and height of either 20 (main icons) or 12 (auto mode icons)
133
- expect(svgs.length).toBeGreaterThan(0);
134
- svgs.forEach(svg => {
135
- const width = svg.getAttribute('width');
136
- const height = svg.getAttribute('height');
137
- expect(['20', '12']).toContain(width);
138
- expect(['20', '12']).toContain(height);
139
- });
140
- });
141
-
142
- test("Sun icon has correct color class", () => {
143
- const { container } = setupTest();
144
- const sunIcon = container.querySelector('span.text-amber-500');
145
- expect(sunIcon).toHaveClass("text-amber-500");
146
- });
147
-
148
- test("Button has overflow hidden for auto icon", () => {
149
- setupTest();
150
- const button = screen.getByRole("button");
151
- expect(button).toHaveClass("overflow-hidden");
152
- });
153
-
154
- test("Auto mode div has absolute positioning", () => {
155
- const { container } = setupTest();
156
- const autoDiv = container.querySelector('div.absolute.inset-0');
157
- expect(autoDiv).toHaveClass("absolute");
158
- expect(autoDiv).toHaveClass("inset-0");
159
- });
160
- });
161
-
162
- describe("DarkModeToggle 3-State Cycle", () => {
163
- beforeEach(() => {
164
- localStorageMock.clear();
165
- Object.defineProperty(window, "localStorage", { value: localStorageMock });
166
- Object.defineProperty(window, "matchMedia", { value: matchMediaMock });
167
- document.documentElement.classList.remove("dark");
168
- });
169
-
170
- afterEach(() => {
171
- vi.clearAllMocks();
172
- });
173
-
174
- test("First click: auto -> light, saves 'light' to localStorage", async () => {
175
- const { user } = setupTest();
176
- const button = screen.getByRole("button");
177
-
178
- await user.click(button);
179
-
180
- expect(localStorageMock.setItem).toHaveBeenCalledWith("theme", "light");
181
- });
182
-
183
- test("Second click: light -> dark, saves 'dark' to localStorage", async () => {
184
- const { user } = setupTest();
185
- const button = screen.getByRole("button");
186
-
187
- // First click: auto -> light
188
- await user.click(button);
189
- // Second click: light -> dark
190
- await user.click(button);
191
-
192
- expect(localStorageMock.setItem).toHaveBeenLastCalledWith("theme", "dark");
193
- });
194
-
195
- test("Third click: dark -> auto, removes theme from localStorage", async () => {
196
- const { user } = setupTest();
197
- const button = screen.getByRole("button");
198
-
199
- // First click: auto -> light
200
- await user.click(button);
201
- // Second click: light -> dark
202
- await user.click(button);
203
- // Third click: dark -> auto
204
- await user.click(button);
205
-
206
- expect(localStorageMock.removeItem).toHaveBeenCalledWith("theme");
207
- });
208
-
209
- test("Full cycle: auto -> light -> dark -> auto", async () => {
210
- const { user, container } = setupTest();
211
- const button = screen.getByRole("button");
212
-
213
- // Initially auto - auto div should exist
214
- expect(container.querySelector("div.absolute.inset-0")).toBeInTheDocument();
215
-
216
- // Click 1: auto -> light
217
- await user.click(button);
218
- const sunSpan = container.querySelector('span.text-amber-500');
219
- // In light mode, sun should be visible (no opacity-0 class)
220
- expect(sunSpan.classList.contains("opacity-0")).toBe(false);
221
- expect(container.querySelector("div.absolute.inset-0")).not.toBeInTheDocument();
222
-
223
- // Click 2: light -> dark
224
- await user.click(button);
225
- // Find the direct child spans of the button (not nested spans)
226
- const buttonSpans = Array.from(button.querySelectorAll(':scope > span'));
227
- const moonSpan = buttonSpans.find(
228
- span => !span.classList.contains('text-amber-500')
229
- );
230
- // In dark mode, moon should be visible (no opacity-0)
231
- expect(moonSpan.classList.contains("opacity-0")).toBe(false);
232
- expect(sunSpan.classList.contains("opacity-0")).toBe(true);
233
-
234
- // Click 3: dark -> auto
235
- await user.click(button);
236
- expect(container.querySelector("div.absolute.inset-0")).toBeInTheDocument();
237
- // After returning to auto, both icons should be hidden
238
- expect(sunSpan.classList.contains("opacity-0")).toBe(true);
239
- });
240
-
241
- test("Dark class applied when switching to dark mode", async () => {
242
- const { user } = setupTest();
243
- const button = screen.getByRole("button");
244
-
245
- // auto -> light
246
- await user.click(button);
247
- expect(document.documentElement.classList.contains("dark")).toBe(false);
248
-
249
- // light -> dark
250
- await user.click(button);
251
- expect(document.documentElement.classList.contains("dark")).toBe(true);
252
- });
253
-
254
- test("Dark class removed when switching back to light", async () => {
255
- const { user } = setupTest();
256
- const button = screen.getByRole("button");
257
-
258
- // auto -> light -> dark
259
- await user.click(button);
260
- await user.click(button);
261
- expect(document.documentElement.classList.contains("dark")).toBe(true);
262
-
263
- // dark -> auto (system is light by default in mock)
264
- await user.click(button);
265
- expect(document.documentElement.classList.contains("dark")).toBe(false);
266
- });
267
- });
268
-
269
- describe("DarkModeToggle Initial State", () => {
270
- beforeEach(() => {
271
- localStorageMock.clear();
272
- Object.defineProperty(window, "localStorage", { value: localStorageMock });
273
- Object.defineProperty(window, "matchMedia", { value: matchMediaMock });
274
- document.documentElement.classList.remove("dark");
275
- });
276
-
277
- afterEach(() => {
278
- vi.clearAllMocks();
279
- });
280
-
281
- test("Initializes to auto mode when no localStorage value", () => {
282
- const { container } = setupTest();
283
- const autoDiv = container.querySelector("div.absolute.inset-0");
284
- expect(autoDiv).toBeInTheDocument();
285
- });
286
-
287
- test("Initializes to dark mode when localStorage has 'dark'", () => {
288
- localStorageMock.store.theme = "dark";
289
- const { container } = setupTest();
290
- const button = screen.getByRole("button");
291
- const buttonSpans = Array.from(button.querySelectorAll(':scope > span'));
292
- const moonSpan = buttonSpans.find(
293
- span => !span.classList.contains('text-amber-500')
294
- );
295
- // In dark mode, moon should be visible (no opacity-0)
296
- expect(moonSpan.classList.contains("opacity-0")).toBe(false);
297
- // Sun should be hidden
298
- const sunSpan = container.querySelector('span.text-amber-500');
299
- expect(sunSpan.classList.contains("opacity-0")).toBe(true);
300
- });
301
-
302
- test("Initializes to light mode when localStorage has 'light'", () => {
303
- localStorageMock.store.theme = "light";
304
- const { container } = setupTest();
305
- const button = screen.getByRole("button");
306
- const sunSpan = container.querySelector('span.text-amber-500');
307
- // In light mode, sun should be visible (no opacity-0)
308
- expect(sunSpan.classList.contains("opacity-0")).toBe(false);
309
- // Moon should be hidden - need to check all spans that have transition-all
310
- const allSpansWithTransition = Array.from(button.querySelectorAll('span.transition-all'));
311
- const moonSpan = allSpansWithTransition.find(
312
- span => !span.classList.contains('text-amber-500')
313
- );
314
- expect(moonSpan).toBeDefined();
315
- expect(moonSpan.classList.contains("opacity-0")).toBe(true);
316
- });
317
-
318
- test("Reads from localStorage on mount", () => {
319
- localStorageMock.store.theme = "dark";
320
- setupTest();
321
- expect(localStorageMock.getItem).toHaveBeenCalledWith("theme");
322
- });
323
-
324
- test("Checks system preference for auto mode", () => {
325
- setupTest();
326
- expect(matchMediaMock).toHaveBeenCalledWith("(prefers-color-scheme: dark)");
327
- });
328
-
329
- test("Auto mode follows system dark preference", () => {
330
- matchMediaMock.mockReturnValue({
331
- matches: true, // System prefers dark
332
- media: "(prefers-color-scheme: dark)",
333
- addEventListener: vi.fn(),
334
- removeEventListener: vi.fn(),
335
- });
336
- setupTest();
337
- // In auto mode with system dark preference, dark class should be applied
338
- expect(document.documentElement.classList.contains("dark")).toBe(true);
339
- });
340
-
341
- test("Auto mode follows system light preference", () => {
342
- matchMediaMock.mockReturnValue({
343
- matches: false, // System prefers light
344
- media: "(prefers-color-scheme: dark)",
345
- addEventListener: vi.fn(),
346
- removeEventListener: vi.fn(),
347
- });
348
- setupTest();
349
- expect(document.documentElement.classList.contains("dark")).toBe(false);
350
- });
351
- });
352
-
353
- describe("DarkModeToggle Title/Tooltip", () => {
354
- beforeEach(() => {
355
- localStorageMock.clear();
356
- Object.defineProperty(window, "localStorage", { value: localStorageMock });
357
- Object.defineProperty(window, "matchMedia", { value: matchMediaMock });
358
- document.documentElement.classList.remove("dark");
359
- });
360
-
361
- afterEach(() => {
362
- vi.clearAllMocks();
363
- });
364
-
365
- test("Shows correct title in auto mode", () => {
366
- setupTest();
367
- const button = screen.getByRole("button");
368
- expect(button.title).toContain("Auto");
369
- expect(button.title).toContain("light mode");
370
- });
371
-
372
- test("Shows correct title in light mode", async () => {
373
- const { user } = setupTest();
374
- const button = screen.getByRole("button");
375
-
376
- await user.click(button); // auto -> light
377
- expect(button.title).toContain("Light mode");
378
- expect(button.title).toContain("dark mode");
379
- });
380
-
381
- test("Shows correct title in dark mode", async () => {
382
- const { user } = setupTest();
383
- const button = screen.getByRole("button");
384
-
385
- await user.click(button); // auto -> light
386
- await user.click(button); // light -> dark
387
- expect(button.title).toContain("Dark mode");
388
- expect(button.title).toContain("auto");
389
- });
390
- });
1
+ import { render, screen, fireEvent } from "@testing-library/svelte";
2
+ import userEvent from "@testing-library/user-event";
3
+ import { expect, describe, test, vi, beforeEach, afterEach } from "vitest";
4
+ import DarkModeToggle from "./DarkModeToggle.svelte";
5
+
6
+ // Mock localStorage
7
+ const localStorageMock = {
8
+ store: {},
9
+ getItem: vi.fn((key) => localStorageMock.store[key] || null),
10
+ setItem: vi.fn((key, value) => {
11
+ localStorageMock.store[key] = value;
12
+ }),
13
+ removeItem: vi.fn((key) => {
14
+ delete localStorageMock.store[key];
15
+ }),
16
+ clear: vi.fn(() => {
17
+ localStorageMock.store = {};
18
+ }),
19
+ };
20
+
21
+ // Mock matchMedia
22
+ const matchMediaMock = vi.fn((query) => ({
23
+ matches: false,
24
+ media: query,
25
+ onchange: null,
26
+ addListener: vi.fn(),
27
+ removeListener: vi.fn(),
28
+ addEventListener: vi.fn(),
29
+ removeEventListener: vi.fn(),
30
+ dispatchEvent: vi.fn(),
31
+ }));
32
+
33
+ function setupTest() {
34
+ const user = userEvent.setup();
35
+ const { component, container } = render(DarkModeToggle);
36
+ return { user, component, container };
37
+ }
38
+
39
+ describe("DarkModeToggle Component Tests", () => {
40
+ beforeEach(() => {
41
+ localStorageMock.clear();
42
+ Object.defineProperty(window, "localStorage", { value: localStorageMock });
43
+ Object.defineProperty(window, "matchMedia", { value: matchMediaMock });
44
+ // Reset any dark class
45
+ document.documentElement.classList.remove("dark");
46
+ });
47
+
48
+ afterEach(() => {
49
+ vi.clearAllMocks();
50
+ });
51
+
52
+ test("Renders toggle button", () => {
53
+ setupTest();
54
+ const button = screen.getByRole("button", { name: /toggle theme/i });
55
+ expect(button).toBeInTheDocument();
56
+ });
57
+
58
+ test("Has aria-label for accessibility", () => {
59
+ setupTest();
60
+ const button = screen.getByRole("button");
61
+ expect(button).toHaveAttribute("aria-label", "Toggle theme");
62
+ });
63
+
64
+ test("Has title attribute for tooltip", () => {
65
+ setupTest();
66
+ const button = screen.getByRole("button");
67
+ expect(button).toHaveAttribute("title");
68
+ });
69
+
70
+ test("Contains sun icon SVG", () => {
71
+ const { container } = setupTest();
72
+ const sunIcon = container.querySelector('span.text-amber-500 svg');
73
+ expect(sunIcon).toBeInTheDocument();
74
+ });
75
+
76
+ test("Contains moon icon SVG", () => {
77
+ const { container } = setupTest();
78
+ const moonIcon = container.querySelector('span:not(.text-amber-500) svg');
79
+ expect(moonIcon).toBeInTheDocument();
80
+ });
81
+
82
+ test("Contains auto mode split icon", () => {
83
+ const { container } = setupTest();
84
+ // In auto mode, there should be a div with absolute positioning containing the split icon
85
+ const autoDiv = container.querySelector('div.absolute.inset-0');
86
+ expect(autoDiv).toBeInTheDocument();
87
+ });
88
+
89
+ test("Auto icon is visible in auto mode (default)", () => {
90
+ const { container } = setupTest();
91
+ const autoDiv = container.querySelector('div.absolute.inset-0');
92
+ expect(autoDiv).toBeInTheDocument();
93
+ });
94
+
95
+ test("Sun and moon single icons have hidden classes in auto mode", () => {
96
+ const { container } = setupTest();
97
+ // In auto mode, the main sun/moon icons should have opacity-0 and scale-0 classes
98
+ const sunIcon = container.querySelector('span.text-amber-500');
99
+ const moonIcon = container.querySelectorAll('span')[1]; // Second span is moon
100
+ expect(sunIcon).toHaveClass('opacity-0');
101
+ expect(sunIcon).toHaveClass('scale-0');
102
+ expect(moonIcon).toHaveClass('opacity-0');
103
+ expect(moonIcon).toHaveClass('scale-0');
104
+ });
105
+
106
+ test("Button has relative class for positioning", () => {
107
+ setupTest();
108
+ const button = screen.getByRole("button");
109
+ expect(button).toHaveClass("relative");
110
+ });
111
+
112
+ test("Button has focus outline removed", () => {
113
+ setupTest();
114
+ const button = screen.getByRole("button");
115
+ expect(button).toHaveClass("focus:outline-hidden");
116
+ });
117
+
118
+ test("Button click handler is called on interaction", async () => {
119
+ const { user } = setupTest();
120
+ const button = screen.getByRole("button");
121
+
122
+ // Verify button is interactive
123
+ await user.click(button);
124
+
125
+ // After click, localStorage should be updated (auto -> light)
126
+ expect(localStorageMock.setItem).toHaveBeenCalled();
127
+ });
128
+
129
+ test("SVG icons have correct dimensions", () => {
130
+ const { container } = setupTest();
131
+ const svgs = container.querySelectorAll('svg');
132
+ // All SVGs should have width and height of either 20 (main icons) or 12 (auto mode icons)
133
+ expect(svgs.length).toBeGreaterThan(0);
134
+ svgs.forEach(svg => {
135
+ const width = svg.getAttribute('width');
136
+ const height = svg.getAttribute('height');
137
+ expect(['20', '12']).toContain(width);
138
+ expect(['20', '12']).toContain(height);
139
+ });
140
+ });
141
+
142
+ test("Sun icon has correct color class", () => {
143
+ const { container } = setupTest();
144
+ const sunIcon = container.querySelector('span.text-amber-500');
145
+ expect(sunIcon).toHaveClass("text-amber-500");
146
+ });
147
+
148
+ test("Button has overflow hidden for auto icon", () => {
149
+ setupTest();
150
+ const button = screen.getByRole("button");
151
+ expect(button).toHaveClass("overflow-hidden");
152
+ });
153
+
154
+ test("Auto mode div has absolute positioning", () => {
155
+ const { container } = setupTest();
156
+ const autoDiv = container.querySelector('div.absolute.inset-0');
157
+ expect(autoDiv).toHaveClass("absolute");
158
+ expect(autoDiv).toHaveClass("inset-0");
159
+ });
160
+ });
161
+
162
+ describe("DarkModeToggle 3-State Cycle", () => {
163
+ beforeEach(() => {
164
+ localStorageMock.clear();
165
+ Object.defineProperty(window, "localStorage", { value: localStorageMock });
166
+ Object.defineProperty(window, "matchMedia", { value: matchMediaMock });
167
+ document.documentElement.classList.remove("dark");
168
+ });
169
+
170
+ afterEach(() => {
171
+ vi.clearAllMocks();
172
+ });
173
+
174
+ test("First click: auto -> light, saves 'light' to localStorage", async () => {
175
+ const { user } = setupTest();
176
+ const button = screen.getByRole("button");
177
+
178
+ await user.click(button);
179
+
180
+ expect(localStorageMock.setItem).toHaveBeenCalledWith("theme", "light");
181
+ });
182
+
183
+ test("Second click: light -> dark, saves 'dark' to localStorage", async () => {
184
+ const { user } = setupTest();
185
+ const button = screen.getByRole("button");
186
+
187
+ // First click: auto -> light
188
+ await user.click(button);
189
+ // Second click: light -> dark
190
+ await user.click(button);
191
+
192
+ expect(localStorageMock.setItem).toHaveBeenLastCalledWith("theme", "dark");
193
+ });
194
+
195
+ test("Third click: dark -> auto, removes theme from localStorage", async () => {
196
+ const { user } = setupTest();
197
+ const button = screen.getByRole("button");
198
+
199
+ // First click: auto -> light
200
+ await user.click(button);
201
+ // Second click: light -> dark
202
+ await user.click(button);
203
+ // Third click: dark -> auto
204
+ await user.click(button);
205
+
206
+ expect(localStorageMock.removeItem).toHaveBeenCalledWith("theme");
207
+ });
208
+
209
+ test("Full cycle: auto -> light -> dark -> auto", async () => {
210
+ const { user, container } = setupTest();
211
+ const button = screen.getByRole("button");
212
+
213
+ // Initially auto - auto div should exist
214
+ expect(container.querySelector("div.absolute.inset-0")).toBeInTheDocument();
215
+
216
+ // Click 1: auto -> light
217
+ await user.click(button);
218
+ const sunSpan = container.querySelector('span.text-amber-500');
219
+ // In light mode, sun should be visible (no opacity-0 class)
220
+ expect(sunSpan.classList.contains("opacity-0")).toBe(false);
221
+ expect(container.querySelector("div.absolute.inset-0")).not.toBeInTheDocument();
222
+
223
+ // Click 2: light -> dark
224
+ await user.click(button);
225
+ // Find the direct child spans of the button (not nested spans)
226
+ const buttonSpans = Array.from(button.querySelectorAll(':scope > span'));
227
+ const moonSpan = buttonSpans.find(
228
+ span => !span.classList.contains('text-amber-500')
229
+ );
230
+ // In dark mode, moon should be visible (no opacity-0)
231
+ expect(moonSpan.classList.contains("opacity-0")).toBe(false);
232
+ expect(sunSpan.classList.contains("opacity-0")).toBe(true);
233
+
234
+ // Click 3: dark -> auto
235
+ await user.click(button);
236
+ expect(container.querySelector("div.absolute.inset-0")).toBeInTheDocument();
237
+ // After returning to auto, both icons should be hidden
238
+ expect(sunSpan.classList.contains("opacity-0")).toBe(true);
239
+ });
240
+
241
+ test("Dark class applied when switching to dark mode", async () => {
242
+ const { user } = setupTest();
243
+ const button = screen.getByRole("button");
244
+
245
+ // auto -> light
246
+ await user.click(button);
247
+ expect(document.documentElement.classList.contains("dark")).toBe(false);
248
+
249
+ // light -> dark
250
+ await user.click(button);
251
+ expect(document.documentElement.classList.contains("dark")).toBe(true);
252
+ });
253
+
254
+ test("Dark class removed when switching back to light", async () => {
255
+ const { user } = setupTest();
256
+ const button = screen.getByRole("button");
257
+
258
+ // auto -> light -> dark
259
+ await user.click(button);
260
+ await user.click(button);
261
+ expect(document.documentElement.classList.contains("dark")).toBe(true);
262
+
263
+ // dark -> auto (system is light by default in mock)
264
+ await user.click(button);
265
+ expect(document.documentElement.classList.contains("dark")).toBe(false);
266
+ });
267
+ });
268
+
269
+ describe("DarkModeToggle Initial State", () => {
270
+ beforeEach(() => {
271
+ localStorageMock.clear();
272
+ Object.defineProperty(window, "localStorage", { value: localStorageMock });
273
+ Object.defineProperty(window, "matchMedia", { value: matchMediaMock });
274
+ document.documentElement.classList.remove("dark");
275
+ });
276
+
277
+ afterEach(() => {
278
+ vi.clearAllMocks();
279
+ });
280
+
281
+ test("Initializes to auto mode when no localStorage value", () => {
282
+ const { container } = setupTest();
283
+ const autoDiv = container.querySelector("div.absolute.inset-0");
284
+ expect(autoDiv).toBeInTheDocument();
285
+ });
286
+
287
+ test("Initializes to dark mode when localStorage has 'dark'", () => {
288
+ localStorageMock.store.theme = "dark";
289
+ const { container } = setupTest();
290
+ const button = screen.getByRole("button");
291
+ const buttonSpans = Array.from(button.querySelectorAll(':scope > span'));
292
+ const moonSpan = buttonSpans.find(
293
+ span => !span.classList.contains('text-amber-500')
294
+ );
295
+ // In dark mode, moon should be visible (no opacity-0)
296
+ expect(moonSpan.classList.contains("opacity-0")).toBe(false);
297
+ // Sun should be hidden
298
+ const sunSpan = container.querySelector('span.text-amber-500');
299
+ expect(sunSpan.classList.contains("opacity-0")).toBe(true);
300
+ });
301
+
302
+ test("Initializes to light mode when localStorage has 'light'", () => {
303
+ localStorageMock.store.theme = "light";
304
+ const { container } = setupTest();
305
+ const button = screen.getByRole("button");
306
+ const sunSpan = container.querySelector('span.text-amber-500');
307
+ // In light mode, sun should be visible (no opacity-0)
308
+ expect(sunSpan.classList.contains("opacity-0")).toBe(false);
309
+ // Moon should be hidden - need to check all spans that have transition-all
310
+ const allSpansWithTransition = Array.from(button.querySelectorAll('span.transition-all'));
311
+ const moonSpan = allSpansWithTransition.find(
312
+ span => !span.classList.contains('text-amber-500')
313
+ );
314
+ expect(moonSpan).toBeDefined();
315
+ expect(moonSpan.classList.contains("opacity-0")).toBe(true);
316
+ });
317
+
318
+ test("Reads from localStorage on mount", () => {
319
+ localStorageMock.store.theme = "dark";
320
+ setupTest();
321
+ expect(localStorageMock.getItem).toHaveBeenCalledWith("theme");
322
+ });
323
+
324
+ test("Checks system preference for auto mode", () => {
325
+ setupTest();
326
+ expect(matchMediaMock).toHaveBeenCalledWith("(prefers-color-scheme: dark)");
327
+ });
328
+
329
+ test("Auto mode follows system dark preference", () => {
330
+ matchMediaMock.mockReturnValue({
331
+ matches: true, // System prefers dark
332
+ media: "(prefers-color-scheme: dark)",
333
+ addEventListener: vi.fn(),
334
+ removeEventListener: vi.fn(),
335
+ });
336
+ setupTest();
337
+ // In auto mode with system dark preference, dark class should be applied
338
+ expect(document.documentElement.classList.contains("dark")).toBe(true);
339
+ });
340
+
341
+ test("Auto mode follows system light preference", () => {
342
+ matchMediaMock.mockReturnValue({
343
+ matches: false, // System prefers light
344
+ media: "(prefers-color-scheme: dark)",
345
+ addEventListener: vi.fn(),
346
+ removeEventListener: vi.fn(),
347
+ });
348
+ setupTest();
349
+ expect(document.documentElement.classList.contains("dark")).toBe(false);
350
+ });
351
+ });
352
+
353
+ describe("DarkModeToggle Title/Tooltip", () => {
354
+ beforeEach(() => {
355
+ localStorageMock.clear();
356
+ Object.defineProperty(window, "localStorage", { value: localStorageMock });
357
+ Object.defineProperty(window, "matchMedia", { value: matchMediaMock });
358
+ document.documentElement.classList.remove("dark");
359
+ });
360
+
361
+ afterEach(() => {
362
+ vi.clearAllMocks();
363
+ });
364
+
365
+ test("Shows correct title in auto mode", () => {
366
+ setupTest();
367
+ const button = screen.getByRole("button");
368
+ expect(button.title).toContain("Auto");
369
+ expect(button.title).toContain("light mode");
370
+ });
371
+
372
+ test("Shows correct title in light mode", async () => {
373
+ const { user } = setupTest();
374
+ const button = screen.getByRole("button");
375
+
376
+ await user.click(button); // auto -> light
377
+ expect(button.title).toContain("Light mode");
378
+ expect(button.title).toContain("dark mode");
379
+ });
380
+
381
+ test("Shows correct title in dark mode", async () => {
382
+ const { user } = setupTest();
383
+ const button = screen.getByRole("button");
384
+
385
+ await user.click(button); // auto -> light
386
+ await user.click(button); // light -> dark
387
+ expect(button.title).toContain("Dark mode");
388
+ expect(button.title).toContain("auto");
389
+ });
390
+ });