@getmicdrop/svelte-components 5.3.12 → 5.3.13
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 +145 -145
- package/dist/calendar/ShowCard/ShowCard.svelte +157 -157
- package/dist/calendar/ShowTimeCard/ShowTimeCard.svelte +61 -61
- package/dist/components/Layout/Grid.svelte +109 -109
- package/dist/components/Layout/Section.svelte +80 -80
- package/dist/components/Layout/Sidebar.svelte +108 -108
- package/dist/components/Layout/Stack.svelte +90 -90
- package/dist/constants/formOptions.js +26 -26
- package/dist/constants/validation.js +91 -91
- package/dist/constants/validation.spec.js +64 -64
- package/dist/datetime/__tests__/format.test.d.ts +2 -0
- package/dist/datetime/__tests__/format.test.d.ts.map +1 -0
- package/dist/datetime/__tests__/format.test.js +268 -0
- package/dist/datetime/__tests__/integration.test.d.ts +2 -0
- package/dist/datetime/__tests__/integration.test.d.ts.map +1 -0
- package/dist/datetime/__tests__/integration.test.js +243 -0
- package/dist/datetime/__tests__/parse.test.d.ts +2 -0
- package/dist/datetime/__tests__/parse.test.d.ts.map +1 -0
- package/dist/datetime/__tests__/parse.test.js +261 -0
- package/dist/datetime/__tests__/timezone.test.d.ts +2 -0
- package/dist/datetime/__tests__/timezone.test.d.ts.map +1 -0
- package/dist/datetime/__tests__/timezone.test.js +214 -0
- package/dist/datetime/constants.d.ts +133 -0
- package/dist/datetime/constants.d.ts.map +1 -0
- package/dist/datetime/constants.js +112 -0
- package/dist/datetime/format.d.ts +158 -0
- package/dist/datetime/format.d.ts.map +1 -0
- package/dist/datetime/format.js +315 -0
- package/dist/datetime/index.d.ts +42 -0
- package/dist/datetime/index.d.ts.map +1 -0
- package/dist/datetime/index.js +44 -0
- package/dist/datetime/parse.d.ts +149 -0
- package/dist/datetime/parse.d.ts.map +1 -0
- package/dist/datetime/parse.js +276 -0
- package/dist/datetime/timezone.d.ts +95 -0
- package/dist/datetime/timezone.d.ts.map +1 -0
- package/dist/datetime/timezone.js +241 -0
- package/dist/datetime/types.d.ts +105 -0
- package/dist/datetime/types.d.ts.map +1 -0
- package/dist/datetime/types.js +31 -0
- package/dist/index.d.ts +10 -0
- package/dist/index.js +232 -218
- package/dist/patterns/data/DataGrid.svelte +45 -45
- package/dist/patterns/data/DataList.svelte +24 -24
- package/dist/patterns/data/DataTable.svelte +40 -40
- package/dist/patterns/forms/FormActions.spec.js +88 -88
- 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.spec.js +203 -203
- package/dist/patterns/forms/FormValidationSummary.stories.svelte +97 -97
- package/dist/patterns/forms/FormValidationSummary.svelte +67 -67
- package/dist/patterns/layout/Grid.svelte +35 -35
- package/dist/patterns/layout/Sidebar.svelte +39 -39
- package/dist/patterns/layout/Stack.svelte +45 -45
- package/dist/patterns/navigation/BottomNav.spec.js +130 -130
- package/dist/patterns/navigation/BottomNav.stories.svelte +117 -117
- package/dist/patterns/navigation/BottomNav.svelte +54 -54
- package/dist/patterns/navigation/Header.spec.js +203 -203
- package/dist/patterns/navigation/Header.stories.svelte +77 -77
- package/dist/patterns/navigation/Header.svelte +240 -240
- package/dist/patterns/page/PageHeader.svelte +36 -36
- package/dist/patterns/page/PageLayout.svelte +40 -40
- package/dist/patterns/page/PageLoader.spec.js +54 -54
- package/dist/patterns/page/PageLoader.stories.svelte +137 -137
- package/dist/patterns/page/PageLoader.svelte +41 -41
- package/dist/patterns/page/SectionHeader.svelte +41 -41
- 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 +61 -61
- package/dist/primitives/Accordion/AccordionItem.svelte +95 -95
- package/dist/primitives/Alert/Alert.spec.js +170 -170
- package/dist/primitives/Alert/Alert.stories.svelte +88 -88
- package/dist/primitives/Alert/Alert.svelte +65 -65
- package/dist/primitives/Avatar/Avatar.stories.svelte +94 -94
- package/dist/primitives/Avatar/Avatar.svelte +66 -66
- package/dist/primitives/Badges/Badge.spec.js +103 -103
- package/dist/primitives/Badges/Badge.stories.svelte +86 -86
- package/dist/primitives/Badges/Badge.svelte +142 -142
- package/dist/primitives/BottomSheet/BottomSheet.spec.js +127 -127
- package/dist/primitives/BottomSheet/BottomSheet.stories.svelte +83 -83
- package/dist/primitives/BottomSheet/BottomSheet.svelte +100 -100
- package/dist/primitives/Breadcrumb/Breadcrumb.spec.js +120 -120
- package/dist/primitives/Breadcrumb/Breadcrumb.stories.svelte +23 -23
- package/dist/primitives/Breadcrumb/Breadcrumb.svelte +89 -89
- package/dist/primitives/Button/Button.spec.js +211 -211
- package/dist/primitives/Button/Button.stories.svelte +76 -76
- package/dist/primitives/Button/Button.svelte +301 -301
- package/dist/primitives/Button/ButtonSaveDemo.spec.js +48 -48
- 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 +357 -357
- package/dist/primitives/DarkModeToggle.stories.svelte +57 -57
- package/dist/primitives/DarkModeToggle.svelte +136 -136
- package/dist/primitives/Drawer/Drawer.stories.svelte +100 -100
- package/dist/primitives/Drawer/Drawer.svelte +214 -214
- package/dist/primitives/Dropdown/Dropdown.stories.svelte +137 -137
- package/dist/primitives/Dropdown/Dropdown.svelte +148 -148
- package/dist/primitives/Dropdown/DropdownItem.svelte +80 -80
- package/dist/primitives/Icons/ArrowLeft.svelte +20 -20
- package/dist/primitives/Icons/ArrowRight.svelte +20 -20
- package/dist/primitives/Icons/Availability.svelte +26 -26
- package/dist/primitives/Icons/Back.svelte +26 -26
- package/dist/primitives/Icons/CheckCircle.svelte +18 -18
- package/dist/primitives/Icons/CheckCircleOutline.svelte +27 -27
- package/dist/primitives/Icons/ChevronLeft.svelte +16 -16
- package/dist/primitives/Icons/ChevronRight.svelte +16 -16
- package/dist/primitives/Icons/Copy.svelte +27 -27
- package/dist/primitives/Icons/Cross.svelte +17 -17
- package/dist/primitives/Icons/DownArrow.svelte +20 -20
- package/dist/primitives/Icons/ErrorCircle.svelte +18 -18
- package/dist/primitives/Icons/FacebookIcon.svelte +13 -13
- package/dist/primitives/Icons/Home.svelte +27 -27
- package/dist/primitives/Icons/Icon.spec.js +175 -175
- package/dist/primitives/Icons/Icon.stories.svelte +100 -100
- package/dist/primitives/Icons/Icon.svelte +63 -63
- package/dist/primitives/Icons/IconGallery.stories.svelte +235 -235
- package/dist/primitives/Icons/ImageOutline.svelte +19 -19
- package/dist/primitives/Icons/Info.svelte +19 -19
- package/dist/primitives/Icons/InstagramIcon.svelte +19 -19
- package/dist/primitives/Icons/LogoInstagram.svelte +15 -15
- package/dist/primitives/Icons/Message.svelte +27 -27
- package/dist/primitives/Icons/MoonIcon.svelte +16 -16
- package/dist/primitives/Icons/More.svelte +33 -33
- package/dist/primitives/Icons/MoreHori.spec.js +67 -67
- package/dist/primitives/Icons/MoreHori.svelte +34 -34
- package/dist/primitives/Icons/Notification.svelte +26 -26
- package/dist/primitives/Icons/Payment.svelte +26 -26
- package/dist/primitives/Icons/Profile.svelte +33 -33
- package/dist/primitives/Icons/Reload.svelte +41 -41
- package/dist/primitives/Icons/Shows.svelte +33 -33
- package/dist/primitives/Icons/Signout.svelte +33 -33
- package/dist/primitives/Icons/SunIcon.svelte +19 -19
- package/dist/primitives/Icons/TiktokIcon.svelte +13 -13
- package/dist/primitives/Icons/TrashBinOutline.svelte +19 -19
- package/dist/primitives/Icons/TwitterIcon.svelte +13 -13
- package/dist/primitives/Icons/WarningIcon.spec.js +30 -30
- package/dist/primitives/Icons/WarningIcon.svelte +24 -24
- package/dist/primitives/Input/Input.spec.js +573 -573
- package/dist/primitives/Input/Input.stories.svelte +139 -139
- package/dist/primitives/Input/Input.svelte +444 -444
- package/dist/primitives/Input/Select.spec.js +218 -218
- package/dist/primitives/Input/Select.stories.svelte +112 -112
- package/dist/primitives/Input/Select.svelte +232 -232
- package/dist/primitives/Input/Textarea.stories.svelte +137 -137
- package/dist/primitives/Input/Textarea.svelte +79 -79
- package/dist/primitives/Label/Label.svelte +37 -37
- package/dist/primitives/Modal/Modal.spec.js +95 -95
- package/dist/primitives/Modal/Modal.stories.svelte +86 -86
- package/dist/primitives/Modal/Modal.svelte +158 -158
- 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 +52 -52
- package/dist/primitives/Spinner/Spinner.spec.js +75 -75
- package/dist/primitives/Spinner/Spinner.stories.svelte +29 -29
- package/dist/primitives/Spinner/Spinner.svelte +57 -57
- package/dist/primitives/Tabs/TabItem.svelte +51 -51
- package/dist/primitives/Tabs/Tabs.stories.svelte +112 -112
- package/dist/primitives/Tabs/Tabs.svelte +128 -128
- package/dist/primitives/Toggle.spec.js +127 -127
- package/dist/primitives/Toggle.stories.svelte +92 -92
- package/dist/primitives/Toggle.svelte +71 -71
- package/dist/primitives/Typography/Typography.svelte +53 -53
- package/dist/primitives/ValidationError.spec.js +103 -103
- package/dist/primitives/ValidationError.stories.svelte +111 -111
- package/dist/primitives/ValidationError.svelte +29 -29
- package/dist/recipes/CropImage/CropImage.spec.js +216 -216
- package/dist/recipes/CropImage/CropImage.stories.svelte +104 -104
- package/dist/recipes/CropImage/CropImage.svelte +238 -238
- package/dist/recipes/ImageUploader/ImageUploader.stories.svelte +125 -125
- package/dist/recipes/ImageUploader/ImageUploader.svelte +980 -980
- package/dist/recipes/Toaster/Toaster.stories.svelte +62 -62
- package/dist/recipes/feedback/EmptyState/EmptyState.svelte +47 -47
- package/dist/recipes/feedback/ErrorDisplay.spec.js +69 -69
- package/dist/recipes/feedback/ErrorDisplay.stories.svelte +112 -112
- package/dist/recipes/feedback/ErrorDisplay.svelte +38 -38
- package/dist/recipes/feedback/StatusIndicator/StatusIndicator.spec.js +129 -129
- package/dist/recipes/feedback/StatusIndicator/StatusIndicator.svelte +167 -167
- 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/inputs/MultiSelect.spec.js +257 -257
- package/dist/recipes/inputs/MultiSelect.stories.svelte +133 -133
- package/dist/recipes/inputs/MultiSelect.svelte +244 -244
- package/dist/recipes/inputs/OTPInput.spec.js +238 -238
- package/dist/recipes/inputs/OTPInput.stories.svelte +162 -162
- package/dist/recipes/inputs/OTPInput.svelte +102 -102
- package/dist/recipes/inputs/PasswordInput.svelte +100 -100
- package/dist/recipes/inputs/PasswordStrengthIndicator/PasswordStrengthIndicator.spec.js +173 -173
- package/dist/recipes/inputs/PasswordStrengthIndicator/PasswordStrengthIndicator.svelte +108 -108
- package/dist/recipes/inputs/PlaceAutocomplete/PlaceAutocomplete.spec.js +300 -300
- package/dist/recipes/inputs/PlaceAutocomplete/PlaceAutocomplete.stories.svelte +165 -165
- package/dist/recipes/inputs/PlaceAutocomplete/PlaceAutocomplete.svelte +337 -337
- package/dist/recipes/inputs/Search.svelte +85 -85
- package/dist/recipes/inputs/SelectDropdown.svelte +161 -161
- package/dist/recipes/modals/AlertModal.svelte +130 -130
- package/dist/recipes/modals/ConfirmationModal.spec.js +191 -191
- 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 +342 -342
- package/dist/stores/auth.js +93 -6
- package/dist/stores/auth.spec.js +310 -2
- 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 +388 -388
- package/dist/stories/PrimitivesGallery.stories.svelte +19 -19
- package/dist/stories/PrimitivesGallery.svelte +752 -752
- package/dist/stories/RecipesGallery.stories.svelte +19 -19
- package/dist/stories/RecipesGallery.svelte +441 -441
- package/dist/stories/button-audit-manifest.json +11186 -11186
- package/dist/tailwind/preset.cjs +82 -82
- package/dist/telemetry.js +357 -357
- package/dist/tokens/tokens.css +87 -87
- package/dist/utils/apiConfig.js +49 -49
- package/dist/utils/utils.js +9 -1
- package/package.json +233 -191
|
@@ -1,137 +1,137 @@
|
|
|
1
|
-
<script module>
|
|
2
|
-
import { defineMeta } from '@storybook/addon-svelte-csf';
|
|
3
|
-
import Dropdown from './Dropdown.svelte';
|
|
4
|
-
import DropdownItem from './DropdownItem.svelte';
|
|
5
|
-
import Button from '../Button/Button.svelte';
|
|
6
|
-
|
|
7
|
-
const { Story } = defineMeta({
|
|
8
|
-
title: 'Primitives/Dropdown',
|
|
9
|
-
component: Dropdown,
|
|
10
|
-
tags: ['autodocs'],
|
|
11
|
-
parameters: {
|
|
12
|
-
docs: {
|
|
13
|
-
description: {
|
|
14
|
-
component: 'Dropdown menu component for actions and navigation.',
|
|
15
|
-
},
|
|
16
|
-
},
|
|
17
|
-
},
|
|
18
|
-
});
|
|
19
|
-
|
|
20
|
-
let open1 = $state(false);
|
|
21
|
-
let open2 = $state(false);
|
|
22
|
-
let open3 = $state(false);
|
|
23
|
-
</script>
|
|
24
|
-
|
|
25
|
-
<Story name="Default">
|
|
26
|
-
{#snippet template()}
|
|
27
|
-
<div class="relative inline-block">
|
|
28
|
-
<Button onclick={() => open1 = !open1}>
|
|
29
|
-
Options ▼
|
|
30
|
-
</Button>
|
|
31
|
-
{#if open1}
|
|
32
|
-
<Dropdown class="absolute mt-1 z-10">
|
|
33
|
-
<DropdownItem onclick={() => { console.log('Edit'); open1 = false; }}>Edit</DropdownItem>
|
|
34
|
-
<DropdownItem onclick={() => { console.log('Duplicate'); open1 = false; }}>Duplicate</DropdownItem>
|
|
35
|
-
<DropdownItem onclick={() => { console.log('Archive'); open1 = false; }}>Archive</DropdownItem>
|
|
36
|
-
</Dropdown>
|
|
37
|
-
{/if}
|
|
38
|
-
</div>
|
|
39
|
-
{/snippet}
|
|
40
|
-
</Story>
|
|
41
|
-
|
|
42
|
-
<Story name="With Dividers">
|
|
43
|
-
{#snippet template()}
|
|
44
|
-
<div class="relative inline-block">
|
|
45
|
-
<Button onclick={() => open2 = !open2}>
|
|
46
|
-
Actions ▼
|
|
47
|
-
</Button>
|
|
48
|
-
{#if open2}
|
|
49
|
-
<Dropdown class="absolute mt-1 z-10">
|
|
50
|
-
<DropdownItem onclick={() => open2 = false}>View Details</DropdownItem>
|
|
51
|
-
<DropdownItem onclick={() => open2 = false}>Edit</DropdownItem>
|
|
52
|
-
<div class="border-t my-1"></div>
|
|
53
|
-
<DropdownItem onclick={() => open2 = false}>Share</DropdownItem>
|
|
54
|
-
<DropdownItem onclick={() => open2 = false}>Export</DropdownItem>
|
|
55
|
-
<div class="border-t my-1"></div>
|
|
56
|
-
<DropdownItem onclick={() => open2 = false} class="text-red-600">Delete</DropdownItem>
|
|
57
|
-
</Dropdown>
|
|
58
|
-
{/if}
|
|
59
|
-
</div>
|
|
60
|
-
{/snippet}
|
|
61
|
-
</Story>
|
|
62
|
-
|
|
63
|
-
<Story name="Right Aligned">
|
|
64
|
-
{#snippet template()}
|
|
65
|
-
<div class="flex justify-end">
|
|
66
|
-
<div class="relative inline-block">
|
|
67
|
-
<Button onclick={() => open3 = !open3}>
|
|
68
|
-
Menu ▼
|
|
69
|
-
</Button>
|
|
70
|
-
{#if open3}
|
|
71
|
-
<Dropdown class="absolute mt-1 right-0 z-10">
|
|
72
|
-
<DropdownItem onclick={() => open3 = false}>Profile</DropdownItem>
|
|
73
|
-
<DropdownItem onclick={() => open3 = false}>Settings</DropdownItem>
|
|
74
|
-
<DropdownItem onclick={() => open3 = false}>Help</DropdownItem>
|
|
75
|
-
<div class="border-t my-1"></div>
|
|
76
|
-
<DropdownItem onclick={() => open3 = false}>Sign Out</DropdownItem>
|
|
77
|
-
</Dropdown>
|
|
78
|
-
{/if}
|
|
79
|
-
</div>
|
|
80
|
-
</div>
|
|
81
|
-
{/snippet}
|
|
82
|
-
</Story>
|
|
83
|
-
|
|
84
|
-
<Story name="With Icons">
|
|
85
|
-
{#snippet template()}
|
|
86
|
-
<div class="relative inline-block">
|
|
87
|
-
<Button onclick={() => open1 = !open1}>
|
|
88
|
-
User Menu ▼
|
|
89
|
-
</Button>
|
|
90
|
-
{#if open1}
|
|
91
|
-
<Dropdown class="absolute mt-1 z-10 min-w-[180px]">
|
|
92
|
-
<DropdownItem onclick={() => open1 = false}>
|
|
93
|
-
<span class="mr-2">👤</span> Profile
|
|
94
|
-
</DropdownItem>
|
|
95
|
-
<DropdownItem onclick={() => open1 = false}>
|
|
96
|
-
<span class="mr-2">⚙️</span> Settings
|
|
97
|
-
</DropdownItem>
|
|
98
|
-
<DropdownItem onclick={() => open1 = false}>
|
|
99
|
-
<span class="mr-2">📊</span> Analytics
|
|
100
|
-
</DropdownItem>
|
|
101
|
-
<div class="border-t my-1"></div>
|
|
102
|
-
<DropdownItem onclick={() => open1 = false}>
|
|
103
|
-
<span class="mr-2">🚪</span> Sign Out
|
|
104
|
-
</DropdownItem>
|
|
105
|
-
</Dropdown>
|
|
106
|
-
{/if}
|
|
107
|
-
</div>
|
|
108
|
-
{/snippet}
|
|
109
|
-
</Story>
|
|
110
|
-
|
|
111
|
-
<Story name="In Card Context">
|
|
112
|
-
{#snippet template()}
|
|
113
|
-
<div class="border rounded-lg p-4 max-w-sm">
|
|
114
|
-
<div class="flex justify-between items-center mb-4">
|
|
115
|
-
<h3 class="font-medium">Card Title</h3>
|
|
116
|
-
<div class="relative">
|
|
117
|
-
<button
|
|
118
|
-
class="p-1 hover:bg-gray-100 rounded"
|
|
119
|
-
onclick={() => open1 = !open1}
|
|
120
|
-
>
|
|
121
|
-
⋮
|
|
122
|
-
</button>
|
|
123
|
-
{#if open1}
|
|
124
|
-
<Dropdown class="absolute mt-1 right-0 z-10">
|
|
125
|
-
<DropdownItem onclick={() => open1 = false}>Edit</DropdownItem>
|
|
126
|
-
<DropdownItem onclick={() => open1 = false}>Share</DropdownItem>
|
|
127
|
-
<DropdownItem onclick={() => open1 = false} class="text-red-600">Delete</DropdownItem>
|
|
128
|
-
</Dropdown>
|
|
129
|
-
{/if}
|
|
130
|
-
</div>
|
|
131
|
-
</div>
|
|
132
|
-
<p class="text-gray-600 text-sm">
|
|
133
|
-
This is a card with a dropdown menu in the header for common actions.
|
|
134
|
-
</p>
|
|
135
|
-
</div>
|
|
136
|
-
{/snippet}
|
|
137
|
-
</Story>
|
|
1
|
+
<script module>
|
|
2
|
+
import { defineMeta } from '@storybook/addon-svelte-csf';
|
|
3
|
+
import Dropdown from './Dropdown.svelte';
|
|
4
|
+
import DropdownItem from './DropdownItem.svelte';
|
|
5
|
+
import Button from '../Button/Button.svelte';
|
|
6
|
+
|
|
7
|
+
const { Story } = defineMeta({
|
|
8
|
+
title: 'Primitives/Dropdown',
|
|
9
|
+
component: Dropdown,
|
|
10
|
+
tags: ['autodocs'],
|
|
11
|
+
parameters: {
|
|
12
|
+
docs: {
|
|
13
|
+
description: {
|
|
14
|
+
component: 'Dropdown menu component for actions and navigation.',
|
|
15
|
+
},
|
|
16
|
+
},
|
|
17
|
+
},
|
|
18
|
+
});
|
|
19
|
+
|
|
20
|
+
let open1 = $state(false);
|
|
21
|
+
let open2 = $state(false);
|
|
22
|
+
let open3 = $state(false);
|
|
23
|
+
</script>
|
|
24
|
+
|
|
25
|
+
<Story name="Default">
|
|
26
|
+
{#snippet template()}
|
|
27
|
+
<div class="relative inline-block">
|
|
28
|
+
<Button onclick={() => open1 = !open1}>
|
|
29
|
+
Options ▼
|
|
30
|
+
</Button>
|
|
31
|
+
{#if open1}
|
|
32
|
+
<Dropdown class="absolute mt-1 z-10">
|
|
33
|
+
<DropdownItem onclick={() => { console.log('Edit'); open1 = false; }}>Edit</DropdownItem>
|
|
34
|
+
<DropdownItem onclick={() => { console.log('Duplicate'); open1 = false; }}>Duplicate</DropdownItem>
|
|
35
|
+
<DropdownItem onclick={() => { console.log('Archive'); open1 = false; }}>Archive</DropdownItem>
|
|
36
|
+
</Dropdown>
|
|
37
|
+
{/if}
|
|
38
|
+
</div>
|
|
39
|
+
{/snippet}
|
|
40
|
+
</Story>
|
|
41
|
+
|
|
42
|
+
<Story name="With Dividers">
|
|
43
|
+
{#snippet template()}
|
|
44
|
+
<div class="relative inline-block">
|
|
45
|
+
<Button onclick={() => open2 = !open2}>
|
|
46
|
+
Actions ▼
|
|
47
|
+
</Button>
|
|
48
|
+
{#if open2}
|
|
49
|
+
<Dropdown class="absolute mt-1 z-10">
|
|
50
|
+
<DropdownItem onclick={() => open2 = false}>View Details</DropdownItem>
|
|
51
|
+
<DropdownItem onclick={() => open2 = false}>Edit</DropdownItem>
|
|
52
|
+
<div class="border-t my-1"></div>
|
|
53
|
+
<DropdownItem onclick={() => open2 = false}>Share</DropdownItem>
|
|
54
|
+
<DropdownItem onclick={() => open2 = false}>Export</DropdownItem>
|
|
55
|
+
<div class="border-t my-1"></div>
|
|
56
|
+
<DropdownItem onclick={() => open2 = false} class="text-red-600">Delete</DropdownItem>
|
|
57
|
+
</Dropdown>
|
|
58
|
+
{/if}
|
|
59
|
+
</div>
|
|
60
|
+
{/snippet}
|
|
61
|
+
</Story>
|
|
62
|
+
|
|
63
|
+
<Story name="Right Aligned">
|
|
64
|
+
{#snippet template()}
|
|
65
|
+
<div class="flex justify-end">
|
|
66
|
+
<div class="relative inline-block">
|
|
67
|
+
<Button onclick={() => open3 = !open3}>
|
|
68
|
+
Menu ▼
|
|
69
|
+
</Button>
|
|
70
|
+
{#if open3}
|
|
71
|
+
<Dropdown class="absolute mt-1 right-0 z-10">
|
|
72
|
+
<DropdownItem onclick={() => open3 = false}>Profile</DropdownItem>
|
|
73
|
+
<DropdownItem onclick={() => open3 = false}>Settings</DropdownItem>
|
|
74
|
+
<DropdownItem onclick={() => open3 = false}>Help</DropdownItem>
|
|
75
|
+
<div class="border-t my-1"></div>
|
|
76
|
+
<DropdownItem onclick={() => open3 = false}>Sign Out</DropdownItem>
|
|
77
|
+
</Dropdown>
|
|
78
|
+
{/if}
|
|
79
|
+
</div>
|
|
80
|
+
</div>
|
|
81
|
+
{/snippet}
|
|
82
|
+
</Story>
|
|
83
|
+
|
|
84
|
+
<Story name="With Icons">
|
|
85
|
+
{#snippet template()}
|
|
86
|
+
<div class="relative inline-block">
|
|
87
|
+
<Button onclick={() => open1 = !open1}>
|
|
88
|
+
User Menu ▼
|
|
89
|
+
</Button>
|
|
90
|
+
{#if open1}
|
|
91
|
+
<Dropdown class="absolute mt-1 z-10 min-w-[180px]">
|
|
92
|
+
<DropdownItem onclick={() => open1 = false}>
|
|
93
|
+
<span class="mr-2">👤</span> Profile
|
|
94
|
+
</DropdownItem>
|
|
95
|
+
<DropdownItem onclick={() => open1 = false}>
|
|
96
|
+
<span class="mr-2">⚙️</span> Settings
|
|
97
|
+
</DropdownItem>
|
|
98
|
+
<DropdownItem onclick={() => open1 = false}>
|
|
99
|
+
<span class="mr-2">📊</span> Analytics
|
|
100
|
+
</DropdownItem>
|
|
101
|
+
<div class="border-t my-1"></div>
|
|
102
|
+
<DropdownItem onclick={() => open1 = false}>
|
|
103
|
+
<span class="mr-2">🚪</span> Sign Out
|
|
104
|
+
</DropdownItem>
|
|
105
|
+
</Dropdown>
|
|
106
|
+
{/if}
|
|
107
|
+
</div>
|
|
108
|
+
{/snippet}
|
|
109
|
+
</Story>
|
|
110
|
+
|
|
111
|
+
<Story name="In Card Context">
|
|
112
|
+
{#snippet template()}
|
|
113
|
+
<div class="border rounded-lg p-4 max-w-sm">
|
|
114
|
+
<div class="flex justify-between items-center mb-4">
|
|
115
|
+
<h3 class="font-medium">Card Title</h3>
|
|
116
|
+
<div class="relative">
|
|
117
|
+
<button
|
|
118
|
+
class="p-1 hover:bg-gray-100 rounded"
|
|
119
|
+
onclick={() => open1 = !open1}
|
|
120
|
+
>
|
|
121
|
+
⋮
|
|
122
|
+
</button>
|
|
123
|
+
{#if open1}
|
|
124
|
+
<Dropdown class="absolute mt-1 right-0 z-10">
|
|
125
|
+
<DropdownItem onclick={() => open1 = false}>Edit</DropdownItem>
|
|
126
|
+
<DropdownItem onclick={() => open1 = false}>Share</DropdownItem>
|
|
127
|
+
<DropdownItem onclick={() => open1 = false} class="text-red-600">Delete</DropdownItem>
|
|
128
|
+
</Dropdown>
|
|
129
|
+
{/if}
|
|
130
|
+
</div>
|
|
131
|
+
</div>
|
|
132
|
+
<p class="text-gray-600 text-sm">
|
|
133
|
+
This is a card with a dropdown menu in the header for common actions.
|
|
134
|
+
</p>
|
|
135
|
+
</div>
|
|
136
|
+
{/snippet}
|
|
137
|
+
</Story>
|
|
@@ -1,148 +1,148 @@
|
|
|
1
|
-
<script>
|
|
2
|
-
import { onMount, onDestroy, tick, setContext } from "svelte";
|
|
3
|
-
|
|
4
|
-
let {
|
|
5
|
-
open = $bindable(false),
|
|
6
|
-
activeUrl = "",
|
|
7
|
-
placement = "bottom",
|
|
8
|
-
class: className = "",
|
|
9
|
-
ariaLabel = "Menu",
|
|
10
|
-
onclose,
|
|
11
|
-
children,
|
|
12
|
-
...restProps
|
|
13
|
-
} = $props();
|
|
14
|
-
|
|
15
|
-
let dropdownRef = $state(null);
|
|
16
|
-
let containerRef = $state(null);
|
|
17
|
-
let focusedIndex = $state(-1);
|
|
18
|
-
let menuItems = $state([]);
|
|
19
|
-
|
|
20
|
-
async function updateMenuItems() {
|
|
21
|
-
await tick();
|
|
22
|
-
if (dropdownRef) {
|
|
23
|
-
menuItems = Array.from(dropdownRef.querySelectorAll('[role="menuitem"]:not([disabled])'));
|
|
24
|
-
}
|
|
25
|
-
}
|
|
26
|
-
|
|
27
|
-
function focusItem(index) {
|
|
28
|
-
if (menuItems.length === 0) return;
|
|
29
|
-
if (index < 0) index = menuItems.length - 1;
|
|
30
|
-
if (index >= menuItems.length) index = 0;
|
|
31
|
-
focusedIndex = index;
|
|
32
|
-
menuItems[focusedIndex]?.focus();
|
|
33
|
-
}
|
|
34
|
-
|
|
35
|
-
function getTriggerElement() {
|
|
36
|
-
const parent = containerRef?.parentElement;
|
|
37
|
-
if (parent) {
|
|
38
|
-
const siblings = Array.from(parent.children);
|
|
39
|
-
const containerIndex = siblings.indexOf(containerRef);
|
|
40
|
-
if (containerIndex > 0) {
|
|
41
|
-
return siblings[containerIndex - 1];
|
|
42
|
-
}
|
|
43
|
-
}
|
|
44
|
-
return null;
|
|
45
|
-
}
|
|
46
|
-
|
|
47
|
-
function handleClickOutside(event) {
|
|
48
|
-
if (!open) return;
|
|
49
|
-
|
|
50
|
-
if (dropdownRef && dropdownRef.contains(event.target)) {
|
|
51
|
-
return;
|
|
52
|
-
}
|
|
53
|
-
|
|
54
|
-
const trigger = getTriggerElement();
|
|
55
|
-
if (trigger && trigger.contains(event.target)) {
|
|
56
|
-
return;
|
|
57
|
-
}
|
|
58
|
-
|
|
59
|
-
open = false;
|
|
60
|
-
onclose?.();
|
|
61
|
-
}
|
|
62
|
-
|
|
63
|
-
function handleKeydown(event) {
|
|
64
|
-
if (!open) return;
|
|
65
|
-
|
|
66
|
-
const trigger = getTriggerElement();
|
|
67
|
-
if (!dropdownRef?.contains(event.target) && !trigger?.contains(event.target)) {
|
|
68
|
-
return;
|
|
69
|
-
}
|
|
70
|
-
|
|
71
|
-
switch (event.key) {
|
|
72
|
-
case "Escape":
|
|
73
|
-
open = false;
|
|
74
|
-
trigger?.focus();
|
|
75
|
-
event.preventDefault();
|
|
76
|
-
onclose?.();
|
|
77
|
-
break;
|
|
78
|
-
case "ArrowDown":
|
|
79
|
-
focusItem(focusedIndex + 1);
|
|
80
|
-
event.preventDefault();
|
|
81
|
-
break;
|
|
82
|
-
case "ArrowUp":
|
|
83
|
-
focusItem(focusedIndex - 1);
|
|
84
|
-
event.preventDefault();
|
|
85
|
-
break;
|
|
86
|
-
case "Home":
|
|
87
|
-
focusItem(0);
|
|
88
|
-
event.preventDefault();
|
|
89
|
-
break;
|
|
90
|
-
case "End":
|
|
91
|
-
focusItem(menuItems.length - 1);
|
|
92
|
-
event.preventDefault();
|
|
93
|
-
break;
|
|
94
|
-
case "Tab":
|
|
95
|
-
open = false;
|
|
96
|
-
onclose?.();
|
|
97
|
-
break;
|
|
98
|
-
}
|
|
99
|
-
}
|
|
100
|
-
|
|
101
|
-
$effect(() => {
|
|
102
|
-
if (open) {
|
|
103
|
-
focusedIndex = -1;
|
|
104
|
-
updateMenuItems();
|
|
105
|
-
}
|
|
106
|
-
});
|
|
107
|
-
|
|
108
|
-
onMount(() => {
|
|
109
|
-
document.addEventListener("mousedown", handleClickOutside, true);
|
|
110
|
-
document.addEventListener("keydown", handleKeydown);
|
|
111
|
-
});
|
|
112
|
-
|
|
113
|
-
onDestroy(() => {
|
|
114
|
-
if (typeof document !== "undefined") {
|
|
115
|
-
document.removeEventListener("mousedown", handleClickOutside, true);
|
|
116
|
-
document.removeEventListener("keydown", handleKeydown);
|
|
117
|
-
}
|
|
118
|
-
});
|
|
119
|
-
|
|
120
|
-
setContext("dropdown", {
|
|
121
|
-
activeUrl,
|
|
122
|
-
close: () => {
|
|
123
|
-
open = false;
|
|
124
|
-
onclose?.();
|
|
125
|
-
}
|
|
126
|
-
});
|
|
127
|
-
|
|
128
|
-
const placementClasses = {
|
|
129
|
-
bottom: "top-full left-0 mt-1",
|
|
130
|
-
top: "bottom-full left-0 mb-1",
|
|
131
|
-
left: "right-full top-0 mr-1",
|
|
132
|
-
right: "left-full top-0 ml-1"
|
|
133
|
-
};
|
|
134
|
-
</script>
|
|
135
|
-
|
|
136
|
-
{#if open}
|
|
137
|
-
<div
|
|
138
|
-
bind:this={containerRef}
|
|
139
|
-
class="absolute z-10 bg-white divide-y divide-gray-100 rounded-lg shadow-lg w-44 dark:bg-gray-700 dark:divide-gray-600 {className}"
|
|
140
|
-
role="menu"
|
|
141
|
-
aria-label={ariaLabel}
|
|
142
|
-
{...restProps}
|
|
143
|
-
>
|
|
144
|
-
<ul bind:this={dropdownRef} class="py-2 text-sm text-gray-700 dark:text-gray-200">
|
|
145
|
-
{#if typeof children === 'function'}{@render children()}{:else if children}{children}{/if}
|
|
146
|
-
</ul>
|
|
147
|
-
</div>
|
|
148
|
-
{/if}
|
|
1
|
+
<script>
|
|
2
|
+
import { onMount, onDestroy, tick, setContext } from "svelte";
|
|
3
|
+
|
|
4
|
+
let {
|
|
5
|
+
open = $bindable(false),
|
|
6
|
+
activeUrl = "",
|
|
7
|
+
placement = "bottom",
|
|
8
|
+
class: className = "",
|
|
9
|
+
ariaLabel = "Menu",
|
|
10
|
+
onclose,
|
|
11
|
+
children,
|
|
12
|
+
...restProps
|
|
13
|
+
} = $props();
|
|
14
|
+
|
|
15
|
+
let dropdownRef = $state(null);
|
|
16
|
+
let containerRef = $state(null);
|
|
17
|
+
let focusedIndex = $state(-1);
|
|
18
|
+
let menuItems = $state([]);
|
|
19
|
+
|
|
20
|
+
async function updateMenuItems() {
|
|
21
|
+
await tick();
|
|
22
|
+
if (dropdownRef) {
|
|
23
|
+
menuItems = Array.from(dropdownRef.querySelectorAll('[role="menuitem"]:not([disabled])'));
|
|
24
|
+
}
|
|
25
|
+
}
|
|
26
|
+
|
|
27
|
+
function focusItem(index) {
|
|
28
|
+
if (menuItems.length === 0) return;
|
|
29
|
+
if (index < 0) index = menuItems.length - 1;
|
|
30
|
+
if (index >= menuItems.length) index = 0;
|
|
31
|
+
focusedIndex = index;
|
|
32
|
+
menuItems[focusedIndex]?.focus();
|
|
33
|
+
}
|
|
34
|
+
|
|
35
|
+
function getTriggerElement() {
|
|
36
|
+
const parent = containerRef?.parentElement;
|
|
37
|
+
if (parent) {
|
|
38
|
+
const siblings = Array.from(parent.children);
|
|
39
|
+
const containerIndex = siblings.indexOf(containerRef);
|
|
40
|
+
if (containerIndex > 0) {
|
|
41
|
+
return siblings[containerIndex - 1];
|
|
42
|
+
}
|
|
43
|
+
}
|
|
44
|
+
return null;
|
|
45
|
+
}
|
|
46
|
+
|
|
47
|
+
function handleClickOutside(event) {
|
|
48
|
+
if (!open) return;
|
|
49
|
+
|
|
50
|
+
if (dropdownRef && dropdownRef.contains(event.target)) {
|
|
51
|
+
return;
|
|
52
|
+
}
|
|
53
|
+
|
|
54
|
+
const trigger = getTriggerElement();
|
|
55
|
+
if (trigger && trigger.contains(event.target)) {
|
|
56
|
+
return;
|
|
57
|
+
}
|
|
58
|
+
|
|
59
|
+
open = false;
|
|
60
|
+
onclose?.();
|
|
61
|
+
}
|
|
62
|
+
|
|
63
|
+
function handleKeydown(event) {
|
|
64
|
+
if (!open) return;
|
|
65
|
+
|
|
66
|
+
const trigger = getTriggerElement();
|
|
67
|
+
if (!dropdownRef?.contains(event.target) && !trigger?.contains(event.target)) {
|
|
68
|
+
return;
|
|
69
|
+
}
|
|
70
|
+
|
|
71
|
+
switch (event.key) {
|
|
72
|
+
case "Escape":
|
|
73
|
+
open = false;
|
|
74
|
+
trigger?.focus();
|
|
75
|
+
event.preventDefault();
|
|
76
|
+
onclose?.();
|
|
77
|
+
break;
|
|
78
|
+
case "ArrowDown":
|
|
79
|
+
focusItem(focusedIndex + 1);
|
|
80
|
+
event.preventDefault();
|
|
81
|
+
break;
|
|
82
|
+
case "ArrowUp":
|
|
83
|
+
focusItem(focusedIndex - 1);
|
|
84
|
+
event.preventDefault();
|
|
85
|
+
break;
|
|
86
|
+
case "Home":
|
|
87
|
+
focusItem(0);
|
|
88
|
+
event.preventDefault();
|
|
89
|
+
break;
|
|
90
|
+
case "End":
|
|
91
|
+
focusItem(menuItems.length - 1);
|
|
92
|
+
event.preventDefault();
|
|
93
|
+
break;
|
|
94
|
+
case "Tab":
|
|
95
|
+
open = false;
|
|
96
|
+
onclose?.();
|
|
97
|
+
break;
|
|
98
|
+
}
|
|
99
|
+
}
|
|
100
|
+
|
|
101
|
+
$effect(() => {
|
|
102
|
+
if (open) {
|
|
103
|
+
focusedIndex = -1;
|
|
104
|
+
updateMenuItems();
|
|
105
|
+
}
|
|
106
|
+
});
|
|
107
|
+
|
|
108
|
+
onMount(() => {
|
|
109
|
+
document.addEventListener("mousedown", handleClickOutside, true);
|
|
110
|
+
document.addEventListener("keydown", handleKeydown);
|
|
111
|
+
});
|
|
112
|
+
|
|
113
|
+
onDestroy(() => {
|
|
114
|
+
if (typeof document !== "undefined") {
|
|
115
|
+
document.removeEventListener("mousedown", handleClickOutside, true);
|
|
116
|
+
document.removeEventListener("keydown", handleKeydown);
|
|
117
|
+
}
|
|
118
|
+
});
|
|
119
|
+
|
|
120
|
+
setContext("dropdown", {
|
|
121
|
+
activeUrl,
|
|
122
|
+
close: () => {
|
|
123
|
+
open = false;
|
|
124
|
+
onclose?.();
|
|
125
|
+
}
|
|
126
|
+
});
|
|
127
|
+
|
|
128
|
+
const placementClasses = {
|
|
129
|
+
bottom: "top-full left-0 mt-1",
|
|
130
|
+
top: "bottom-full left-0 mb-1",
|
|
131
|
+
left: "right-full top-0 mr-1",
|
|
132
|
+
right: "left-full top-0 ml-1"
|
|
133
|
+
};
|
|
134
|
+
</script>
|
|
135
|
+
|
|
136
|
+
{#if open}
|
|
137
|
+
<div
|
|
138
|
+
bind:this={containerRef}
|
|
139
|
+
class="absolute z-10 bg-white divide-y divide-gray-100 rounded-lg shadow-lg w-44 dark:bg-gray-700 dark:divide-gray-600 {className}"
|
|
140
|
+
role="menu"
|
|
141
|
+
aria-label={ariaLabel}
|
|
142
|
+
{...restProps}
|
|
143
|
+
>
|
|
144
|
+
<ul bind:this={dropdownRef} class="py-2 text-sm text-gray-700 dark:text-gray-200">
|
|
145
|
+
{#if typeof children === 'function'}{@render children()}{:else if children}{children}{/if}
|
|
146
|
+
</ul>
|
|
147
|
+
</div>
|
|
148
|
+
{/if}
|