@getmicdrop/svelte-components 5.5.0 → 5.5.4

This diff represents the content of publicly available package versions that have been released to one of the supported registries. The information contained in this diff is provided for informational purposes only and reflects changes between package versions as they appear in their respective public registries.
Files changed (214) hide show
  1. package/dist/calendar/AboutShow/AboutShow.svelte +172 -172
  2. package/dist/calendar/Calendar/MiniMonthCalendar.svelte +782 -782
  3. package/dist/calendar/FAQs/FAQs.svelte +75 -75
  4. package/dist/calendar/MonthSwitcher/MonthSwitcher.svelte +126 -126
  5. package/dist/calendar/OrderSummary/OrderSummary.svelte +367 -367
  6. package/dist/calendar/PublicCard/PublicCard.svelte +134 -134
  7. package/dist/calendar/ShowCard/ShowCard.svelte +157 -157
  8. package/dist/calendar/ShowTimeCard/ShowTimeCard.svelte +61 -61
  9. package/dist/components/Layout/Grid.svelte +4 -4
  10. package/dist/components/Layout/Section.svelte +80 -80
  11. package/dist/components/Layout/Sidebar.svelte +108 -108
  12. package/dist/components/Layout/Stack.svelte +6 -6
  13. package/dist/constants/validation.js +91 -91
  14. package/dist/constants/validation.spec.js +64 -64
  15. package/dist/index.d.ts +1 -1
  16. package/dist/index.js +226 -226
  17. package/dist/patterns/data/DataGrid.svelte +45 -45
  18. package/dist/patterns/data/DataList.svelte +24 -24
  19. package/dist/patterns/data/DataTable.svelte +36 -36
  20. package/dist/patterns/forms/FormActions.spec.js +88 -88
  21. package/dist/patterns/forms/FormActions.stories.svelte +97 -97
  22. package/dist/patterns/forms/FormActions.svelte +46 -46
  23. package/dist/patterns/forms/FormGrid.svelte +33 -33
  24. package/dist/patterns/forms/FormSection.svelte +32 -32
  25. package/dist/patterns/forms/FormValidationSummary.stories.svelte +83 -83
  26. package/dist/patterns/forms/FormValidationSummary.svelte +33 -33
  27. package/dist/patterns/layout/Sidebar.svelte +39 -39
  28. package/dist/patterns/navigation/BottomNav.stories.svelte +117 -117
  29. package/dist/patterns/navigation/BottomNav.svelte +20 -20
  30. package/dist/patterns/navigation/Header.stories.svelte +77 -77
  31. package/dist/patterns/navigation/Header.svelte +193 -193
  32. package/dist/patterns/page/PageHeader.svelte +18 -18
  33. package/dist/patterns/page/PageLayout.svelte +40 -40
  34. package/dist/patterns/page/PageLoader.spec.js +54 -54
  35. package/dist/patterns/page/PageLoader.stories.svelte +137 -137
  36. package/dist/patterns/page/PageLoader.svelte +24 -24
  37. package/dist/patterns/page/SectionHeader.svelte +29 -29
  38. package/dist/presets/badges.js +112 -112
  39. package/dist/presets/buttons.js +76 -76
  40. package/dist/presets/index.js +9 -9
  41. package/dist/primitives/Accordion/Accordion.stories.svelte +75 -75
  42. package/dist/primitives/Accordion/Accordion.svelte +42 -42
  43. package/dist/primitives/Accordion/AccordionItem.svelte +95 -95
  44. package/dist/primitives/Alert/Alert.spec.js +170 -170
  45. package/dist/primitives/Alert/Alert.stories.svelte +88 -88
  46. package/dist/primitives/Alert/Alert.svelte +27 -27
  47. package/dist/primitives/Avatar/Avatar.stories.svelte +94 -94
  48. package/dist/primitives/Avatar/Avatar.svelte +66 -66
  49. package/dist/primitives/Badges/Badge.spec.js +103 -103
  50. package/dist/primitives/Badges/Badge.stories.svelte +86 -86
  51. package/dist/primitives/Badges/Badge.svelte +79 -79
  52. package/dist/primitives/BottomSheet/BottomSheet.spec.js +127 -127
  53. package/dist/primitives/BottomSheet/BottomSheet.stories.svelte +83 -83
  54. package/dist/primitives/BottomSheet/BottomSheet.svelte +100 -100
  55. package/dist/primitives/Breadcrumb/Breadcrumb.spec.js +120 -120
  56. package/dist/primitives/Breadcrumb/Breadcrumb.stories.svelte +23 -23
  57. package/dist/primitives/Breadcrumb/Breadcrumb.svelte +89 -89
  58. package/dist/primitives/Button/Button.spec.js +211 -211
  59. package/dist/primitives/Button/Button.stories.svelte +76 -76
  60. package/dist/primitives/Button/Button.svelte +270 -270
  61. package/dist/primitives/Button/ButtonSaveDemo.spec.js +48 -48
  62. package/dist/primitives/Button/ButtonSaveDemo.svelte +25 -25
  63. package/dist/primitives/Button/ButtonVariantShowcase.svelte +129 -129
  64. package/dist/primitives/Card.spec.js +49 -49
  65. package/dist/primitives/Card.stories.svelte +22 -22
  66. package/dist/primitives/Card.svelte +28 -28
  67. package/dist/primitives/Checkbox/Checkbox.stories.svelte +84 -84
  68. package/dist/primitives/Checkbox/Checkbox.svelte +88 -88
  69. package/dist/primitives/DarkModeToggle.spec.js +357 -357
  70. package/dist/primitives/DarkModeToggle.stories.svelte +57 -57
  71. package/dist/primitives/DarkModeToggle.svelte +136 -136
  72. package/dist/primitives/Drawer/Drawer.stories.svelte +80 -80
  73. package/dist/primitives/Drawer/Drawer.svelte +120 -120
  74. package/dist/primitives/Dropdown/Dropdown.stories.svelte +137 -137
  75. package/dist/primitives/Dropdown/Dropdown.svelte +14 -14
  76. package/dist/primitives/Dropdown/DropdownItem.svelte +80 -80
  77. package/dist/primitives/Icons/ArrowLeft.svelte +8 -8
  78. package/dist/primitives/Icons/ArrowRight.svelte +8 -8
  79. package/dist/primitives/Icons/Availability.svelte +14 -14
  80. package/dist/primitives/Icons/Back.svelte +14 -14
  81. package/dist/primitives/Icons/CheckCircle.svelte +6 -6
  82. package/dist/primitives/Icons/CheckCircleOutline.svelte +15 -15
  83. package/dist/primitives/Icons/ChevronLeft.svelte +4 -4
  84. package/dist/primitives/Icons/ChevronRight.svelte +4 -4
  85. package/dist/primitives/Icons/Copy.svelte +15 -15
  86. package/dist/primitives/Icons/Cross.svelte +5 -5
  87. package/dist/primitives/Icons/DownArrow.svelte +8 -8
  88. package/dist/primitives/Icons/ErrorCircle.svelte +6 -6
  89. package/dist/primitives/Icons/FacebookIcon.svelte +2 -2
  90. package/dist/primitives/Icons/Home.svelte +15 -15
  91. package/dist/primitives/Icons/Icon.spec.js +169 -169
  92. package/dist/primitives/Icons/Icon.stories.svelte +100 -100
  93. package/dist/primitives/Icons/Icon.svelte +52 -52
  94. package/dist/primitives/Icons/IconGallery.stories.svelte +235 -235
  95. package/dist/primitives/Icons/Info.svelte +7 -7
  96. package/dist/primitives/Icons/InstagramIcon.svelte +4 -4
  97. package/dist/primitives/Icons/LogoInstagram.svelte +2 -2
  98. package/dist/primitives/Icons/Message.svelte +15 -15
  99. package/dist/primitives/Icons/MoonIcon.svelte +5 -5
  100. package/dist/primitives/Icons/More.svelte +21 -21
  101. package/dist/primitives/Icons/MoreHori.spec.js +61 -61
  102. package/dist/primitives/Icons/MoreHori.svelte +22 -22
  103. package/dist/primitives/Icons/Notification.svelte +14 -14
  104. package/dist/primitives/Icons/Payment.svelte +14 -14
  105. package/dist/primitives/Icons/Profile.svelte +21 -21
  106. package/dist/primitives/Icons/Reload.svelte +29 -29
  107. package/dist/primitives/Icons/Shows.svelte +21 -21
  108. package/dist/primitives/Icons/Signout.svelte +21 -21
  109. package/dist/primitives/Icons/SunIcon.svelte +8 -8
  110. package/dist/primitives/Icons/TiktokIcon.svelte +2 -2
  111. package/dist/primitives/Icons/TwitterIcon.svelte +2 -2
  112. package/dist/primitives/Icons/WarningIcon.spec.js +18 -18
  113. package/dist/primitives/Icons/WarningIcon.svelte +5 -5
  114. package/dist/primitives/Input/Input.spec.js +573 -573
  115. package/dist/primitives/Input/Input.stories.svelte +139 -139
  116. package/dist/primitives/Input/Input.svelte +392 -392
  117. package/dist/primitives/Input/Select.spec.js +218 -218
  118. package/dist/primitives/Input/Select.stories.svelte +112 -112
  119. package/dist/primitives/Input/Select.svelte +128 -128
  120. package/dist/primitives/Input/Textarea.stories.svelte +137 -137
  121. package/dist/primitives/Input/Textarea.svelte +35 -35
  122. package/dist/primitives/Label/Label.svelte +37 -37
  123. package/dist/primitives/Modal/Modal.spec.js +95 -95
  124. package/dist/primitives/Modal/Modal.stories.svelte +86 -86
  125. package/dist/primitives/Modal/Modal.svelte +158 -158
  126. package/dist/primitives/NumberInput/NumberInput.svelte +106 -106
  127. package/dist/primitives/Pagination/Pagination.stories.svelte +76 -76
  128. package/dist/primitives/Pagination/Pagination.svelte +261 -261
  129. package/dist/primitives/Radio/Radio.stories.svelte +80 -80
  130. package/dist/primitives/Radio/Radio.svelte +67 -67
  131. package/dist/primitives/Skeleton/CardPlaceholder.svelte +87 -87
  132. package/dist/primitives/Skeleton/ImagePlaceholder.svelte +59 -59
  133. package/dist/primitives/Skeleton/ListPlaceholder.svelte +76 -76
  134. package/dist/primitives/Skeleton/Skeleton.stories.svelte +151 -151
  135. package/dist/primitives/Skeleton/Skeleton.svelte +26 -26
  136. package/dist/primitives/Spinner/Spinner.spec.js +75 -75
  137. package/dist/primitives/Spinner/Spinner.stories.svelte +29 -29
  138. package/dist/primitives/Spinner/Spinner.svelte +20 -20
  139. package/dist/primitives/Tabs/TabItem.svelte +49 -49
  140. package/dist/primitives/Tabs/Tabs.stories.svelte +112 -112
  141. package/dist/primitives/Tabs/Tabs.svelte +123 -123
  142. package/dist/primitives/Toggle.spec.js +127 -127
  143. package/dist/primitives/Toggle.stories.svelte +92 -92
  144. package/dist/primitives/Toggle.svelte +71 -71
  145. package/dist/primitives/Typography/Typography.svelte +53 -53
  146. package/dist/primitives/ValidationError.spec.js +103 -103
  147. package/dist/primitives/ValidationError.stories.svelte +69 -69
  148. package/dist/primitives/ValidationError.svelte +29 -29
  149. package/dist/recipes/CropImage/CropImage.spec.js +216 -216
  150. package/dist/recipes/CropImage/CropImage.stories.svelte +104 -104
  151. package/dist/recipes/CropImage/CropImage.svelte +238 -238
  152. package/dist/recipes/ImageUploader/ImageUploader.stories.svelte +125 -125
  153. package/dist/recipes/ImageUploader/ImageUploader.svelte +804 -804
  154. package/dist/recipes/SuperLogin/SuperLogin.svelte +7 -6
  155. package/dist/recipes/SuperLogin/SuperLogin.svelte.d.ts.map +1 -1
  156. package/dist/recipes/Toaster/Toaster.stories.svelte +62 -62
  157. package/dist/recipes/feedback/EmptyState/EmptyState.svelte +1 -1
  158. package/dist/recipes/feedback/ErrorDisplay.spec.js +69 -69
  159. package/dist/recipes/feedback/ErrorDisplay.stories.svelte +101 -101
  160. package/dist/recipes/feedback/ErrorDisplay.svelte +1 -1
  161. package/dist/recipes/feedback/StatusIndicator/StatusIndicator.spec.js +129 -129
  162. package/dist/recipes/feedback/StatusIndicator/StatusIndicator.svelte +157 -157
  163. package/dist/recipes/fields/CheckboxField.svelte +85 -85
  164. package/dist/recipes/fields/FormField.svelte +58 -58
  165. package/dist/recipes/fields/RadioGroup.svelte +95 -95
  166. package/dist/recipes/fields/SelectField.svelte +80 -80
  167. package/dist/recipes/fields/TextareaField.svelte +97 -97
  168. package/dist/recipes/fields/ToggleField.svelte +60 -60
  169. package/dist/recipes/fields/index.js +7 -7
  170. package/dist/recipes/inputs/MultiSelect.spec.js +257 -257
  171. package/dist/recipes/inputs/MultiSelect.stories.svelte +133 -133
  172. package/dist/recipes/inputs/MultiSelect.svelte +249 -249
  173. package/dist/recipes/inputs/OTPInput.spec.js +238 -238
  174. package/dist/recipes/inputs/OTPInput.stories.svelte +162 -162
  175. package/dist/recipes/inputs/OTPInput.svelte +29 -29
  176. package/dist/recipes/inputs/PasswordInput.svelte +22 -22
  177. package/dist/recipes/inputs/PasswordStrengthIndicator/PasswordStrengthIndicator.spec.js +173 -173
  178. package/dist/recipes/inputs/PasswordStrengthIndicator/PasswordStrengthIndicator.svelte +43 -43
  179. package/dist/recipes/inputs/PlaceAutocomplete/PlaceAutocomplete.spec.js +300 -300
  180. package/dist/recipes/inputs/PlaceAutocomplete/PlaceAutocomplete.stories.svelte +123 -123
  181. package/dist/recipes/inputs/PlaceAutocomplete/PlaceAutocomplete.svelte +326 -326
  182. package/dist/recipes/inputs/Search.svelte +37 -37
  183. package/dist/recipes/inputs/SelectDropdown.svelte +57 -57
  184. package/dist/recipes/modals/AlertModal.svelte +130 -130
  185. package/dist/recipes/modals/ConfirmationModal.spec.js +191 -191
  186. package/dist/recipes/modals/ConfirmationModal.stories.svelte +119 -119
  187. package/dist/recipes/modals/ConfirmationModal.svelte +152 -152
  188. package/dist/recipes/modals/InputModal.svelte +182 -182
  189. package/dist/recipes/modals/ModalStateManager.spec.js +100 -100
  190. package/dist/recipes/modals/ModalStateManager.svelte +77 -77
  191. package/dist/recipes/modals/ModalTestWrapper.svelte +65 -65
  192. package/dist/recipes/modals/StatusModal.svelte +206 -206
  193. package/dist/services/EventService.js +75 -75
  194. package/dist/services/EventService.spec.js +217 -217
  195. package/dist/services/ShowService.spec.js +342 -342
  196. package/dist/stores/auth.js +36 -36
  197. package/dist/stores/auth.spec.js +139 -139
  198. package/dist/stores/toaster.js +13 -13
  199. package/dist/stories/ButtonAuditReview.stories.svelte +14 -14
  200. package/dist/stories/ButtonAuditReview.svelte +427 -427
  201. package/dist/stories/PatternsGallery.stories.svelte +19 -19
  202. package/dist/stories/PatternsGallery.svelte +206 -206
  203. package/dist/stories/PrimitivesGallery.stories.svelte +19 -19
  204. package/dist/stories/PrimitivesGallery.svelte +725 -725
  205. package/dist/stories/RecipesGallery.stories.svelte +19 -19
  206. package/dist/stories/RecipesGallery.svelte +271 -271
  207. package/dist/stories/button-audit-manifest.json +11186 -11186
  208. package/dist/tailwind/preset.cjs +82 -82
  209. package/dist/tokens/tokens.css +87 -87
  210. package/dist/utils/transitions.d.ts +24 -0
  211. package/dist/utils/transitions.d.ts.map +1 -0
  212. package/dist/utils/transitions.js +62 -0
  213. package/dist/utils/utils.js +354 -354
  214. package/package.json +283 -283
