@getmicdrop/svelte-components 5.8.1 → 5.8.2

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 (228) 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/Heading.svelte +60 -58
  10. package/dist/components/Heading.svelte.d.ts +1 -0
  11. package/dist/components/Heading.svelte.d.ts.map +1 -1
  12. package/dist/components/Layout/AppShell.svelte +104 -104
  13. package/dist/components/Layout/ContentSection.svelte +80 -80
  14. package/dist/components/Layout/Grid.svelte +4 -4
  15. package/dist/components/Layout/Heading.svelte +81 -81
  16. package/dist/components/Layout/PageContainer.svelte +69 -69
  17. package/dist/components/Layout/Responsive.svelte +75 -75
  18. package/dist/components/Layout/Section.svelte +80 -80
  19. package/dist/components/Layout/ShowOnDesktop.svelte +37 -37
  20. package/dist/components/Layout/ShowOnMobile.svelte +37 -37
  21. package/dist/components/Layout/Sidebar.svelte +108 -108
  22. package/dist/components/Layout/Stack.spec.js +1 -1
  23. package/dist/components/Layout/Stack.svelte +6 -6
  24. package/dist/components/Layout/Text.svelte +87 -87
  25. package/dist/components/Layout/TwoColumn.svelte +108 -108
  26. package/dist/components/Text.svelte +53 -53
  27. package/dist/constants/validation.js +91 -91
  28. package/dist/constants/validation.spec.js +64 -64
  29. package/dist/datetime/__tests__/format.test.js +1 -1
  30. package/dist/datetime/__tests__/parse.test.js +1 -1
  31. package/dist/datetime/__tests__/timezone.test.js +1 -1
  32. package/dist/datetime/parse.js +1 -1
  33. package/dist/forms/createFormStore.svelte.js +1 -0
  34. package/dist/index.js +50 -50
  35. package/dist/patterns/data/DataGrid.svelte +45 -45
  36. package/dist/patterns/data/DataList.svelte +24 -24
  37. package/dist/patterns/data/DataTable.svelte +36 -36
  38. package/dist/patterns/forms/FormActions.spec.js +95 -95
  39. package/dist/patterns/forms/FormActions.stories.svelte +97 -97
  40. package/dist/patterns/forms/FormActions.svelte +46 -46
  41. package/dist/patterns/forms/FormGrid.svelte +33 -33
  42. package/dist/patterns/forms/FormSection.svelte +32 -32
  43. package/dist/patterns/forms/FormValidationSummary.stories.svelte +83 -83
  44. package/dist/patterns/forms/FormValidationSummary.svelte +74 -74
  45. package/dist/patterns/layout/Sidebar.svelte +39 -39
  46. package/dist/patterns/layout/index.js +29 -29
  47. package/dist/patterns/navigation/BottomNav.stories.svelte +117 -117
  48. package/dist/patterns/navigation/BottomNav.svelte +74 -74
  49. package/dist/patterns/navigation/Header.stories.svelte +77 -77
  50. package/dist/patterns/navigation/Header.svelte +193 -193
  51. package/dist/patterns/page/PageHeader.svelte +18 -18
  52. package/dist/patterns/page/PageLayout.svelte +40 -40
  53. package/dist/patterns/page/PageLoader.spec.js +57 -57
  54. package/dist/patterns/page/PageLoader.stories.svelte +137 -137
  55. package/dist/patterns/page/PageLoader.svelte +24 -24
  56. package/dist/patterns/page/SectionHeader.svelte +29 -29
  57. package/dist/presets/badges.js +112 -112
  58. package/dist/presets/buttons.js +76 -76
  59. package/dist/presets/index.js +9 -9
  60. package/dist/primitives/Accordion/Accordion.stories.svelte +75 -75
  61. package/dist/primitives/Accordion/Accordion.svelte +42 -42
  62. package/dist/primitives/Accordion/AccordionItem.svelte +95 -95
  63. package/dist/primitives/Alert/Alert.spec.js +173 -173
  64. package/dist/primitives/Alert/Alert.stories.svelte +88 -88
  65. package/dist/primitives/Alert/Alert.svelte +27 -27
  66. package/dist/primitives/Avatar/Avatar.stories.svelte +94 -94
  67. package/dist/primitives/Avatar/Avatar.svelte +66 -66
  68. package/dist/primitives/Badges/Badge.spec.js +144 -144
  69. package/dist/primitives/Badges/Badge.stories.svelte +86 -86
  70. package/dist/primitives/Badges/Badge.svelte +79 -79
  71. package/dist/primitives/BottomSheet/BottomSheet.spec.js +136 -136
  72. package/dist/primitives/BottomSheet/BottomSheet.stories.svelte +83 -83
  73. package/dist/primitives/BottomSheet/BottomSheet.svelte +100 -100
  74. package/dist/primitives/Breadcrumb/Breadcrumb.spec.js +122 -122
  75. package/dist/primitives/Breadcrumb/Breadcrumb.stories.svelte +23 -23
  76. package/dist/primitives/Breadcrumb/Breadcrumb.svelte +89 -89
  77. package/dist/primitives/Button/Button.spec.js +223 -223
  78. package/dist/primitives/Button/Button.stories.svelte +76 -76
  79. package/dist/primitives/Button/Button.svelte +270 -270
  80. package/dist/primitives/Button/ButtonSaveDemo.spec.js +146 -146
  81. package/dist/primitives/Button/ButtonSaveDemo.svelte +25 -25
  82. package/dist/primitives/Button/ButtonVariantShowcase.svelte +129 -129
  83. package/dist/primitives/Card.spec.js +49 -49
  84. package/dist/primitives/Card.stories.svelte +22 -22
  85. package/dist/primitives/Card.svelte +28 -28
  86. package/dist/primitives/Checkbox/Checkbox.stories.svelte +84 -84
  87. package/dist/primitives/Checkbox/Checkbox.svelte +88 -88
  88. package/dist/primitives/DarkModeToggle.spec.js +390 -390
  89. package/dist/primitives/DarkModeToggle.stories.svelte +57 -57
  90. package/dist/primitives/DarkModeToggle.svelte +136 -136
  91. package/dist/primitives/Drawer/Drawer.stories.svelte +80 -80
  92. package/dist/primitives/Drawer/Drawer.svelte +120 -120
  93. package/dist/primitives/Dropdown/Dropdown.stories.svelte +137 -137
  94. package/dist/primitives/Dropdown/Dropdown.svelte +14 -14
  95. package/dist/primitives/Dropdown/DropdownDivider.svelte +9 -9
  96. package/dist/primitives/Dropdown/DropdownItem.svelte +80 -80
  97. package/dist/primitives/Helper/Helper.svelte +33 -33
  98. package/dist/primitives/Icons/ArrowLeft.svelte +8 -8
  99. package/dist/primitives/Icons/ArrowRight.svelte +8 -8
  100. package/dist/primitives/Icons/Availability.svelte +14 -14
  101. package/dist/primitives/Icons/Back.svelte +14 -14
  102. package/dist/primitives/Icons/CheckCircle.svelte +6 -6
  103. package/dist/primitives/Icons/CheckCircleOutline.svelte +15 -15
  104. package/dist/primitives/Icons/ChevronLeft.svelte +4 -4
  105. package/dist/primitives/Icons/ChevronRight.svelte +4 -4
  106. package/dist/primitives/Icons/Copy.svelte +15 -15
  107. package/dist/primitives/Icons/Cross.svelte +5 -5
  108. package/dist/primitives/Icons/DownArrow.svelte +8 -8
  109. package/dist/primitives/Icons/ErrorCircle.svelte +6 -6
  110. package/dist/primitives/Icons/FacebookIcon.svelte +2 -2
  111. package/dist/primitives/Icons/Home.svelte +15 -15
  112. package/dist/primitives/Icons/Icon.spec.js +169 -169
  113. package/dist/primitives/Icons/Icon.stories.svelte +100 -100
  114. package/dist/primitives/Icons/Icon.svelte +52 -52
  115. package/dist/primitives/Icons/IconGallery.stories.svelte +235 -235
  116. package/dist/primitives/Icons/Info.svelte +7 -7
  117. package/dist/primitives/Icons/InstagramIcon.svelte +4 -4
  118. package/dist/primitives/Icons/LogoInstagram.svelte +2 -2
  119. package/dist/primitives/Icons/Message.svelte +15 -15
  120. package/dist/primitives/Icons/MoonIcon.svelte +5 -5
  121. package/dist/primitives/Icons/More.svelte +21 -21
  122. package/dist/primitives/Icons/MoreHori.spec.js +61 -61
  123. package/dist/primitives/Icons/MoreHori.svelte +22 -22
  124. package/dist/primitives/Icons/Notification.svelte +14 -14
  125. package/dist/primitives/Icons/Payment.svelte +14 -14
  126. package/dist/primitives/Icons/Profile.svelte +21 -21
  127. package/dist/primitives/Icons/Reload.svelte +29 -29
  128. package/dist/primitives/Icons/Shows.svelte +21 -21
  129. package/dist/primitives/Icons/Signout.svelte +21 -21
  130. package/dist/primitives/Icons/SunIcon.svelte +8 -8
  131. package/dist/primitives/Icons/TiktokIcon.svelte +2 -2
  132. package/dist/primitives/Icons/TwitterIcon.svelte +2 -2
  133. package/dist/primitives/Icons/WarningIcon.spec.js +18 -18
  134. package/dist/primitives/Icons/WarningIcon.svelte +5 -5
  135. package/dist/primitives/Input/Input.spec.js +573 -573
  136. package/dist/primitives/Input/Input.stories.svelte +139 -139
  137. package/dist/primitives/Input/Select.spec.js +212 -212
  138. package/dist/primitives/Input/Select.stories.svelte +112 -112
  139. package/dist/primitives/Input/Select.svelte +128 -128
  140. package/dist/primitives/Input/Textarea.stories.svelte +137 -137
  141. package/dist/primitives/Input/Textarea.svelte +35 -35
  142. package/dist/primitives/Label/Label.svelte +37 -37
  143. package/dist/primitives/Modal/Modal.spec.js +99 -99
  144. package/dist/primitives/Modal/Modal.stories.svelte +86 -86
  145. package/dist/primitives/Pagination/Pagination.stories.svelte +76 -76
  146. package/dist/primitives/Pagination/Pagination.svelte +261 -261
  147. package/dist/primitives/Radio/Radio.stories.svelte +80 -80
  148. package/dist/primitives/Radio/Radio.svelte +67 -67
  149. package/dist/primitives/Skeleton/CardPlaceholder.svelte +87 -87
  150. package/dist/primitives/Skeleton/ImagePlaceholder.svelte +59 -59
  151. package/dist/primitives/Skeleton/ListPlaceholder.svelte +76 -76
  152. package/dist/primitives/Skeleton/Skeleton.stories.svelte +151 -151
  153. package/dist/primitives/Skeleton/Skeleton.svelte +26 -26
  154. package/dist/primitives/Spinner/Spinner.spec.js +71 -71
  155. package/dist/primitives/Spinner/Spinner.stories.svelte +29 -29
  156. package/dist/primitives/Spinner/Spinner.svelte +20 -20
  157. package/dist/primitives/Tabs/TabItem.svelte +49 -49
  158. package/dist/primitives/Tabs/Tabs.stories.svelte +112 -112
  159. package/dist/primitives/Tabs/Tabs.svelte +123 -123
  160. package/dist/primitives/Toggle.spec.js +143 -143
  161. package/dist/primitives/Toggle.stories.svelte +92 -92
  162. package/dist/primitives/Tooltip/Tooltip.svelte +83 -83
  163. package/dist/primitives/Typography/Typography.svelte +53 -53
  164. package/dist/primitives/ValidationError.spec.js +103 -103
  165. package/dist/primitives/ValidationError.stories.svelte +69 -69
  166. package/dist/primitives/ValidationError.svelte +29 -29
  167. package/dist/primitives/index.js +91 -91
  168. package/dist/recipes/CropImage/CropImage.spec.js +208 -208
  169. package/dist/recipes/CropImage/CropImage.stories.svelte +104 -104
  170. package/dist/recipes/CropImage/CropImage.svelte +238 -238
  171. package/dist/recipes/ImageUploader/ImageUploader.stories.svelte +125 -125
  172. package/dist/recipes/ImageUploader/ImageUploader.svelte +804 -804
  173. package/dist/recipes/Toaster/Toaster.stories.svelte +62 -62
  174. package/dist/recipes/feedback/EmptyState/EmptyState.svelte +1 -1
  175. package/dist/recipes/feedback/ErrorDisplay.spec.js +69 -69
  176. package/dist/recipes/feedback/ErrorDisplay.stories.svelte +101 -101
  177. package/dist/recipes/feedback/ErrorDisplay.svelte +1 -1
  178. package/dist/recipes/feedback/StatusIndicator/StatusIndicator.spec.js +133 -133
  179. package/dist/recipes/feedback/StatusIndicator/StatusIndicator.svelte +157 -157
  180. package/dist/recipes/fields/CheckboxField.svelte +85 -85
  181. package/dist/recipes/fields/FormField.svelte +58 -58
  182. package/dist/recipes/fields/RadioGroup.svelte +95 -95
  183. package/dist/recipes/fields/SelectField.svelte +80 -80
  184. package/dist/recipes/fields/TextareaField.svelte +97 -97
  185. package/dist/recipes/fields/ToggleField.svelte +60 -60
  186. package/dist/recipes/fields/index.js +7 -7
  187. package/dist/recipes/inputs/MultiSelect.spec.js +258 -258
  188. package/dist/recipes/inputs/MultiSelect.stories.svelte +133 -133
  189. package/dist/recipes/inputs/OTPInput.spec.js +251 -251
  190. package/dist/recipes/inputs/OTPInput.stories.svelte +162 -162
  191. package/dist/recipes/inputs/OTPInput.svelte +29 -29
  192. package/dist/recipes/inputs/PasswordInput.svelte +22 -22
  193. package/dist/recipes/inputs/PasswordStrengthIndicator/PasswordStrengthIndicator.svelte +117 -117
  194. package/dist/recipes/inputs/PlaceAutocomplete/PlaceAutocomplete.stories.svelte +123 -123
  195. package/dist/recipes/inputs/Search.svelte +37 -37
  196. package/dist/recipes/inputs/SelectDropdown.svelte +57 -57
  197. package/dist/recipes/modals/AlertModal.svelte +130 -130
  198. package/dist/recipes/modals/ConfirmationModal.spec.js +206 -206
  199. package/dist/recipes/modals/ConfirmationModal.stories.svelte +119 -119
  200. package/dist/recipes/modals/ConfirmationModal.svelte +152 -152
  201. package/dist/recipes/modals/InputModal.svelte +182 -182
  202. package/dist/recipes/modals/ModalStateManager.spec.js +100 -100
  203. package/dist/recipes/modals/ModalStateManager.svelte +77 -77
  204. package/dist/recipes/modals/ModalTestWrapper.svelte +65 -65
  205. package/dist/recipes/modals/StatusModal.svelte +206 -206
  206. package/dist/services/EventService.js +75 -75
  207. package/dist/services/EventService.spec.js +217 -217
  208. package/dist/services/ShowService.spec.js +345 -345
  209. package/dist/stores/toaster.js +13 -13
  210. package/dist/stories/ButtonAuditReview.stories.svelte +14 -14
  211. package/dist/stories/ButtonAuditReview.svelte +427 -427
  212. package/dist/stories/PatternsGallery.stories.svelte +19 -19
  213. package/dist/stories/PatternsGallery.svelte +206 -206
  214. package/dist/stories/PrimitivesGallery.stories.svelte +19 -19
  215. package/dist/stories/PrimitivesGallery.svelte +725 -725
  216. package/dist/stories/RecipesGallery.stories.svelte +19 -19
  217. package/dist/stories/RecipesGallery.svelte +271 -271
  218. package/dist/stories/button-audit-manifest.json +11186 -11186
  219. package/dist/tailwind/preset.cjs +82 -82
  220. package/dist/telemetry.js +405 -405
  221. package/dist/telemetry.spec.js +1169 -1169
  222. package/dist/tokens/tokens.css +87 -87
  223. package/dist/tokens/typography-base.css +163 -163
  224. package/dist/tokens/utilities.css +353 -0
  225. package/dist/utils/apiConfig.spec.js +219 -219
  226. package/dist/utils/transitions.js +62 -62
  227. package/dist/utils/utils.js +354 -354
  228. package/package.json +296 -295
@@ -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}