@getmicdrop/svelte-components 5.19.0 → 5.20.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.
- package/dist/base.css +18 -0
- package/dist/calendar/AboutShow/AboutShow.svelte +191 -191
- package/dist/calendar/Calendar/MiniMonthCalendar.svelte +803 -801
- package/dist/calendar/Calendar/MiniMonthCalendar.svelte.d.ts.map +1 -1
- package/dist/calendar/FAQs/FAQs.svelte +88 -88
- package/dist/calendar/MonthSwitcher/MonthSwitcher.svelte +140 -140
- package/dist/calendar/OrderSummary/OrderSummary.svelte +461 -461
- package/dist/calendar/PublicCard/PublicCard.svelte +164 -164
- package/dist/calendar/ShowCard/ShowCard.svelte +180 -180
- package/dist/calendar/ShowTimeCard/ShowTimeCard.svelte +80 -80
- package/dist/calendar/index.js +15 -15
- package/dist/components/Heading.spec.js +89 -89
- package/dist/components/Heading.svelte +66 -60
- package/dist/components/Heading.svelte.d.ts +1 -0
- package/dist/components/Heading.svelte.d.ts.map +1 -1
- package/dist/components/Layout/AppShell.svelte +104 -104
- package/dist/components/Layout/ContentSection.svelte +80 -80
- package/dist/components/Layout/Grid.svelte +101 -101
- package/dist/components/Layout/Heading.svelte +81 -81
- package/dist/components/Layout/PageContainer.svelte +69 -69
- package/dist/components/Layout/Responsive.svelte +75 -75
- package/dist/components/Layout/Section.svelte +80 -80
- package/dist/components/Layout/ShowOnDesktop.svelte +37 -37
- package/dist/components/Layout/ShowOnMobile.svelte +37 -37
- package/dist/components/Layout/Sidebar.svelte +108 -108
- package/dist/components/Layout/Stack.spec.js +1 -1
- package/dist/components/Layout/Stack.svelte +52 -52
- package/dist/components/Layout/Text.svelte +87 -87
- package/dist/components/Layout/TwoColumn.svelte +108 -108
- package/dist/components/Text.spec.js +89 -89
- package/dist/components/Text.svelte +64 -53
- package/dist/components/Text.svelte.d.ts +2 -1
- package/dist/components/Text.svelte.d.ts.map +1 -1
- package/dist/config.js +151 -151
- package/dist/config.spec.js +29 -29
- package/dist/constants/formOptions.d.ts +2 -5
- package/dist/constants/formOptions.d.ts.map +1 -1
- package/dist/constants/formOptions.js +48 -48
- package/dist/constants/formOptions.spec.js +2 -7
- package/dist/constants/validation.js +91 -91
- package/dist/constants/validation.spec.js +64 -64
- package/dist/datetime/README.md +323 -323
- package/dist/datetime/__tests__/format.test.js +1 -1
- package/dist/datetime/__tests__/parse.test.js +1 -1
- package/dist/datetime/__tests__/timezone.test.js +1 -1
- package/dist/datetime/parse.js +1 -1
- package/dist/forms/createFormStore.svelte.js +0 -1
- package/dist/forms/createFormStore.svelte.spec.js +0 -1
- package/dist/index.d.ts +6 -2
- package/dist/index.d.ts.map +1 -1
- package/dist/index.js +85 -66
- package/dist/index.spec.js +369 -369
- package/dist/patterns/chat/ChatActivityNotice.spec.js +59 -59
- package/dist/patterns/chat/ChatActivityNotice.svelte +41 -41
- package/dist/patterns/chat/ChatBubble.spec.js +91 -91
- package/dist/patterns/chat/ChatBubble.svelte +103 -103
- package/dist/patterns/chat/ChatContainer.spec.js +30 -30
- package/dist/patterns/chat/ChatContainer.svelte +46 -46
- package/dist/patterns/chat/ChatDateDivider.spec.js +30 -30
- package/dist/patterns/chat/ChatDateDivider.svelte +27 -27
- package/dist/patterns/chat/ChatInvitationBubble.spec.js +46 -46
- package/dist/patterns/chat/ChatInvitationBubble.svelte +46 -46
- package/dist/patterns/chat/ChatInvitationNotice.spec.js +32 -32
- package/dist/patterns/chat/ChatInvitationNotice.svelte +36 -36
- package/dist/patterns/chat/ChatMessageGroup.spec.js +58 -58
- package/dist/patterns/chat/ChatMessageGroup.svelte +57 -57
- package/dist/patterns/chat/ChatSlotUpdate.spec.js +65 -65
- package/dist/patterns/chat/ChatSlotUpdate.svelte +46 -46
- package/dist/patterns/chat/ChatStatusBadge.spec.js +79 -79
- package/dist/patterns/chat/ChatStatusBadge.svelte +91 -91
- package/dist/patterns/chat/ChatStatusTransition.spec.js +81 -81
- package/dist/patterns/chat/ChatStatusTransition.svelte +64 -64
- package/dist/patterns/chat/ChatTextBubble.spec.js +35 -35
- package/dist/patterns/chat/ChatTextBubble.svelte +41 -41
- package/dist/patterns/chat/index.js +22 -22
- package/dist/patterns/data/DataGrid.svelte +45 -45
- package/dist/patterns/data/DataList.svelte +24 -24
- package/dist/patterns/data/DataTable.svelte +45 -45
- package/dist/patterns/data/index.js +4 -4
- package/dist/patterns/forms/FormActions.spec.js +95 -95
- package/dist/patterns/forms/FormActions.stories.svelte +97 -97
- package/dist/patterns/forms/FormActions.svelte +46 -46
- package/dist/patterns/forms/FormGrid.svelte +33 -33
- package/dist/patterns/forms/FormSection.svelte +33 -32
- package/dist/patterns/forms/FormSection.svelte.d.ts +2 -0
- package/dist/patterns/forms/FormSection.svelte.d.ts.map +1 -1
- package/dist/patterns/forms/FormValidationSummary.stories.svelte +97 -97
- package/dist/patterns/forms/FormValidationSummary.svelte +82 -82
- package/dist/patterns/forms/index.js +5 -5
- package/dist/patterns/index.js +21 -21
- package/dist/patterns/layout/Sidebar.svelte +39 -39
- package/dist/patterns/layout/SidebarTestWrapper.svelte +34 -34
- package/dist/patterns/layout/Stack.svelte +61 -61
- package/dist/patterns/layout/index.js +29 -29
- package/dist/patterns/navigation/BottomNav.stories.svelte +117 -117
- package/dist/patterns/navigation/BottomNav.svelte +82 -82
- package/dist/patterns/navigation/Header.stories.svelte +77 -77
- package/dist/patterns/navigation/Header.svelte +263 -263
- package/dist/patterns/navigation/index.js +3 -3
- package/dist/patterns/page/PageHeader.svelte +49 -49
- package/dist/patterns/page/PageLayout.svelte +40 -40
- package/dist/patterns/page/PageLoader.spec.js +57 -57
- package/dist/patterns/page/PageLoader.stories.svelte +137 -137
- package/dist/patterns/page/PageLoader.svelte +62 -62
- package/dist/patterns/page/SectionHeader.svelte +51 -51
- package/dist/patterns/page/index.js +5 -5
- package/dist/presets/badges.js +112 -112
- package/dist/presets/buttons.js +76 -76
- package/dist/presets/index.js +9 -9
- package/dist/primitives/Accordion/Accordion.stories.svelte +75 -75
- package/dist/primitives/Accordion/Accordion.svelte +62 -62
- package/dist/primitives/Accordion/AccordionItem.svelte +103 -103
- package/dist/primitives/Accordion/AccordionItemWrapper.test.svelte +107 -107
- package/dist/primitives/Accordion/AccordionToggleWrapper.test.svelte +28 -28
- package/dist/primitives/Alert/Alert.spec.js +173 -173
- package/dist/primitives/Alert/Alert.stories.svelte +88 -88
- package/dist/primitives/Alert/Alert.svelte +72 -72
- package/dist/primitives/Avatar/Avatar.stories.svelte +94 -94
- package/dist/primitives/Avatar/Avatar.svelte +66 -66
- package/dist/primitives/AvatarButton/AvatarButton.svelte +57 -57
- package/dist/primitives/Badges/Badge.spec.js +144 -144
- package/dist/primitives/Badges/Badge.stories.svelte +86 -86
- package/dist/primitives/Badges/Badge.svelte +99 -99
- package/dist/primitives/BottomSheet/BottomSheet.spec.js +238 -238
- package/dist/primitives/BottomSheet/BottomSheet.stories.svelte +83 -83
- package/dist/primitives/BottomSheet/BottomSheet.svelte +115 -115
- package/dist/primitives/BottomSheet/BottomSheetWithActions.test.svelte +20 -20
- package/dist/primitives/Breadcrumb/Breadcrumb.spec.js +123 -123
- package/dist/primitives/Breadcrumb/Breadcrumb.stories.svelte +23 -23
- package/dist/primitives/Breadcrumb/Breadcrumb.svelte +107 -107
- package/dist/primitives/Button/Button.spec.js +225 -225
- package/dist/primitives/Button/Button.stories.svelte +76 -76
- package/dist/primitives/Button/Button.svelte +278 -278
- package/dist/primitives/Button/ButtonGroup.spec.js +44 -44
- package/dist/primitives/Button/ButtonGroup.svelte +50 -50
- package/dist/primitives/Button/ButtonSaveDemo.spec.js +146 -146
- package/dist/primitives/Button/ButtonSaveDemo.svelte +25 -25
- package/dist/primitives/Button/ButtonVariantShowcase.svelte +129 -129
- package/dist/primitives/Card.spec.js +49 -49
- package/dist/primitives/Card.stories.svelte +22 -22
- package/dist/primitives/Card.svelte +28 -28
- package/dist/primitives/CardAction/CardAction.svelte +68 -68
- package/dist/primitives/Checkbox/Checkbox.spec.js +4 -15
- package/dist/primitives/Checkbox/Checkbox.stories.svelte +84 -84
- package/dist/primitives/Checkbox/Checkbox.svelte +88 -88
- package/dist/primitives/DarkModeToggle.spec.js +390 -390
- package/dist/primitives/DarkModeToggle.stories.svelte +57 -57
- package/dist/primitives/DarkModeToggle.svelte +147 -147
- package/dist/primitives/Drawer/Drawer.stories.svelte +100 -100
- package/dist/primitives/Drawer/Drawer.svelte +224 -224
- package/dist/primitives/Drawer/DrawerTestWrapper.svelte +86 -86
- package/dist/primitives/Dropdown/Dropdown.stories.svelte +137 -137
- package/dist/primitives/Dropdown/Dropdown.svelte +179 -179
- package/dist/primitives/Dropdown/DropdownDivider.spec.js +30 -30
- package/dist/primitives/Dropdown/DropdownDivider.svelte +9 -9
- package/dist/primitives/Dropdown/DropdownItem.svelte +80 -80
- package/dist/primitives/Dropdown/DropdownItemTestWrapper.svelte +43 -43
- package/dist/primitives/Helper/Helper.spec.js +57 -57
- package/dist/primitives/Helper/Helper.svelte +33 -33
- package/dist/primitives/Icons/ArrowLeft.svelte +21 -21
- package/dist/primitives/Icons/ArrowRight.svelte +21 -21
- package/dist/primitives/Icons/Availability.svelte +27 -27
- package/dist/primitives/Icons/Back.svelte +27 -27
- package/dist/primitives/Icons/CheckCircle.svelte +19 -19
- package/dist/primitives/Icons/CheckCircleOutline.svelte +28 -28
- package/dist/primitives/Icons/ChevronLeft.svelte +17 -17
- package/dist/primitives/Icons/ChevronRight.svelte +17 -17
- package/dist/primitives/Icons/Copy.svelte +28 -28
- package/dist/primitives/Icons/Cross.svelte +18 -18
- package/dist/primitives/Icons/DownArrow.svelte +21 -21
- package/dist/primitives/Icons/ErrorCircle.svelte +19 -19
- package/dist/primitives/Icons/FacebookIcon.svelte +15 -15
- package/dist/primitives/Icons/Home.svelte +28 -28
- package/dist/primitives/Icons/Icon.spec.js +175 -175
- package/dist/primitives/Icons/Icon.stories.svelte +100 -100
- package/dist/primitives/Icons/Icon.svelte +79 -79
- package/dist/primitives/Icons/IconGallery.stories.svelte +235 -235
- package/dist/primitives/Icons/ImageOutline.svelte +21 -21
- package/dist/primitives/Icons/Info.svelte +20 -20
- package/dist/primitives/Icons/InstagramIcon.svelte +21 -21
- package/dist/primitives/Icons/LogoInstagram.svelte +15 -15
- package/dist/primitives/Icons/Message.svelte +28 -28
- package/dist/primitives/Icons/MoonIcon.svelte +18 -18
- package/dist/primitives/Icons/More.svelte +34 -34
- package/dist/primitives/Icons/MoreHori.spec.js +67 -67
- package/dist/primitives/Icons/MoreHori.svelte +35 -35
- package/dist/primitives/Icons/Notification.svelte +27 -27
- package/dist/primitives/Icons/Payment.svelte +27 -27
- package/dist/primitives/Icons/Profile.svelte +34 -34
- package/dist/primitives/Icons/Reload.svelte +42 -42
- package/dist/primitives/Icons/Shows.svelte +34 -34
- package/dist/primitives/Icons/Signout.svelte +34 -34
- package/dist/primitives/Icons/SunIcon.svelte +21 -21
- package/dist/primitives/Icons/TiktokIcon.svelte +15 -15
- package/dist/primitives/Icons/TrashBinOutline.svelte +21 -21
- package/dist/primitives/Icons/TwitterIcon.svelte +15 -15
- package/dist/primitives/Icons/WarningIcon.spec.js +30 -30
- package/dist/primitives/Icons/WarningIcon.svelte +24 -24
- package/dist/primitives/Input/Input.spec.js +1237 -1237
- package/dist/primitives/Input/Input.stories.svelte +139 -139
- package/dist/primitives/Input/Input.svelte +444 -444
- package/dist/primitives/Input/Select.spec.js +632 -632
- package/dist/primitives/Input/Select.stories.svelte +112 -112
- package/dist/primitives/Input/Select.svelte +252 -252
- package/dist/primitives/Input/Textarea.stories.svelte +137 -137
- package/dist/primitives/Input/Textarea.svelte +105 -105
- package/dist/primitives/Label/Label.svelte +37 -37
- package/dist/primitives/LandingButton/LandingButton.spec.js +61 -61
- package/dist/primitives/LandingButton/LandingButton.svelte +92 -92
- package/dist/primitives/MenuItem/MenuItem.spec.js +130 -130
- package/dist/primitives/MenuItem/MenuItem.svelte +85 -85
- package/dist/primitives/Modal/Modal.spec.js +314 -314
- package/dist/primitives/Modal/Modal.stories.svelte +86 -86
- package/dist/primitives/Modal/Modal.svelte +181 -181
- package/dist/primitives/NavItem/NavItem.spec.js +97 -97
- package/dist/primitives/NavItem/NavItem.svelte +75 -75
- package/dist/primitives/NumberInput/NumberInput.svelte +113 -113
- package/dist/primitives/Pagination/DotIndicator.svelte +66 -0
- package/dist/primitives/Pagination/DotIndicator.svelte.d.ts +18 -0
- package/dist/primitives/Pagination/DotIndicator.svelte.d.ts.map +1 -0
- package/dist/primitives/Pagination/Pagination.stories.svelte +76 -76
- package/dist/primitives/Pagination/Pagination.svelte +275 -275
- package/dist/primitives/Radio/Radio.stories.svelte +80 -80
- package/dist/primitives/Radio/Radio.svelte +67 -67
- package/dist/primitives/SearchResultItem/SearchResultItem.spec.js +78 -78
- package/dist/primitives/SearchResultItem/SearchResultItem.svelte +109 -109
- package/dist/primitives/SidebarToggle/SidebarToggle.spec.js +61 -61
- package/dist/primitives/SidebarToggle/SidebarToggle.svelte +55 -55
- package/dist/primitives/Skeleton/CardPlaceholder.svelte +96 -96
- package/dist/primitives/Skeleton/ImagePlaceholder.svelte +68 -68
- package/dist/primitives/Skeleton/ListPlaceholder.svelte +85 -85
- package/dist/primitives/Skeleton/Skeleton.stories.svelte +151 -151
- package/dist/primitives/Skeleton/Skeleton.svelte +55 -55
- package/dist/primitives/Spinner/Spinner.spec.js +84 -84
- package/dist/primitives/Spinner/Spinner.stories.svelte +29 -29
- package/dist/primitives/Spinner/Spinner.svelte +52 -52
- package/dist/primitives/Tabs/TabItem.svelte +52 -52
- package/dist/primitives/Tabs/Tabs.stories.svelte +112 -112
- package/dist/primitives/Tabs/Tabs.svelte +137 -137
- package/dist/primitives/Toggle.spec.js +221 -221
- package/dist/primitives/Toggle.stories.svelte +92 -92
- package/dist/primitives/Toggle.svelte +141 -141
- package/dist/primitives/ToggleTestWrapper.svelte +30 -30
- package/dist/primitives/Tooltip/Tooltip.spec.js +126 -126
- package/dist/primitives/Tooltip/Tooltip.svelte +83 -83
- package/dist/primitives/Typography/Typography.svelte +53 -53
- package/dist/primitives/ValidationError.spec.js +103 -103
- package/dist/primitives/ValidationError.stories.svelte +112 -112
- package/dist/primitives/ValidationError.svelte +29 -29
- package/dist/primitives/index.d.ts +1 -0
- package/dist/primitives/index.js +114 -113
- package/dist/recipes/CropImage/CropImage.spec.js +208 -208
- package/dist/recipes/CropImage/CropImage.stories.svelte +104 -104
- package/dist/recipes/CropImage/CropImage.svelte +241 -241
- package/dist/recipes/ImageUploader/ImageUploader.stories.svelte +125 -125
- package/dist/recipes/ImageUploader/ImageUploader.svelte +994 -994
- package/dist/recipes/Toaster/Toaster.stories.svelte +62 -62
- package/dist/recipes/feedback/EmptyState/EmptyState.svelte +75 -75
- package/dist/recipes/feedback/ErrorDisplay.spec.js +69 -69
- package/dist/recipes/feedback/ErrorDisplay.stories.svelte +113 -113
- package/dist/recipes/feedback/ErrorDisplay.svelte +67 -67
- package/dist/recipes/feedback/StatusIndicator/StatusIndicator.spec.js +133 -133
- package/dist/recipes/feedback/StatusIndicator/StatusIndicator.svelte +176 -176
- package/dist/recipes/feedback/index.js +4 -4
- package/dist/recipes/fields/CheckboxField.svelte +85 -85
- package/dist/recipes/fields/FormField.svelte +58 -58
- package/dist/recipes/fields/RadioGroup.svelte +95 -95
- package/dist/recipes/fields/SelectField.svelte +82 -82
- package/dist/recipes/fields/TextareaField.svelte +101 -101
- package/dist/recipes/fields/ToggleField.svelte +60 -60
- package/dist/recipes/fields/index.js +7 -7
- package/dist/recipes/index.js +24 -24
- package/dist/recipes/inputs/MultiSelect.spec.js +263 -263
- package/dist/recipes/inputs/MultiSelect.stories.svelte +133 -133
- package/dist/recipes/inputs/MultiSelect.svelte +291 -291
- package/dist/recipes/inputs/OTPInput.spec.js +251 -251
- package/dist/recipes/inputs/OTPInput.stories.svelte +162 -162
- package/dist/recipes/inputs/OTPInput.svelte +128 -128
- package/dist/recipes/inputs/PasswordInput.svelte +130 -130
- package/dist/recipes/inputs/PasswordStrengthIndicator/PasswordStrengthIndicator.svelte +142 -142
- package/dist/recipes/inputs/PhoneInput.svelte +254 -254
- package/dist/recipes/inputs/PlaceAutocomplete/PlaceAutocomplete.stories.svelte +170 -170
- package/dist/recipes/inputs/PlaceAutocomplete/PlaceAutocomplete.svelte +349 -349
- package/dist/recipes/inputs/Search.svelte +110 -110
- package/dist/recipes/inputs/index.js +8 -8
- package/dist/recipes/inputs/phoneInput/CountrySelector.svelte +240 -240
- package/dist/recipes/modals/AlertModal.svelte +139 -139
- package/dist/recipes/modals/ConfirmationModal.spec.js +396 -396
- package/dist/recipes/modals/ConfirmationModal.stories.svelte +119 -119
- package/dist/recipes/modals/ConfirmationModal.svelte +169 -169
- package/dist/recipes/modals/FeedbackModal.svelte +205 -0
- package/dist/recipes/modals/FeedbackModal.svelte.d.ts +24 -0
- package/dist/recipes/modals/FeedbackModal.svelte.d.ts.map +1 -0
- package/dist/recipes/modals/InputModal.svelte +194 -194
- package/dist/recipes/modals/ModalStateManager.spec.js +100 -100
- package/dist/recipes/modals/ModalStateManager.svelte +77 -77
- package/dist/recipes/modals/ModalTestWrapper.svelte +65 -65
- package/dist/recipes/modals/StatusModal.svelte +216 -216
- package/dist/recipes/modals/index.d.ts +1 -0
- package/dist/recipes/modals/index.js +8 -7
- package/dist/schemas/common.js +1 -1
- package/dist/schemas/event.js +1 -1
- package/dist/schemas/order.js +0 -1
- package/dist/schemas/performer.js +1 -1
- package/dist/schemas/promo.js +2 -2
- package/dist/schemas/ticket.js +1 -1
- package/dist/schemas/user.js +1 -2
- package/dist/services/show.service.d.ts +46 -46
- package/dist/stores/auth.d.ts +8 -8
- package/dist/stores/auth.svelte.spec.js +1 -1
- package/dist/stores/index.js +9 -9
- package/dist/stores/toaster.d.ts +3 -3
- package/dist/stores/toaster.js +13 -13
- package/dist/stores/toaster.spec.js +59 -59
- package/dist/stories/ButtonAuditReview.stories.svelte +14 -14
- package/dist/stories/ButtonAuditReview.svelte +427 -427
- package/dist/stories/PatternsGallery.stories.svelte +19 -19
- package/dist/stories/PatternsGallery.svelte +399 -399
- package/dist/stories/PrimitivesGallery.stories.svelte +19 -19
- package/dist/stories/PrimitivesGallery.svelte +756 -756
- package/dist/stories/RecipesGallery.stories.svelte +19 -19
- package/dist/stories/RecipesGallery.svelte +454 -454
- package/dist/stories/button-audit-manifest.json +11186 -11186
- package/dist/tailwind/preset.cjs +87 -82
- package/dist/tailwind/preset.d.cts +2 -0
- package/dist/tailwind/preset.d.cts.map +1 -1
- package/dist/telemetry.js +402 -402
- package/dist/telemetry.server.js +212 -212
- package/dist/telemetry.server.spec.js +437 -437
- package/dist/telemetry.spec.js +1173 -1173
- package/dist/tokens/__tests__/spacing.test.js +2 -2
- package/dist/tokens/__tests__/variants.test.js +12 -4
- package/dist/tokens/base-resets.css +124 -0
- package/dist/tokens/spacing.d.ts +2 -0
- package/dist/tokens/spacing.d.ts.map +1 -1
- package/dist/tokens/spacing.js +1 -0
- package/dist/tokens/tokens.css +87 -87
- package/dist/tokens/typography-base.css +163 -163
- package/dist/tokens/utilities.css +430 -353
- package/dist/tokens/variants.d.ts +4 -4
- package/dist/tokens/variants.js +4 -4
- package/dist/utils/__tests__/auth.test.js +431 -431
- package/dist/utils/apiConfig.d.ts +29 -29
- package/dist/utils/apiConfig.js +117 -117
- package/dist/utils/apiConfig.spec.js +226 -219
- package/dist/utils/auth.d.ts +46 -46
- package/dist/utils/auth.js +195 -195
- package/dist/utils/clickOutside.d.ts +4 -0
- package/dist/utils/clickOutside.d.ts.map +1 -0
- package/dist/utils/clickOutside.js +13 -0
- package/dist/utils/feedbackContext.d.ts +24 -0
- package/dist/utils/feedbackContext.d.ts.map +1 -0
- package/dist/utils/feedbackContext.js +19 -0
- package/dist/utils/fetchHelpers.d.ts.map +1 -1
- package/dist/utils/fetchHelpers.js +1 -0
- package/dist/utils/focusTrap.d.ts +20 -0
- package/dist/utils/focusTrap.d.ts.map +1 -0
- package/dist/utils/focusTrap.js +130 -0
- package/dist/utils/formatters.js +1 -1
- package/dist/utils/greetings.js +187 -187
- package/dist/utils/greetings.spec.js +337 -337
- package/dist/utils/haptic.spec.js +1 -1
- package/dist/utils/imageValidation.js +121 -121
- package/dist/utils/imageValidation.spec.js +223 -223
- package/dist/utils/logger.d.ts +25 -1
- package/dist/utils/logger.d.ts.map +1 -1
- package/dist/utils/logger.js +59 -1
- package/dist/utils/logger.spec.js +99 -1
- package/dist/utils/portal.d.ts +11 -11
- package/dist/utils/portal.js +25 -25
- package/dist/utils/portal.spec.js +143 -143
- package/dist/utils/transitions.d.ts +99 -1
- package/dist/utils/transitions.d.ts.map +1 -1
- package/dist/utils/transitions.js +148 -6
- package/dist/utils/utils/utils.d.ts +2 -2
- package/dist/utils/utils/utils.js +3 -3
- package/dist/utils/utils/utils.spec.js +698 -698
- package/dist/utils/utils.d.ts +41 -41
- package/dist/utils/utils.js +59 -59
- package/dist/utils/utils.spec.js +643 -643
- package/package.json +310 -301
|
@@ -1,337 +1,337 @@
|
|
|
1
|
-
import { describe, it, expect, vi, beforeEach, afterEach } from 'vitest';
|
|
2
|
-
import { getGreeting, getFirstName, clearGreetingCache } from './greetings';
|
|
3
|
-
|
|
4
|
-
describe('greetings', () => {
|
|
5
|
-
beforeEach(() => {
|
|
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
|
-
Object.defineProperty(window, 'localStorage', {
|
|
21
|
-
value: localStorageMock,
|
|
22
|
-
writable: true,
|
|
23
|
-
});
|
|
24
|
-
|
|
25
|
-
// Clear cache before each test
|
|
26
|
-
clearGreetingCache();
|
|
27
|
-
});
|
|
28
|
-
|
|
29
|
-
afterEach(() => {
|
|
30
|
-
vi.restoreAllMocks();
|
|
31
|
-
});
|
|
32
|
-
|
|
33
|
-
describe('getFirstName', () => {
|
|
34
|
-
it('returns first name from full name', () => {
|
|
35
|
-
expect(getFirstName('John Doe')).toBe('John');
|
|
36
|
-
});
|
|
37
|
-
|
|
38
|
-
it('handles single name', () => {
|
|
39
|
-
expect(getFirstName('John')).toBe('John');
|
|
40
|
-
});
|
|
41
|
-
|
|
42
|
-
it('handles multiple names', () => {
|
|
43
|
-
expect(getFirstName('John Michael Doe')).toBe('John');
|
|
44
|
-
});
|
|
45
|
-
|
|
46
|
-
it('returns empty string for empty input', () => {
|
|
47
|
-
expect(getFirstName('')).toBe('');
|
|
48
|
-
});
|
|
49
|
-
|
|
50
|
-
it('returns empty string for null', () => {
|
|
51
|
-
expect(getFirstName(null)).toBe('');
|
|
52
|
-
});
|
|
53
|
-
|
|
54
|
-
it('returns empty string for undefined', () => {
|
|
55
|
-
expect(getFirstName(undefined)).toBe('');
|
|
56
|
-
});
|
|
57
|
-
});
|
|
58
|
-
|
|
59
|
-
describe('getGreeting', () => {
|
|
60
|
-
it('returns a string containing the name', () => {
|
|
61
|
-
const greeting = getGreeting('Alex');
|
|
62
|
-
expect(greeting).toContain('Alex');
|
|
63
|
-
});
|
|
64
|
-
|
|
65
|
-
it('returns different greetings for different time periods', () => {
|
|
66
|
-
// Run multiple times to increase chance of getting different greetings
|
|
67
|
-
const greetings = new Set();
|
|
68
|
-
for (let i = 0; i < 20; i++) {
|
|
69
|
-
clearGreetingCache();
|
|
70
|
-
greetings.add(getGreeting('Test'));
|
|
71
|
-
}
|
|
72
|
-
// Should have at least one greeting (randomness means we might get the same one)
|
|
73
|
-
expect(greetings.size).toBeGreaterThanOrEqual(1);
|
|
74
|
-
});
|
|
75
|
-
|
|
76
|
-
it('caches greeting in localStorage', () => {
|
|
77
|
-
getGreeting('Alex');
|
|
78
|
-
expect(window.localStorage.setItem).toHaveBeenCalled();
|
|
79
|
-
});
|
|
80
|
-
|
|
81
|
-
it('returns cached greeting for same user', () => {
|
|
82
|
-
const greeting1 = getGreeting('Alex');
|
|
83
|
-
const greeting2 = getGreeting('Alex');
|
|
84
|
-
expect(greeting1).toBe(greeting2);
|
|
85
|
-
});
|
|
86
|
-
|
|
87
|
-
it('generates new greeting for different user', () => {
|
|
88
|
-
const greeting1 = getGreeting('Alex');
|
|
89
|
-
clearGreetingCache();
|
|
90
|
-
const greeting2 = getGreeting('Beth');
|
|
91
|
-
expect(greeting2).toContain('Beth');
|
|
92
|
-
});
|
|
93
|
-
|
|
94
|
-
it('handles localStorage errors gracefully', () => {
|
|
95
|
-
window.localStorage.getItem.mockImplementation(() => {
|
|
96
|
-
throw new Error('localStorage error');
|
|
97
|
-
});
|
|
98
|
-
|
|
99
|
-
// Should not throw
|
|
100
|
-
expect(() => getGreeting('Alex')).not.toThrow();
|
|
101
|
-
const greeting = getGreeting('Alex');
|
|
102
|
-
expect(greeting).toContain('Alex');
|
|
103
|
-
});
|
|
104
|
-
|
|
105
|
-
it('handles invalid cached JSON gracefully', () => {
|
|
106
|
-
window.localStorage.getItem.mockReturnValue('invalid json');
|
|
107
|
-
|
|
108
|
-
// Should not throw and should generate new greeting
|
|
109
|
-
expect(() => getGreeting('Alex')).not.toThrow();
|
|
110
|
-
});
|
|
111
|
-
});
|
|
112
|
-
|
|
113
|
-
describe('clearGreetingCache', () => {
|
|
114
|
-
it('removes greeting from localStorage', () => {
|
|
115
|
-
getGreeting('Alex');
|
|
116
|
-
clearGreetingCache();
|
|
117
|
-
expect(window.localStorage.removeItem).toHaveBeenCalledWith('micdrop_greeting');
|
|
118
|
-
});
|
|
119
|
-
|
|
120
|
-
it('handles localStorage errors gracefully', () => {
|
|
121
|
-
window.localStorage.removeItem.mockImplementation(() => {
|
|
122
|
-
throw new Error('localStorage error');
|
|
123
|
-
});
|
|
124
|
-
|
|
125
|
-
// Should not throw
|
|
126
|
-
expect(() => clearGreetingCache()).not.toThrow();
|
|
127
|
-
});
|
|
128
|
-
});
|
|
129
|
-
|
|
130
|
-
describe('greeting content', () => {
|
|
131
|
-
it('greeting always includes user name', () => {
|
|
132
|
-
for (let i = 0; i < 10; i++) {
|
|
133
|
-
clearGreetingCache();
|
|
134
|
-
const greeting = getGreeting('TestUser');
|
|
135
|
-
expect(greeting).toContain('TestUser');
|
|
136
|
-
}
|
|
137
|
-
});
|
|
138
|
-
|
|
139
|
-
it('handles names with special characters', () => {
|
|
140
|
-
const greeting = getGreeting("O'Brien");
|
|
141
|
-
expect(greeting).toContain("O'Brien");
|
|
142
|
-
});
|
|
143
|
-
|
|
144
|
-
it('handles names with numbers', () => {
|
|
145
|
-
const greeting = getGreeting('User123');
|
|
146
|
-
expect(greeting).toContain('User123');
|
|
147
|
-
});
|
|
148
|
-
});
|
|
149
|
-
|
|
150
|
-
describe('localStorage setItem error handling', () => {
|
|
151
|
-
it('handles localStorage setItem error gracefully', () => {
|
|
152
|
-
window.localStorage.setItem.mockImplementation(() => {
|
|
153
|
-
throw new Error('localStorage setItem error');
|
|
154
|
-
});
|
|
155
|
-
|
|
156
|
-
// Should not throw even when setItem fails
|
|
157
|
-
expect(() => getGreeting('Alex')).not.toThrow();
|
|
158
|
-
const greeting = getGreeting('Alex');
|
|
159
|
-
expect(greeting).toContain('Alex');
|
|
160
|
-
});
|
|
161
|
-
});
|
|
162
|
-
|
|
163
|
-
describe('SSR behavior', () => {
|
|
164
|
-
it('works without window (SSR simulation)', () => {
|
|
165
|
-
// The module checks typeof window === 'undefined' for SSR
|
|
166
|
-
// In JSDOM window exists, but we can test by checking that function works
|
|
167
|
-
const greeting = getGreeting('TestUser');
|
|
168
|
-
expect(typeof greeting).toBe('string');
|
|
169
|
-
expect(greeting).toContain('TestUser');
|
|
170
|
-
});
|
|
171
|
-
});
|
|
172
|
-
|
|
173
|
-
describe('time period coverage', () => {
|
|
174
|
-
// Test all time period branches by mocking Date
|
|
175
|
-
const RealDate = Date;
|
|
176
|
-
|
|
177
|
-
afterEach(() => {
|
|
178
|
-
global.Date = RealDate;
|
|
179
|
-
});
|
|
180
|
-
|
|
181
|
-
it('generates greeting for night period (21-24)', () => {
|
|
182
|
-
// Mock Date to return 22:00
|
|
183
|
-
global.Date = class extends RealDate {
|
|
184
|
-
constructor() {
|
|
185
|
-
super();
|
|
186
|
-
return new RealDate('2025-06-15T22:00:00');
|
|
187
|
-
}
|
|
188
|
-
getHours() { return 22; }
|
|
189
|
-
getDay() { return 0; }
|
|
190
|
-
};
|
|
191
|
-
|
|
192
|
-
clearGreetingCache();
|
|
193
|
-
const greeting = getGreeting('NightUser');
|
|
194
|
-
expect(greeting).toContain('NightUser');
|
|
195
|
-
});
|
|
196
|
-
|
|
197
|
-
it('generates greeting for late night period (0-5 AM)', () => {
|
|
198
|
-
// Mock Date to return 03:00
|
|
199
|
-
global.Date = class extends RealDate {
|
|
200
|
-
constructor() {
|
|
201
|
-
super();
|
|
202
|
-
return new RealDate('2025-06-15T03:00:00');
|
|
203
|
-
}
|
|
204
|
-
getHours() { return 3; }
|
|
205
|
-
getDay() { return 0; }
|
|
206
|
-
};
|
|
207
|
-
|
|
208
|
-
clearGreetingCache();
|
|
209
|
-
const greeting = getGreeting('LateNightUser');
|
|
210
|
-
expect(greeting).toContain('LateNightUser');
|
|
211
|
-
});
|
|
212
|
-
|
|
213
|
-
it('generates greeting for early morning period (5-7)', () => {
|
|
214
|
-
global.Date = class extends RealDate {
|
|
215
|
-
constructor() {
|
|
216
|
-
super();
|
|
217
|
-
return new RealDate('2025-06-15T06:00:00');
|
|
218
|
-
}
|
|
219
|
-
getHours() { return 6; }
|
|
220
|
-
getDay() { return 1; }
|
|
221
|
-
};
|
|
222
|
-
|
|
223
|
-
clearGreetingCache();
|
|
224
|
-
const greeting = getGreeting('EarlyUser');
|
|
225
|
-
expect(greeting).toContain('EarlyUser');
|
|
226
|
-
});
|
|
227
|
-
|
|
228
|
-
it('generates greeting for morning period (7-12)', () => {
|
|
229
|
-
global.Date = class extends RealDate {
|
|
230
|
-
constructor() {
|
|
231
|
-
super();
|
|
232
|
-
return new RealDate('2025-06-15T09:00:00');
|
|
233
|
-
}
|
|
234
|
-
getHours() { return 9; }
|
|
235
|
-
getDay() { return 2; }
|
|
236
|
-
};
|
|
237
|
-
|
|
238
|
-
clearGreetingCache();
|
|
239
|
-
const greeting = getGreeting('MorningUser');
|
|
240
|
-
expect(greeting).toContain('MorningUser');
|
|
241
|
-
});
|
|
242
|
-
|
|
243
|
-
it('generates greeting for afternoon period (12-17)', () => {
|
|
244
|
-
global.Date = class extends RealDate {
|
|
245
|
-
constructor() {
|
|
246
|
-
super();
|
|
247
|
-
return new RealDate('2025-06-15T14:00:00');
|
|
248
|
-
}
|
|
249
|
-
getHours() { return 14; }
|
|
250
|
-
getDay() { return 3; }
|
|
251
|
-
};
|
|
252
|
-
|
|
253
|
-
clearGreetingCache();
|
|
254
|
-
const greeting = getGreeting('AfternoonUser');
|
|
255
|
-
expect(greeting).toContain('AfternoonUser');
|
|
256
|
-
});
|
|
257
|
-
|
|
258
|
-
it('generates greeting for evening period (17-21)', () => {
|
|
259
|
-
global.Date = class extends RealDate {
|
|
260
|
-
constructor() {
|
|
261
|
-
super();
|
|
262
|
-
return new RealDate('2025-06-15T19:00:00');
|
|
263
|
-
}
|
|
264
|
-
getHours() { return 19; }
|
|
265
|
-
getDay() { return 4; }
|
|
266
|
-
};
|
|
267
|
-
|
|
268
|
-
clearGreetingCache();
|
|
269
|
-
const greeting = getGreeting('EveningUser');
|
|
270
|
-
expect(greeting).toContain('EveningUser');
|
|
271
|
-
});
|
|
272
|
-
});
|
|
273
|
-
|
|
274
|
-
describe('time-based greeting generation', () => {
|
|
275
|
-
it('generates consistent greeting with cached data for same period and day', () => {
|
|
276
|
-
// First call generates and caches
|
|
277
|
-
const greeting1 = getGreeting('Alex');
|
|
278
|
-
|
|
279
|
-
// Second call should return cached
|
|
280
|
-
const greeting2 = getGreeting('Alex');
|
|
281
|
-
|
|
282
|
-
expect(greeting1).toBe(greeting2);
|
|
283
|
-
});
|
|
284
|
-
|
|
285
|
-
it('generates greeting containing name for various scenarios', () => {
|
|
286
|
-
// Test with longer names
|
|
287
|
-
clearGreetingCache();
|
|
288
|
-
const longNameGreeting = getGreeting('AlexanderTheGreat');
|
|
289
|
-
expect(longNameGreeting).toContain('AlexanderTheGreat');
|
|
290
|
-
|
|
291
|
-
// Test with short names
|
|
292
|
-
clearGreetingCache();
|
|
293
|
-
const shortNameGreeting = getGreeting('Jo');
|
|
294
|
-
expect(shortNameGreeting).toContain('Jo');
|
|
295
|
-
});
|
|
296
|
-
|
|
297
|
-
it('respects cache when period and day match', () => {
|
|
298
|
-
const greeting1 = getGreeting('TestUser');
|
|
299
|
-
|
|
300
|
-
// Mock getting a valid cached value
|
|
301
|
-
const cachedValue = window.localStorage.store['micdrop_greeting'];
|
|
302
|
-
expect(cachedValue).toBeDefined();
|
|
303
|
-
|
|
304
|
-
// Parse to verify structure
|
|
305
|
-
const parsed = JSON.parse(cachedValue);
|
|
306
|
-
expect(parsed).toHaveProperty('greeting');
|
|
307
|
-
expect(parsed).toHaveProperty('period');
|
|
308
|
-
expect(parsed).toHaveProperty('day');
|
|
309
|
-
expect(parsed).toHaveProperty('cachedName');
|
|
310
|
-
expect(parsed.cachedName).toBe('TestUser');
|
|
311
|
-
});
|
|
312
|
-
});
|
|
313
|
-
|
|
314
|
-
describe('SSR mode', () => {
|
|
315
|
-
it('generates greeting without caching when window is undefined', async () => {
|
|
316
|
-
vi.resetModules();
|
|
317
|
-
|
|
318
|
-
// Save original window
|
|
319
|
-
const originalWindow = global.window;
|
|
320
|
-
|
|
321
|
-
// Delete window to simulate SSR
|
|
322
|
-
delete global.window;
|
|
323
|
-
|
|
324
|
-
// Re-import module in SSR context
|
|
325
|
-
const { getGreeting: ssrGetGreeting } = await import('./greetings.js');
|
|
326
|
-
|
|
327
|
-
// Should still generate a greeting
|
|
328
|
-
const greeting = ssrGetGreeting('TestUser');
|
|
329
|
-
expect(greeting).toBeDefined();
|
|
330
|
-
expect(typeof greeting).toBe('string');
|
|
331
|
-
expect(greeting).toContain('TestUser');
|
|
332
|
-
|
|
333
|
-
// Restore window
|
|
334
|
-
global.window = originalWindow;
|
|
335
|
-
});
|
|
336
|
-
});
|
|
337
|
-
});
|
|
1
|
+
import { describe, it, expect, vi, beforeEach, afterEach } from 'vitest';
|
|
2
|
+
import { getGreeting, getFirstName, clearGreetingCache } from './greetings';
|
|
3
|
+
|
|
4
|
+
describe('greetings', () => {
|
|
5
|
+
beforeEach(() => {
|
|
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
|
+
Object.defineProperty(window, 'localStorage', {
|
|
21
|
+
value: localStorageMock,
|
|
22
|
+
writable: true,
|
|
23
|
+
});
|
|
24
|
+
|
|
25
|
+
// Clear cache before each test
|
|
26
|
+
clearGreetingCache();
|
|
27
|
+
});
|
|
28
|
+
|
|
29
|
+
afterEach(() => {
|
|
30
|
+
vi.restoreAllMocks();
|
|
31
|
+
});
|
|
32
|
+
|
|
33
|
+
describe('getFirstName', () => {
|
|
34
|
+
it('returns first name from full name', () => {
|
|
35
|
+
expect(getFirstName('John Doe')).toBe('John');
|
|
36
|
+
});
|
|
37
|
+
|
|
38
|
+
it('handles single name', () => {
|
|
39
|
+
expect(getFirstName('John')).toBe('John');
|
|
40
|
+
});
|
|
41
|
+
|
|
42
|
+
it('handles multiple names', () => {
|
|
43
|
+
expect(getFirstName('John Michael Doe')).toBe('John');
|
|
44
|
+
});
|
|
45
|
+
|
|
46
|
+
it('returns empty string for empty input', () => {
|
|
47
|
+
expect(getFirstName('')).toBe('');
|
|
48
|
+
});
|
|
49
|
+
|
|
50
|
+
it('returns empty string for null', () => {
|
|
51
|
+
expect(getFirstName(null)).toBe('');
|
|
52
|
+
});
|
|
53
|
+
|
|
54
|
+
it('returns empty string for undefined', () => {
|
|
55
|
+
expect(getFirstName(undefined)).toBe('');
|
|
56
|
+
});
|
|
57
|
+
});
|
|
58
|
+
|
|
59
|
+
describe('getGreeting', () => {
|
|
60
|
+
it('returns a string containing the name', () => {
|
|
61
|
+
const greeting = getGreeting('Alex');
|
|
62
|
+
expect(greeting).toContain('Alex');
|
|
63
|
+
});
|
|
64
|
+
|
|
65
|
+
it('returns different greetings for different time periods', () => {
|
|
66
|
+
// Run multiple times to increase chance of getting different greetings
|
|
67
|
+
const greetings = new Set();
|
|
68
|
+
for (let i = 0; i < 20; i++) {
|
|
69
|
+
clearGreetingCache();
|
|
70
|
+
greetings.add(getGreeting('Test'));
|
|
71
|
+
}
|
|
72
|
+
// Should have at least one greeting (randomness means we might get the same one)
|
|
73
|
+
expect(greetings.size).toBeGreaterThanOrEqual(1);
|
|
74
|
+
});
|
|
75
|
+
|
|
76
|
+
it('caches greeting in localStorage', () => {
|
|
77
|
+
getGreeting('Alex');
|
|
78
|
+
expect(window.localStorage.setItem).toHaveBeenCalled();
|
|
79
|
+
});
|
|
80
|
+
|
|
81
|
+
it('returns cached greeting for same user', () => {
|
|
82
|
+
const greeting1 = getGreeting('Alex');
|
|
83
|
+
const greeting2 = getGreeting('Alex');
|
|
84
|
+
expect(greeting1).toBe(greeting2);
|
|
85
|
+
});
|
|
86
|
+
|
|
87
|
+
it('generates new greeting for different user', () => {
|
|
88
|
+
const greeting1 = getGreeting('Alex');
|
|
89
|
+
clearGreetingCache();
|
|
90
|
+
const greeting2 = getGreeting('Beth');
|
|
91
|
+
expect(greeting2).toContain('Beth');
|
|
92
|
+
});
|
|
93
|
+
|
|
94
|
+
it('handles localStorage errors gracefully', () => {
|
|
95
|
+
window.localStorage.getItem.mockImplementation(() => {
|
|
96
|
+
throw new Error('localStorage error');
|
|
97
|
+
});
|
|
98
|
+
|
|
99
|
+
// Should not throw
|
|
100
|
+
expect(() => getGreeting('Alex')).not.toThrow();
|
|
101
|
+
const greeting = getGreeting('Alex');
|
|
102
|
+
expect(greeting).toContain('Alex');
|
|
103
|
+
});
|
|
104
|
+
|
|
105
|
+
it('handles invalid cached JSON gracefully', () => {
|
|
106
|
+
window.localStorage.getItem.mockReturnValue('invalid json');
|
|
107
|
+
|
|
108
|
+
// Should not throw and should generate new greeting
|
|
109
|
+
expect(() => getGreeting('Alex')).not.toThrow();
|
|
110
|
+
});
|
|
111
|
+
});
|
|
112
|
+
|
|
113
|
+
describe('clearGreetingCache', () => {
|
|
114
|
+
it('removes greeting from localStorage', () => {
|
|
115
|
+
getGreeting('Alex');
|
|
116
|
+
clearGreetingCache();
|
|
117
|
+
expect(window.localStorage.removeItem).toHaveBeenCalledWith('micdrop_greeting');
|
|
118
|
+
});
|
|
119
|
+
|
|
120
|
+
it('handles localStorage errors gracefully', () => {
|
|
121
|
+
window.localStorage.removeItem.mockImplementation(() => {
|
|
122
|
+
throw new Error('localStorage error');
|
|
123
|
+
});
|
|
124
|
+
|
|
125
|
+
// Should not throw
|
|
126
|
+
expect(() => clearGreetingCache()).not.toThrow();
|
|
127
|
+
});
|
|
128
|
+
});
|
|
129
|
+
|
|
130
|
+
describe('greeting content', () => {
|
|
131
|
+
it('greeting always includes user name', () => {
|
|
132
|
+
for (let i = 0; i < 10; i++) {
|
|
133
|
+
clearGreetingCache();
|
|
134
|
+
const greeting = getGreeting('TestUser');
|
|
135
|
+
expect(greeting).toContain('TestUser');
|
|
136
|
+
}
|
|
137
|
+
});
|
|
138
|
+
|
|
139
|
+
it('handles names with special characters', () => {
|
|
140
|
+
const greeting = getGreeting("O'Brien");
|
|
141
|
+
expect(greeting).toContain("O'Brien");
|
|
142
|
+
});
|
|
143
|
+
|
|
144
|
+
it('handles names with numbers', () => {
|
|
145
|
+
const greeting = getGreeting('User123');
|
|
146
|
+
expect(greeting).toContain('User123');
|
|
147
|
+
});
|
|
148
|
+
});
|
|
149
|
+
|
|
150
|
+
describe('localStorage setItem error handling', () => {
|
|
151
|
+
it('handles localStorage setItem error gracefully', () => {
|
|
152
|
+
window.localStorage.setItem.mockImplementation(() => {
|
|
153
|
+
throw new Error('localStorage setItem error');
|
|
154
|
+
});
|
|
155
|
+
|
|
156
|
+
// Should not throw even when setItem fails
|
|
157
|
+
expect(() => getGreeting('Alex')).not.toThrow();
|
|
158
|
+
const greeting = getGreeting('Alex');
|
|
159
|
+
expect(greeting).toContain('Alex');
|
|
160
|
+
});
|
|
161
|
+
});
|
|
162
|
+
|
|
163
|
+
describe('SSR behavior', () => {
|
|
164
|
+
it('works without window (SSR simulation)', () => {
|
|
165
|
+
// The module checks typeof window === 'undefined' for SSR
|
|
166
|
+
// In JSDOM window exists, but we can test by checking that function works
|
|
167
|
+
const greeting = getGreeting('TestUser');
|
|
168
|
+
expect(typeof greeting).toBe('string');
|
|
169
|
+
expect(greeting).toContain('TestUser');
|
|
170
|
+
});
|
|
171
|
+
});
|
|
172
|
+
|
|
173
|
+
describe('time period coverage', () => {
|
|
174
|
+
// Test all time period branches by mocking Date
|
|
175
|
+
const RealDate = Date;
|
|
176
|
+
|
|
177
|
+
afterEach(() => {
|
|
178
|
+
global.Date = RealDate;
|
|
179
|
+
});
|
|
180
|
+
|
|
181
|
+
it('generates greeting for night period (21-24)', () => {
|
|
182
|
+
// Mock Date to return 22:00
|
|
183
|
+
global.Date = class extends RealDate {
|
|
184
|
+
constructor() {
|
|
185
|
+
super();
|
|
186
|
+
return new RealDate('2025-06-15T22:00:00');
|
|
187
|
+
}
|
|
188
|
+
getHours() { return 22; }
|
|
189
|
+
getDay() { return 0; }
|
|
190
|
+
};
|
|
191
|
+
|
|
192
|
+
clearGreetingCache();
|
|
193
|
+
const greeting = getGreeting('NightUser');
|
|
194
|
+
expect(greeting).toContain('NightUser');
|
|
195
|
+
});
|
|
196
|
+
|
|
197
|
+
it('generates greeting for late night period (0-5 AM)', () => {
|
|
198
|
+
// Mock Date to return 03:00
|
|
199
|
+
global.Date = class extends RealDate {
|
|
200
|
+
constructor() {
|
|
201
|
+
super();
|
|
202
|
+
return new RealDate('2025-06-15T03:00:00');
|
|
203
|
+
}
|
|
204
|
+
getHours() { return 3; }
|
|
205
|
+
getDay() { return 0; }
|
|
206
|
+
};
|
|
207
|
+
|
|
208
|
+
clearGreetingCache();
|
|
209
|
+
const greeting = getGreeting('LateNightUser');
|
|
210
|
+
expect(greeting).toContain('LateNightUser');
|
|
211
|
+
});
|
|
212
|
+
|
|
213
|
+
it('generates greeting for early morning period (5-7)', () => {
|
|
214
|
+
global.Date = class extends RealDate {
|
|
215
|
+
constructor() {
|
|
216
|
+
super();
|
|
217
|
+
return new RealDate('2025-06-15T06:00:00');
|
|
218
|
+
}
|
|
219
|
+
getHours() { return 6; }
|
|
220
|
+
getDay() { return 1; }
|
|
221
|
+
};
|
|
222
|
+
|
|
223
|
+
clearGreetingCache();
|
|
224
|
+
const greeting = getGreeting('EarlyUser');
|
|
225
|
+
expect(greeting).toContain('EarlyUser');
|
|
226
|
+
});
|
|
227
|
+
|
|
228
|
+
it('generates greeting for morning period (7-12)', () => {
|
|
229
|
+
global.Date = class extends RealDate {
|
|
230
|
+
constructor() {
|
|
231
|
+
super();
|
|
232
|
+
return new RealDate('2025-06-15T09:00:00');
|
|
233
|
+
}
|
|
234
|
+
getHours() { return 9; }
|
|
235
|
+
getDay() { return 2; }
|
|
236
|
+
};
|
|
237
|
+
|
|
238
|
+
clearGreetingCache();
|
|
239
|
+
const greeting = getGreeting('MorningUser');
|
|
240
|
+
expect(greeting).toContain('MorningUser');
|
|
241
|
+
});
|
|
242
|
+
|
|
243
|
+
it('generates greeting for afternoon period (12-17)', () => {
|
|
244
|
+
global.Date = class extends RealDate {
|
|
245
|
+
constructor() {
|
|
246
|
+
super();
|
|
247
|
+
return new RealDate('2025-06-15T14:00:00');
|
|
248
|
+
}
|
|
249
|
+
getHours() { return 14; }
|
|
250
|
+
getDay() { return 3; }
|
|
251
|
+
};
|
|
252
|
+
|
|
253
|
+
clearGreetingCache();
|
|
254
|
+
const greeting = getGreeting('AfternoonUser');
|
|
255
|
+
expect(greeting).toContain('AfternoonUser');
|
|
256
|
+
});
|
|
257
|
+
|
|
258
|
+
it('generates greeting for evening period (17-21)', () => {
|
|
259
|
+
global.Date = class extends RealDate {
|
|
260
|
+
constructor() {
|
|
261
|
+
super();
|
|
262
|
+
return new RealDate('2025-06-15T19:00:00');
|
|
263
|
+
}
|
|
264
|
+
getHours() { return 19; }
|
|
265
|
+
getDay() { return 4; }
|
|
266
|
+
};
|
|
267
|
+
|
|
268
|
+
clearGreetingCache();
|
|
269
|
+
const greeting = getGreeting('EveningUser');
|
|
270
|
+
expect(greeting).toContain('EveningUser');
|
|
271
|
+
});
|
|
272
|
+
});
|
|
273
|
+
|
|
274
|
+
describe('time-based greeting generation', () => {
|
|
275
|
+
it('generates consistent greeting with cached data for same period and day', () => {
|
|
276
|
+
// First call generates and caches
|
|
277
|
+
const greeting1 = getGreeting('Alex');
|
|
278
|
+
|
|
279
|
+
// Second call should return cached
|
|
280
|
+
const greeting2 = getGreeting('Alex');
|
|
281
|
+
|
|
282
|
+
expect(greeting1).toBe(greeting2);
|
|
283
|
+
});
|
|
284
|
+
|
|
285
|
+
it('generates greeting containing name for various scenarios', () => {
|
|
286
|
+
// Test with longer names
|
|
287
|
+
clearGreetingCache();
|
|
288
|
+
const longNameGreeting = getGreeting('AlexanderTheGreat');
|
|
289
|
+
expect(longNameGreeting).toContain('AlexanderTheGreat');
|
|
290
|
+
|
|
291
|
+
// Test with short names
|
|
292
|
+
clearGreetingCache();
|
|
293
|
+
const shortNameGreeting = getGreeting('Jo');
|
|
294
|
+
expect(shortNameGreeting).toContain('Jo');
|
|
295
|
+
});
|
|
296
|
+
|
|
297
|
+
it('respects cache when period and day match', () => {
|
|
298
|
+
const greeting1 = getGreeting('TestUser');
|
|
299
|
+
|
|
300
|
+
// Mock getting a valid cached value
|
|
301
|
+
const cachedValue = window.localStorage.store['micdrop_greeting'];
|
|
302
|
+
expect(cachedValue).toBeDefined();
|
|
303
|
+
|
|
304
|
+
// Parse to verify structure
|
|
305
|
+
const parsed = JSON.parse(cachedValue);
|
|
306
|
+
expect(parsed).toHaveProperty('greeting');
|
|
307
|
+
expect(parsed).toHaveProperty('period');
|
|
308
|
+
expect(parsed).toHaveProperty('day');
|
|
309
|
+
expect(parsed).toHaveProperty('cachedName');
|
|
310
|
+
expect(parsed.cachedName).toBe('TestUser');
|
|
311
|
+
});
|
|
312
|
+
});
|
|
313
|
+
|
|
314
|
+
describe('SSR mode', () => {
|
|
315
|
+
it('generates greeting without caching when window is undefined', async () => {
|
|
316
|
+
vi.resetModules();
|
|
317
|
+
|
|
318
|
+
// Save original window
|
|
319
|
+
const originalWindow = global.window;
|
|
320
|
+
|
|
321
|
+
// Delete window to simulate SSR
|
|
322
|
+
delete global.window;
|
|
323
|
+
|
|
324
|
+
// Re-import module in SSR context
|
|
325
|
+
const { getGreeting: ssrGetGreeting } = await import('./greetings.js');
|
|
326
|
+
|
|
327
|
+
// Should still generate a greeting
|
|
328
|
+
const greeting = ssrGetGreeting('TestUser');
|
|
329
|
+
expect(greeting).toBeDefined();
|
|
330
|
+
expect(typeof greeting).toBe('string');
|
|
331
|
+
expect(greeting).toContain('TestUser');
|
|
332
|
+
|
|
333
|
+
// Restore window
|
|
334
|
+
global.window = originalWindow;
|
|
335
|
+
});
|
|
336
|
+
});
|
|
337
|
+
});
|