@getmicdrop/svelte-components 5.6.1 → 5.7.1

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