@getmicdrop/svelte-components 5.6.0 → 5.7.1
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/calendar/AboutShow/AboutShow.svelte +172 -172
- package/dist/calendar/Calendar/MiniMonthCalendar.svelte +782 -782
- package/dist/calendar/FAQs/FAQs.svelte +75 -75
- package/dist/calendar/MonthSwitcher/MonthSwitcher.svelte +126 -126
- package/dist/calendar/OrderSummary/OrderSummary.svelte +367 -367
- package/dist/calendar/PublicCard/PublicCard.svelte +134 -134
- package/dist/calendar/ShowCard/ShowCard.svelte +157 -157
- package/dist/calendar/ShowTimeCard/ShowTimeCard.svelte +61 -61
- package/dist/components/Heading.svelte +57 -0
- package/dist/components/Heading.svelte.d.ts +12 -0
- package/dist/components/Heading.svelte.d.ts.map +1 -0
- package/dist/components/Layout/AppShell.svelte +104 -104
- package/dist/components/Layout/ContentSection.svelte +80 -80
- package/dist/components/Layout/Grid.svelte +4 -4
- 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 +6 -6
- package/dist/components/Layout/Text.svelte +87 -87
- package/dist/components/Layout/TwoColumn.svelte +108 -108
- package/dist/components/Layout/__tests__/AppShell.test.d.ts +2 -0
- package/dist/components/Layout/__tests__/AppShell.test.d.ts.map +1 -0
- package/dist/components/Layout/__tests__/AppShell.test.js +95 -0
- package/dist/components/Layout/__tests__/ContentSection.test.d.ts +2 -0
- package/dist/components/Layout/__tests__/ContentSection.test.d.ts.map +1 -0
- package/dist/components/Layout/__tests__/ContentSection.test.js +112 -0
- package/dist/components/Layout/__tests__/PageContainer.test.d.ts +2 -0
- package/dist/components/Layout/__tests__/PageContainer.test.d.ts.map +1 -0
- package/dist/components/Layout/__tests__/PageContainer.test.js +133 -0
- package/dist/components/Layout/__tests__/Responsive.test.d.ts +2 -0
- package/dist/components/Layout/__tests__/Responsive.test.d.ts.map +1 -0
- package/dist/components/Layout/__tests__/Responsive.test.js +123 -0
- package/dist/components/Layout/index.d.ts +11 -0
- package/dist/components/Layout/index.d.ts.map +1 -0
- package/dist/components/Layout/index.js +14 -0
- package/dist/components/Text.svelte +40 -0
- package/dist/components/Text.svelte.d.ts +11 -0
- package/dist/components/Text.svelte.d.ts.map +1 -0
- package/dist/components/index.d.ts +4 -0
- package/dist/components/index.d.ts.map +1 -0
- package/dist/components/index.js +12 -0
- package/dist/constants/validation.js +91 -91
- package/dist/constants/validation.spec.js +64 -64
- 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/index.d.ts +1 -0
- package/dist/index.js +50 -40
- package/dist/patterns/data/DataGrid.svelte +45 -45
- package/dist/patterns/data/DataList.svelte +24 -24
- package/dist/patterns/data/DataTable.svelte +36 -36
- 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 +32 -32
- package/dist/patterns/forms/FormValidationSummary.stories.svelte +83 -83
- package/dist/patterns/forms/FormValidationSummary.svelte +74 -74
- package/dist/patterns/layout/Sidebar.svelte +39 -39
- package/dist/patterns/navigation/BottomNav.stories.svelte +117 -117
- package/dist/patterns/navigation/BottomNav.svelte +74 -64
- package/dist/patterns/navigation/BottomNav.svelte.d.ts.map +1 -1
- package/dist/patterns/navigation/Header.stories.svelte +77 -77
- package/dist/patterns/navigation/Header.svelte +193 -193
- package/dist/patterns/page/PageHeader.svelte +18 -18
- 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 +24 -24
- package/dist/patterns/page/SectionHeader.svelte +29 -29
- 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 +42 -42
- package/dist/primitives/Accordion/AccordionItem.svelte +95 -95
- package/dist/primitives/Accordion/AccordionItemWrapper.test.svelte +107 -107
- package/dist/primitives/Accordion/AccordionItemWrapper.test.svelte.d.ts +2 -2
- package/dist/primitives/Accordion/AccordionItemWrapper.test.svelte.d.ts.map +1 -1
- package/dist/primitives/Alert/Alert.spec.js +173 -173
- package/dist/primitives/Alert/Alert.stories.svelte +88 -88
- package/dist/primitives/Alert/Alert.svelte +27 -27
- package/dist/primitives/Avatar/Avatar.stories.svelte +94 -94
- package/dist/primitives/Avatar/Avatar.svelte +66 -66
- package/dist/primitives/Badges/Badge.spec.js +144 -144
- package/dist/primitives/Badges/Badge.stories.svelte +86 -86
- package/dist/primitives/Badges/Badge.svelte +79 -79
- package/dist/primitives/BottomSheet/BottomSheet.spec.js +136 -136
- package/dist/primitives/BottomSheet/BottomSheet.stories.svelte +83 -83
- package/dist/primitives/BottomSheet/BottomSheet.svelte +100 -100
- package/dist/primitives/Breadcrumb/Breadcrumb.spec.js +122 -122
- package/dist/primitives/Breadcrumb/Breadcrumb.stories.svelte +23 -23
- package/dist/primitives/Breadcrumb/Breadcrumb.svelte +89 -89
- package/dist/primitives/Button/Button.spec.js +223 -223
- package/dist/primitives/Button/Button.stories.svelte +76 -76
- package/dist/primitives/Button/Button.svelte +270 -270
- 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/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 +136 -136
- package/dist/primitives/Drawer/Drawer.stories.svelte +80 -80
- package/dist/primitives/Drawer/Drawer.svelte +120 -120
- package/dist/primitives/Dropdown/Dropdown.stories.svelte +137 -137
- package/dist/primitives/Dropdown/Dropdown.svelte +14 -14
- package/dist/primitives/Dropdown/DropdownDivider.svelte +9 -0
- package/dist/primitives/Dropdown/DropdownDivider.svelte.d.ts +7 -0
- package/dist/primitives/Dropdown/DropdownDivider.svelte.d.ts.map +1 -0
- package/dist/primitives/Dropdown/DropdownItem.svelte +80 -80
- package/dist/primitives/Helper/Helper.svelte +33 -0
- package/dist/primitives/Helper/Helper.svelte.d.ts +18 -0
- package/dist/primitives/Helper/Helper.svelte.d.ts.map +1 -0
- package/dist/primitives/Icons/ArrowLeft.svelte +8 -8
- package/dist/primitives/Icons/ArrowRight.svelte +8 -8
- package/dist/primitives/Icons/Availability.svelte +14 -14
- package/dist/primitives/Icons/Back.svelte +14 -14
- package/dist/primitives/Icons/CheckCircle.svelte +6 -6
- package/dist/primitives/Icons/CheckCircleOutline.svelte +15 -15
- package/dist/primitives/Icons/ChevronLeft.svelte +4 -4
- package/dist/primitives/Icons/ChevronRight.svelte +4 -4
- package/dist/primitives/Icons/Copy.svelte +15 -15
- package/dist/primitives/Icons/Cross.svelte +5 -5
- package/dist/primitives/Icons/DownArrow.svelte +8 -8
- package/dist/primitives/Icons/ErrorCircle.svelte +6 -6
- package/dist/primitives/Icons/FacebookIcon.svelte +2 -2
- package/dist/primitives/Icons/Home.svelte +15 -15
- package/dist/primitives/Icons/Icon.spec.js +169 -169
- package/dist/primitives/Icons/Icon.stories.svelte +100 -100
- package/dist/primitives/Icons/Icon.svelte +52 -52
- package/dist/primitives/Icons/IconGallery.stories.svelte +235 -235
- package/dist/primitives/Icons/Info.svelte +7 -7
- package/dist/primitives/Icons/InstagramIcon.svelte +4 -4
- package/dist/primitives/Icons/LogoInstagram.svelte +2 -2
- package/dist/primitives/Icons/Message.svelte +15 -15
- package/dist/primitives/Icons/MoonIcon.svelte +5 -5
- package/dist/primitives/Icons/More.svelte +21 -21
- package/dist/primitives/Icons/MoreHori.spec.js +61 -61
- package/dist/primitives/Icons/MoreHori.svelte +22 -22
- package/dist/primitives/Icons/Notification.svelte +14 -14
- package/dist/primitives/Icons/Payment.svelte +14 -14
- package/dist/primitives/Icons/Profile.svelte +21 -21
- package/dist/primitives/Icons/Reload.svelte +29 -29
- package/dist/primitives/Icons/Shows.svelte +21 -21
- package/dist/primitives/Icons/Signout.svelte +21 -21
- package/dist/primitives/Icons/SunIcon.svelte +8 -8
- package/dist/primitives/Icons/TiktokIcon.svelte +2 -2
- package/dist/primitives/Icons/TwitterIcon.svelte +2 -2
- package/dist/primitives/Icons/WarningIcon.spec.js +18 -18
- package/dist/primitives/Icons/WarningIcon.svelte +5 -5
- package/dist/primitives/Input/Input.spec.js +573 -573
- package/dist/primitives/Input/Input.stories.svelte +139 -139
- package/dist/primitives/Input/Input.svelte +416 -417
- package/dist/primitives/Input/Input.svelte.d.ts +2 -4
- package/dist/primitives/Input/Input.svelte.d.ts.map +1 -1
- package/dist/primitives/Input/Select.spec.js +212 -212
- package/dist/primitives/Input/Select.stories.svelte +112 -112
- package/dist/primitives/Input/Select.svelte +128 -128
- package/dist/primitives/Input/Textarea.stories.svelte +137 -137
- package/dist/primitives/Input/Textarea.svelte +35 -35
- package/dist/primitives/Label/Label.svelte +37 -37
- package/dist/primitives/Modal/Modal.spec.js +99 -99
- package/dist/primitives/Modal/Modal.stories.svelte +86 -86
- package/dist/primitives/Modal/Modal.svelte +157 -158
- package/dist/primitives/Modal/Modal.svelte.d.ts +6 -8
- package/dist/primitives/Modal/Modal.svelte.d.ts.map +1 -1
- package/dist/primitives/NumberInput/NumberInput.svelte +105 -106
- package/dist/primitives/NumberInput/NumberInput.svelte.d.ts +0 -2
- package/dist/primitives/NumberInput/NumberInput.svelte.d.ts.map +1 -1
- package/dist/primitives/Pagination/Pagination.stories.svelte +76 -76
- package/dist/primitives/Pagination/Pagination.svelte +261 -261
- package/dist/primitives/Radio/Radio.stories.svelte +80 -80
- package/dist/primitives/Radio/Radio.svelte +67 -67
- package/dist/primitives/Skeleton/CardPlaceholder.svelte +87 -87
- package/dist/primitives/Skeleton/ImagePlaceholder.svelte +59 -59
- package/dist/primitives/Skeleton/ListPlaceholder.svelte +76 -76
- package/dist/primitives/Skeleton/Skeleton.stories.svelte +151 -151
- package/dist/primitives/Skeleton/Skeleton.svelte +26 -26
- package/dist/primitives/Spinner/Spinner.spec.js +71 -71
- package/dist/primitives/Spinner/Spinner.stories.svelte +29 -29
- package/dist/primitives/Spinner/Spinner.svelte +20 -20
- package/dist/primitives/Tabs/TabItem.svelte +49 -49
- package/dist/primitives/Tabs/Tabs.stories.svelte +112 -112
- package/dist/primitives/Tabs/Tabs.svelte +123 -123
- package/dist/primitives/Toggle.spec.js +143 -143
- package/dist/primitives/Toggle.stories.svelte +92 -92
- package/dist/primitives/Toggle.svelte +70 -71
- package/dist/primitives/Toggle.svelte.d.ts +2 -4
- package/dist/primitives/Toggle.svelte.d.ts.map +1 -1
- package/dist/primitives/Tooltip/Tooltip.svelte +83 -0
- package/dist/primitives/Tooltip/Tooltip.svelte.d.ts +15 -0
- package/dist/primitives/Tooltip/Tooltip.svelte.d.ts.map +1 -0
- package/dist/primitives/Typography/Typography.svelte +53 -53
- package/dist/primitives/ValidationError.spec.js +103 -103
- package/dist/primitives/ValidationError.stories.svelte +69 -69
- package/dist/primitives/ValidationError.svelte +29 -29
- package/dist/primitives/index.d.ts +3 -0
- package/dist/primitives/index.js +91 -84
- package/dist/recipes/CropImage/CropImage.spec.js +208 -208
- package/dist/recipes/CropImage/CropImage.stories.svelte +104 -104
- package/dist/recipes/CropImage/CropImage.svelte +238 -238
- package/dist/recipes/ImageUploader/ImageUploader.spec.js +6 -5
- package/dist/recipes/ImageUploader/ImageUploader.stories.svelte +125 -125
- package/dist/recipes/ImageUploader/ImageUploader.svelte +804 -804
- package/dist/recipes/Toaster/Toaster.stories.svelte +62 -62
- package/dist/recipes/feedback/EmptyState/EmptyState.svelte +1 -1
- package/dist/recipes/feedback/ErrorDisplay.spec.js +69 -69
- package/dist/recipes/feedback/ErrorDisplay.stories.svelte +101 -101
- package/dist/recipes/feedback/ErrorDisplay.svelte +1 -1
- package/dist/recipes/feedback/StatusIndicator/StatusIndicator.spec.js +133 -133
- package/dist/recipes/feedback/StatusIndicator/StatusIndicator.svelte +157 -157
- 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 +80 -80
- package/dist/recipes/fields/TextareaField.svelte +97 -97
- package/dist/recipes/fields/ToggleField.svelte +60 -60
- package/dist/recipes/fields/index.js +7 -7
- package/dist/recipes/inputs/MultiSelect.spec.js +258 -258
- package/dist/recipes/inputs/MultiSelect.stories.svelte +133 -133
- package/dist/recipes/inputs/MultiSelect.svelte +277 -256
- package/dist/recipes/inputs/MultiSelect.svelte.d.ts +54 -21
- package/dist/recipes/inputs/MultiSelect.svelte.d.ts.map +1 -1
- package/dist/recipes/inputs/OTPInput.spec.js +251 -251
- package/dist/recipes/inputs/OTPInput.stories.svelte +162 -162
- package/dist/recipes/inputs/OTPInput.svelte +29 -29
- package/dist/recipes/inputs/PasswordInput.svelte +22 -22
- package/dist/recipes/inputs/PasswordStrengthIndicator/PasswordStrengthIndicator.svelte +117 -117
- package/dist/recipes/inputs/PlaceAutocomplete/PlaceAutocomplete.spec.js +9 -4
- package/dist/recipes/inputs/PlaceAutocomplete/PlaceAutocomplete.stories.svelte +123 -123
- package/dist/recipes/inputs/PlaceAutocomplete/PlaceAutocomplete.svelte +332 -327
- package/dist/recipes/inputs/PlaceAutocomplete/PlaceAutocomplete.svelte.d.ts +12 -1
- package/dist/recipes/inputs/PlaceAutocomplete/PlaceAutocomplete.svelte.d.ts.map +1 -1
- package/dist/recipes/inputs/Search.svelte +37 -37
- package/dist/recipes/inputs/SelectDropdown.svelte +57 -57
- package/dist/recipes/modals/AlertModal.svelte +130 -130
- package/dist/recipes/modals/ConfirmationModal.spec.js +206 -206
- package/dist/recipes/modals/ConfirmationModal.stories.svelte +119 -119
- package/dist/recipes/modals/ConfirmationModal.svelte +152 -152
- package/dist/recipes/modals/InputModal.svelte +182 -182
- 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 +206 -206
- package/dist/services/EventService.js +75 -75
- package/dist/services/EventService.spec.js +217 -217
- package/dist/services/ShowService.spec.js +345 -345
- package/dist/stores/auth.js +36 -36
- package/dist/stores/auth.spec.js +139 -139
- package/dist/stores/toaster.js +13 -13
- 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 +206 -206
- package/dist/stories/PrimitivesGallery.stories.svelte +19 -19
- package/dist/stories/PrimitivesGallery.svelte +725 -725
- package/dist/stories/RecipesGallery.stories.svelte +19 -19
- package/dist/stories/RecipesGallery.svelte +271 -271
- package/dist/stories/button-audit-manifest.json +11186 -11186
- package/dist/tailwind/preset.cjs +82 -82
- package/dist/telemetry.js +405 -405
- package/dist/telemetry.server.spec.js +11 -8
- package/dist/telemetry.spec.js +1169 -1144
- package/dist/tokens/tokens.css +87 -87
- package/dist/tokens/typography-base.css +163 -163
- package/dist/utils/apiConfig.spec.js +219 -219
- package/dist/utils/imageValidation.spec.js +62 -59
- package/dist/utils/logger.d.ts +19 -0
- package/dist/utils/logger.d.ts.map +1 -0
- package/dist/utils/logger.js +47 -0
- package/dist/utils/transitions.js +62 -62
- package/dist/utils/utils.js +354 -354
- package/package.json +296 -292
|
@@ -1,137 +1,137 @@
|
|
|
1
|
-
<script module>
|
|
2
|
-
import { defineMeta } from '@storybook/addon-svelte-csf';
|
|
3
|
-
import Textarea from './Textarea.svelte';
|
|
4
|
-
|
|
5
|
-
const { Story } = defineMeta({
|
|
6
|
-
title: 'Primitives/Textarea',
|
|
7
|
-
component: Textarea,
|
|
8
|
-
tags: ['autodocs'],
|
|
9
|
-
argTypes: {
|
|
10
|
-
label: { control: 'text' },
|
|
11
|
-
placeholder: { control: 'text' },
|
|
12
|
-
rows: { control: 'number' },
|
|
13
|
-
disabled: { control: 'boolean' },
|
|
14
|
-
required: { control: 'boolean' },
|
|
15
|
-
maxlength: { control: 'number' },
|
|
16
|
-
errorText: { control: 'text' },
|
|
17
|
-
},
|
|
18
|
-
parameters: {
|
|
19
|
-
docs: {
|
|
20
|
-
description: {
|
|
21
|
-
component: 'Multi-line text input component.',
|
|
22
|
-
},
|
|
23
|
-
},
|
|
24
|
-
},
|
|
25
|
-
});
|
|
26
|
-
|
|
27
|
-
let value = $state('');
|
|
28
|
-
</script>
|
|
29
|
-
|
|
30
|
-
<Story name="Default">
|
|
31
|
-
{#snippet template()}
|
|
32
|
-
<div class="max-w-md">
|
|
33
|
-
<Textarea
|
|
34
|
-
label="Description"
|
|
35
|
-
placeholder="Enter your description..."
|
|
36
|
-
/>
|
|
37
|
-
</div>
|
|
38
|
-
{/snippet}
|
|
39
|
-
</Story>
|
|
40
|
-
|
|
41
|
-
<Story name="With Value">
|
|
42
|
-
{#snippet template()}
|
|
43
|
-
<div class="max-w-md">
|
|
44
|
-
<Textarea
|
|
45
|
-
label="Bio"
|
|
46
|
-
placeholder="Tell us about yourself..."
|
|
47
|
-
bind:value={value}
|
|
48
|
-
/>
|
|
49
|
-
<p class="text-sm text-gray-500 mt-2">{value.length} characters</p>
|
|
50
|
-
</div>
|
|
51
|
-
{/snippet}
|
|
52
|
-
</Story>
|
|
53
|
-
|
|
54
|
-
<Story name="With Max Length">
|
|
55
|
-
{#snippet template()}
|
|
56
|
-
<div class="max-w-md">
|
|
57
|
-
<Textarea
|
|
58
|
-
label="Short Bio"
|
|
59
|
-
placeholder="Keep it brief..."
|
|
60
|
-
maxlength={200}
|
|
61
|
-
helperText="Maximum 200 characters"
|
|
62
|
-
/>
|
|
63
|
-
</div>
|
|
64
|
-
{/snippet}
|
|
65
|
-
</Story>
|
|
66
|
-
|
|
67
|
-
<Story name="With Error">
|
|
68
|
-
{#snippet template()}
|
|
69
|
-
<div class="max-w-md">
|
|
70
|
-
<Textarea
|
|
71
|
-
label="Comments"
|
|
72
|
-
placeholder="Enter your comments..."
|
|
73
|
-
errorText="Please provide at least 10 characters"
|
|
74
|
-
/>
|
|
75
|
-
</div>
|
|
76
|
-
{/snippet}
|
|
77
|
-
</Story>
|
|
78
|
-
|
|
79
|
-
<Story name="Custom Rows">
|
|
80
|
-
{#snippet template()}
|
|
81
|
-
<div class="space-y-6 max-w-md">
|
|
82
|
-
<Textarea
|
|
83
|
-
label="Small (2 rows)"
|
|
84
|
-
placeholder="Two rows..."
|
|
85
|
-
rows={2}
|
|
86
|
-
/>
|
|
87
|
-
<Textarea
|
|
88
|
-
label="Medium (4 rows)"
|
|
89
|
-
placeholder="Four rows..."
|
|
90
|
-
rows={4}
|
|
91
|
-
/>
|
|
92
|
-
<Textarea
|
|
93
|
-
label="Large (8 rows)"
|
|
94
|
-
placeholder="Eight rows..."
|
|
95
|
-
rows={8}
|
|
96
|
-
/>
|
|
97
|
-
</div>
|
|
98
|
-
{/snippet}
|
|
99
|
-
</Story>
|
|
100
|
-
|
|
101
|
-
<Story name="Disabled">
|
|
102
|
-
{#snippet template()}
|
|
103
|
-
<div class="max-w-md">
|
|
104
|
-
<Textarea
|
|
105
|
-
label="Disabled Textarea"
|
|
106
|
-
placeholder="Cannot edit this..."
|
|
107
|
-
disabled
|
|
108
|
-
/>
|
|
109
|
-
</div>
|
|
110
|
-
{/snippet}
|
|
111
|
-
</Story>
|
|
112
|
-
|
|
113
|
-
<Story name="All States">
|
|
114
|
-
{#snippet template()}
|
|
115
|
-
<div class="grid grid-cols-2 gap-6 max-w-2xl">
|
|
116
|
-
<Textarea
|
|
117
|
-
label="Default"
|
|
118
|
-
placeholder="Enter text..."
|
|
119
|
-
/>
|
|
120
|
-
<Textarea
|
|
121
|
-
label="With Error"
|
|
122
|
-
placeholder="Enter text..."
|
|
123
|
-
errorText="This field is required"
|
|
124
|
-
/>
|
|
125
|
-
<Textarea
|
|
126
|
-
label="Disabled"
|
|
127
|
-
placeholder="Cannot edit..."
|
|
128
|
-
disabled
|
|
129
|
-
/>
|
|
130
|
-
<Textarea
|
|
131
|
-
label="With Helper"
|
|
132
|
-
placeholder="Enter text..."
|
|
133
|
-
helperText="Optional additional information"
|
|
134
|
-
/>
|
|
135
|
-
</div>
|
|
136
|
-
{/snippet}
|
|
137
|
-
</Story>
|
|
1
|
+
<script module>
|
|
2
|
+
import { defineMeta } from '@storybook/addon-svelte-csf';
|
|
3
|
+
import Textarea from './Textarea.svelte';
|
|
4
|
+
|
|
5
|
+
const { Story } = defineMeta({
|
|
6
|
+
title: 'Primitives/Textarea',
|
|
7
|
+
component: Textarea,
|
|
8
|
+
tags: ['autodocs'],
|
|
9
|
+
argTypes: {
|
|
10
|
+
label: { control: 'text' },
|
|
11
|
+
placeholder: { control: 'text' },
|
|
12
|
+
rows: { control: 'number' },
|
|
13
|
+
disabled: { control: 'boolean' },
|
|
14
|
+
required: { control: 'boolean' },
|
|
15
|
+
maxlength: { control: 'number' },
|
|
16
|
+
errorText: { control: 'text' },
|
|
17
|
+
},
|
|
18
|
+
parameters: {
|
|
19
|
+
docs: {
|
|
20
|
+
description: {
|
|
21
|
+
component: 'Multi-line text input component.',
|
|
22
|
+
},
|
|
23
|
+
},
|
|
24
|
+
},
|
|
25
|
+
});
|
|
26
|
+
|
|
27
|
+
let value = $state('');
|
|
28
|
+
</script>
|
|
29
|
+
|
|
30
|
+
<Story name="Default">
|
|
31
|
+
{#snippet template()}
|
|
32
|
+
<div class="max-w-md">
|
|
33
|
+
<Textarea
|
|
34
|
+
label="Description"
|
|
35
|
+
placeholder="Enter your description..."
|
|
36
|
+
/>
|
|
37
|
+
</div>
|
|
38
|
+
{/snippet}
|
|
39
|
+
</Story>
|
|
40
|
+
|
|
41
|
+
<Story name="With Value">
|
|
42
|
+
{#snippet template()}
|
|
43
|
+
<div class="max-w-md">
|
|
44
|
+
<Textarea
|
|
45
|
+
label="Bio"
|
|
46
|
+
placeholder="Tell us about yourself..."
|
|
47
|
+
bind:value={value}
|
|
48
|
+
/>
|
|
49
|
+
<p class="text-sm text-gray-500 mt-2">{value.length} characters</p>
|
|
50
|
+
</div>
|
|
51
|
+
{/snippet}
|
|
52
|
+
</Story>
|
|
53
|
+
|
|
54
|
+
<Story name="With Max Length">
|
|
55
|
+
{#snippet template()}
|
|
56
|
+
<div class="max-w-md">
|
|
57
|
+
<Textarea
|
|
58
|
+
label="Short Bio"
|
|
59
|
+
placeholder="Keep it brief..."
|
|
60
|
+
maxlength={200}
|
|
61
|
+
helperText="Maximum 200 characters"
|
|
62
|
+
/>
|
|
63
|
+
</div>
|
|
64
|
+
{/snippet}
|
|
65
|
+
</Story>
|
|
66
|
+
|
|
67
|
+
<Story name="With Error">
|
|
68
|
+
{#snippet template()}
|
|
69
|
+
<div class="max-w-md">
|
|
70
|
+
<Textarea
|
|
71
|
+
label="Comments"
|
|
72
|
+
placeholder="Enter your comments..."
|
|
73
|
+
errorText="Please provide at least 10 characters"
|
|
74
|
+
/>
|
|
75
|
+
</div>
|
|
76
|
+
{/snippet}
|
|
77
|
+
</Story>
|
|
78
|
+
|
|
79
|
+
<Story name="Custom Rows">
|
|
80
|
+
{#snippet template()}
|
|
81
|
+
<div class="space-y-6 max-w-md">
|
|
82
|
+
<Textarea
|
|
83
|
+
label="Small (2 rows)"
|
|
84
|
+
placeholder="Two rows..."
|
|
85
|
+
rows={2}
|
|
86
|
+
/>
|
|
87
|
+
<Textarea
|
|
88
|
+
label="Medium (4 rows)"
|
|
89
|
+
placeholder="Four rows..."
|
|
90
|
+
rows={4}
|
|
91
|
+
/>
|
|
92
|
+
<Textarea
|
|
93
|
+
label="Large (8 rows)"
|
|
94
|
+
placeholder="Eight rows..."
|
|
95
|
+
rows={8}
|
|
96
|
+
/>
|
|
97
|
+
</div>
|
|
98
|
+
{/snippet}
|
|
99
|
+
</Story>
|
|
100
|
+
|
|
101
|
+
<Story name="Disabled">
|
|
102
|
+
{#snippet template()}
|
|
103
|
+
<div class="max-w-md">
|
|
104
|
+
<Textarea
|
|
105
|
+
label="Disabled Textarea"
|
|
106
|
+
placeholder="Cannot edit this..."
|
|
107
|
+
disabled
|
|
108
|
+
/>
|
|
109
|
+
</div>
|
|
110
|
+
{/snippet}
|
|
111
|
+
</Story>
|
|
112
|
+
|
|
113
|
+
<Story name="All States">
|
|
114
|
+
{#snippet template()}
|
|
115
|
+
<div class="grid grid-cols-2 gap-6 max-w-2xl">
|
|
116
|
+
<Textarea
|
|
117
|
+
label="Default"
|
|
118
|
+
placeholder="Enter text..."
|
|
119
|
+
/>
|
|
120
|
+
<Textarea
|
|
121
|
+
label="With Error"
|
|
122
|
+
placeholder="Enter text..."
|
|
123
|
+
errorText="This field is required"
|
|
124
|
+
/>
|
|
125
|
+
<Textarea
|
|
126
|
+
label="Disabled"
|
|
127
|
+
placeholder="Cannot edit..."
|
|
128
|
+
disabled
|
|
129
|
+
/>
|
|
130
|
+
<Textarea
|
|
131
|
+
label="With Helper"
|
|
132
|
+
placeholder="Enter text..."
|
|
133
|
+
helperText="Optional additional information"
|
|
134
|
+
/>
|
|
135
|
+
</div>
|
|
136
|
+
{/snippet}
|
|
137
|
+
</Story>
|
|
@@ -68,38 +68,38 @@
|
|
|
68
68
|
onfocus?.(event);
|
|
69
69
|
}
|
|
70
70
|
</script>
|
|
71
|
-
|
|
72
|
-
<div class="flex flex-col gap-2 w-full">
|
|
73
|
-
{#if label}
|
|
74
|
-
<label for={id || name} class={`${typography.label} leading-tight`}>
|
|
75
|
-
{label}{#if required}<span class="text-red-500 font-medium text-sm ml-0.5">*</span>{/if}
|
|
76
|
-
</label>
|
|
77
|
-
{/if}
|
|
78
|
-
|
|
79
|
-
<textarea
|
|
80
|
-
{id}
|
|
81
|
-
{name}
|
|
82
|
-
{placeholder}
|
|
83
|
-
{rows}
|
|
84
|
-
{disabled}
|
|
85
|
-
{readonly}
|
|
86
|
-
{maxlength}
|
|
87
|
-
{minlength}
|
|
88
|
-
class="{typography.sm} w-full p-2.5 bg-gray-50 dark:bg-gray-800 leading-normal border rounded-lg resize-y transition-colors focus:outline-none focus:ring-4 focus:ring-blue-300 dark:focus:ring-blue-800 placeholder-gray-500 dark:placeholder-gray-400 {error ? 'border-red-500' : 'border-gray-300 dark:border-gray-600 hover:border-blue-500 focus:border-blue-500'} {disabled ? 'opacity-50 cursor-not-allowed' : ''} {className}"
|
|
89
|
-
bind:value
|
|
90
|
-
oninput={handleInput}
|
|
91
|
-
onchange={handleChange}
|
|
92
|
-
onblur={handleBlur}
|
|
93
|
-
onfocus={handleFocus}
|
|
94
|
-
{onkeydown}
|
|
95
|
-
{onkeyup}
|
|
96
|
-
{onkeypress}
|
|
97
|
-
aria-required={required}
|
|
98
|
-
aria-invalid={!!error}
|
|
99
|
-
{...restProps}
|
|
100
|
-
></textarea>
|
|
101
|
-
|
|
102
|
-
{#if error}
|
|
103
|
-
<p class={typography.error}>{error}</p>
|
|
104
|
-
{/if}
|
|
105
|
-
</div>
|
|
71
|
+
|
|
72
|
+
<div class="flex flex-col gap-2 w-full">
|
|
73
|
+
{#if label}
|
|
74
|
+
<label for={id || name} class={`${typography.label} leading-tight`}>
|
|
75
|
+
{label}{#if required}<span class="text-red-500 font-medium text-sm ml-0.5">*</span>{/if}
|
|
76
|
+
</label>
|
|
77
|
+
{/if}
|
|
78
|
+
|
|
79
|
+
<textarea
|
|
80
|
+
{id}
|
|
81
|
+
{name}
|
|
82
|
+
{placeholder}
|
|
83
|
+
{rows}
|
|
84
|
+
{disabled}
|
|
85
|
+
{readonly}
|
|
86
|
+
{maxlength}
|
|
87
|
+
{minlength}
|
|
88
|
+
class="{typography.sm} w-full p-2.5 bg-gray-50 dark:bg-gray-800 leading-normal border rounded-lg resize-y transition-colors focus:outline-none focus:ring-4 focus:ring-blue-300 dark:focus:ring-blue-800 placeholder-gray-500 dark:placeholder-gray-400 {error ? 'border-red-500' : 'border-gray-300 dark:border-gray-600 hover:border-blue-500 focus:border-blue-500'} {disabled ? 'opacity-50 cursor-not-allowed' : ''} {className}"
|
|
89
|
+
bind:value
|
|
90
|
+
oninput={handleInput}
|
|
91
|
+
onchange={handleChange}
|
|
92
|
+
onblur={handleBlur}
|
|
93
|
+
onfocus={handleFocus}
|
|
94
|
+
{onkeydown}
|
|
95
|
+
{onkeyup}
|
|
96
|
+
{onkeypress}
|
|
97
|
+
aria-required={required}
|
|
98
|
+
aria-invalid={!!error}
|
|
99
|
+
{...restProps}
|
|
100
|
+
></textarea>
|
|
101
|
+
|
|
102
|
+
{#if error}
|
|
103
|
+
<p class={typography.error}>{error}</p>
|
|
104
|
+
{/if}
|
|
105
|
+
</div>
|
|
@@ -1,37 +1,37 @@
|
|
|
1
|
-
<script lang="ts">
|
|
2
|
-
import type { Snippet } from 'svelte';
|
|
3
|
-
|
|
4
|
-
interface Props {
|
|
5
|
-
color?: 'gray' | 'green' | 'red' | 'disabled';
|
|
6
|
-
defaultClass?: string;
|
|
7
|
-
show?: boolean;
|
|
8
|
-
class?: string;
|
|
9
|
-
children?: Snippet;
|
|
10
|
-
[key: string]: unknown;
|
|
11
|
-
}
|
|
12
|
-
|
|
13
|
-
let {
|
|
14
|
-
color = "gray",
|
|
15
|
-
defaultClass = "text-sm rtl:text-right font-medium block",
|
|
16
|
-
show = true,
|
|
17
|
-
class: className = "",
|
|
18
|
-
children,
|
|
19
|
-
...restProps
|
|
20
|
-
}: Props = $props();
|
|
21
|
-
|
|
22
|
-
const colorClasses = {
|
|
23
|
-
gray: "text-gray-900 dark:text-gray-300",
|
|
24
|
-
green: "text-green-700 dark:text-green-500",
|
|
25
|
-
red: "text-red-700 dark:text-red-500",
|
|
26
|
-
disabled: "text-gray-400 dark:text-gray-500"
|
|
27
|
-
};
|
|
28
|
-
|
|
29
|
-
let labelClass = $derived(`${defaultClass} ${colorClasses[color] || colorClasses.gray} ${className}`.trim());
|
|
30
|
-
</script>
|
|
31
|
-
|
|
32
|
-
{#if show}
|
|
33
|
-
<!-- svelte-ignore a11y_label_has_associated_control -->
|
|
34
|
-
<label {...restProps} class={labelClass}>{#if children}{@render children()}{/if}</label>
|
|
35
|
-
{:else}
|
|
36
|
-
{#if children}{@render children()}{/if}
|
|
37
|
-
{/if}
|
|
1
|
+
<script lang="ts">
|
|
2
|
+
import type { Snippet } from 'svelte';
|
|
3
|
+
|
|
4
|
+
interface Props {
|
|
5
|
+
color?: 'gray' | 'green' | 'red' | 'disabled';
|
|
6
|
+
defaultClass?: string;
|
|
7
|
+
show?: boolean;
|
|
8
|
+
class?: string;
|
|
9
|
+
children?: Snippet;
|
|
10
|
+
[key: string]: unknown;
|
|
11
|
+
}
|
|
12
|
+
|
|
13
|
+
let {
|
|
14
|
+
color = "gray",
|
|
15
|
+
defaultClass = "text-sm rtl:text-right font-medium block",
|
|
16
|
+
show = true,
|
|
17
|
+
class: className = "",
|
|
18
|
+
children,
|
|
19
|
+
...restProps
|
|
20
|
+
}: Props = $props();
|
|
21
|
+
|
|
22
|
+
const colorClasses = {
|
|
23
|
+
gray: "text-gray-900 dark:text-gray-300",
|
|
24
|
+
green: "text-green-700 dark:text-green-500",
|
|
25
|
+
red: "text-red-700 dark:text-red-500",
|
|
26
|
+
disabled: "text-gray-400 dark:text-gray-500"
|
|
27
|
+
};
|
|
28
|
+
|
|
29
|
+
let labelClass = $derived(`${defaultClass} ${colorClasses[color] || colorClasses.gray} ${className}`.trim());
|
|
30
|
+
</script>
|
|
31
|
+
|
|
32
|
+
{#if show}
|
|
33
|
+
<!-- svelte-ignore a11y_label_has_associated_control -->
|
|
34
|
+
<label {...restProps} class={labelClass}>{#if children}{@render children()}{/if}</label>
|
|
35
|
+
{:else}
|
|
36
|
+
{#if children}{@render children()}{/if}
|
|
37
|
+
{/if}
|
|
@@ -1,99 +1,99 @@
|
|
|
1
|
-
import { render, screen, fireEvent } from '@testing-library/svelte';
|
|
2
|
-
import userEvent from '@testing-library/user-event';
|
|
3
|
-
import { expect, describe, test, vi } from 'vitest';
|
|
4
|
-
import ModalTestWrapper from './ModalTestWrapper.svelte';
|
|
5
|
-
|
|
6
|
-
function setupTest(args) {
|
|
7
|
-
const user = userEvent.setup();
|
|
8
|
-
const { component, rerender } = render(ModalTestWrapper, { props: { ...args } });
|
|
9
|
-
return { user, component, rerender };
|
|
10
|
-
}
|
|
11
|
-
|
|
12
|
-
describe('Modal Component Tests', () => {
|
|
13
|
-
// Note: Modal renders both mobile sheet and desktop centered versions
|
|
14
|
-
// CSS hides one based on viewport, but in tests both are in DOM
|
|
15
|
-
// Use getAllByText and check that at least one exists
|
|
16
|
-
|
|
17
|
-
test('Renders Modal when show is true', () => {
|
|
18
|
-
setupTest({
|
|
19
|
-
show: true,
|
|
20
|
-
title: 'Test Title',
|
|
21
|
-
description: 'Test Description',
|
|
22
|
-
warningText: 'Test Warning'
|
|
23
|
-
});
|
|
24
|
-
// Modal renders both mobile and desktop versions, so use getAllByText
|
|
25
|
-
expect(screen.getAllByText((content, element) => content.includes('Test Title')).length).toBeGreaterThan(0);
|
|
26
|
-
expect(screen.getAllByText((content, element) => content.includes('Test Description')).length).toBeGreaterThan(0);
|
|
27
|
-
expect(screen.getAllByText((content, element) => content.includes('Test Warning')).length).toBeGreaterThan(0);
|
|
28
|
-
expect(screen.getAllByRole('button', { name: /Cancel/i }).length).toBeGreaterThan(0);
|
|
29
|
-
expect(screen.getAllByRole('button', { name: /Confirm/i }).length).toBeGreaterThan(0);
|
|
30
|
-
});
|
|
31
|
-
|
|
32
|
-
test('Does not render Modal when show is false', () => {
|
|
33
|
-
setupTest({
|
|
34
|
-
show: false,
|
|
35
|
-
title: 'Test Title',
|
|
36
|
-
description: 'Test Description',
|
|
37
|
-
warningText: 'Test Warning'
|
|
38
|
-
});
|
|
39
|
-
expect(screen.queryByText('Test Title')).not.toBeInTheDocument();
|
|
40
|
-
expect(screen.queryByText('Test Description')).not.toBeInTheDocument();
|
|
41
|
-
expect(screen.queryByText('Test Warning')).not.toBeInTheDocument();
|
|
42
|
-
});
|
|
43
|
-
|
|
44
|
-
test('Displays the correct title, description, and warning text', () => {
|
|
45
|
-
const title = 'Correct Title';
|
|
46
|
-
const description = 'Correct Description';
|
|
47
|
-
const warningText = 'Correct Warning';
|
|
48
|
-
setupTest({
|
|
49
|
-
show: true,
|
|
50
|
-
title,
|
|
51
|
-
description,
|
|
52
|
-
warningText,
|
|
53
|
-
});
|
|
54
|
-
// Modal renders both mobile and desktop versions
|
|
55
|
-
expect(screen.getAllByText((content) => content.includes(title)).length).toBeGreaterThan(0);
|
|
56
|
-
expect(screen.getAllByText((content) => content.includes(description)).length).toBeGreaterThan(0);
|
|
57
|
-
expect(screen.getAllByText((content) => content.includes(warningText)).length).toBeGreaterThan(0);
|
|
58
|
-
});
|
|
59
|
-
|
|
60
|
-
test('Dispatches cancel event on escape key press', async () => {
|
|
61
|
-
// Create a mock function to track cancel calls
|
|
62
|
-
const oncancel = vi.fn();
|
|
63
|
-
|
|
64
|
-
const { user } = setupTest({
|
|
65
|
-
show: true,
|
|
66
|
-
title: 'Test Title',
|
|
67
|
-
oncancel
|
|
68
|
-
});
|
|
69
|
-
|
|
70
|
-
// Verify modal is open
|
|
71
|
-
expect(screen.getAllByText((content) => content.includes('Test Title')).length).toBeGreaterThan(0);
|
|
72
|
-
|
|
73
|
-
// Press Escape key
|
|
74
|
-
await fireEvent.keyDown(window, { key: 'Escape' });
|
|
75
|
-
|
|
76
|
-
// Cancel callback should be called
|
|
77
|
-
expect(oncancel).toHaveBeenCalled();
|
|
78
|
-
});
|
|
79
|
-
|
|
80
|
-
test('Prevents propagation of click events within modal', async () => {
|
|
81
|
-
const { user } = setupTest({
|
|
82
|
-
show: true,
|
|
83
|
-
title: 'Test Title'
|
|
84
|
-
});
|
|
85
|
-
|
|
86
|
-
// Find the modal content (first instance)
|
|
87
|
-
const modalElements = screen.getAllByText((content) => content.includes('Test Title'));
|
|
88
|
-
expect(modalElements.length).toBeGreaterThan(0);
|
|
89
|
-
|
|
90
|
-
const modalContent = modalElements[0].closest('.md\\:hidden, .hidden');
|
|
91
|
-
if (modalContent) {
|
|
92
|
-
const clickEvent = new MouseEvent('click', { bubbles: true });
|
|
93
|
-
modalContent.dispatchEvent(clickEvent);
|
|
94
|
-
|
|
95
|
-
// Modal should still be visible
|
|
96
|
-
expect(screen.getAllByText((content) => content.includes('Test Title')).length).toBeGreaterThan(0);
|
|
97
|
-
}
|
|
98
|
-
});
|
|
99
|
-
});
|
|
1
|
+
import { render, screen, fireEvent } from '@testing-library/svelte';
|
|
2
|
+
import userEvent from '@testing-library/user-event';
|
|
3
|
+
import { expect, describe, test, vi } from 'vitest';
|
|
4
|
+
import ModalTestWrapper from './ModalTestWrapper.svelte';
|
|
5
|
+
|
|
6
|
+
function setupTest(args) {
|
|
7
|
+
const user = userEvent.setup();
|
|
8
|
+
const { component, rerender } = render(ModalTestWrapper, { props: { ...args } });
|
|
9
|
+
return { user, component, rerender };
|
|
10
|
+
}
|
|
11
|
+
|
|
12
|
+
describe('Modal Component Tests', () => {
|
|
13
|
+
// Note: Modal renders both mobile sheet and desktop centered versions
|
|
14
|
+
// CSS hides one based on viewport, but in tests both are in DOM
|
|
15
|
+
// Use getAllByText and check that at least one exists
|
|
16
|
+
|
|
17
|
+
test('Renders Modal when show is true', () => {
|
|
18
|
+
setupTest({
|
|
19
|
+
show: true,
|
|
20
|
+
title: 'Test Title',
|
|
21
|
+
description: 'Test Description',
|
|
22
|
+
warningText: 'Test Warning'
|
|
23
|
+
});
|
|
24
|
+
// Modal renders both mobile and desktop versions, so use getAllByText
|
|
25
|
+
expect(screen.getAllByText((content, element) => content.includes('Test Title')).length).toBeGreaterThan(0);
|
|
26
|
+
expect(screen.getAllByText((content, element) => content.includes('Test Description')).length).toBeGreaterThan(0);
|
|
27
|
+
expect(screen.getAllByText((content, element) => content.includes('Test Warning')).length).toBeGreaterThan(0);
|
|
28
|
+
expect(screen.getAllByRole('button', { name: /Cancel/i }).length).toBeGreaterThan(0);
|
|
29
|
+
expect(screen.getAllByRole('button', { name: /Confirm/i }).length).toBeGreaterThan(0);
|
|
30
|
+
});
|
|
31
|
+
|
|
32
|
+
test('Does not render Modal when show is false', () => {
|
|
33
|
+
setupTest({
|
|
34
|
+
show: false,
|
|
35
|
+
title: 'Test Title',
|
|
36
|
+
description: 'Test Description',
|
|
37
|
+
warningText: 'Test Warning'
|
|
38
|
+
});
|
|
39
|
+
expect(screen.queryByText('Test Title')).not.toBeInTheDocument();
|
|
40
|
+
expect(screen.queryByText('Test Description')).not.toBeInTheDocument();
|
|
41
|
+
expect(screen.queryByText('Test Warning')).not.toBeInTheDocument();
|
|
42
|
+
});
|
|
43
|
+
|
|
44
|
+
test('Displays the correct title, description, and warning text', () => {
|
|
45
|
+
const title = 'Correct Title';
|
|
46
|
+
const description = 'Correct Description';
|
|
47
|
+
const warningText = 'Correct Warning';
|
|
48
|
+
setupTest({
|
|
49
|
+
show: true,
|
|
50
|
+
title,
|
|
51
|
+
description,
|
|
52
|
+
warningText,
|
|
53
|
+
});
|
|
54
|
+
// Modal renders both mobile and desktop versions
|
|
55
|
+
expect(screen.getAllByText((content) => content.includes(title)).length).toBeGreaterThan(0);
|
|
56
|
+
expect(screen.getAllByText((content) => content.includes(description)).length).toBeGreaterThan(0);
|
|
57
|
+
expect(screen.getAllByText((content) => content.includes(warningText)).length).toBeGreaterThan(0);
|
|
58
|
+
});
|
|
59
|
+
|
|
60
|
+
test('Dispatches cancel event on escape key press', async () => {
|
|
61
|
+
// Create a mock function to track cancel calls
|
|
62
|
+
const oncancel = vi.fn();
|
|
63
|
+
|
|
64
|
+
const { user } = setupTest({
|
|
65
|
+
show: true,
|
|
66
|
+
title: 'Test Title',
|
|
67
|
+
oncancel
|
|
68
|
+
});
|
|
69
|
+
|
|
70
|
+
// Verify modal is open
|
|
71
|
+
expect(screen.getAllByText((content) => content.includes('Test Title')).length).toBeGreaterThan(0);
|
|
72
|
+
|
|
73
|
+
// Press Escape key
|
|
74
|
+
await fireEvent.keyDown(window, { key: 'Escape' });
|
|
75
|
+
|
|
76
|
+
// Cancel callback should be called
|
|
77
|
+
expect(oncancel).toHaveBeenCalled();
|
|
78
|
+
});
|
|
79
|
+
|
|
80
|
+
test('Prevents propagation of click events within modal', async () => {
|
|
81
|
+
const { user } = setupTest({
|
|
82
|
+
show: true,
|
|
83
|
+
title: 'Test Title'
|
|
84
|
+
});
|
|
85
|
+
|
|
86
|
+
// Find the modal content (first instance)
|
|
87
|
+
const modalElements = screen.getAllByText((content) => content.includes('Test Title'));
|
|
88
|
+
expect(modalElements.length).toBeGreaterThan(0);
|
|
89
|
+
|
|
90
|
+
const modalContent = modalElements[0].closest('.md\\:hidden, .hidden');
|
|
91
|
+
if (modalContent) {
|
|
92
|
+
const clickEvent = new MouseEvent('click', { bubbles: true });
|
|
93
|
+
modalContent.dispatchEvent(clickEvent);
|
|
94
|
+
|
|
95
|
+
// Modal should still be visible
|
|
96
|
+
expect(screen.getAllByText((content) => content.includes('Test Title')).length).toBeGreaterThan(0);
|
|
97
|
+
}
|
|
98
|
+
});
|
|
99
|
+
});
|