@@ -1,367 +1,367 @@
1
- <script>
2
- import { fly, fade } from 'svelte/transition';
3
- import { cubicOut } from 'svelte/easing';
4
- import { ChevronDownOutline, CloseOutline } from '../../primitives/Icons';
5
- import Spinner from '../../primitives/Spinner/Spinner.svelte';
6
- import { typography } from '../../tokens/typography';
7
-
8
- let {
9
- loading = false,
10
- quantities = {},
11
- donationAmounts = {}, // Map of ticketId -> donation amount string
12
- eventTickets = [],
13
- checkoutTicket = null,
14
- isAgreed = true,
15
- btnText = 'Checkout',
16
- promoApplied = false,
17
- promoDiscount = 0,
18
- currentPromoRule = null,
19
- executePurchase = null,
20
- elements = null,
21
- venueServiceCharge = {
22
- serviceFeeCents: 0,
23
- serviceFeePercentage: 0,
24
- serviceFeeChargeType: 'both',
25
- maxServiceFeeCents: 0,
26
- taxPercentage: 0,
27
- },
28
- onPriceUpdate,
29
- } = $props();
30
-
31
- // Helper to get effective price for a ticket (handles donation tickets)
32
- function getEffectivePrice(ticket) {
33
- // Donation ticket (type 2): use user-entered donation amount
34
- if (isDonationTicket(ticket)) {
35
- const donationAmount = donationAmounts[ticket.ID];
36
- return parseFloat(donationAmount) || 0;
37
- }
38
- // Regular ticket: use ticket price
39
- return parseFloat(ticket.price) || 0;
40
- }
41
-
42
- // Check if ticket is a donation ticket
43
- // Handle both 'type' and 'ticketType' fields, and coerce to number for comparison
44
- function isDonationTicket(ticket) {
45
- const ticketType = ticket.type ?? ticket.ticketType ?? 0;
46
- return Number(ticketType) === 2;
47
- }
48
-
49
- let showOrderSummaryOnMobile = $state(false);
50
-
51
- function feeFor(price) {
52
- const chargeType = venueServiceCharge.serviceFeeChargeType || 'both';
53
- let fee = 0;
54
-
55
- if (chargeType === 'flat' || chargeType === 'both') {
56
- fee += (venueServiceCharge.serviceFeeCents || 0) / 100;
57
- }
58
- if (chargeType === 'percent' || chargeType === 'both') {
59
- fee += ((venueServiceCharge.serviceFeePercentage || 0) / 100) * price;
60
- }
61
-
62
- const maxFee = venueServiceCharge.maxServiceFeeCents || 0;
63
- if (maxFee > 0 && fee > maxFee / 100) {
64
- fee = maxFee / 100;
65
- }
66
-
67
- return fee;
68
- }
69
-
70
- function makeOrderSummaryVisible() {
71
- showOrderSummaryOnMobile = !showOrderSummaryOnMobile;
72
- }
73
-
74
- function getDiscountedPrice(ticket) {
75
- if (!currentPromoRule?.provideDiscount) return null;
76
-
77
- const discountTicketIds = currentPromoRule.discountTicketIds || [];
78
- if (discountTicketIds.length > 0 && !discountTicketIds.includes(ticket.ID)) {
79
- return null;
80
- }
81
-
82
- const basePrice = parseFloat(ticket.price) || 0;
83
- const discountAmount = parseFloat(currentPromoRule?.amount || '0') || 0;
84
- if (currentPromoRule.discountType === '%') {
85
- return (basePrice * (1 - discountAmount / 100)).toFixed(2);
86
- } else if (currentPromoRule.discountType === '$') {
87
- return Math.max(0, basePrice - discountAmount).toFixed(2);
88
- }
89
- return null;
90
- }
91
-
92
- let totalQuantity = $derived(Object.values(quantities).reduce((sum, q) => sum + q, 0));
93
- let showFooter = $derived(totalQuantity > 0);
94
-
95
- let subtotalWithoutDiscount = $derived(Object.keys(quantities).reduce((acc, key) => {
96
- const ticket = eventTickets.find(t => t.ID == key);
97
- if (!ticket) return acc;
98
- // For donation tickets, use the donation amount; otherwise use ticket price
99
- const effectivePrice = getEffectivePrice(ticket);
100
- return acc + quantities[key] * effectivePrice;
101
- }, 0));
102
-
103
- let subtotal = $derived(Object.keys(quantities).reduce((acc, key) => {
104
- const ticket = eventTickets.find(t => t.ID == key);
105
- if (!ticket) return acc;
106
- // For donation tickets, use the donation amount (no discounts apply)
107
- if (isDonationTicket(ticket)) {
108
- const effectivePrice = getEffectivePrice(ticket);
109
- return acc + quantities[key] * effectivePrice;
110
- }
111
- // For regular tickets, apply discounts as usual
112
- const discountedPrice = getDiscountedPrice(ticket);
113
- const priceToUse = discountedPrice !== null ? parseFloat(discountedPrice) : ticket.price;
114
- return acc + quantities[key] * priceToUse;
115
- }, 0));
116
-
117
- let promoSavings = $derived(currentPromoRule?.provideDiscount ? (subtotalWithoutDiscount - subtotal) : 0);
118
-
119
- let fees = $derived(Object.keys(quantities).reduce((acc, key) => {
120
- const ticket = eventTickets.find(t => t.ID == key);
121
- // Skip fees for: no ticket, free tickets, and donation tickets (type 2)
122
- if (!ticket || ticket.price == 0 || isDonationTicket(ticket)) return acc;
123
- const discountedPrice = getDiscountedPrice(ticket);
124
- const priceToUse = discountedPrice !== null ? parseFloat(discountedPrice) : ticket.price;
125
- return acc + quantities[key] * feeFor(priceToUse);
126
- }, 0));
127
-
128
- let taxRate = $derived((venueServiceCharge.taxPercentage || 0) / 100);
129
- let taxes = $derived(subtotal > 0 ? subtotal * taxRate : 0);
130
- let total = $derived(Math.max(0, subtotal + fees + taxes - (promoApplied && !currentPromoRule?.provideDiscount ? promoDiscount : 0)));
131
-
132
- $effect(() => {
133
- onPriceUpdate?.({ subtotal, fees, taxes, total, promoSavings });
134
- });
135
- </script>
136
-
137
- <div
138
- id="orderSummary"
139
- class="hidden min-[872px]:block h-fit rounded-lg bg-white dark:bg-gray-800 border border-gray-200 dark:border-gray-600"
140
- >
141
- <div class="px-5 py-4 border-b border-gray-200 dark:border-gray-600">
142
- <h3 class={typography.h3}>
143
- Order summary
144
- </h3>
145
- </div>
146
-
147
- <div class="px-5 pb-5">
148
- {#if totalQuantity === 0}
149
- <div class="py-8 text-center">
150
- <p class="{typography.smMuted}">Select tickets to continue</p>
151
- </div>
152
- {:else}
153
- {#each Object.keys(quantities) as key}
154
- {#if quantities[key] > 0}
155
- {#each eventTickets as ticket}
156
- {#if ticket.ID == key}
157
- {@const effectivePrice = getEffectivePrice(ticket)}
158
- {@const isDonation = isDonationTicket(ticket)}
159
- <div class="flex justify-between py-3 border-b border-gray-200/50 dark:border-gray-600/50">
160
- <div>
161
- <p class="{typography.label}">{ticket.name}</p>
162
- <p class="{typography.smMuted}">
163
- {#if ticket.price === 0 && !isDonation}
164
- Free x {quantities[key]}
165
- {:else if isDonation}
166
- ${effectivePrice.toFixed(2)} x {quantities[key]}
167
- {:else}
168
- ${ticket.price.toFixed(2)} x {quantities[key]}
169
- {/if}
170
- </p>
171
- </div>
172
- <div class="{typography.label}">
173
- {#if ticket.price === 0 && !isDonation}
174
- Free
175
- {:else}
176
- ${(effectivePrice * quantities[key]).toFixed(2)}
177
- {/if}
178
- </div>
179
- </div>
180
- {/if}
181
- {/each}
182
- {/if}
183
- {/each}
184
-
185
- <div class="{`${typography.sm} flex flex-col py-3 gap-2`}">
186
- {#if promoSavings > 0 || (promoDiscount > 0 && !currentPromoRule?.provideDiscount)}
187
- <div class="flex justify-between text-gray-600 dark:text-gray-300">
188
- <span>Full Price</span><span>${subtotalWithoutDiscount.toFixed(2)}</span>
189
- </div>
190
- {/if}
191
- {#if promoSavings > 0}
192
- <div class="flex justify-between text-green-600 dark:text-green-500">
193
- <span>Discount</span><span>-${promoSavings.toFixed(2)}</span>
194
- </div>
195
- {:else if promoDiscount > 0 && !currentPromoRule?.provideDiscount}
196
- <div class="flex justify-between text-green-600 dark:text-green-500">
197
- <span>Discount</span><span>-${promoDiscount.toFixed(2)}</span>
198
- </div>
199
- {/if}
200
- <div class="flex justify-between text-gray-600 dark:text-gray-300">
201
- <span>Subtotal</span><span>${subtotal.toFixed(2)}</span>
202
- </div>
203
- <div class="flex justify-between text-gray-600 dark:text-gray-300">
204
- <span>Fees</span><span>${fees.toFixed(2)}</span>
205
- </div>
206
- <div class="flex justify-between text-gray-600 dark:text-gray-300">
207
- <span>Taxes</span><span>${taxes.toFixed(2)}</span>
208
- </div>
209
- </div>
210
-
211
- <div class="flex justify-between {typography.h3} py-4 text-lg border-t border-gray-200 dark:border-gray-600">
212
- <span>Total</span><span>${total.toFixed(2)}</span>
213
- </div>
214
- {/if}
215
-
216
- {#if totalQuantity > 0 && btnText === 'Place order'}
217
- <p class="{typography.xsMuted} text-center mb-3">
218
- By selecting Place order, I agree to the <a href="https://get-micdrop.com/tos" class="text-blue-700 dark:text-blue-500 underline hover:opacity-80" target="_blank" rel="noopener noreferrer">terms of service</a>
219
- </p>
220
- {/if}
221
-
222
- <button
223
- class="w-full h-12 font-semibold rounded-lg flex items-center justify-center transition-colors select-none touch-manipulation {totalQuantity === 0 ? 'bg-gray-200 dark:bg-gray-700 text-gray-500 dark:text-gray-400 cursor-not-allowed' : 'bg-blue-700 dark:bg-blue-600 text-white hover:bg-blue-800 dark:hover:bg-blue-700'}"
224
- onclick={() => {
225
- if (totalQuantity === 0) return;
226
- if (executePurchase) {
227
- executePurchase(elements);
228
- } else if (checkoutTicket) {
229
- checkoutTicket();
230
- }
231
- }}
232
- disabled={totalQuantity === 0}
233
- >
234
- {#if loading}
235
- <Spinner size="sm" color="white" />
236
- {:else}
237
- <span translate="no">{btnText}</span>
238
- {/if}
239
- </button>
240
- </div>
241
- </div>
242
-
243
- {#if showOrderSummaryOnMobile}
244
- <button
245
- class="min-[872px]:hidden fixed inset-0 bg-black/50 z-40 border-none cursor-pointer"
246
- onclick={() => (showOrderSummaryOnMobile = false)}
247
- aria-label="Close order summary"
248
- transition:fade={{ duration: 200 }}
249
- ></button>
250
- <div
251
- in:fly={{ y: 800, duration: 300, easing: cubicOut }}
252
- out:fly={{ y: 800, duration: 300, easing: cubicOut }}
253
- class="min-[872px]:hidden fixed bottom-0 left-0 w-full overflow-x-hidden overflow-y-auto z-50 max-h-[80vh] rounded-t-lg shadow-xl bg-white dark:bg-gray-800"
254
- >
255
- <div class="flex flex-row justify-between items-center px-5 py-4 border-b border-gray-200 dark:border-gray-600">
256
- <h2 class="{typography.h2} text-xl">
257
- Order summary
258
- </h2>
259
- <button
260
- onclick={() => (showOrderSummaryOnMobile = false)}
261
- class="transition-colors p-2 rounded-lg text-gray-500 dark:text-gray-400 hover:text-gray-900 dark:hover:text-white hover:bg-gray-50 dark:hover:bg-gray-700"
262
- aria-label="Close order summary"
263
- >
264
- <CloseOutline class="w-7 h-7" />
265
- </button>
266
- </div>
267
-
268
- <div class="px-5 pb-5">
269
- {#each Object.keys(quantities) as key}
270
- {#if quantities[key] > 0}
271
- {#each eventTickets as ticket}
272
- {#if ticket.ID == key}
273
- {@const effectivePrice = getEffectivePrice(ticket)}
274
- {@const isDonation = isDonationTicket(ticket)}
275
- <div class="flex justify-between py-4 border-b border-gray-200 dark:border-gray-600">
276
- <div>
277
- <p class="{typography.h5}">{ticket.name}</p>
278
- <p class="{typography.smMuted}">
279
- {#if ticket.price === 0 && !isDonation}
280
- Free x {quantities[key]}
281
- {:else if isDonation}
282
- ${effectivePrice.toFixed(2)} x {quantities[key]}
283
- {:else}
284
- ${ticket.price.toFixed(2)} x {quantities[key]}
285
- {/if}
286
- </p>
287
- </div>
288
- <div class="{typography.h5}">
289
- {#if ticket.price === 0 && !isDonation}
290
- Free
291
- {:else}
292
- ${(effectivePrice * quantities[key]).toFixed(2)}
293
- {/if}
294
- </div>
295
- </div>
296
- {/if}
297
- {/each}
298
- {/if}
299
- {/each}
300
-
301
- <div class="flex flex-col py-4 gap-3 text-gray-600 dark:text-gray-300">
302
- {#if promoSavings > 0 || (promoDiscount > 0 && !currentPromoRule?.provideDiscount)}
303
- <div class="flex justify-between">
304
- <span>Full Price</span><span>${subtotalWithoutDiscount.toFixed(2)}</span>
305
- </div>
306
- {/if}
307
- {#if promoSavings > 0}
308
- <div class="flex justify-between text-green-600 dark:text-green-500">
309
- <span>Discount</span><span>-${promoSavings.toFixed(2)}</span>
310
- </div>
311
- {:else if promoDiscount > 0 && !currentPromoRule?.provideDiscount}
312
- <div class="flex justify-between text-green-600 dark:text-green-500">
313
- <span>Discount</span><span>-${promoDiscount.toFixed(2)}</span>
314
- </div>
315
- {/if}
316
- <div class="flex justify-between">
317
- <span>Subtotal</span><span>${subtotal.toFixed(2)}</span>
318
- </div>
319
- <div class="flex justify-between">
320
- <span>Fees</span><span>${fees.toFixed(2)}</span>
321
- </div>
322
- <div class="flex justify-between">
323
- <span>Taxes</span><span>${taxes.toFixed(2)}</span>
324
- </div>
325
- </div>
326
-
327
- <div class="flex justify-between {typography.h3} py-5 border-t border-gray-200 dark:border-gray-600">
328
- <span>Total</span><span>${total.toFixed(2)}</span>
329
- </div>
330
- </div>
331
- </div>
332
- {/if}
333
-
334
- {#if showFooter}
335
- <div
336
- transition:fly={{ y: 100, duration: 200 }}
337
- class="min-[872px]:hidden fixed bottom-0 left-0 right-0 z-40 bg-white dark:bg-gray-800 border-t border-gray-200 dark:border-gray-600 shadow-xl p-4 pb-[max(16px,calc(env(safe-area-inset-bottom)+8px))] select-none touch-manipulation"
338
- >
339
- <div class="flex items-stretch gap-3">
340
- <button class="flex flex-col justify-between items-start shrink-0 whitespace-nowrap bg-transparent border-none p-0 cursor-pointer touch-manipulation" onclick={makeOrderSummaryVisible}>
341
- <span class="{`${typography.sm} flex items-center gap-1 text-gray-600 dark:text-gray-300`}">
342
- {totalQuantity} {totalQuantity > 1 ? 'tickets' : 'ticket'}
343
- <ChevronDownOutline class={`w-4 h-4 ${typography.iconMuted} transition-transform duration-200 ${showOrderSummaryOnMobile ? 'rotate-180' : ''}`} />
344
- </span>
345
- <span class="{typography.h2} text-xl">${total.toFixed(2)}</span>
346
- </button>
347
-
348
- <button
349
- class="{`${typography.label} flex-1 min-w-36 h-12 rounded-lg touch-manipulation flex items-center justify-center ${totalQuantity === 0 || !isAgreed ? 'bg-gray-200 dark:bg-gray-700 text-gray-500 dark:text-gray-400 cursor-not-allowed' : 'bg-blue-700 dark:bg-blue-600 text-white hover:bg-blue-800 dark:hover:bg-blue-700'}`}"
350
- onclick={() => {
351
- if (executePurchase) {
352
- executePurchase(elements);
353
- } else if (checkoutTicket) {
354
- checkoutTicket();
355
- }
356
- }}
357
- disabled={totalQuantity === 0 || !isAgreed}
358
- >
359
- {#if loading}
360
- <Spinner size="sm" color="white" />
361
- {:else}
362
- <span translate="no">Checkout</span>
363
- {/if}
364
- </button>
365
- </div>
366
- </div>
367
- {/if}
1
+ <script>
2
+ import { fly, fade } from 'svelte/transition';
3
+ import { cubicOut } from 'svelte/easing';
4
+ import { ChevronDownOutline, CloseOutline } from '../../primitives/Icons';
5
+ import Spinner from '../../primitives/Spinner/Spinner.svelte';
6
+ import { typography } from '../../tokens/typography';
7
+
8
+ let {
9
+ loading = false,
10
+ quantities = {},
11
+ donationAmounts = {}, // Map of ticketId -> donation amount string
12
+ eventTickets = [],
13
+ checkoutTicket = null,
14
+ isAgreed = true,
15
+ btnText = 'Checkout',
16
+ promoApplied = false,
17
+ promoDiscount = 0,
18
+ currentPromoRule = null,
19
+ executePurchase = null,
20
+ elements = null,
21
+ venueServiceCharge = {
22
+ serviceFeeCents: 0,
23
+ serviceFeePercentage: 0,
24
+ serviceFeeChargeType: 'both',
25
+ maxServiceFeeCents: 0,
26
+ taxPercentage: 0,
27
+ },
28
+ onPriceUpdate,
29
+ } = $props();
30
+
31
+ // Helper to get effective price for a ticket (handles donation tickets)
32
+ function getEffectivePrice(ticket) {
33
+ // Donation ticket (type 2): use user-entered donation amount
34
+ if (isDonationTicket(ticket)) {
35
+ const donationAmount = donationAmounts[ticket.ID];
36
+ return parseFloat(donationAmount) || 0;
37
+ }
38
+ // Regular ticket: use ticket price
39
+ return parseFloat(ticket.price) || 0;
40
+ }
41
+
42
+ // Check if ticket is a donation ticket
43
+ // Handle both 'type' and 'ticketType' fields, and coerce to number for comparison
44
+ function isDonationTicket(ticket) {
45
+ const ticketType = ticket.type ?? ticket.ticketType ?? 0;
46
+ return Number(ticketType) === 2;
47
+ }
48
+
49
+ let showOrderSummaryOnMobile = $state(false);
50
+
51
+ function feeFor(price) {
52
+ const chargeType = venueServiceCharge.serviceFeeChargeType || 'both';
53
+ let fee = 0;
54
+
55
+ if (chargeType === 'flat' || chargeType === 'both') {
56
+ fee += (venueServiceCharge.serviceFeeCents || 0) / 100;
57
+ }
58
+ if (chargeType === 'percent' || chargeType === 'both') {
59
+ fee += ((venueServiceCharge.serviceFeePercentage || 0) / 100) * price;
60
+ }
61
+
62
+ const maxFee = venueServiceCharge.maxServiceFeeCents || 0;
63
+ if (maxFee > 0 && fee > maxFee / 100) {
64
+ fee = maxFee / 100;
65
+ }
66
+
67
+ return fee;
68
+ }
69
+
70
+ function makeOrderSummaryVisible() {
71
+ showOrderSummaryOnMobile = !showOrderSummaryOnMobile;
72
+ }
73
+
74
+ function getDiscountedPrice(ticket) {
75
+ if (!currentPromoRule?.provideDiscount) return null;
76
+
77
+ const discountTicketIds = currentPromoRule.discountTicketIds || [];
78
+ if (discountTicketIds.length > 0 && !discountTicketIds.includes(ticket.ID)) {
79
+ return null;
80
+ }
81
+
82
+ const basePrice = parseFloat(ticket.price) || 0;
83
+ const discountAmount = parseFloat(currentPromoRule?.amount || '0') || 0;
84
+ if (currentPromoRule.discountType === '%') {
85
+ return (basePrice * (1 - discountAmount / 100)).toFixed(2);
86
+ } else if (currentPromoRule.discountType === '$') {
87
+ return Math.max(0, basePrice - discountAmount).toFixed(2);
88
+ }
89
+ return null;
90
+ }
91
+
92
+ let totalQuantity = $derived(Object.values(quantities).reduce((sum, q) => sum + q, 0));
93
+ let showFooter = $derived(totalQuantity > 0);
94
+
95
+ let subtotalWithoutDiscount = $derived(Object.keys(quantities).reduce((acc, key) => {
96
+ const ticket = eventTickets.find(t => t.ID == key);
97
+ if (!ticket) return acc;
98
+ // For donation tickets, use the donation amount; otherwise use ticket price
99
+ const effectivePrice = getEffectivePrice(ticket);
100
+ return acc + quantities[key] * effectivePrice;
101
+ }, 0));
102
+
103
+ let subtotal = $derived(Object.keys(quantities).reduce((acc, key) => {
104
+ const ticket = eventTickets.find(t => t.ID == key);
105
+ if (!ticket) return acc;
106
+ // For donation tickets, use the donation amount (no discounts apply)
107
+ if (isDonationTicket(ticket)) {
108
+ const effectivePrice = getEffectivePrice(ticket);
109
+ return acc + quantities[key] * effectivePrice;
110
+ }
111
+ // For regular tickets, apply discounts as usual
112
+ const discountedPrice = getDiscountedPrice(ticket);
113
+ const priceToUse = discountedPrice !== null ? parseFloat(discountedPrice) : ticket.price;
114
+ return acc + quantities[key] * priceToUse;
115
+ }, 0));
116
+
117
+ let promoSavings = $derived(currentPromoRule?.provideDiscount ? (subtotalWithoutDiscount - subtotal) : 0);
118
+
119
+ let fees = $derived(Object.keys(quantities).reduce((acc, key) => {
120
+ const ticket = eventTickets.find(t => t.ID == key);
121
+ // Skip fees for: no ticket, free tickets, and donation tickets (type 2)
122
+ if (!ticket || ticket.price == 0 || isDonationTicket(ticket)) return acc;
123
+ const discountedPrice = getDiscountedPrice(ticket);
124
+ const priceToUse = discountedPrice !== null ? parseFloat(discountedPrice) : ticket.price;
125
+ return acc + quantities[key] * feeFor(priceToUse);
126
+ }, 0));
127
+
128
+ let taxRate = $derived((venueServiceCharge.taxPercentage || 0) / 100);
129
+ let taxes = $derived(subtotal > 0 ? subtotal * taxRate : 0);
130
+ let total = $derived(Math.max(0, subtotal + fees + taxes - (promoApplied && !currentPromoRule?.provideDiscount ? promoDiscount : 0)));
131
+
132
+ $effect(() => {
133
+ onPriceUpdate?.({ subtotal, fees, taxes, total, promoSavings });
134
+ });
135
+ </script>
136
+
137
+ <div
138
+ id="orderSummary"
139
+ class="hidden min-[872px]:block h-fit rounded-lg bg-white dark:bg-gray-800 border border-gray-200 dark:border-gray-600"
140
+ >
141
+ <div class="px-5 py-4 border-b border-gray-200 dark:border-gray-600">
142
+ <h3 class={typography.h3}>
143
+ Order summary
144
+ </h3>
145
+ </div>
146
+
147
+ <div class="px-5 pb-5">
148
+ {#if totalQuantity === 0}
149
+ <div class="py-8 text-center">
150
+ <p class="{typography.smMuted}">Select tickets to continue</p>
151
+ </div>
152
+ {:else}
153
+ {#each Object.keys(quantities) as key}
154
+ {#if quantities[key] > 0}
155
+ {#each eventTickets as ticket}
156
+ {#if ticket.ID == key}
157
+ {@const effectivePrice = getEffectivePrice(ticket)}
158
+ {@const isDonation = isDonationTicket(ticket)}
159
+ <div class="flex justify-between py-3 border-b border-gray-200/50 dark:border-gray-600/50">
160
+ <div>
161
+ <p class="{typography.label}">{ticket.name}</p>
162
+ <p class="{typography.smMuted}">
163
+ {#if ticket.price === 0 && !isDonation}
164
+ Free x {quantities[key]}
165
+ {:else if isDonation}
166
+ ${effectivePrice.toFixed(2)} x {quantities[key]}
167
+ {:else}
168
+ ${ticket.price.toFixed(2)} x {quantities[key]}
169
+ {/if}
170
+ </p>
171
+ </div>
172
+ <div class="{typography.label}">
173
+ {#if ticket.price === 0 && !isDonation}
174
+ Free
175
+ {:else}
176
+ ${(effectivePrice * quantities[key]).toFixed(2)}
177
+ {/if}
178
+ </div>
179
+ </div>
180
+ {/if}
181
+ {/each}
182
+ {/if}
183
+ {/each}
184
+
185
+ <div class="{`${typography.sm} flex flex-col py-3 gap-2`}">
186
+ {#if promoSavings > 0 || (promoDiscount > 0 && !currentPromoRule?.provideDiscount)}
187
+ <div class="flex justify-between text-gray-600 dark:text-gray-300">
188
+ <span>Full Price</span><span>${subtotalWithoutDiscount.toFixed(2)}</span>
189
+ </div>
190
+ {/if}
191
+ {#if promoSavings > 0}
192
+ <div class="flex justify-between text-green-600 dark:text-green-500">
193
+ <span>Discount</span><span>-${promoSavings.toFixed(2)}</span>
194
+ </div>
195
+ {:else if promoDiscount > 0 && !currentPromoRule?.provideDiscount}
196
+ <div class="flex justify-between text-green-600 dark:text-green-500">
197
+ <span>Discount</span><span>-${promoDiscount.toFixed(2)}</span>
198
+ </div>
199
+ {/if}
200
+ <div class="flex justify-between text-gray-600 dark:text-gray-300">
201
+ <span>Subtotal</span><span>${subtotal.toFixed(2)}</span>
202
+ </div>
203
+ <div class="flex justify-between text-gray-600 dark:text-gray-300">
204
+ <span>Fees</span><span>${fees.toFixed(2)}</span>
205
+ </div>
206
+ <div class="flex justify-between text-gray-600 dark:text-gray-300">
207
+ <span>Taxes</span><span>${taxes.toFixed(2)}</span>
208
+ </div>
209
+ </div>
210
+
211
+ <div class="flex justify-between {typography.h3} py-4 text-lg border-t border-gray-200 dark:border-gray-600">
212
+ <span>Total</span><span>${total.toFixed(2)}</span>
213
+ </div>
214
+ {/if}
215
+
216
+ {#if totalQuantity > 0 && btnText === 'Place order'}
217
+ <p class="{typography.xsMuted} text-center mb-3">
218
+ By selecting Place order, I agree to the <a href="https://get-micdrop.com/tos" class="text-blue-700 dark:text-blue-500 underline hover:opacity-80" target="_blank" rel="noopener noreferrer">terms of service</a>
219
+ </p>
220
+ {/if}
221
+
222
+ <button
223
+ class="w-full h-12 font-semibold rounded-lg flex items-center justify-center transition-colors select-none touch-manipulation {totalQuantity === 0 ? 'bg-gray-200 dark:bg-gray-700 text-gray-500 dark:text-gray-400 cursor-not-allowed' : 'bg-blue-700 dark:bg-blue-600 text-white hover:bg-blue-800 dark:hover:bg-blue-700'}"
224
+ onclick={() => {
225
+ if (totalQuantity === 0) return;
226
+ if (executePurchase) {
227
+ executePurchase(elements);
228
+ } else if (checkoutTicket) {
229
+ checkoutTicket();
230
+ }
231
+ }}
232
+ disabled={totalQuantity === 0}
233
+ >
234
+ {#if loading}
235
+ <Spinner size="sm" color="white" />
236
+ {:else}
237
+ <span translate="no">{btnText}</span>
238
+ {/if}
239
+ </button>
240
+ </div>
241
+ </div>
242
+
243
+ {#if showOrderSummaryOnMobile}
244
+ <button
245
+ class="min-[872px]:hidden fixed inset-0 bg-black/50 z-40 border-none cursor-pointer"
246
+ onclick={() => (showOrderSummaryOnMobile = false)}
247
+ aria-label="Close order summary"
248
+ transition:fade={{ duration: 200 }}
249
+ ></button>
250
+ <div
251
+ in:fly={{ y: 800, duration: 300, easing: cubicOut }}
252
+ out:fly={{ y: 800, duration: 300, easing: cubicOut }}
253
+ class="min-[872px]:hidden fixed bottom-0 left-0 w-full overflow-x-hidden overflow-y-auto z-50 max-h-[80vh] rounded-t-lg shadow-xl bg-white dark:bg-gray-800"
254
+ >
255
+ <div class="flex flex-row justify-between items-center px-5 py-4 border-b border-gray-200 dark:border-gray-600">
256
+ <h2 class="{typography.h2} text-xl">
257
+ Order summary
258
+ </h2>
259
+ <button
260
+ onclick={() => (showOrderSummaryOnMobile = false)}
261
+ class="transition-colors p-2 rounded-lg text-gray-500 dark:text-gray-400 hover:text-gray-900 dark:hover:text-white hover:bg-gray-50 dark:hover:bg-gray-700"
262
+ aria-label="Close order summary"
263
+ >
264
+ <CloseOutline class="w-7 h-7" />
265
+ </button>
266
+ </div>
267
+
268
+ <div class="px-5 pb-5">
269
+ {#each Object.keys(quantities) as key}
270
+ {#if quantities[key] > 0}
271
+ {#each eventTickets as ticket}
272
+ {#if ticket.ID == key}
273
+ {@const effectivePrice = getEffectivePrice(ticket)}
274
+ {@const isDonation = isDonationTicket(ticket)}
275
+ <div class="flex justify-between py-4 border-b border-gray-200 dark:border-gray-600">
276
+ <div>
277
+ <p class="{typography.h5}">{ticket.name}</p>
278
+ <p class="{typography.smMuted}">
279
+ {#if ticket.price === 0 && !isDonation}
280
+ Free x {quantities[key]}
281
+ {:else if isDonation}
282
+ ${effectivePrice.toFixed(2)} x {quantities[key]}
283
+ {:else}
284
+ ${ticket.price.toFixed(2)} x {quantities[key]}
285
+ {/if}
286
+ </p>
287
+ </div>
288
+ <div class="{typography.h5}">
289
+ {#if ticket.price === 0 && !isDonation}
290
+ Free
291
+ {:else}
292
+ ${(effectivePrice * quantities[key]).toFixed(2)}
293
+ {/if}
294
+ </div>
295
+ </div>
296
+ {/if}
297
+ {/each}
298
+ {/if}
299
+ {/each}
300
+
301
+ <div class="flex flex-col py-4 gap-3 text-gray-600 dark:text-gray-300">
302
+ {#if promoSavings > 0 || (promoDiscount > 0 && !currentPromoRule?.provideDiscount)}
303
+ <div class="flex justify-between">
304
+ <span>Full Price</span><span>${subtotalWithoutDiscount.toFixed(2)}</span>
305
+ </div>
306
+ {/if}
307
+ {#if promoSavings > 0}
308
+ <div class="flex justify-between text-green-600 dark:text-green-500">
309
+ <span>Discount</span><span>-${promoSavings.toFixed(2)}</span>
310
+ </div>
311
+ {:else if promoDiscount > 0 && !currentPromoRule?.provideDiscount}
312
+ <div class="flex justify-between text-green-600 dark:text-green-500">
313
+ <span>Discount</span><span>-${promoDiscount.toFixed(2)}</span>
314
+ </div>
315
+ {/if}
316
+ <div class="flex justify-between">
317
+ <span>Subtotal</span><span>${subtotal.toFixed(2)}</span>
318
+ </div>
319
+ <div class="flex justify-between">
320
+ <span>Fees</span><span>${fees.toFixed(2)}</span>
321
+ </div>
322
+ <div class="flex justify-between">
323
+ <span>Taxes</span><span>${taxes.toFixed(2)}</span>
324
+ </div>
325
+ </div>
326
+
327
+ <div class="flex justify-between {typography.h3} py-5 border-t border-gray-200 dark:border-gray-600">
328
+ <span>Total</span><span>${total.toFixed(2)}</span>
329
+ </div>
330
+ </div>
331
+ </div>
332
+ {/if}
333
+
334
+ {#if showFooter}
335
+ <div
336
+ transition:fly={{ y: 100, duration: 200 }}
337
+ class="min-[872px]:hidden fixed bottom-0 left-0 right-0 z-40 bg-white dark:bg-gray-800 border-t border-gray-200 dark:border-gray-600 shadow-xl p-4 pb-[max(16px,calc(env(safe-area-inset-bottom)+8px))] select-none touch-manipulation"
338
+ >
339
+ <div class="flex items-stretch gap-3">
340
+ <button class="flex flex-col justify-between items-start shrink-0 whitespace-nowrap bg-transparent border-none p-0 cursor-pointer touch-manipulation" onclick={makeOrderSummaryVisible}>
341
+ <span class="{`${typography.sm} flex items-center gap-1 text-gray-600 dark:text-gray-300`}">
342
+ {totalQuantity} {totalQuantity > 1 ? 'tickets' : 'ticket'}
343
+ <ChevronDownOutline class={`w-4 h-4 ${typography.iconMuted} transition-transform duration-200 ${showOrderSummaryOnMobile ? 'rotate-180' : ''}`} />
344
+ </span>
345
+ <span class="{typography.h2} text-xl">${total.toFixed(2)}</span>
346
+ </button>
347
+
348
+ <button
349
+ class="{`${typography.label} flex-1 min-w-36 h-12 rounded-lg touch-manipulation flex items-center justify-center ${totalQuantity === 0 || !isAgreed ? 'bg-gray-200 dark:bg-gray-700 text-gray-500 dark:text-gray-400 cursor-not-allowed' : 'bg-blue-700 dark:bg-blue-600 text-white hover:bg-blue-800 dark:hover:bg-blue-700'}`}"
350
+ onclick={() => {
351
+ if (executePurchase) {
352
+ executePurchase(elements);
353
+ } else if (checkoutTicket) {
354
+ checkoutTicket();
355
+ }
356
+ }}
357
+ disabled={totalQuantity === 0 || !isAgreed}
358
+ >
359
+ {#if loading}
360
+ <Spinner size="sm" color="white" />
361
+ {:else}
362
+ <span translate="no">Checkout</span>
363
+ {/if}
364
+ </button>
365
+ </div>
366
+ </div>
367
+ {/if}