@getmicdrop/venue-calendar 4.0.83 → 4.0.84

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 (47) hide show
  1. package/dist/{CartView-DBrUqLSR.js → CartView-DwAGELwG.js} +5 -5
  2. package/dist/{CartView-DBrUqLSR.js.map → CartView-DwAGELwG.js.map} +1 -1
  3. package/dist/{Checkout-BAtOdBIM.js → Checkout-CHo11MsU.js} +7 -7
  4. package/dist/{Checkout-BAtOdBIM.js.map → Checkout-CHo11MsU.js.map} +1 -1
  5. package/dist/{Checkout-BEcmGOZm.js → Checkout-_NuoBAKt.js} +3 -3
  6. package/dist/{Checkout-BEcmGOZm.js.map → Checkout-_NuoBAKt.js.map} +1 -1
  7. package/dist/{CheckoutTimer-CayvrAa9.js → CheckoutTimer-CAGR7L-7.js} +4 -4
  8. package/dist/{CheckoutTimer-CayvrAa9.js.map → CheckoutTimer-CAGR7L-7.js.map} +1 -1
  9. package/dist/{CollectionView-kk90OMFq.js → CollectionView-Dstgu8sR.js} +5 -5
  10. package/dist/{CollectionView-kk90OMFq.js.map → CollectionView-Dstgu8sR.js.map} +1 -1
  11. package/dist/{Event-N959l6uT.js → Event-kfRjkMzQ.js} +1125 -1125
  12. package/dist/Event-kfRjkMzQ.js.map +1 -0
  13. package/dist/{EventPage-DaEMiLjY.js → EventPage-CnhccVCI.js} +7 -7
  14. package/dist/{EventPage-DaEMiLjY.js.map → EventPage-CnhccVCI.js.map} +1 -1
  15. package/dist/{Heading-A_nixT3L.js → Heading-Ielp7dS-.js} +2 -2
  16. package/dist/{Heading-A_nixT3L.js.map → Heading-Ielp7dS-.js.map} +1 -1
  17. package/dist/{ModalHeader-CxMVzApa.js → ModalHeader-Q7levd4l.js} +2 -2
  18. package/dist/{ModalHeader-CxMVzApa.js.map → ModalHeader-Q7levd4l.js.map} +1 -1
  19. package/dist/{ScarcityBadge-CiF-JsPl.js → ScarcityBadge-CG7Mz5YN.js} +2 -2
  20. package/dist/{ScarcityBadge-CiF-JsPl.js.map → ScarcityBadge-CG7Mz5YN.js.map} +1 -1
  21. package/dist/{SeriesPage-CQJHrpTH.js → SeriesPage-C8fIvt63.js} +5 -5
  22. package/dist/{SeriesPage-CQJHrpTH.js.map → SeriesPage-C8fIvt63.js.map} +1 -1
  23. package/dist/{Success-R-1NhLjx.js → Success-SZNXo-Ou.js} +7 -7
  24. package/dist/{Success-R-1NhLjx.js.map → Success-SZNXo-Ou.js.map} +1 -1
  25. package/dist/{Text-F6D2QvI6.js → Text-DHgQP4rA.js} +2 -2
  26. package/dist/{Text-F6D2QvI6.js.map → Text-DHgQP4rA.js.map} +1 -1
  27. package/dist/{VenueCalendar-DdRkxjxd.js → VenueCalendar-DTOyAhDU.js} +444 -445
  28. package/dist/VenueCalendar-DTOyAhDU.js.map +1 -0
  29. package/dist/{ViewTicketsEmbed-BwvS8VgE.js → ViewTicketsEmbed-DBEOjijY.js} +4 -4
  30. package/dist/{ViewTicketsEmbed-BwvS8VgE.js.map → ViewTicketsEmbed-DBEOjijY.js.map} +1 -1
  31. package/dist/{data-toggle-store.svelte-D9XyVwDW.js → data-toggle-store.svelte-E2JyoGTV.js} +2 -2
  32. package/dist/{data-toggle-store.svelte-D9XyVwDW.js.map → data-toggle-store.svelte-E2JyoGTV.js.map} +1 -1
  33. package/dist/{labels-DG5wh_vD.js → labels-nQIooadc.js} +3 -3
  34. package/dist/{labels-DG5wh_vD.js.map → labels-nQIooadc.js.map} +1 -1
  35. package/dist/seo/seo.cjs.map +1 -1
  36. package/dist/seo/seo.mjs.map +1 -1
  37. package/dist/{transform-C5S2acqv.js → transform-C1Rmuzlt.js} +2 -2
  38. package/dist/{transform-C5S2acqv.js.map → transform-C1Rmuzlt.js.map} +1 -1
  39. package/dist/venue-calendar.css +1 -1
  40. package/dist/venue-calendar.es.js +2 -2
  41. package/dist/venue-calendar.iife.js +11 -11
  42. package/dist/venue-calendar.iife.js.map +1 -1
  43. package/dist/venue-calendar.umd.js +10 -10
  44. package/dist/venue-calendar.umd.js.map +1 -1
  45. package/package.json +3 -4
  46. package/dist/Event-N959l6uT.js.map +0 -1
  47. package/dist/VenueCalendar-DdRkxjxd.js.map +0 -1
@@ -1 +1 @@
1
- {"version":3,"file":"Checkout-BAtOdBIM.js","sources":["../node_modules/@getmicdrop/svelte-components/dist/cookies/cookies.js","../node_modules/@getmicdrop/svelte-components/dist/primitives/Checkbox/Checkbox.svelte","../node_modules/@getmicdrop/svelte-components/dist/utils/storage/storage.js","../node_modules/@getmicdrop/svelte-components/dist/utils/storage/registry.js","../node_modules/@getmicdrop/svelte-components/dist/utils/storage/themeStorage.js","../__SKIP_ENVIRONMENT__","../node_modules/@getmicdrop/svelte-components/dist/stripe/useStripeTheme.svelte.js","../src/lib/public-calendar-flow/CheckoutForm.svelte","../src/lib/public-calendar-flow/CheckoutSidebar.svelte","../src/lib/public-calendar-flow/CheckoutMobileFooter.svelte","../src/lib/public-calendar-flow/Checkout.svelte"],"sourcesContent":["/**\n * Cookies canonical — the ONE sanctioned surface for browser cookie access\n * across all four Micdrop apps. Lifts micdrop-frontend's typed cookies layer\n * (registry + per-event builders + HttpOnly diagnostic) onto svelte-components'\n * battle-tested `document.cookie` implementation.\n *\n * CANONICAL DESIGN DECISION (Atlas convergence — utility-cookies):\n * The runtime behaviour is **byte-identical to SC's pre-audit-09\n * `cookieHelpers.ts`** — `getCookie` returns `string | null` (not js-cookie's\n * `undefined`), `setCookie` takes the `{ maxAge, path }` options shape and\n * writes `document.cookie` with the `.get-micdrop.com` production-domain check\n * and hardcoded `samesite=lax`, `deleteCookie` sets `max-age=0`,\n * `setAuthCookies` / `clearAuthCookies` are unchanged. MF's *typing* layer\n * (the `COOKIE_NAMES` registry, the per-event templated name builders, the\n * HttpOnly write diagnostic, and the `getCookieStrict` variant) is folded in\n * around that behaviour. SC's exported function names keep resolving so no\n * consumer breaks. (Rubric: prevent bugs > no user-facing harm > consistency.)\n *\n * Reads NEVER throw on a missing cookie — `getCookie` returns `null`, matching\n * SC's historical contract. Use `getCookieStrict` when absence is a programmer\n * error (it throws `CookieError` with code `NOT_SET`).\n *\n * Names: prefer `COOKIE_NAMES.X` and the builder helpers over string literals\n * so renames stay atomic and typos surface at compile time — the read/write\n * surface is typed to `CookieName`, which only the registry + builders +\n * `toCookieName` mint.\n */\nimport { CookieError } from '../errors/classes/CookieError.js';\nimport { asCookieName, toCookieName } from '../types/brands.js';\nimport { createLogger } from '../utils/logger.js';\nconst logger = createLogger('cookies');\n// ── Name registry ──────────────────────────────────────────────────────────\n/**\n * Registry of cookie names Micdrop apps read/write from JS.\n *\n * Covers the union of MF's `cookies.ts` registry and SC's pre-audit-09\n * `cookieHelpers.ts` literals. Prefer `COOKIE_NAMES.X` over raw strings so\n * renames are atomic and typos surface at compile time. Each entry is a\n * branded `CookieName` (minted via `asCookieName`), so values flow straight\n * into the read/write surface below without re-branding. For per-event\n * templated names (`checkout-promocode-${eventID}` etc.), use the builder\n * helpers below.\n */\nexport const COOKIE_NAMES = {\n // ── Operator (admin) auth — server-managed, HttpOnly ──\n // The server sets these on Set-Cookie; JS can read a presence shadow but\n // cannot write or delete them. Listed for type-safety in the client paths\n // that reference them.\n OPERATOR_TOKEN: asCookieName('operator_token'),\n REFRESH_TOKEN: asCookieName('refresh_token'),\n // ── Operator auth state (non-HttpOnly) ──\n TOKEN_EXPIRY: asCookieName('token_expiry'),\n ACTIVE_CONTEXT_ID: asCookieName('active_context_id'),\n // ── Performer (portal) auth — SC's cookieHelpers / performers-portal ──\n PERFORMER_TOKEN: asCookieName('performer_token'),\n PERFORMER_REFRESH_TOKEN: asCookieName('performer_refresh_token'),\n PERFORMER_TOKEN_EXPIRY: asCookieName('performer_token_expiry'),\n USER_DETAILS: asCookieName('userDetails'),\n SUPPRESSED_ALERTS: asCookieName('suppressedAlerts'),\n // ── Cart / checkout (non-templated) ──\n SESSION_ID: asCookieName('session_id'),\n CHECKOUT_CARTID: asCookieName('checkout-cartid'),\n CHECKOUT_PROMOCODE: asCookieName('checkout-promocode'),\n CHECKOUT_PROMO_DISCOUNT: asCookieName('checkout-promo-discount'),\n // ── Browser-resolved IANA timezone, set by hooks.client on app init ──\n USER_TZ: asCookieName('user_tz'),\n};\n/**\n * Cookies the server sets with HttpOnly. JS writes targeting these names are\n * effectively no-ops (the browser ignores `document.cookie =` writes that try\n * to overwrite an HttpOnly cookie). `setCookie` / `deleteCookie` emit a\n * dev-mode warning when one is targeted client-side — the historical bug was a\n * fake-delete of `operator_token` that silently did nothing. The fix is a\n * server endpoint that calls `cookies.delete()`.\n */\nconst HTTP_ONLY_COOKIES = new Set([\n COOKIE_NAMES.OPERATOR_TOKEN,\n COOKIE_NAMES.REFRESH_TOKEN,\n]);\nfunction warnIfHttpOnly(name, op) {\n if (!HTTP_ONLY_COOKIES.has(name))\n return;\n logger.warn(`${op}Cookie('${name}'): this cookie is HttpOnly; browsers ignore JS writes. ` +\n `Use a server endpoint (e.g. /api/auth/logout) that calls cookies.${op === 'set' ? 'set' : 'delete'}() instead.`);\n}\n// ── Per-event templated name builders ────────────────────────────────────────\n// Keep these as builders rather than raw template strings at call sites so\n// renames remain atomic. Each returns a branded `CookieName`.\n/** `checkout-quantities-${eventID}` */\nexport const checkoutQuantitiesCookie = (eventID) => toCookieName(`checkout-quantities-${eventID}`);\n/** `checkout-promocode-${eventID}` */\nexport const checkoutPromocodeCookie = (eventID) => toCookieName(`checkout-promocode-${eventID}`);\n/** `checkout-tickets-${eventID}` */\nexport const checkoutCartCookie = (eventID) => toCookieName(`checkout-tickets-${eventID}`);\n/**\n * @deprecated Alias of {@link checkoutCartCookie} — MF named this builder\n * `checkoutTicketsCookie`. Kept so the lift is name-stable for MF consumers;\n * the canonical name is `checkoutCartCookie` (matches the proposal).\n */\nexport const checkoutTicketsCookie = checkoutCartCookie;\n// ── Read / parse ──────────────────────────────────────────────────────────────\n/**\n * Parse all cookies into an object. SSR-safe (returns `{}` when `document` is\n * undefined). Behaviour is byte-identical to SC's pre-audit-09 `parseCookies`.\n */\nexport function parseCookies() {\n if (typeof document === 'undefined')\n return {};\n return document.cookie\n .split('; ')\n .reduce((acc, cookie) => {\n if (!cookie)\n return acc;\n const [name, ...valueParts] = cookie.split('=');\n acc[name] = decodeURIComponent(valueParts.join('='));\n return acc;\n }, {});\n}\n/**\n * Get a specific cookie value by name. Returns `null` for a missing value\n * (SC's historical contract — NOT js-cookie's `undefined`). Never throws.\n */\n// eslint-disable-next-line no-restricted-syntax -- this file IS the canonical getCookie; the rule points other call sites here\nexport function getCookie(name) {\n const cookies = parseCookies();\n return cookies[name] || null;\n}\n/**\n * Get a cookie value or throw if it is not set. Use when a missing cookie is a\n * programmer error rather than an expected state. Throws `CookieError` with\n * code `NOT_SET`.\n */\nexport function getCookieStrict(name) {\n const value = getCookie(name);\n if (value === null) {\n throw CookieError.notSet(name);\n }\n return value;\n}\n// ── Write / delete ────────────────────────────────────────────────────────────\n/**\n * Set a cookie. Behaviour is byte-identical to SC's pre-audit-09 `setCookie`:\n * encodes the value, derives the `.get-micdrop.com` production domain + `secure`\n * flag from `window.location`, hardcodes `samesite=lax`, defaults `path` to `/`.\n * Adds MF's HttpOnly-write diagnostic (a dev warning, no behaviour change).\n */\n// eslint-disable-next-line no-restricted-syntax -- this file IS the canonical setCookie; the rule points other call sites here\nexport function setCookie(name, value, options = {}) {\n warnIfHttpOnly(name, 'set');\n const isProd = typeof window !== 'undefined' &&\n window.location.hostname.includes('get-micdrop.com');\n const domainAttr = isProd ? 'domain=.get-micdrop.com;' : '';\n const secureAttr = typeof window !== 'undefined' && window.location.protocol === 'https:'\n ? 'secure;'\n : '';\n const maxAge = options.maxAge ? `max-age=${options.maxAge};` : '';\n const path = options.path || '/';\n document.cookie = `${name}=${encodeURIComponent(value)}; path=${path}; ${maxAge} ${domainAttr} ${secureAttr} samesite=lax`;\n}\n/**\n * Delete a cookie by name. Behaviour is byte-identical to SC's pre-audit-09\n * `deleteCookie`: SSR-safe no-op, sets `max-age=0` with the same domain/path\n * used by `setCookie` so the deletion takes effect on prod + dev. Adds MF's\n * HttpOnly-write diagnostic.\n */\nexport function deleteCookie(name, options = {}) {\n if (typeof document === 'undefined')\n return;\n warnIfHttpOnly(name, 'delete');\n const isProd = typeof window !== 'undefined' &&\n window.location.hostname.includes('get-micdrop.com');\n const domainAttr = isProd ? 'domain=.get-micdrop.com;' : '';\n const path = options.path || '/';\n document.cookie = `${name}=; path=${path}; max-age=0; ${domainAttr}`;\n}\n// ── Performer-token / details / alert helpers (SC pre-audit-09) ──────────────\n/** Get the performer token from cookies. */\nexport function getPerformerToken() {\n return getCookie(COOKIE_NAMES.PERFORMER_TOKEN);\n}\n/**\n * @deprecated R-STORAGE-07: the `userDetails` PII cookie has been removed.\n * Identity comes from `GET /api/auth/me` / the SSR profile, never from\n * `document.cookie`. Always returns null; retained only for API compatibility.\n */\nexport function getUserDetails() {\n return null;\n}\n/** Get suppressed alert IDs from cookies. Returns [] on miss / bad JSON. */\nexport function getSuppressedAlerts() {\n const alerts = getCookie(COOKIE_NAMES.SUPPRESSED_ALERTS);\n if (!alerts)\n return [];\n try {\n return JSON.parse(alerts);\n }\n catch {\n return [];\n }\n}\n/** Save suppressed alert IDs to cookies (1 year expiry). */\nexport function saveSuppressedAlerts(alertIds) {\n setCookie(COOKIE_NAMES.SUPPRESSED_ALERTS, JSON.stringify(alertIds), {\n maxAge: 31536000,\n });\n}\n// ── Auth cookie helpers (SC pre-audit-09 — LOAD-BEARING, byte-identical) ─────\n// SC's index.js exports `setAuthCookies` / `clearAuthCookies`; consumers rely\n// on the exact `document.cookie` writes below. Do NOT reroute these through\n// setCookie/deleteCookie — the raw writes (performer_token, userDetails,\n// performer_refresh_token, the 30-day max-age, the domain/secure derivation)\n// must stay identical.\n/**\n * Set authentication cookies with proper domain and security settings.\n *\n * SECURITY (R-STORAGE-07): this NO LONGER writes the `userDetails` PII cookie.\n * A `{firstName,lastName,email}` blob in a JS-readable cookie is XSS-extractable;\n * identity now comes from the cookie-authed `GET /api/auth/me` endpoint (or the\n * SSR profile), never from `document.cookie`. The `userDetails` param is retained\n * for signature compatibility but is intentionally ignored. The token cookies\n * (performer_token / performer_refresh_token) are unchanged.\n */\nexport function setAuthCookies(token, _userDetails, rememberMe = true, refreshToken = null) {\n void _userDetails; // R-STORAGE-07: never persist identity PII to document.cookie\n const isProd = window.location.hostname.includes('get-micdrop.com');\n const domainAttr = isProd ? 'domain=.get-micdrop.com;' : '';\n const secureAttr = window.location.protocol === 'https:' ? 'secure;' : '';\n const maxAge = rememberMe ? 'max-age=2592000;' : ''; // 30 days or session\n document.cookie = `performer_token=${token}; path=/; ${maxAge} ${domainAttr} ${secureAttr} samesite=lax`;\n if (refreshToken) {\n document.cookie = `performer_refresh_token=${refreshToken}; path=/; ${maxAge} ${domainAttr} ${secureAttr} samesite=lax`;\n }\n}\n/**\n * Clear authentication cookies.\n * Byte-identical to SC's pre-audit-09 `clearAuthCookies`.\n */\nexport function clearAuthCookies() {\n const isProd = window.location.hostname.includes('get-micdrop.com');\n const domainAttr = isProd ? 'domain=.get-micdrop.com;' : '';\n document.cookie = `userDetails=; path=/; max-age=0; ${domainAttr}`;\n document.cookie = `performer_token=; path=/; max-age=0; ${domainAttr}`;\n document.cookie = `performer_refresh_token=; path=/; max-age=0; ${domainAttr}`;\n}\n","<script lang=\"ts\">\r\n import type { Snippet } from 'svelte';\r\n\r\n interface Props {\r\n checked?: boolean;\r\n indeterminate?: boolean;\r\n value?: string;\r\n name?: string;\r\n disabled?: boolean;\r\n color?: 'blue' | 'red' | 'green' | 'purple' | 'orange' | 'yellow';\r\n class?: string;\r\n id?: string;\r\n 'aria-label'?: string;\r\n 'aria-labelledby'?: string;\r\n 'aria-describedby'?: string;\r\n // eslint-disable-next-line no-unused-vars -- intentionally retained (prop/slot/forward-compat or Svelte-5 reactive binding eslint cannot see)\n onchange?: (detail: { checked: boolean; value: string }) => void;\r\n onclick?: () => void;\r\n onfocus?: () => void;\r\n onblur?: () => void;\r\n // eslint-disable-next-line no-unused-vars -- intentionally retained (prop/slot/forward-compat or Svelte-5 reactive binding eslint cannot see)\n onkeydown?: (e: KeyboardEvent) => void;\r\n // eslint-disable-next-line no-unused-vars -- intentionally retained (prop/slot/forward-compat or Svelte-5 reactive binding eslint cannot see)\n onkeyup?: (e: KeyboardEvent) => void;\r\n children?: Snippet;\r\n }\r\n\r\n let {\r\n checked = $bindable(false),\r\n indeterminate = false,\r\n value = \"\",\r\n name = \"\",\r\n disabled = false,\r\n color = \"blue\",\r\n class: className = \"\",\r\n id,\r\n 'aria-label': ariaLabel,\r\n 'aria-labelledby': ariaLabelledBy,\r\n 'aria-describedby': ariaDescribedBy,\r\n onchange,\r\n onclick,\r\n onfocus,\r\n onblur,\r\n onkeydown,\r\n onkeyup,\r\n children,\r\n }: Props = $props();\r\n\r\n let inputEl = $state<HTMLInputElement>();\r\n // `indeterminate` is a DOM property, not a reflected HTML attribute,\r\n // so it has to be set imperatively after mount/update.\r\n $effect(() => {\r\n if (inputEl) inputEl.indeterminate = indeterminate;\r\n });\r\n\r\n function handleChange() {\r\n // Note: checked is already updated by bind:checked before this handler runs\r\n // We just need to call the onchange callback with the current values\r\n onchange?.({ checked, value });\r\n }\r\n\r\n const colorClasses = {\r\n blue: \"text-brand-primary\",\r\n red: \"text-accent-danger\",\r\n green: \"text-accent-success\",\r\n purple: \"text-brand-primary\",\r\n orange: \"text-accent-warning\",\r\n yellow: \"text-accent-warning\"\r\n };\r\n\r\n let colorClass = $derived(colorClasses[color] || colorClasses.blue);\r\n\r\n let inputClasses = $derived([\r\n \"w-4 h-4 shrink-0 rounded\",\r\n \"border-stroke-primary\",\r\n \"focus:outline-none focus:ring-0 focus:ring-offset-0 focus:shadow-none focus-visible:outline-none\",\r\n colorClass,\r\n disabled ? \"cursor-not-allowed opacity-50\" : \"cursor-pointer\"\r\n ].join(\" \"));\r\n</script>\r\n\r\n<label\r\n class=\"inline-flex items-start gap-2 {disabled ? 'cursor-not-allowed' : 'cursor-pointer'} select-none {className}\"\r\n {id}\r\n>\r\n <input\r\n bind:this={inputEl}\r\n type=\"checkbox\"\r\n {name}\r\n {value}\r\n {disabled}\r\n bind:checked\r\n onchange={handleChange}\r\n {onclick}\r\n {onfocus}\r\n {onblur}\r\n {onkeydown}\r\n {onkeyup}\r\n aria-label={ariaLabel}\r\n aria-labelledby={ariaLabelledBy}\r\n aria-describedby={ariaDescribedBy}\r\n class={inputClasses}\r\n />\r\n {#if children}\r\n <span class=\"text-sm font-medium text-text-primary\">\r\n {@render children()}\r\n </span>\r\n {/if}\r\n</label>\r\n","/**\n * Storage canonical — SSR-safe, JSON-coded, quota-guarded localStorage /\n * sessionStorage primitives. The ONE sanctioned path for storage access across\n * all four Micdrop apps (R-STORAGE-04/05/06).\n *\n * Consumers NEVER touch `localStorage` / `sessionStorage` / `JSON.parse` /\n * `JSON.stringify` directly — the `no-direct-localstorage` lint rule enforces\n * this and points here. Use, in order of preference:\n * - `createPersistedState` (reactive rune store) for state that drives UI\n * - `getLocalStorage` / `setLocalStorage` / `removeLocalStorage` (imperative kv)\n * - `getSessionStorage` / `setSessionStorage` / `removeSessionStorage`\n * - `clearAppStorageOnLogout` to wipe `clearOnLogout`-flagged keys\n *\n * Every read returns a fallback (never throws on corrupt JSON, R-STORAGE-03);\n * every write is quota-guarded and logged, never crashing (R-STORAGE-06);\n * every access is SSR-safe (R-STORAGE-05).\n */\nimport { logger } from '../logger.js';\n/** Resolve the backend, or null on SSR / private-mode property-access throw. */\nfunction getBackend(kind) {\n if (typeof window === 'undefined')\n return null; // SSR — R-STORAGE-05\n try {\n return kind === 'local' ? window.localStorage : window.sessionStorage;\n }\n catch {\n // Some browsers throw merely accessing the property in private mode.\n return null;\n }\n}\n/**\n * Whether the given storage backend can actually be written to. False on SSR\n * and in Safari Private Mode / quota-exhausted contexts (R-STORAGE-06). Use to\n * gate a degradation banner on high-impact features.\n */\nexport function isStorageAvailable(kind = 'local') {\n const backend = getBackend(kind);\n if (!backend)\n return false;\n const probe = '__micdrop_probe__';\n try {\n backend.setItem(probe, '1');\n backend.removeItem(probe);\n return true;\n }\n catch {\n return false;\n }\n}\n/**\n * Parse a JSON string defensively. Returns `fallback` on null / empty / corrupt\n * input instead of throwing (R-STORAGE-03). The single safe decode entry point\n * for any persisted string — storage values AND cookie values.\n */\nexport function safeJsonParse(raw, fallback) {\n if (raw === null || raw === undefined || raw === '')\n return fallback;\n try {\n return JSON.parse(raw);\n }\n catch {\n return fallback;\n }\n}\nfunction read(kind, key, fallback) {\n const backend = getBackend(kind);\n if (!backend)\n return fallback;\n let raw;\n try {\n raw = backend.getItem(key);\n }\n catch {\n return fallback;\n }\n return safeJsonParse(raw, fallback);\n}\nfunction write(kind, key, value) {\n const backend = getBackend(kind);\n if (!backend)\n return false;\n try {\n backend.setItem(key, JSON.stringify(value));\n return true;\n }\n catch (error) {\n // QuotaExceededError / private-mode — log + continue, never crash the page.\n logger.warn(`storage: failed to write \"${key}\"`, error);\n return false;\n }\n}\nfunction remove(kind, key) {\n const backend = getBackend(kind);\n if (!backend)\n return;\n try {\n backend.removeItem(key);\n }\n catch {\n // ignore — removal is best-effort\n }\n}\n/**\n * Read a JSON-coded value from localStorage. Returns `fallback` (default `null`)\n * on miss / SSR / corrupt JSON.\n */\nexport function getLocalStorage(key, fallback = null) {\n return read('local', key, fallback);\n}\n/** Write a JSON-coded value to localStorage. Returns false if unavailable (never throws). */\nexport function setLocalStorage(key, value) {\n return write('local', key, value);\n}\n/** Remove a key from localStorage. No-op on SSR. */\nexport function removeLocalStorage(key) {\n remove('local', key);\n}\n/**\n * Read a JSON-coded value from sessionStorage. Returns `fallback` (default\n * `null`) on miss / SSR / corrupt JSON.\n */\nexport function getSessionStorage(key, fallback = null) {\n return read('session', key, fallback);\n}\n/** Write a JSON-coded value to sessionStorage. Returns false if unavailable (never throws). */\nexport function setSessionStorage(key, value) {\n return write('session', key, value);\n}\n/** Remove a key from sessionStorage. No-op on SSR. */\nexport function removeSessionStorage(key) {\n remove('session', key);\n}\n","/**\n * Storage key registry + canonical namespace (R-STORAGE-01/02/08).\n *\n * Every persisted key is colon-namespaced `micdrop:<feature>:<detail>` and\n * registered with its lifecycle flags so a single `clearAppStorageOnLogout()`\n * can walk the registry instead of every logout flow hand-clearing keys.\n */\nimport { removeLocalStorage, removeSessionStorage, } from './storage.js';\nconst NAMESPACE_RE = /^micdrop:[a-z0-9]+(:[a-z0-9-]+)+$/;\n/**\n * Build a canonical colon-namespaced storage key `micdrop:<feature>:<detail...>`\n * (R-STORAGE-02). Lowercases tokens and throws on empty / malformed segments so\n * a bad key fails at the builder, not silently at runtime.\n *\n * @example buildStorageKey('checkout', 'state', eventId) // micdrop:checkout:state:<id>\n */\nexport function buildStorageKey(feature, ...detail) {\n const segments = [feature, ...detail.map(String)]\n .map((s) => s.trim().toLowerCase())\n .filter((s) => s.length > 0);\n if (segments.length < 2) {\n throw new Error(`buildStorageKey requires a feature and at least one detail segment (got: ${JSON.stringify([feature, ...detail])})`);\n }\n const key = `micdrop:${segments.join(':')}`;\n if (!NAMESPACE_RE.test(key)) {\n throw new Error(`buildStorageKey produced a non-canonical key: \"${key}\"`);\n }\n return key;\n}\n/** Whether a key matches the canonical `micdrop:<feature>:<detail>` namespace. */\nexport function isCanonicalStorageKey(key) {\n return NAMESPACE_RE.test(key);\n}\nconst registry = new Map();\n/**\n * Register a storage key + its lifecycle flags. Idempotent per (kind, key).\n * Returns the key so a caller can `const K = registerStorageKey({...})`.\n */\nexport function registerStorageKey(descriptor) {\n registry.set(`${descriptor.kind}:${descriptor.key}`, descriptor);\n return descriptor.key;\n}\n/** Read-only snapshot of all registered descriptors. */\nexport function getStorageRegistry() {\n return [...registry.values()];\n}\n/**\n * Clear every registered key flagged `clearOnLogout` (R-STORAGE-08). The single\n * canonical logout cleanup — logout flows call this instead of inline clears,\n * so a new persisted key is wiped on logout the moment it declares the flag.\n */\nexport function clearAppStorageOnLogout(options = {}) {\n for (const descriptor of registry.values()) {\n if (!descriptor.clearOnLogout)\n continue;\n if (options.onlyKey && descriptor.key !== options.onlyKey)\n continue;\n if (descriptor.kind === 'local')\n removeLocalStorage(descriptor.key);\n else\n removeSessionStorage(descriptor.key);\n }\n}\n","/**\n * Theme persistence onto the storage canonical (`micdrop:ui:theme`).\n *\n * The theme override is TRINARY: an explicit `'light'` / `'dark'` choice, or its\n * ABSENCE meaning \"auto — follow the system preference\". Absence is load-bearing:\n * removing the key is how `DarkModeToggle` expresses auto mode, so this layer\n * stores ONLY explicit choices and treats a missing key as auto. It never\n * writes a sentinel for auto.\n *\n * Encoding: the storage canonical JSON-codes every value, so the explicit\n * override is persisted JSON-coded (`\"dark\"`) under the canonical key and read\n * back through `getLocalStorage` like any other canonical value.\n *\n * Legacy migration (the codex trap): before the canonical existed, all three\n * theme consumers read/wrote a bare un-namespaced `\"theme\"` key holding a PLAIN\n * string (`dark` / `light`), NOT JSON. That legacy value is unreadable through\n * `getLocalStorage` because `JSON.parse('dark')` throws and falls back to null.\n * `readThemeOverride()` therefore one-time-adopts a legacy bare value into the\n * canonical key (JSON-coded) and deletes the legacy key, so existing users keep\n * their exact theme with zero visual change. The read goes through the shared\n * helper, so whichever of the three consumers mounts first performs the adoption.\n */\nimport { getLocalStorage, setLocalStorage, removeLocalStorage, } from './storage.js';\nimport { registerStorageKey } from './registry.js';\n/**\n * Canonical key for the persisted theme override. Registered (not\n * clearOnLogout — a theme preference outlives the session, matching the\n * pre-canonical behavior where the bare `theme` key was never logout-cleared).\n */\nexport const THEME_STORAGE_KEY = registerStorageKey({\n key: 'micdrop:ui:theme',\n kind: 'local',\n});\n/** Pre-canonical bare key. Held a PLAIN string (`dark` / `light`), not JSON. */\nconst LEGACY_THEME_KEY = 'theme';\n/** Read the legacy bare `theme` value (plain string), SSR/private-mode safe. */\nfunction readLegacyTheme() {\n if (typeof window === 'undefined')\n return null;\n let raw;\n try {\n raw = window.localStorage.getItem(LEGACY_THEME_KEY);\n }\n catch {\n return null;\n }\n return raw === 'dark' || raw === 'light' ? raw : null;\n}\n/** Best-effort removal of the legacy bare key after adoption. */\nfunction removeLegacyTheme() {\n if (typeof window === 'undefined')\n return;\n try {\n window.localStorage.removeItem(LEGACY_THEME_KEY);\n }\n catch {\n // best-effort — removal must never throw\n }\n}\n/**\n * Read the explicit theme override, or `null` for \"auto / follow system\".\n *\n * One-time-migrates a legacy bare `theme` value into the canonical key and\n * deletes the legacy key. Idempotent: once migrated, subsequent calls just read\n * the canonical key. Canonical value always wins over a stale legacy key.\n */\nexport function readThemeOverride() {\n const canonical = getLocalStorage(THEME_STORAGE_KEY);\n const legacy = readLegacyTheme();\n if (canonical === 'dark' || canonical === 'light') {\n // Canonical wins; retire any stale legacy bare key still hanging around.\n if (legacy)\n removeLegacyTheme();\n return canonical;\n }\n if (legacy) {\n // Adopt the legacy choice into the canonical key, then drop the bare key.\n setLocalStorage(THEME_STORAGE_KEY, legacy);\n removeLegacyTheme();\n return legacy;\n }\n return null;\n}\n/** Persist an explicit theme override (`'light'` / `'dark'`). */\nexport function writeThemeOverride(theme) {\n setLocalStorage(THEME_STORAGE_KEY, theme);\n}\n/** Clear the override — back to auto / follow system. Also drops the legacy key. */\nexport function clearThemeOverride() {\n removeLocalStorage(THEME_STORAGE_KEY);\n removeLegacyTheme();\n}\n","\n export const browser = typeof window !== 'undefined';\n export const dev = false;\n export const building = false;\n export const version = '';\n ","/**\n * Stripe Theme Utility\n *\n * Provides reactive dark mode detection for Stripe Elements.\n * Automatically watches for theme changes via MutationObserver.\n *\n * @example\n * ```svelte\n * <script>\n * import { useStripeTheme } from '@getmicdrop/svelte-components/stripe';\n *\n * const stripeTheme = useStripeTheme();\n * // stripeTheme.current is 'stripe' | 'night'\n * // Automatically updates when theme changes\n * </script>\n *\n * <Elements theme={stripeTheme.current} {stripe} clientSecret={paymentIntent}>\n * <PaymentElement />\n * </Elements>\n * ```\n */\nimport { onMount, onDestroy } from 'svelte';\nimport { readThemeOverride } from '../utils/storage/index.js';\n/**\n * Detects if dark mode is active.\n *\n * Checks in order:\n * 1. localStorage 'theme' value\n * 2. Container element classes (.dark, [data-theme=\"dark\"])\n * 3. System preference (prefers-color-scheme)\n */\nexport function detectDarkMode(containerSelector) {\n if (typeof window === 'undefined')\n return false;\n try {\n // Check the persisted theme override first (auto/system → null).\n const saved = readThemeOverride();\n if (saved === 'dark')\n return true;\n if (saved === 'light')\n return false;\n // Check for data-theme on common containers\n const selector = containerSelector || '[data-theme=\"dark\"], [data-theme=\"light\"], .micdrop, [data-theme]';\n const containers = document.querySelectorAll(selector);\n for (const container of containers) {\n if (container.getAttribute('data-theme') === 'dark')\n return true;\n }\n // Check document root\n if (document.documentElement.getAttribute('data-theme') === 'dark')\n return true;\n // Fallback to system preference\n return window.matchMedia?.('(prefers-color-scheme: dark)')?.matches ?? false;\n }\n catch {\n return false;\n }\n}\n/**\n * Maps dark mode state to Stripe theme.\n */\nexport function getStripeTheme(isDark) {\n return isDark ? 'night' : 'stripe';\n}\n/**\n * Creates a reactive Stripe theme that watches for dark mode changes.\n *\n * Returns an object with a `current` property that is reactive and\n * automatically updates when the theme changes.\n *\n * @example\n * ```svelte\n * <script>\n * import { useStripeTheme } from '@getmicdrop/svelte-components/stripe';\n * const stripeTheme = useStripeTheme();\n * </script>\n *\n * <Elements theme={stripeTheme.current} ...>\n * ```\n */\nexport function useStripeTheme(options = {}) {\n const { containerSelector = '[data-theme=\"dark\"], [data-theme=\"light\"], .micdrop, [data-theme]', watchChanges = true, fallback = 'stripe', } = options;\n let current = $state(fallback);\n let observer = null;\n function updateTheme() {\n current = getStripeTheme(detectDarkMode(containerSelector));\n }\n onMount(() => {\n // Initial detection\n updateTheme();\n if (watchChanges && typeof MutationObserver !== 'undefined') {\n // Watch for class changes on potential theme containers\n const containers = document.querySelectorAll(containerSelector);\n const root = document.documentElement;\n observer = new MutationObserver(() => {\n updateTheme();\n });\n // Observe document root\n observer.observe(root, {\n attributes: true,\n attributeFilter: ['class', 'data-theme'],\n });\n // Observe other containers\n containers.forEach(container => {\n observer?.observe(container, {\n attributes: true,\n attributeFilter: ['class', 'data-theme'],\n });\n });\n // Also listen for system preference changes\n const mediaQuery = window.matchMedia('(prefers-color-scheme: dark)');\n mediaQuery.addEventListener('change', updateTheme);\n }\n });\n onDestroy(() => {\n if (observer) {\n observer.disconnect();\n observer = null;\n }\n });\n return {\n get current() {\n return current;\n },\n /** Force refresh the theme detection */\n refresh: updateTheme,\n };\n}\n/**\n * Simple synchronous dark mode detection without reactivity.\n * Use this when you only need a one-time check.\n */\nexport function getInitialStripeTheme(containerSelector) {\n return getStripeTheme(detectDarkMode(containerSelector));\n}\n","<script lang=\"ts\">\r\n import { onMount, onDestroy } from 'svelte';\r\n import { browser } from '$app/environment';\r\n import { ChevronLeft, Warning } from 'carbon-icons-svelte';\r\n import {\r\n Button,\r\n Heading,\r\n Text,\r\n Input,\r\n Checkbox,\r\n isEmailAddress,\r\n } from '@getmicdrop/svelte-components';\r\n import { useStripeTheme } from '@getmicdrop/svelte-components/stripe';\r\n import type {\r\n EventData,\r\n OrderTotals,\r\n CheckoutFormData,\r\n } from '$lib/public-calendar-flow/types';\r\n import CheckoutTimer from './CheckoutTimer.svelte';\r\n\r\n interface Props {\r\n event: EventData;\r\n orderTotals: OrderTotals;\r\n timerSeconds: number;\r\n isExpired: boolean;\r\n isRegistration: boolean;\r\n isFreeOrder: boolean;\r\n promoApplied: boolean;\r\n promoDiscount: number;\r\n onBack: () => void;\r\n onPlaceOrder: (formData: CheckoutFormData) => void;\r\n processing: boolean;\r\n /** Incremented by parent to trigger form validation + submit from external buttons (sidebar, mobile footer) */\r\n submitRequested: number;\r\n /** Stripe instance for production mode (absence = showcase mock inputs) */\r\n stripe?: any;\r\n /** Stripe client secret for PaymentElement */\r\n paymentIntent?: string;\r\n /** Bindable Stripe elements reference for the wrapper to use during payment confirmation */\r\n elements?: any;\r\n /** Payment error message from the wrapper */\r\n paymentError?: string;\r\n labels?: Record<string, string>;\r\n }\r\n\r\n let {\r\n event,\r\n orderTotals,\r\n timerSeconds,\r\n isExpired,\r\n isRegistration,\r\n isFreeOrder,\r\n promoApplied,\r\n promoDiscount,\r\n onBack,\r\n onPlaceOrder,\r\n processing,\r\n submitRequested,\r\n stripe = undefined,\r\n paymentIntent = undefined,\r\n elements = $bindable(undefined),\r\n paymentError = '',\r\n labels = {},\r\n }: Props = $props();\r\n\r\n // Use direct Stripe.js API instead of svelte-stripe (Svelte 5 compatibility bug).\r\n const stripeTheme = browser\r\n ? useStripeTheme()\r\n : { current: 'stripe' as const };\r\n let paymentMountDiv = $state<HTMLElement | null>(null);\r\n let paymentElement: any = null;\r\n // Track which client secret the mounted element is bound to. Re-mounting only\r\n // the Stripe element (not the whole CheckoutForm) when the PaymentIntent changes\r\n // lets a partial gift card swap in a reduced-amount PI WITHOUT wiping the\r\n // buyer's typed name/email/attendee fields (the {#key} approach remounted the\r\n // entire form and blanked them).\r\n let mountedSecret = $state<string | null>(null);\r\n\r\n function mountStripePayment() {\r\n if (!browser || !stripe || !paymentIntent || !paymentMountDiv) return;\r\n // Tear down any element bound to a stale secret before re-creating.\r\n if (paymentElement) {\r\n try {\r\n paymentElement.destroy();\r\n } catch (_) {\r\n // ignore\r\n }\r\n paymentElement = null;\r\n }\r\n elements = stripe.elements({\r\n clientSecret: paymentIntent,\r\n appearance: { theme: stripeTheme.current },\r\n });\r\n paymentElement = elements.create('payment');\r\n paymentElement.mount(paymentMountDiv);\r\n mountedSecret = paymentIntent;\r\n }\r\n\r\n function destroyStripePayment() {\r\n if (paymentElement) {\r\n try {\r\n paymentElement.destroy();\r\n } catch (_) {\r\n // ignore\r\n }\r\n paymentElement = null;\r\n }\r\n mountedSecret = null;\r\n }\r\n\r\n $effect(() => {\r\n // (Re)mount whenever the PaymentIntent secret changes — initial mount, or a\r\n // new reduced-amount PI after a partial gift card is applied.\r\n if (\r\n stripe &&\r\n paymentIntent &&\r\n paymentMountDiv &&\r\n paymentIntent !== mountedSecret\r\n ) {\r\n mountStripePayment();\r\n } else if ((!stripe || !paymentIntent) && mountedSecret) {\r\n // No longer needs Stripe (e.g. a gift card now fully covers the order, so\r\n // the wrapper cleared stripe/paymentIntent and the payment section is\r\n // hidden) — tear down the mounted element instead of orphaning the iframe.\r\n destroyStripePayment();\r\n }\r\n });\r\n\r\n // Re-skin Stripe Elements when the theme changes (data-theme toggle).\r\n $effect(() => {\r\n if (mountedSecret && elements && stripeTheme.current) {\r\n try {\r\n elements.update({ appearance: { theme: stripeTheme.current } });\r\n } catch (_) {\r\n // Stripe Elements is finicky about post-mount updates; ignore.\r\n }\r\n }\r\n });\r\n\r\n onDestroy(() => {\r\n if (paymentElement) {\r\n paymentElement.destroy();\r\n paymentElement = null;\r\n }\r\n mountedSecret = null;\r\n });\r\n\r\n // --- Form state ---\r\n let firstName = $state('');\r\n let lastName = $state('');\r\n let email = $state('');\r\n let keepUpdated = $state(false);\r\n let agreeTerms = $state(false);\r\n\r\n // Payment fields (mock)\r\n let cardNumber = $state('');\r\n let cardExpiry = $state('');\r\n let cardCvc = $state('');\r\n\r\n // Per-ticket attendee fields (for registration events)\r\n let attendees = $state<\r\n Array<{\r\n firstName: string;\r\n lastName: string;\r\n email: string;\r\n sameAsPurchaser: boolean;\r\n }>\r\n >([]);\r\n\r\n // No Stripe charge is needed when the order is free OR fully covered by a gift\r\n // card (total 0 but subtotal > 0, so isFreeOrder alone is false). In both cases\r\n // we must NOT render/validate the showcase mock-card inputs — otherwise a\r\n // fully-gift-card-covered order can't be submitted (the wrapper routes it to\r\n // executeGiftCardOnlyPurchase once onPlaceOrder fires).\r\n let noPaymentRequired = $derived(isFreeOrder || orderTotals.total === 0);\r\n\r\n // Validation state\r\n let submitted = $state(false);\r\n let errors = $derived(\r\n (() => {\r\n if (!submitted) return {};\r\n const e: Record<string, string> = {};\r\n if (!firstName.trim())\r\n e.firstName = labels.firstNameRequired || 'First name is required';\r\n if (!lastName.trim())\r\n e.lastName = labels.lastNameRequired || 'Last name is required';\r\n if (!email.trim()) e.email = labels.emailRequired || 'Email is required';\r\n else if (!isEmailAddress(email))\r\n e.email = labels.emailInvalid || 'Enter a valid email address';\r\n if (!noPaymentRequired && !stripe) {\r\n // Only validate mock card fields in showcase mode (Stripe absent AND a\r\n // real balance is due — not free / not fully gift-card covered)\r\n if (!cardNumber.trim())\r\n e.cardNumber = labels.cardNumberRequired || 'Card number is required';\r\n if (!cardExpiry.trim())\r\n e.cardExpiry = labels.expiryRequired || 'Expiry date is required';\r\n if (!cardCvc.trim())\r\n e.cardCvc = labels.cvcRequired || 'CVC is required';\r\n }\r\n if (!agreeTerms)\r\n e.agreeTerms = labels.mustAgreeTerms || 'You must agree to the terms';\r\n\r\n // Validate attendee fields for registration events\r\n if (isRegistration) {\r\n for (let i = 0; i < attendees.length; i++) {\r\n const att = attendees[i];\r\n if (!att.sameAsPurchaser) {\r\n if (!att.firstName.trim())\r\n e[`attendee_${i}_firstName`] = 'First name is required';\r\n if (!att.lastName.trim())\r\n e[`attendee_${i}_lastName`] = 'Last name is required';\r\n if (!att.email.trim())\r\n e[`attendee_${i}_email`] =\r\n labels.emailRequired || 'Email is required';\r\n else if (!isEmailAddress(att.email))\r\n e[`attendee_${i}_email`] =\r\n labels.emailInvalidShort || 'Enter a valid email';\r\n }\r\n }\r\n }\r\n\r\n return e;\r\n })()\r\n );\r\n\r\n let hasErrors = $derived(Object.keys(errors).length > 0);\r\n\r\n let isTimerUrgent = $derived(timerSeconds > 0 && timerSeconds <= 120);\r\n\r\n // Initialize attendee list based on order lines\r\n $effect(() => {\r\n if (isRegistration) {\r\n const totalTickets = orderTotals.lines.reduce(\r\n (sum, l) => sum + l.quantity,\r\n 0\r\n );\r\n if (attendees.length !== totalTickets) {\r\n attendees = Array.from({ length: totalTickets }, (_, i) => ({\r\n firstName: '',\r\n lastName: '',\r\n email: '',\r\n sameAsPurchaser: i === 0,\r\n }));\r\n }\r\n }\r\n });\r\n\r\n // Watch for external submit requests (from sidebar/mobile footer buttons)\r\n let lastSubmitRequest = $state(0);\r\n $effect(() => {\r\n if (submitRequested > lastSubmitRequest) {\r\n lastSubmitRequest = submitRequested;\r\n handleSubmit();\r\n }\r\n });\r\n\r\n function handleSameAsPurchaser(index: number, checked: boolean) {\r\n attendees = attendees.map((att, i) => {\r\n if (i !== index) return att;\r\n return {\r\n ...att,\r\n sameAsPurchaser: checked,\r\n firstName: checked ? firstName : att.firstName,\r\n lastName: checked ? lastName : att.lastName,\r\n email: checked ? email : att.email,\r\n };\r\n });\r\n }\r\n\r\n function handleSubmit() {\r\n submitted = true;\r\n if (hasErrors) return;\r\n onPlaceOrder({\r\n firstName,\r\n lastName,\r\n email,\r\n keepMeUpdated: keepUpdated,\r\n attendees: isRegistration\r\n ? attendees.map(a => ({\r\n firstName: a.sameAsPurchaser ? firstName : a.firstName,\r\n lastName: a.sameAsPurchaser ? lastName : a.lastName,\r\n email: a.sameAsPurchaser ? email : a.email,\r\n sameAsPurchaser: a.sameAsPurchaser,\r\n }))\r\n : undefined,\r\n });\r\n }\r\n</script>\r\n\r\n<!-- Form sits on a Boxes (surface-secondary) panel, not the page Background, so\r\n form controls — especially the bare Checkbox boxes — keep a consistent,\r\n readable surface on any themed scheme (matches the order-summary sidebar\r\n Card). The timer-header was already built to be this panel's flush header\r\n (it bleeds to the edges and rounds only its top). -->\r\n<div class=\"checkout-form-panel space-y-6 bg-surface-secondary border-default\">\r\n <!-- Timer header bar -->\r\n <div class=\"timer-header border-default bg-surface-secondary\">\r\n <Button\r\n variant=\"icon\"\r\n size=\"icon-sm\"\r\n onclick={onBack}\r\n aria-label={labels.goBack || 'Go back'}\r\n >\r\n <ChevronLeft size={20} />\r\n </Button>\r\n <CheckoutTimer seconds={timerSeconds} isUrgent={isTimerUrgent} />\r\n </div>\r\n\r\n <!-- Personal details -->\r\n <section class=\"space-y-4\">\r\n <Heading level={2} size=\"lg\" weight=\"semibold\">\r\n {labels.yourDetails || 'Your details'}\r\n </Heading>\r\n\r\n <div class=\"grid grid-cols-1 sm:grid-cols-2 gap-4\">\r\n <Input\r\n label={labels.firstName || 'First name'}\r\n placeholder={labels.firstName || 'First name'}\r\n required\r\n bind:value={firstName}\r\n errorText={errors.firstName ?? ''}\r\n color={errors.firstName ? 'red' : 'base'}\r\n />\r\n <Input\r\n label={labels.lastName || 'Last name'}\r\n placeholder={labels.lastName || 'Last name'}\r\n required\r\n bind:value={lastName}\r\n errorText={errors.lastName ?? ''}\r\n color={errors.lastName ? 'red' : 'base'}\r\n />\r\n </div>\r\n\r\n <Input\r\n label={labels.emailAddress || 'Email address'}\r\n type=\"email\"\r\n placeholder={labels.emailPlaceholder || 'you@example.com'}\r\n required\r\n bind:value={email}\r\n errorText={errors.email ?? ''}\r\n color={errors.email ? 'red' : 'base'}\r\n />\r\n\r\n <Checkbox bind:checked={keepUpdated}>\r\n {labels.keepMeUpdated ||\r\n 'Keep me updated about this event and similar events'}\r\n </Checkbox>\r\n </section>\r\n\r\n <!-- Per-ticket attendee details (registration events) -->\r\n {#if isRegistration && attendees.length > 0}\r\n <section class=\"space-y-5\">\r\n <Heading level={2} size=\"lg\" weight=\"semibold\">\r\n {labels.attendeeDetails || 'Attendee details'}\r\n </Heading>\r\n\r\n {#each attendees as attendee, i}\r\n <div class=\"attendee-card border-default bg-surface-secondary\">\r\n <div class=\"flex items-center justify-between mb-3\">\r\n <Text size=\"sm\" class=\"font-semibold\">\r\n {labels.attendee || 'Attendee'}\r\n {i + 1}\r\n </Text>\r\n <Checkbox\r\n checked={attendee.sameAsPurchaser}\r\n onchange={({ checked }) => handleSameAsPurchaser(i, checked)}\r\n >\r\n {labels.sameAsPurchaser || 'Same as purchaser'}\r\n </Checkbox>\r\n </div>\r\n\r\n <div class=\"space-y-3\">\r\n <div class=\"grid grid-cols-1 sm:grid-cols-2 gap-3\">\r\n <Input\r\n label={labels.firstName || 'First name'}\r\n placeholder={labels.firstName || 'First name'}\r\n required\r\n disabled={attendee.sameAsPurchaser}\r\n bind:value={attendee.firstName}\r\n errorText={!attendee.sameAsPurchaser\r\n ? (errors[`attendee_${i}_firstName`] ?? '')\r\n : ''}\r\n color={!attendee.sameAsPurchaser &&\r\n errors[`attendee_${i}_firstName`]\r\n ? 'red'\r\n : 'base'}\r\n />\r\n <Input\r\n label={labels.lastName || 'Last name'}\r\n placeholder={labels.lastName || 'Last name'}\r\n required\r\n disabled={attendee.sameAsPurchaser}\r\n bind:value={attendee.lastName}\r\n errorText={!attendee.sameAsPurchaser\r\n ? (errors[`attendee_${i}_lastName`] ?? '')\r\n : ''}\r\n color={!attendee.sameAsPurchaser &&\r\n errors[`attendee_${i}_lastName`]\r\n ? 'red'\r\n : 'base'}\r\n />\r\n </div>\r\n <Input\r\n label={labels.email || 'Email'}\r\n type=\"email\"\r\n placeholder={labels.attendeeEmailPlaceholder ||\r\n 'attendee@example.com'}\r\n required\r\n disabled={attendee.sameAsPurchaser}\r\n bind:value={attendee.email}\r\n errorText={!attendee.sameAsPurchaser\r\n ? (errors[`attendee_${i}_email`] ?? '')\r\n : ''}\r\n color={!attendee.sameAsPurchaser && errors[`attendee_${i}_email`]\r\n ? 'red'\r\n : 'base'}\r\n />\r\n </div>\r\n </div>\r\n {/each}\r\n </section>\r\n {/if}\r\n\r\n <!-- Payment section -->\r\n <section class=\"space-y-4\">\r\n {#if noPaymentRequired}\r\n <div class=\"free-order-notice border-default bg-surface-secondary\">\r\n <Text size=\"sm\" color=\"secondary\" class=\"font-medium block\">\r\n {labels.noPaymentRequired || 'No payment required'}\r\n </Text>\r\n <Text size=\"xs\" color=\"muted\" class=\"block mt-1\">\r\n {labels.freeEventMessage ||\r\n 'This is a free event. Complete the form above to reserve your spot.'}\r\n </Text>\r\n </div>\r\n {:else if stripe && paymentIntent}\r\n <!-- Production: Real Stripe PaymentElement -->\r\n <Heading level={2} size=\"lg\" weight=\"semibold\">Payment</Heading>\r\n {#if browser}\r\n <div bind:this={paymentMountDiv}></div>\r\n {#if !mountedSecret}\r\n <p class=\"text-sm text-color-muted py-2\">\r\n {labels.loadingPaymentForm || 'Loading secure payment form...'}\r\n </p>\r\n {/if}\r\n {:else}\r\n <p class=\"text-sm text-color-muted py-2\">\r\n {labels.loadingPaymentForm || 'Loading secure payment form...'}\r\n </p>\r\n {/if}\r\n {#if paymentError}\r\n <p class=\"text-sm text-accent-danger mt-2\">{paymentError}</p>\r\n {/if}\r\n {:else}\r\n <!-- Showcase: Mock card inputs -->\r\n <Heading level={2} size=\"lg\" weight=\"semibold\">Payment</Heading>\r\n\r\n <Input\r\n label={labels.cardNumber || 'Card number'}\r\n placeholder={labels.cardPlaceholder || '4242 4242 4242 4242'}\r\n type=\"creditCardNumber\"\r\n required\r\n bind:value={cardNumber}\r\n errorText={errors.cardNumber ?? ''}\r\n color={errors.cardNumber ? 'red' : 'base'}\r\n />\r\n\r\n <div class=\"grid grid-cols-2 gap-4\">\r\n <Input\r\n label={labels.expiryDate || 'Expiry date'}\r\n placeholder={labels.expiryPlaceholder || 'MM / YY'}\r\n required\r\n bind:value={cardExpiry}\r\n errorText={errors.cardExpiry ?? ''}\r\n color={errors.cardExpiry ? 'red' : 'base'}\r\n />\r\n <Input\r\n label={labels.cvc || 'CVC'}\r\n placeholder={labels.cvcPlaceholder || '123'}\r\n required\r\n maxlength={4}\r\n bind:value={cardCvc}\r\n errorText={errors.cardCvc ?? ''}\r\n color={errors.cardCvc ? 'red' : 'base'}\r\n />\r\n </div>\r\n {#if paymentError}\r\n <p class=\"text-sm text-accent-danger mt-2\">{paymentError}</p>\r\n {/if}\r\n {/if}\r\n </section>\r\n\r\n <!-- Terms -->\r\n <section class=\"space-y-2\">\r\n <Checkbox bind:checked={agreeTerms}>\r\n {labels.agreeToTerms || 'I agree to the terms and conditions'}\r\n </Checkbox>\r\n {#if errors.agreeTerms}\r\n <div class=\"flex items-center gap-1.5\">\r\n <Warning size={16} class=\"text-accent-danger shrink-0\" />\r\n <Text size=\"xs\" class=\"text-accent-danger\">{errors.agreeTerms}</Text>\r\n </div>\r\n {/if}\r\n </section>\r\n\r\n <!-- Mobile submit is handled by CheckoutMobileFooter -->\r\n</div>\r\n\r\n<style>\r\n /* Boxes-coloured panel the whole form sits on. 1rem padding pairs with the\r\n timer-header's -1rem bleed so the header spans the panel's full width and\r\n caps its top. rounded/border match the order-summary sidebar Card. */\r\n .checkout-form-panel {\r\n padding: 1rem;\r\n border-radius: 0.5rem;\r\n border-width: 1px;\r\n border-style: solid;\r\n overflow: hidden;\r\n }\r\n\r\n .timer-header {\r\n display: flex;\r\n align-items: center;\r\n gap: 0.75rem;\r\n padding: 0.75rem 1rem;\r\n margin-top: -1rem;\r\n margin-left: -1rem;\r\n margin-right: -1rem;\r\n border-bottom-width: 1px;\r\n border-bottom-style: solid;\r\n border-radius: 0.5rem 0.5rem 0 0;\r\n }\r\n\r\n .attendee-card {\r\n padding: 1rem;\r\n border-radius: 0.5rem;\r\n border-width: 1px;\r\n border-style: solid;\r\n }\r\n\r\n .free-order-notice {\r\n padding: 1rem;\r\n border-radius: 0.5rem;\r\n border-width: 1px;\r\n border-style: solid;\r\n }\r\n</style>\r\n","<script lang=\"ts\">\r\n\timport { Calendar } from 'carbon-icons-svelte';\r\n\timport type { USD } from '@getmicdrop/svelte-components';\r\n\timport { Button, Card, Image, Text, formatCurrency } from '@getmicdrop/svelte-components';\r\n\timport type { EventData, OrderTotals } from '$lib/public-calendar-flow/types';\r\n\timport { formatBrowseDate, formatEventTime } from '$lib/public-calendar-flow/defaults';\r\n\timport { PLACEHOLDER_IMAGE } from '$lib/utils/constants.js';\r\n\timport OrderSummary from './OrderSummary.svelte';\r\n\r\n\tinterface Props {\r\n\t\tevent: EventData;\r\n\t\torderTotals: OrderTotals;\r\n\t\tisFreeOrder: boolean;\r\n\t\tonPlaceOrder: () => void;\r\n\t\tprocessing: boolean;\r\n\t\t\tlabels?: Record<string, string>;\r\n\t}\r\n\r\n\tlet { event, orderTotals, isFreeOrder, onPlaceOrder, processing, labels = {}, }: Props = $props();\r\n</script>\r\n\r\n<!--\r\n\tCap the sidebar card to the viewport so the Place Order button never\r\n\tfalls below the fold when many ticket types + fees + taxes + gift-card\r\n\tlines stack up. Three sections, top-to-bottom:\r\n\t 1. Event info — pinned at top so the buyer always sees what they're\r\n\t paying for.\r\n\t 2. Ticket-line list — scrolls when capacity exceeds viewport.\r\n\t 3. Totals + Place Order button — pinned at bottom so summary + CTA\r\n\t are always visible.\r\n-->\r\n<Card border padding={false} class=\"flex flex-col max-h-[calc(100vh-1.5rem)] rounded-xl shadow-lg overflow-hidden bg-surface-secondary\">\r\n\t<!-- Pinned: event info -->\r\n\t<div class=\"p-5 shrink-0\">\r\n\t\t<div class=\"flex gap-3\">\r\n\t\t\t<div class=\"shrink-0 w-20 h-14 rounded-lg overflow-hidden bg-surface-tertiary\">\r\n\t\t\t\t<Image\r\n\t\t\t\t\tsrc={event.imageUrl}\r\n\t\t\t\t\talt={event.title}\r\n\t\t\t\t\tfallback={PLACEHOLDER_IMAGE}\r\n\t\t\t\t\tfit=\"cover\"\r\n\t\t\t\t\tclass=\"w-full h-full\"\r\n\t\t\t\t/>\r\n\t\t\t</div>\r\n\t\t\t<div class=\"flex-1 min-w-0\">\r\n\t\t\t\t<Text size=\"sm\" class=\"font-semibold block leading-tight line-clamp-2\">\r\n\t\t\t\t\t{event.title}\r\n\t\t\t\t</Text>\r\n\t\t\t\t<div class=\"flex items-center gap-1.5 mt-1\">\r\n\t\t\t\t\t<Calendar size={16} class=\"shrink-0 icon-muted\" />\r\n\t\t\t\t\t<Text size=\"xs\" color=\"muted\">\r\n\t\t\t\t\t\t{formatBrowseDate(event.startDateTime, event.timezone)}\r\n\t\t\t\t\t\t{#if event.displayStartTime !== false}\r\n\t\t\t\t\t\t\t&middot; {formatEventTime(event.startDateTime, event.timezone)}\r\n\t\t\t\t\t\t{/if}\r\n\t\t\t\t\t</Text>\r\n\t\t\t\t</div>\r\n\t\t\t\t<Text size=\"xs\" color=\"muted\" class=\"block mt-0.5\">\r\n\t\t\t\t\t{event.venue.name}\r\n\t\t\t\t</Text>\r\n\t\t\t</div>\r\n\t\t</div>\r\n\t</div>\r\n\r\n\t<!-- Scrollable: ticket-line items only -->\r\n\t<div class=\"px-5 pb-3 flex-1 overflow-y-auto min-h-0 border-t border-default pt-3\">\r\n\t\t<OrderSummary {orderTotals} mode=\"lines\" {labels} />\r\n\t</div>\r\n\r\n\t<!-- Pinned: totals + Place Order button -->\r\n\t<div class=\"px-5 pt-3 pb-5 shrink-0 border-t border-default space-y-3\">\r\n\t\t<OrderSummary {orderTotals} mode=\"totals\" {labels} />\r\n\t\t<Button\r\n\t\t\tvariant=\"default\"\r\n\t\t\tsize=\"lg\"\r\n\t\t\tclass=\"w-full\"\r\n\t\t\tdisabled={processing}\r\n\t\t\tloading={processing}\r\n\t\t\tonclick={onPlaceOrder}\r\n\t\t>\r\n\t\t\t{#if processing}\r\n\t\t\t\t{labels.placingOrder || 'Placing order...'}\r\n\t\t\t{:else if isFreeOrder}\r\n\t\t\t\t{labels.completeOrder || 'Complete order'} &middot; {labels.free || 'Free'}\r\n\t\t\t{:else}\r\n\t\t\t\t{labels.placeOrder || 'Place order'} &middot; {formatCurrency(orderTotals.total as USD)}\r\n\t\t\t{/if}\r\n\t\t</Button>\r\n\t</div>\r\n</Card>\r\n\r\n","<!-- @style-escape: widget-intentional. -->\r\n<script lang=\"ts\">\r\n\timport { Button, Text, formatCurrency } from '@getmicdrop/svelte-components';\r\n\timport type { USD } from '@getmicdrop/svelte-components';\r\n\timport type { OrderTotals } from '$lib/public-calendar-flow/types';\r\n\r\n\tinterface Props {\r\n\t\torderTotals: OrderTotals;\r\n\t\tisFreeOrder: boolean;\r\n\t\tonPlaceOrder: () => void;\r\n\t\tprocessing: boolean;\r\n\t\t\tlabels?: Record<string, string>;\r\n\t}\r\n\r\n\tlet { orderTotals, isFreeOrder, onPlaceOrder, processing, labels = {}, }: Props = $props();\r\n\r\n\tlet totalTicketCount = $derived(\r\n\t\torderTotals.lines.reduce((sum, l) => sum + l.quantity, 0)\r\n\t);\r\n</script>\r\n\r\n<div\r\n\tclass=\"fixed bottom-0 left-0 right-0 z-40 md:hidden bg-surface-primary border-t border-default shadow-lg\"\r\n\tstyle=\"padding-bottom: env(safe-area-inset-bottom, 0);\"\r\n>\r\n\t<div class=\"flex items-center justify-between gap-3 px-4 py-3\">\r\n\t\t<!-- Left: total info -->\r\n\t\t<div class=\"flex-1 min-w-0\">\r\n\t\t\t<Text size=\"sm\" class=\"font-semibold tabular-nums block\">\r\n\t\t\t\t{#if isFreeOrder}\r\n\t\t\t\t\t{totalTicketCount} {totalTicketCount === 1 ? (labels.ticket || 'ticket') : (labels.tickets || 'tickets')} &middot; Free\r\n\t\t\t\t{:else}\r\n\t\t\t\t\t{formatCurrency(orderTotals.total as USD /* FIXME(canonical-cleanup:toFixed): this cast bypasses the brand boundary; replace with toCents/toUSD at the API-response-transform layer. */)}\r\n\t\t\t\t{/if}\r\n\t\t\t</Text>\r\n\t\t</div>\r\n\r\n\t\t<!-- Right: Place order button -->\r\n\t\t<Button\r\n\t\t\tvariant=\"default\"\r\n\t\t\tsize=\"lg\"\r\n\t\t\tdisabled={processing}\r\n\t\t\tloading={processing}\r\n\t\t\tonclick={onPlaceOrder}\r\n\t\t\tclass=\"shrink-0 min-w-36\"\r\n\t\t>\r\n\t\t\t{#if processing}\r\n\t\t\t\t{labels.placingOrder || 'Placing order...'}\r\n\t\t\t{:else if isFreeOrder}\r\n\t\t\t\t{labels.completeOrder || 'Complete order'}\r\n\t\t\t{:else}\r\n\t\t\t\t{labels.placeOrder || 'Place order'}\r\n\t\t\t{/if}\r\n\t\t</Button>\r\n\t</div>\r\n</div>","<!--\r\n @form-handling-escape: `processing` is a Stripe payment-confirmation\r\n reentrancy guard (`if (processing) return; processing = true` around\r\n confirmPayment), NOT form-submission/validation state. createFormStore\r\n cannot express a payment-intent confirmation flow, and forcing it here\r\n would risk double-charge regressions for zero user-visible benefit.\r\n Same call as PP#193. Per Atlas behavior-form-handling convergence record.\r\n-->\r\n<script module>\r\n import {\r\n type USD,\r\n createLogger,\r\n formatCurrency,\r\n } from '@getmicdrop/svelte-components';\r\n const logger = createLogger('CheckoutMain');\r\n</script>\r\n\r\n<script lang=\"ts\">\r\n import { Warning, Time, Calendar } from 'carbon-icons-svelte';\r\n import {\r\n AppError,\r\n Button,\r\n Heading,\r\n Image,\r\n Modal,\r\n Text,\r\n toTicketId,\r\n roundCurrency,\r\n getCookie,\r\n COOKIE_NAMES,\r\n } from '@getmicdrop/svelte-components';\r\n import { toast } from '@getmicdrop/svelte-components/toast';\r\n import { getSessionStatus } from '$lib/api';\r\n import {\r\n formatBrowseDate,\r\n formatEventTime,\r\n } from '$lib/public-calendar-flow/defaults';\r\n import type {\r\n EventData,\r\n OrderTotals,\r\n OrderLine,\r\n CheckoutConfig,\r\n CheckoutFormData,\r\n } from '$lib/public-calendar-flow/types';\r\n import {\r\n mockEvent,\r\n mockFreeEvent,\r\n mockDonationEvent,\r\n mockRegistrationEvent,\r\n } from '$lib/public-calendar-flow/mock-data';\r\n import { calculateTax } from '$lib/public-calendar-flow/defaults';\r\n import {\r\n dataToggleStore,\r\n type ToggleOption,\r\n } from '$lib/public-calendar-flow/data-toggle-store.svelte';\r\n\r\n import CheckoutForm from './CheckoutForm.svelte';\r\n import CheckoutSidebar from './CheckoutSidebar.svelte';\r\n import CheckoutMobileFooter from './CheckoutMobileFooter.svelte';\r\n import OrderSummary from './OrderSummary.svelte';\r\n import { mergeLabels } from '$lib/public-calendar-flow/i18n/labels.js';\r\n\r\n // mergedLabels initialized after props destructuring below\r\n\r\n // --- DataToggle types ---\r\n type CheckoutVariant =\r\n | 'default'\r\n | 'free'\r\n | 'donation'\r\n | 'registration'\r\n | 'timer_warning'\r\n | 'expired'\r\n | 'promo';\r\n\r\n const checkoutStateOptions: ToggleOption[] = [\r\n { value: 'default', label: 'Standard', color: 'bg-accent-success' },\r\n { value: 'free', label: 'Free', color: 'bg-sky-500' },\r\n { value: 'donation', label: 'Donation', color: 'bg-accent-danger' },\r\n { value: 'registration', label: 'Registration', color: 'bg-brand-primary' },\r\n {\r\n value: 'timer_warning',\r\n label: 'Timer warning',\r\n color: 'bg-status-warning-bg0',\r\n },\r\n { value: 'expired', label: 'Expired', color: 'bg-accent-danger' },\r\n { value: 'promo', label: 'Promo discount', color: 'bg-accent-success' },\r\n ];\r\n\r\n let dataVariant = $derived(dataToggleStore.current as CheckoutVariant);\r\n\r\n // --- Props (dual-mode: showcase vs production) ---\r\n interface Props {\r\n /** Pre-built checkout config for production mode. Absence = showcase. */\r\n config?: CheckoutConfig;\r\n /** Production order submission handler. Receives form data for the wrapper to submit. */\r\n onPlaceOrder?: (formData: CheckoutFormData) => Promise<void>;\r\n /** Production back navigation handler. */\r\n onBack?: () => void;\r\n /** Production retry handler for expired sessions. */\r\n onRetryExpired?: () => void;\r\n /** Production session extension handler. Returns extension result with newExpiryTime and remainingExtensions. */\r\n onExtendSession?: () => Promise<{\r\n newExpiryTime?: string;\r\n remainingExtensions?: number;\r\n } | null | void>;\r\n /** Stripe instance passed through to CheckoutForm */\r\n stripe?: any;\r\n /** Stripe client secret passed through to CheckoutForm */\r\n paymentIntent?: string;\r\n /** Bindable Stripe elements for wrapper to access */\r\n elements?: any;\r\n /** Payment error message to display */\r\n paymentError?: string;\r\n labels?: Record<string, string>;\r\n }\r\n\r\n let {\r\n config: configProp = undefined,\r\n onPlaceOrder: onPlaceOrderProp = undefined,\r\n onBack: onBackProp = undefined,\r\n onRetryExpired: onRetryExpiredProp = undefined,\r\n onExtendSession: onExtendSessionProp = undefined,\r\n stripe = undefined,\r\n paymentIntent = undefined,\r\n elements = $bindable(undefined),\r\n paymentError = '',\r\n labels = {},\r\n }: Props = $props();\r\n\r\n const mergedLabels = $derived(mergeLabels(labels));\r\n let isShowcaseMode = $derived(!configProp);\r\n\r\n $effect(() => {\r\n if (!isShowcaseMode) return;\r\n dataToggleStore.options = checkoutStateOptions;\r\n dataToggleStore.current = 'default';\r\n dataToggleStore.isActive = true;\r\n return () => {\r\n dataToggleStore.isActive = false;\r\n dataToggleStore.options = [];\r\n };\r\n });\r\n\r\n // --- Mock order configurations per variant ---\r\n function buildConfig(variant: CheckoutVariant): CheckoutConfig {\r\n switch (variant) {\r\n case 'free':\r\n return {\r\n event: mockFreeEvent,\r\n lines: [\r\n {\r\n ticketId: toTicketId(201),\r\n ticketName: 'Free Entry',\r\n quantity: 2,\r\n unitPrice: 0,\r\n fee: 0,\r\n subtotal: 0,\r\n },\r\n ],\r\n timerStart: 900,\r\n promoApplied: false,\r\n promoDiscount: 0,\r\n forceExpired: false,\r\n };\r\n\r\n case 'donation':\r\n return {\r\n event: mockDonationEvent,\r\n lines: [\r\n {\r\n ticketId: toTicketId(401),\r\n ticketName: 'Donation',\r\n quantity: 1,\r\n unitPrice: 25,\r\n fee: 0,\r\n subtotal: 25,\r\n },\r\n {\r\n ticketId: toTicketId(402),\r\n ticketName: 'Standard Admission',\r\n quantity: 1,\r\n unitPrice: 30,\r\n fee: 3,\r\n subtotal: 30,\r\n },\r\n ],\r\n timerStart: 900,\r\n promoApplied: false,\r\n promoDiscount: 0,\r\n forceExpired: false,\r\n };\r\n\r\n case 'registration':\r\n return {\r\n event: mockRegistrationEvent,\r\n lines: [\r\n {\r\n ticketId: toTicketId(301),\r\n ticketName: 'Workshop Seat',\r\n quantity: 1,\r\n unitPrice: 75,\r\n fee: 7.5,\r\n subtotal: 75,\r\n },\r\n ],\r\n timerStart: 900,\r\n promoApplied: false,\r\n promoDiscount: 0,\r\n forceExpired: false,\r\n };\r\n\r\n case 'timer_warning':\r\n return {\r\n event: mockEvent,\r\n lines: [\r\n {\r\n ticketId: toTicketId(101),\r\n ticketName: 'General Admission',\r\n quantity: 2,\r\n unitPrice: 25,\r\n fee: 2.5,\r\n subtotal: 50,\r\n },\r\n {\r\n ticketId: toTicketId(102),\r\n ticketName: 'VIP Front Row',\r\n quantity: 1,\r\n unitPrice: 55,\r\n fee: 5.5,\r\n subtotal: 55,\r\n },\r\n ],\r\n timerStart: 33, // Just over 30s — warning modal triggers at 30s\r\n promoApplied: false,\r\n promoDiscount: 0,\r\n forceExpired: false,\r\n };\r\n\r\n case 'expired':\r\n return {\r\n event: mockEvent,\r\n lines: [\r\n {\r\n ticketId: toTicketId(101),\r\n ticketName: 'General Admission',\r\n quantity: 2,\r\n unitPrice: 25,\r\n fee: 2.5,\r\n subtotal: 50,\r\n },\r\n {\r\n ticketId: toTicketId(102),\r\n ticketName: 'VIP Front Row',\r\n quantity: 1,\r\n unitPrice: 55,\r\n fee: 5.5,\r\n subtotal: 55,\r\n },\r\n ],\r\n timerStart: 0,\r\n promoApplied: false,\r\n promoDiscount: 0,\r\n forceExpired: true,\r\n };\r\n\r\n case 'promo':\r\n return {\r\n event: mockEvent,\r\n lines: [\r\n {\r\n ticketId: toTicketId(101),\r\n ticketName: 'General Admission',\r\n quantity: 2,\r\n unitPrice: 25,\r\n fee: 2.5,\r\n subtotal: 50,\r\n },\r\n {\r\n ticketId: toTicketId(102),\r\n ticketName: 'VIP Front Row',\r\n quantity: 1,\r\n unitPrice: 55,\r\n fee: 5.5,\r\n subtotal: 55,\r\n },\r\n ],\r\n timerStart: 900,\r\n promoApplied: true,\r\n promoDiscount: 10,\r\n forceExpired: false,\r\n };\r\n\r\n default:\r\n // Standard checkout\r\n return {\r\n event: mockEvent,\r\n lines: [\r\n {\r\n ticketId: toTicketId(101),\r\n ticketName: 'General Admission',\r\n quantity: 2,\r\n unitPrice: 25,\r\n fee: 2.5,\r\n subtotal: 50,\r\n },\r\n {\r\n ticketId: toTicketId(102),\r\n ticketName: 'VIP Front Row',\r\n quantity: 1,\r\n unitPrice: 55,\r\n fee: 5.5,\r\n subtotal: 55,\r\n },\r\n ],\r\n timerStart: 900,\r\n promoApplied: false,\r\n promoDiscount: 0,\r\n forceExpired: false,\r\n };\r\n }\r\n }\r\n\r\n // --- State ---\r\n let config: CheckoutConfig = $derived(\r\n isShowcaseMode ? buildConfig(dataVariant) : configProp!\r\n );\r\n // Deadline-based timer model.\r\n //\r\n // We track an absolute deadline timestamp (ms since epoch) and compute\r\n // the displayed seconds from `deadlineMs - Date.now()` on each tick.\r\n // This is immune to backgrounded-tab throttling: when the browser\r\n // suspends our setInterval, the displayed value falls behind, but the\r\n // very next tick (or the visibilitychange handler) recomputes from\r\n // wall-clock time and the display jumps to truth. Previously we\r\n // decremented `timerSeconds -= 1`, which drifted arbitrarily far from\r\n // reality whenever the tab was backgrounded (Chrome throttles 1Hz\r\n // intervals to ~1/min after a few minutes, then pauses them entirely).\r\n //\r\n // `timerSeconds` remains an explicit state var (rather than `$derived`)\r\n // because legacy code paths still write to it (legacy demo modes); the\r\n // canonical writer is now the deadline-driven tick effect below.\r\n let deadlineMs = $state<number | null>(null);\r\n let timerSeconds = $state(900);\r\n let isExpired = $state(false);\r\n let processing = $state(false);\r\n let submitRequested = $state(0);\r\n\r\n // Cart expiration warning/expired modal states\r\n let showWarningModal = $state(false);\r\n let showExpiredModal = $state(false);\r\n let warningShown = $state(false);\r\n let isExtendingSession = $state(false);\r\n let remainingExtensions = $state(3);\r\n let canExtend = $state(true);\r\n\r\n const WARNING_THRESHOLD = 30; // Show warning modal at 30 seconds remaining\r\n\r\n // Formatted timer display for the modal\r\n let timerDisplay = $derived(\r\n (() => {\r\n const m = Math.floor(timerSeconds / 60);\r\n const s = timerSeconds % 60;\r\n return `${m.toString().padStart(2, '0')}:${s.toString().padStart(2, '0')}`;\r\n })()\r\n );\r\n\r\n // Order totals — the server's payment-intent response is the source\r\n // of truth (it's what Stripe will actually charge). Fall back to a\r\n // local estimate only when the server hasn't responded yet (showcase\r\n // mode, or while the initial /payment-intent is in flight) so the\r\n // skeleton has something to render.\r\n let orderTotals: OrderTotals = $derived(\r\n (() => {\r\n const lines = config.lines;\r\n if (config.serverTotals) {\r\n return {\r\n lines,\r\n subtotal: config.serverTotals.subtotal,\r\n fees: config.serverTotals.fees,\r\n taxes: config.serverTotals.taxes,\r\n total: config.serverTotals.total,\r\n promoDiscount: config.serverTotals.promoDiscount,\r\n giftCardAmount: config.serverTotals.giftCardAmount,\r\n };\r\n }\r\n const subtotal = roundCurrency(\r\n lines.reduce((sum, l) => sum + l.subtotal, 0)\r\n );\r\n const fees = roundCurrency(\r\n lines.reduce((sum, l) => sum + l.fee * l.quantity, 0)\r\n );\r\n const promoDiscount = config.promoApplied ? config.promoDiscount : 0;\r\n const adjustedSubtotal = roundCurrency(\r\n Math.max(0, subtotal - promoDiscount)\r\n );\r\n const taxes = calculateTax(\r\n adjustedSubtotal + fees,\r\n config.event.venue.taxPercentage\r\n );\r\n return {\r\n lines,\r\n subtotal,\r\n fees,\r\n taxes,\r\n total: roundCurrency(adjustedSubtotal + fees + taxes),\r\n promoDiscount,\r\n giftCardAmount: 0,\r\n };\r\n })()\r\n );\r\n\r\n // Disable Place-order until the server has confirmed the totals.\r\n // Showing \"Place order · $55\" while the actual Stripe charge will be\r\n // $63.55 is worse than briefly disabling the button. Free orders\r\n // never call the payment-intent endpoint, so they're exempt.\r\n // Showcase mode has no server, so its mock totals are final — treat them as\r\n // locked, otherwise the Place-order button is perpetually \"Placing order…\".\r\n let totalsLocked = $derived(isShowcaseMode || !!config.serverTotals);\r\n\r\n let isFreeOrder = $derived(\r\n orderTotals.total === 0 && orderTotals.subtotal === 0\r\n );\r\n let isRegistration = $derived(config.event.isRegistrationEvent);\r\n\r\n // Set timer ONCE on initial mount — not on every config change.\r\n // Config changes frequently (Stripe loads, quantities update, etc.)\r\n // and must not reset the timer.\r\n let timerInitialized = false;\r\n $effect(() => {\r\n const c = config;\r\n if (!timerInitialized) {\r\n timerInitialized = true;\r\n // Convert the incoming \"seconds from now\" into an absolute deadline.\r\n // From this point on, deadlineMs is the source of truth.\r\n deadlineMs = Date.now() + c.timerStart * 1000;\r\n timerSeconds = c.timerStart;\r\n isExpired = c.forceExpired;\r\n }\r\n // These are safe to reset on config change (UI state, not timer)\r\n processing = false;\r\n submitRequested = 0;\r\n });\r\n\r\n function getCartIdFromCookie(): string | null {\r\n if (typeof document === 'undefined') return null;\r\n return getCookie(COOKIE_NAMES.CHECKOUT_CARTID);\r\n }\r\n\r\n function recomputeFromDeadline(): number {\r\n if (deadlineMs == null) return timerSeconds;\r\n return Math.max(0, Math.floor((deadlineMs - Date.now()) / 1000));\r\n }\r\n\r\n /**\r\n * Re-sync the deadline from the orders-service. Called on tab focus —\r\n * a backgrounded tab might have missed an extension granted in another\r\n * tab, or the server may have already expired the cart while we slept.\r\n *\r\n * Silent on any error: the existing deadlineMs continues to drive the\r\n * display, which is no worse than the pre-resync state.\r\n */\r\n async function resyncDeadlineFromServer(): Promise<void> {\r\n const cartId = getCartIdFromCookie();\r\n if (!cartId) return;\r\n try {\r\n const status = await getSessionStatus(cartId);\r\n if (status.notFound) {\r\n // Server has already released the reservation. Surface the\r\n // expired modal so the user isn't staring at a phantom timer.\r\n deadlineMs = Date.now();\r\n timerSeconds = 0;\r\n isExpired = true;\r\n showWarningModal = false;\r\n showExpiredModal = true;\r\n return;\r\n }\r\n if (status.expiresAt) {\r\n deadlineMs = new Date(status.expiresAt).getTime();\r\n timerSeconds = recomputeFromDeadline();\r\n }\r\n } catch {\r\n /* network error — keep current deadlineMs */\r\n }\r\n }\r\n\r\n // Timer countdown with warning and expired modal triggers.\r\n //\r\n // The interval ticks at 1 Hz, but its job is to *recompute* timerSeconds\r\n // from deadlineMs, not to decrement a counter. Backgrounded tabs throttle\r\n // this interval — that's fine; when the tab regains focus the\r\n // visibilitychange handler runs the recompute immediately and the display\r\n // jumps to truth.\r\n $effect(() => {\r\n if (isExpired || timerSeconds <= 0) {\r\n if (timerSeconds <= 0 && !isExpired) {\r\n isExpired = true;\r\n showWarningModal = false;\r\n showExpiredModal = true;\r\n }\r\n return;\r\n }\r\n\r\n const tick = () => {\r\n const next = recomputeFromDeadline();\r\n timerSeconds = next;\r\n\r\n // Persist deadline to localStorage every 10 displayed seconds.\r\n // (Storing the deadline directly — not a recomputed end-time —\r\n // keeps reads consistent across tabs.)\r\n if (\r\n next % 10 === 0 &&\r\n deadlineMs != null &&\r\n typeof localStorage !== 'undefined'\r\n ) {\r\n try {\r\n const cartId = getCartIdFromCookie();\r\n if (cartId) {\r\n localStorage.setItem(\r\n // @storage-escape: checkout-deadline timer persistence (intentional raw localStorage)\r\n `checkout-expiry-${cartId}`,\r\n String(deadlineMs)\r\n );\r\n }\r\n } catch {\r\n /* ignore */\r\n }\r\n }\r\n\r\n // Show warning modal at 30 seconds remaining (only once per session).\r\n // We compare with <= because a backgrounded tab may have skipped over\r\n // the exact === WARNING_THRESHOLD boundary while throttled.\r\n if (next <= WARNING_THRESHOLD && next > 0 && !warningShown) {\r\n showWarningModal = true;\r\n warningShown = true;\r\n }\r\n\r\n if (next <= 0) {\r\n isExpired = true;\r\n showWarningModal = false;\r\n showExpiredModal = true;\r\n clearInterval(interval);\r\n }\r\n };\r\n\r\n const interval = setInterval(tick, 1000);\r\n\r\n // Recompute immediately whenever the tab becomes visible — a\r\n // throttled/paused interval may have left the display many minutes\r\n // behind reality. Also kick a server resync to catch the case where\r\n // the reservation already expired (or was extended in another tab).\r\n const onVisible = () => {\r\n if (\r\n typeof document !== 'undefined' &&\r\n document.visibilityState === 'visible'\r\n ) {\r\n tick();\r\n void resyncDeadlineFromServer();\r\n }\r\n };\r\n if (typeof document !== 'undefined') {\r\n document.addEventListener('visibilitychange', onVisible);\r\n }\r\n if (typeof window !== 'undefined') {\r\n window.addEventListener('pageshow', onVisible);\r\n }\r\n\r\n // Initial server resync — catches the case where the user landed on\r\n // checkout with a stale cookie whose cart has already been released.\r\n void resyncDeadlineFromServer();\r\n\r\n return () => {\r\n clearInterval(interval);\r\n if (typeof document !== 'undefined') {\r\n document.removeEventListener('visibilitychange', onVisible);\r\n }\r\n if (typeof window !== 'undefined') {\r\n window.removeEventListener('pageshow', onVisible);\r\n }\r\n };\r\n });\r\n\r\n // --- Handlers ---\r\n function handleBack() {\r\n if (onBackProp) {\r\n onBackProp();\r\n } else {\r\n toast.info('Back button pressed — would navigate to ticket selection');\r\n }\r\n }\r\n\r\n /** Called from sidebar/mobile footer — signals the form to validate + submit */\r\n function requestPlaceOrder() {\r\n submitRequested += 1;\r\n }\r\n\r\n /** Called by CheckoutForm after successful validation */\r\n function handlePlaceOrder(formData: CheckoutFormData) {\r\n if (processing) return;\r\n processing = true;\r\n\r\n if (onPlaceOrderProp) {\r\n // Production mode: delegate to wrapper with form data\r\n onPlaceOrderProp(formData)\r\n .catch(() => {\r\n // Error handling is done by the wrapper; just reset processing\r\n })\r\n .finally(() => {\r\n processing = false;\r\n });\r\n return;\r\n }\r\n\r\n // Showcase mode: existing mock delay\r\n setTimeout(() => {\r\n processing = false;\r\n const count = orderTotals.lines.reduce((s, l) => s + l.quantity, 0);\r\n toast.success(\r\n `Order placed! ${count} ticket(s) for ${formatCurrency(orderTotals.total as USD /* FIXME(canonical-cleanup:toFixed): this cast bypasses the brand boundary; replace with toCents/toUSD at the API-response-transform layer. */)}`\r\n );\r\n }, 1500);\r\n }\r\n\r\n function handleRetryExpired() {\r\n showExpiredModal = false;\r\n if (onRetryExpiredProp) {\r\n onRetryExpiredProp();\r\n } else {\r\n toast.info('Would navigate back to ticket selection to start over');\r\n }\r\n }\r\n\r\n /** Extend the checkout session via the parent wrapper's onExtendSession callback */\r\n async function handleExtendSession() {\r\n if (!canExtend || isExtendingSession) return;\r\n isExtendingSession = true;\r\n\r\n try {\r\n if (onExtendSessionProp) {\r\n const result = await onExtendSessionProp();\r\n // The parent wrapper (Checkout.svelte) returns `null` when the\r\n // extend-session API call fails — e.g. when the reservation\r\n // has already expired server-side past its grace window, or\r\n // the user has used all 3 extensions. Treat null/undefined as\r\n // an unambiguous failure: don't reset the timer to +15min\r\n // locally (that would falsely show the user they still have a\r\n // valid cart when the server has already released the tickets).\r\n if (result == null) {\r\n throw new AppError(\r\n 'Extension request returned no result',\r\n 'lib/premium-ticket-experience/variants/v3-airbnb-split/CheckoutMain/handleExtendSession'\r\n );\r\n }\r\n if (typeof result === 'object') {\r\n if (result.newExpiryTime) {\r\n // Server gave us an absolute new expiry — use it directly.\r\n deadlineMs = new Date(result.newExpiryTime).getTime();\r\n } else {\r\n // Fallback: add 15 minutes from now\r\n deadlineMs = Date.now() + 15 * 60 * 1000;\r\n }\r\n timerSeconds = recomputeFromDeadline();\r\n if (result.remainingExtensions !== undefined) {\r\n remainingExtensions = result.remainingExtensions;\r\n canExtend = remainingExtensions > 0;\r\n }\r\n } else {\r\n // Truthy non-object: legacy callers that just return `true`\r\n // on success. Treat as a simple 15-minute extension.\r\n deadlineMs = Date.now() + 15 * 60 * 1000;\r\n timerSeconds = recomputeFromDeadline();\r\n }\r\n } else {\r\n // Showcase mode: just add 15 minutes\r\n deadlineMs = Date.now() + 15 * 60 * 1000;\r\n timerSeconds = recomputeFromDeadline();\r\n }\r\n\r\n isExpired = false;\r\n warningShown = false;\r\n showWarningModal = false;\r\n showExpiredModal = false;\r\n } catch (err) {\r\n logger.error('Failed to extend session:', err);\r\n // Extension failed — close the warning but keep the expired\r\n // modal open so the user is offered \"Select tickets again\".\r\n // Disable further extend attempts on this dead cart.\r\n showWarningModal = false;\r\n canExtend = false;\r\n if (timerSeconds <= 0) {\r\n isExpired = true;\r\n showExpiredModal = true;\r\n }\r\n } finally {\r\n isExtendingSession = false;\r\n }\r\n }\r\n\r\n /** Dismiss warning modal without extending (user will hurry) */\r\n function dismissWarning() {\r\n showWarningModal = false;\r\n }\r\n</script>\r\n\r\n<div class=\"w-full min-h-screen bg-surface-primary text-color-primary\">\r\n <div class=\"max-w-6xl mx-auto px-4 py-6 md:px-6 md:py-8\">\r\n <!-- Single CheckoutForm shared between desktop and mobile layouts.\r\n\t\t Previously there were two instances (one hidden md:grid, one md:hidden)\r\n\t\t which caused duplicate Stripe elements and broken form submission. -->\r\n <div class=\"max-w-6xl\">\r\n <!-- Desktop: two-column grid; Mobile: single column -->\r\n <div class=\"md:grid md:grid-cols-booking-split gap-8 items-start\">\r\n <!-- LEFT COLUMN (or full width on mobile): Checkout form -->\r\n <div class=\"pb-24 md:pb-0\">\r\n <!-- CheckoutForm re-mounts ONLY the Stripe element internally when\r\n `paymentIntent` changes (e.g. a partial gift card swaps in a\r\n reduced-amount PI), so Elements always binds to the current secret\r\n WITHOUT remounting the whole form and wiping the buyer's typed\r\n fields. (An earlier {#key paymentIntent} here blanked the form.) -->\r\n <CheckoutForm\r\n event={config.event}\r\n {orderTotals}\r\n {timerSeconds}\r\n {isExpired}\r\n {isRegistration}\r\n {isFreeOrder}\r\n promoApplied={config.promoApplied}\r\n promoDiscount={config.promoDiscount}\r\n onBack={handleBack}\r\n onPlaceOrder={handlePlaceOrder}\r\n {processing}\r\n {submitRequested}\r\n {stripe}\r\n {paymentIntent}\r\n bind:elements\r\n {paymentError}\r\n labels={mergedLabels}\r\n />\r\n\r\n <!-- Inline order summary for mobile (below form). Lead with the\r\n event header (poster + title + date + venue) so the buyer\r\n has visual confirmation of what they're paying for as they\r\n approach the sticky Place order button. Mirrors the desktop\r\n CheckoutSidebar layout. -->\r\n <section\r\n class=\"md:hidden mobile-order-summary border-default bg-surface-secondary mt-6\"\r\n >\r\n <div class=\"flex gap-3 pb-3 mb-3 border-b border-default\">\r\n <div\r\n class=\"shrink-0 w-20 h-14 rounded-lg overflow-hidden bg-surface-tertiary\"\r\n >\r\n <Image\r\n src={config.event.imageUrl}\r\n alt={config.event.title}\r\n class=\"w-full h-full\"\r\n />\r\n </div>\r\n <div class=\"flex-1 min-w-0\">\r\n <Text\r\n size=\"sm\"\r\n class=\"font-semibold block leading-tight line-clamp-2\"\r\n >\r\n {config.event.title}\r\n </Text>\r\n <div class=\"flex items-center gap-1.5 mt-1\">\r\n <Calendar size={16} class=\"shrink-0 icon-muted\" />\r\n <Text size=\"xs\" color=\"muted\">\r\n {formatBrowseDate(\r\n config.event.startDateTime,\r\n config.event.timezone\r\n )}\r\n {#if config.event.displayStartTime !== false}\r\n &middot; {formatEventTime(\r\n config.event.startDateTime,\r\n config.event.timezone\r\n )}\r\n {/if}\r\n </Text>\r\n </div>\r\n <Text size=\"xs\" color=\"muted\" class=\"block mt-0.5\">\r\n {config.event.venue.name}\r\n </Text>\r\n </div>\r\n </div>\r\n <OrderSummary {orderTotals} labels={mergedLabels} />\r\n </section>\r\n </div>\r\n\r\n <!-- RIGHT COLUMN: Sticky order summary sidebar (desktop only) -->\r\n <div class=\"hidden md:block sticky top-6\">\r\n <CheckoutSidebar\r\n event={config.event}\r\n {orderTotals}\r\n {isFreeOrder}\r\n onPlaceOrder={requestPlaceOrder}\r\n processing={processing || (!isFreeOrder && !totalsLocked)}\r\n labels={mergedLabels}\r\n />\r\n </div>\r\n </div>\r\n </div>\r\n\r\n <!-- Mobile: Fixed bottom bar -->\r\n <div class=\"md:hidden\">\r\n <CheckoutMobileFooter\r\n {orderTotals}\r\n {isFreeOrder}\r\n onPlaceOrder={requestPlaceOrder}\r\n processing={processing || (!isFreeOrder && !totalsLocked)}\r\n labels={mergedLabels}\r\n />\r\n </div>\r\n </div>\r\n\r\n <!-- Session Warning Modal (30 seconds remaining).\r\n Wrapped in `{#if}` (instead of relying on `bind:open` alone) so\r\n the Modal fully mounts/unmounts each time showWarningModal\r\n transitions. Surfaced by a wall-clock test where, after the\r\n user clicked \"Continue checkout\" on a first warning, the Modal\r\n component's internal open-state never reset cleanly — on the\r\n NEXT 30s threshold the timer text in DOM updated but the modal\r\n stayed visually hidden, so the user lost the cart with no\r\n chance to re-extend. Forcing remount with #if sidesteps that. -->\r\n {#if showWarningModal}\r\n <Modal\r\n open={true}\r\n oncancel={() => (showWarningModal = false)}\r\n size=\"sm\"\r\n aria-labelledby=\"warning-modal-title\"\r\n >\r\n {#snippet body()}\r\n <div class=\"timer-modal\" id=\"warning-modal-description\">\r\n <div class=\"timer-modal-icon warning\">\r\n <Warning size={32} />\r\n </div>\r\n <h3 id=\"warning-modal-title\" class=\"timer-modal-title\">\r\n {mergedLabels.sessionExpiringSoon || 'Session expiring soon'}\r\n </h3>\r\n <p class=\"timer-modal-text\">\r\n {(\r\n mergedLabels.sessionExpiringSoonMessage ||\r\n 'Your checkout session will expire in {time}. Your selected tickets will be released back to inventory.'\r\n ).replace('{time}', timerDisplay)}\r\n </p>\r\n {#if canExtend && remainingExtensions > 0}\r\n <p class=\"timer-modal-hint\">\r\n {(\r\n mergedLabels.extensionsRemaining ||\r\n '{count} extension{plural} remaining'\r\n )\r\n .replace('{count}', String(remainingExtensions))\r\n .replace('{plural}', remainingExtensions === 1 ? '' : 's')}\r\n </p>\r\n {/if}\r\n </div>\r\n {/snippet}\r\n {#snippet footer()}\r\n <div class=\"timer-modal-actions\">\r\n {#if canExtend}\r\n <Button\r\n variant=\"default\"\r\n size=\"lg\"\r\n class=\"w-full\"\r\n onclick={handleExtendSession}\r\n disabled={isExtendingSession}\r\n loading={isExtendingSession}\r\n >\r\n {isExtendingSession\r\n ? mergedLabels.extendingSession || 'Extending...'\r\n : mergedLabels.continueCheckout || 'Continue checkout'}\r\n </Button>\r\n {:else}\r\n <Button\r\n variant=\"default\"\r\n size=\"lg\"\r\n class=\"w-full\"\r\n onclick={dismissWarning}\r\n >\r\n {mergedLabels.okIllHurry || \"OK, I'll hurry\"}\r\n </Button>\r\n <p class=\"timer-modal-hint\">\r\n {mergedLabels.maxExtensionsReached ||\r\n 'Maximum extensions reached'}\r\n </p>\r\n {/if}\r\n </div>\r\n {/snippet}\r\n </Modal>\r\n {/if}\r\n\r\n <!-- Session Expired Modal (\"Want to keep your tickets?\").\r\n Same `{#if}` wrap as the warning modal — forces a fresh\r\n Modal mount each time so reopening after a previous extend\r\n cycle actually re-displays the dialog. -->\r\n {#if showExpiredModal}\r\n <Modal\r\n open={true}\r\n oncancel={() => (showExpiredModal = false)}\r\n size=\"sm\"\r\n aria-labelledby=\"expired-modal-title\"\r\n >\r\n {#snippet body()}\r\n <div class=\"timer-modal\" id=\"expired-modal-description\">\r\n <div class=\"timer-modal-icon expired\">\r\n <Time size={32} />\r\n </div>\r\n <h3 id=\"expired-modal-title\" class=\"timer-modal-title\">\r\n {canExtend\r\n ? mergedLabels.wantToKeepTickets || 'Want to keep your tickets?'\r\n : mergedLabels.sessionExpired || 'Session expired'}\r\n </h3>\r\n <p class=\"timer-modal-text\">\r\n {canExtend\r\n ? mergedLabels.sessionExpiredMessage ||\r\n 'Your checkout session has expired and your selected tickets have been released.'\r\n : mergedLabels.reservationTimedOut ||\r\n 'Your reservation has timed out. Please select your tickets again to continue.'}\r\n </p>\r\n </div>\r\n {/snippet}\r\n {#snippet footer()}\r\n <div class=\"timer-modal-actions\">\r\n {#if canExtend}\r\n <Button\r\n variant=\"default\"\r\n size=\"lg\"\r\n class=\"w-full\"\r\n onclick={handleExtendSession}\r\n disabled={isExtendingSession}\r\n loading={isExtendingSession}\r\n >\r\n {isExtendingSession\r\n ? mergedLabels.extendingSession || 'Extending...'\r\n : mergedLabels.keepMyTickets || 'Keep my tickets'}\r\n </Button>\r\n <Button\r\n variant=\"alternative\"\r\n size=\"lg\"\r\n class=\"w-full\"\r\n onclick={handleRetryExpired}\r\n >\r\n {mergedLabels.selectTicketsAgain || 'Select tickets again'}\r\n </Button>\r\n {:else}\r\n <Button\r\n variant=\"default\"\r\n size=\"lg\"\r\n class=\"w-full\"\r\n onclick={handleRetryExpired}\r\n >\r\n {mergedLabels.selectTicketsAgain || 'Select tickets again'}\r\n </Button>\r\n {/if}\r\n </div>\r\n {/snippet}\r\n </Modal>\r\n {/if}\r\n</div>\r\n\r\n<style>\r\n /* Mobile order summary */\r\n .mobile-order-summary {\r\n padding: 1rem;\r\n border-radius: 0.5rem;\r\n border-width: 1px;\r\n border-style: solid;\r\n }\r\n\r\n /* Timer modals */\r\n .timer-modal {\r\n text-align: center;\r\n padding: 0.5rem 0;\r\n }\r\n\r\n .timer-modal-icon {\r\n display: inline-flex;\r\n align-items: center;\r\n justify-content: center;\r\n width: 3.5rem;\r\n height: 3.5rem;\r\n border-radius: 9999px;\r\n margin: 0 auto 1rem;\r\n }\r\n\r\n .timer-modal-icon.warning {\r\n background-color: hsl(var(--accent-warning) / 8%);\r\n color: hsl(var(--accent-warning));\r\n }\r\n\r\n .timer-modal-icon.expired {\r\n background-color: hsl(var(--accent-danger) / 5%);\r\n color: hsl(var(--accent-danger));\r\n }\r\n\r\n .timer-modal-title {\r\n font-size: 1.125rem;\r\n font-weight: 600;\r\n margin-bottom: 0.5rem;\r\n }\r\n\r\n .timer-modal-text {\r\n font-size: 0.875rem;\r\n line-height: 1.4;\r\n color: hsl(var(--text-tertiary)); /* gray-500 */\r\n }\r\n\r\n .timer-modal-hint {\r\n font-size: 0.75rem;\r\n color: hsl(var(--text-head)); /* gray-400 */\r\n margin-top: 0.5rem;\r\n }\r\n\r\n .timer-modal-actions {\r\n display: flex;\r\n flex-direction: column;\r\n gap: 0.5rem;\r\n }\r\n /* The Button component's class=\"w-full\" is a Tailwind utility that\r\n isn't always emitted in the venue-calendar bundle (no scanner\r\n pass over node_modules), so force full width at the layout\r\n level. Applies to both <button> and the anchor variant. */\r\n .timer-modal-actions :global(button),\r\n .timer-modal-actions :global(a) {\r\n width: 100%;\r\n }\r\n</style>\r\n"],"names":["createLogger","COOKIE_NAMES","asCookieName","parseCookies","acc","cookie","name","valueParts","getCookie","checked","$","$$props","indeterminate","value","disabled","color","className","inputEl","handleChange","colorClasses","colorClass","inputClasses","label","root_1","input","$$value","span","root","$$args","getBackend","kind","safeJsonParse","raw","fallback","read","key","backend","write","error","logger","getLocalStorage","setLocalStorage","registry","registerStorageKey","descriptor","THEME_STORAGE_KEY","LEGACY_THEME_KEY","readLegacyTheme","removeLegacyTheme","readThemeOverride","canonical","legacy","browser","detectDarkMode","containerSelector","saved","selector","containers","container","getStripeTheme","isDark","useStripeTheme","options","watchChanges","current","observer","updateTheme","onMount","onDestroy","stripe","paymentIntent","elements","paymentError","labels","stripeTheme","paymentMountDiv","paymentElement","mountedSecret","mountStripePayment","destroyStripePayment","firstName","lastName","email","keepUpdated","agreeTerms","cardNumber","cardExpiry","cardCvc","attendees","noPaymentRequired","submitted","errors","e","isEmailAddress","i","att","hasErrors","isTimerUrgent","totalTickets","sum","_","lastSubmitRequest","handleSubmit","handleSameAsPurchaser","index","a","div","root_9","div_1","Button","node","ChevronLeft","CheckoutTimer","node_1","section","Heading","node_2","div_2","$3","Input","node_3","node_4","node_5","Checkbox","node_6","section_1","node_8","node_9","attendee","div_3","div_4","Text","node_10","node_11","div_5","div_6","node_12","node_13","node_14","$$render","consequent","section_2","node_7","div_7","root_2","node_16","node_17","node_18","div_8","fragment_9","p","root_3","consequent_2","p_1","consequent_3","alternate","p_2","root_5","consequent_4","node_22","node_23","div_9","node_24","node_25","p_3","consequent_6","consequent_1","consequent_5","alternate_1","section_3","node_27","div_10","root_8","Warning","node_30","text_14","consequent_7","Card","$$anchor","fragment_1","Image","PLACEHOLDER_IMAGE","Calendar","formatEventTime","formatBrowseDate","text_3","OrderSummary","$0","text_6","formatCurrency","totalTicketCount","l","text_1","checkoutStateOptions","dataVariant","dataToggleStore","configProp","onPlaceOrderProp","onBackProp","onRetryExpiredProp","onExtendSessionProp","mergedLabels","mergeLabels","isShowcaseMode","buildConfig","variant","mockFreeEvent","toTicketId","mockDonationEvent","mockRegistrationEvent","mockEvent","config","deadlineMs","timerSeconds","isExpired","processing","submitRequested","showWarningModal","showExpiredModal","warningShown","isExtendingSession","remainingExtensions","canExtend","WARNING_THRESHOLD","timerDisplay","m","s","orderTotals","lines","subtotal","roundCurrency","fees","promoDiscount","adjustedSubtotal","taxes","calculateTax","totalsLocked","isFreeOrder","isRegistration","timerInitialized","c","getCartIdFromCookie","recomputeFromDeadline","resyncDeadlineFromServer","cartId","status","getSessionStatus","tick","next","interval","onVisible","handleBack","toast","requestPlaceOrder","handlePlaceOrder","formData","count","handleRetryExpired","handleExtendSession","result","AppError","err","dismissWarning","root_7","CheckoutForm","CheckoutSidebar","CheckoutMobileFooter","Modal","div_11","div_12","h3","div_13","root_4","div_14","div_15","Time","h3_1","div_16","node_19"],"mappings":"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","x_google_ignoreList":[0,1,2,3,4,6]}
1
+ {"version":3,"file":"Checkout-CHo11MsU.js","sources":["../node_modules/@getmicdrop/svelte-components/dist/cookies/cookies.js","../node_modules/@getmicdrop/svelte-components/dist/primitives/Checkbox/Checkbox.svelte","../node_modules/@getmicdrop/svelte-components/dist/utils/storage/storage.js","../node_modules/@getmicdrop/svelte-components/dist/utils/storage/registry.js","../node_modules/@getmicdrop/svelte-components/dist/utils/storage/themeStorage.js","../__SKIP_ENVIRONMENT__","../node_modules/@getmicdrop/svelte-components/dist/stripe/useStripeTheme.svelte.js","../src/lib/public-calendar-flow/CheckoutForm.svelte","../src/lib/public-calendar-flow/CheckoutSidebar.svelte","../src/lib/public-calendar-flow/CheckoutMobileFooter.svelte","../src/lib/public-calendar-flow/Checkout.svelte"],"sourcesContent":["/**\n * Cookies canonical — the ONE sanctioned surface for browser cookie access\n * across all four Micdrop apps. Lifts micdrop-frontend's typed cookies layer\n * (registry + per-event builders + HttpOnly diagnostic) onto svelte-components'\n * battle-tested `document.cookie` implementation.\n *\n * CANONICAL DESIGN DECISION (Atlas convergence — utility-cookies):\n * The runtime behaviour is **byte-identical to SC's pre-audit-09\n * `cookieHelpers.ts`** — `getCookie` returns `string | null` (not js-cookie's\n * `undefined`), `setCookie` takes the `{ maxAge, path }` options shape and\n * writes `document.cookie` with the `.get-micdrop.com` production-domain check\n * and hardcoded `samesite=lax`, `deleteCookie` sets `max-age=0`,\n * `setAuthCookies` / `clearAuthCookies` are unchanged. MF's *typing* layer\n * (the `COOKIE_NAMES` registry, the per-event templated name builders, the\n * HttpOnly write diagnostic, and the `getCookieStrict` variant) is folded in\n * around that behaviour. SC's exported function names keep resolving so no\n * consumer breaks. (Rubric: prevent bugs > no user-facing harm > consistency.)\n *\n * Reads NEVER throw on a missing cookie — `getCookie` returns `null`, matching\n * SC's historical contract. Use `getCookieStrict` when absence is a programmer\n * error (it throws `CookieError` with code `NOT_SET`).\n *\n * Names: prefer `COOKIE_NAMES.X` and the builder helpers over string literals\n * so renames stay atomic and typos surface at compile time — the read/write\n * surface is typed to `CookieName`, which only the registry + builders +\n * `toCookieName` mint.\n */\nimport { CookieError } from '../errors/classes/CookieError.js';\nimport { asCookieName, toCookieName } from '../types/brands.js';\nimport { createLogger } from '../utils/logger.js';\nconst logger = createLogger('cookies');\n// ── Name registry ──────────────────────────────────────────────────────────\n/**\n * Registry of cookie names Micdrop apps read/write from JS.\n *\n * Covers the union of MF's `cookies.ts` registry and SC's pre-audit-09\n * `cookieHelpers.ts` literals. Prefer `COOKIE_NAMES.X` over raw strings so\n * renames are atomic and typos surface at compile time. Each entry is a\n * branded `CookieName` (minted via `asCookieName`), so values flow straight\n * into the read/write surface below without re-branding. For per-event\n * templated names (`checkout-promocode-${eventID}` etc.), use the builder\n * helpers below.\n */\nexport const COOKIE_NAMES = {\n // ── Operator (admin) auth — server-managed, HttpOnly ──\n // The server sets these on Set-Cookie; JS can read a presence shadow but\n // cannot write or delete them. Listed for type-safety in the client paths\n // that reference them.\n OPERATOR_TOKEN: asCookieName('operator_token'),\n REFRESH_TOKEN: asCookieName('refresh_token'),\n // ── Operator auth state (non-HttpOnly) ──\n TOKEN_EXPIRY: asCookieName('token_expiry'),\n ACTIVE_CONTEXT_ID: asCookieName('active_context_id'),\n // ── Performer (portal) auth — SC's cookieHelpers / performers-portal ──\n PERFORMER_TOKEN: asCookieName('performer_token'),\n PERFORMER_REFRESH_TOKEN: asCookieName('performer_refresh_token'),\n PERFORMER_TOKEN_EXPIRY: asCookieName('performer_token_expiry'),\n USER_DETAILS: asCookieName('userDetails'),\n SUPPRESSED_ALERTS: asCookieName('suppressedAlerts'),\n // ── Cart / checkout (non-templated) ──\n SESSION_ID: asCookieName('session_id'),\n CHECKOUT_CARTID: asCookieName('checkout-cartid'),\n CHECKOUT_PROMOCODE: asCookieName('checkout-promocode'),\n CHECKOUT_PROMO_DISCOUNT: asCookieName('checkout-promo-discount'),\n // ── Browser-resolved IANA timezone, set by hooks.client on app init ──\n USER_TZ: asCookieName('user_tz'),\n};\n/**\n * Cookies the server sets with HttpOnly. JS writes targeting these names are\n * effectively no-ops (the browser ignores `document.cookie =` writes that try\n * to overwrite an HttpOnly cookie). `setCookie` / `deleteCookie` emit a\n * dev-mode warning when one is targeted client-side — the historical bug was a\n * fake-delete of `operator_token` that silently did nothing. The fix is a\n * server endpoint that calls `cookies.delete()`.\n */\nconst HTTP_ONLY_COOKIES = new Set([\n COOKIE_NAMES.OPERATOR_TOKEN,\n COOKIE_NAMES.REFRESH_TOKEN,\n]);\nfunction warnIfHttpOnly(name, op) {\n if (!HTTP_ONLY_COOKIES.has(name))\n return;\n logger.warn(`${op}Cookie('${name}'): this cookie is HttpOnly; browsers ignore JS writes. ` +\n `Use a server endpoint (e.g. /api/auth/logout) that calls cookies.${op === 'set' ? 'set' : 'delete'}() instead.`);\n}\n// ── Per-event templated name builders ────────────────────────────────────────\n// Keep these as builders rather than raw template strings at call sites so\n// renames remain atomic. Each returns a branded `CookieName`.\n/** `checkout-quantities-${eventID}` */\nexport const checkoutQuantitiesCookie = (eventID) => toCookieName(`checkout-quantities-${eventID}`);\n/** `checkout-promocode-${eventID}` */\nexport const checkoutPromocodeCookie = (eventID) => toCookieName(`checkout-promocode-${eventID}`);\n/** `checkout-tickets-${eventID}` */\nexport const checkoutCartCookie = (eventID) => toCookieName(`checkout-tickets-${eventID}`);\n/**\n * @deprecated Alias of {@link checkoutCartCookie} — MF named this builder\n * `checkoutTicketsCookie`. Kept so the lift is name-stable for MF consumers;\n * the canonical name is `checkoutCartCookie` (matches the proposal).\n */\nexport const checkoutTicketsCookie = checkoutCartCookie;\n// ── Read / parse ──────────────────────────────────────────────────────────────\n/**\n * Parse all cookies into an object. SSR-safe (returns `{}` when `document` is\n * undefined). Behaviour is byte-identical to SC's pre-audit-09 `parseCookies`.\n */\nexport function parseCookies() {\n if (typeof document === 'undefined')\n return {};\n return document.cookie\n .split('; ')\n .reduce((acc, cookie) => {\n if (!cookie)\n return acc;\n const [name, ...valueParts] = cookie.split('=');\n acc[name] = decodeURIComponent(valueParts.join('='));\n return acc;\n }, {});\n}\n/**\n * Get a specific cookie value by name. Returns `null` for a missing value\n * (SC's historical contract — NOT js-cookie's `undefined`). Never throws.\n */\n// eslint-disable-next-line no-restricted-syntax -- this file IS the canonical getCookie; the rule points other call sites here\nexport function getCookie(name) {\n const cookies = parseCookies();\n return cookies[name] || null;\n}\n/**\n * Get a cookie value or throw if it is not set. Use when a missing cookie is a\n * programmer error rather than an expected state. Throws `CookieError` with\n * code `NOT_SET`.\n */\nexport function getCookieStrict(name) {\n const value = getCookie(name);\n if (value === null) {\n throw CookieError.notSet(name);\n }\n return value;\n}\n// ── Write / delete ────────────────────────────────────────────────────────────\n/**\n * Set a cookie. Behaviour is byte-identical to SC's pre-audit-09 `setCookie`:\n * encodes the value, derives the `.get-micdrop.com` production domain + `secure`\n * flag from `window.location`, hardcodes `samesite=lax`, defaults `path` to `/`.\n * Adds MF's HttpOnly-write diagnostic (a dev warning, no behaviour change).\n */\n// eslint-disable-next-line no-restricted-syntax -- this file IS the canonical setCookie; the rule points other call sites here\nexport function setCookie(name, value, options = {}) {\n warnIfHttpOnly(name, 'set');\n const isProd = typeof window !== 'undefined' &&\n window.location.hostname.includes('get-micdrop.com');\n const domainAttr = isProd ? 'domain=.get-micdrop.com;' : '';\n const secureAttr = typeof window !== 'undefined' && window.location.protocol === 'https:'\n ? 'secure;'\n : '';\n const maxAge = options.maxAge ? `max-age=${options.maxAge};` : '';\n const path = options.path || '/';\n document.cookie = `${name}=${encodeURIComponent(value)}; path=${path}; ${maxAge} ${domainAttr} ${secureAttr} samesite=lax`;\n}\n/**\n * Delete a cookie by name. Behaviour is byte-identical to SC's pre-audit-09\n * `deleteCookie`: SSR-safe no-op, sets `max-age=0` with the same domain/path\n * used by `setCookie` so the deletion takes effect on prod + dev. Adds MF's\n * HttpOnly-write diagnostic.\n */\nexport function deleteCookie(name, options = {}) {\n if (typeof document === 'undefined')\n return;\n warnIfHttpOnly(name, 'delete');\n const isProd = typeof window !== 'undefined' &&\n window.location.hostname.includes('get-micdrop.com');\n const domainAttr = isProd ? 'domain=.get-micdrop.com;' : '';\n const path = options.path || '/';\n document.cookie = `${name}=; path=${path}; max-age=0; ${domainAttr}`;\n}\n// ── Performer-token / details / alert helpers (SC pre-audit-09) ──────────────\n/** Get the performer token from cookies. */\nexport function getPerformerToken() {\n return getCookie(COOKIE_NAMES.PERFORMER_TOKEN);\n}\n/**\n * @deprecated R-STORAGE-07: the `userDetails` PII cookie has been removed.\n * Identity comes from `GET /api/auth/me` / the SSR profile, never from\n * `document.cookie`. Always returns null; retained only for API compatibility.\n */\nexport function getUserDetails() {\n return null;\n}\n/** Get suppressed alert IDs from cookies. Returns [] on miss / bad JSON. */\nexport function getSuppressedAlerts() {\n const alerts = getCookie(COOKIE_NAMES.SUPPRESSED_ALERTS);\n if (!alerts)\n return [];\n try {\n return JSON.parse(alerts);\n }\n catch {\n return [];\n }\n}\n/** Save suppressed alert IDs to cookies (1 year expiry). */\nexport function saveSuppressedAlerts(alertIds) {\n setCookie(COOKIE_NAMES.SUPPRESSED_ALERTS, JSON.stringify(alertIds), {\n maxAge: 31536000,\n });\n}\n// ── Auth cookie helpers (SC pre-audit-09 — LOAD-BEARING, byte-identical) ─────\n// SC's index.js exports `setAuthCookies` / `clearAuthCookies`; consumers rely\n// on the exact `document.cookie` writes below. Do NOT reroute these through\n// setCookie/deleteCookie — the raw writes (performer_token, userDetails,\n// performer_refresh_token, the 30-day max-age, the domain/secure derivation)\n// must stay identical.\n/**\n * Set authentication cookies with proper domain and security settings.\n *\n * SECURITY (R-STORAGE-07): this NO LONGER writes the `userDetails` PII cookie.\n * A `{firstName,lastName,email}` blob in a JS-readable cookie is XSS-extractable;\n * identity now comes from the cookie-authed `GET /api/auth/me` endpoint (or the\n * SSR profile), never from `document.cookie`. The `userDetails` param is retained\n * for signature compatibility but is intentionally ignored. The token cookies\n * (performer_token / performer_refresh_token) are unchanged.\n */\nexport function setAuthCookies(token, _userDetails, rememberMe = true, refreshToken = null) {\n void _userDetails; // R-STORAGE-07: never persist identity PII to document.cookie\n const isProd = window.location.hostname.includes('get-micdrop.com');\n const domainAttr = isProd ? 'domain=.get-micdrop.com;' : '';\n const secureAttr = window.location.protocol === 'https:' ? 'secure;' : '';\n const maxAge = rememberMe ? 'max-age=2592000;' : ''; // 30 days or session\n document.cookie = `performer_token=${token}; path=/; ${maxAge} ${domainAttr} ${secureAttr} samesite=lax`;\n if (refreshToken) {\n document.cookie = `performer_refresh_token=${refreshToken}; path=/; ${maxAge} ${domainAttr} ${secureAttr} samesite=lax`;\n }\n}\n/**\n * Clear authentication cookies.\n * Byte-identical to SC's pre-audit-09 `clearAuthCookies`.\n */\nexport function clearAuthCookies() {\n const isProd = window.location.hostname.includes('get-micdrop.com');\n const domainAttr = isProd ? 'domain=.get-micdrop.com;' : '';\n document.cookie = `userDetails=; path=/; max-age=0; ${domainAttr}`;\n document.cookie = `performer_token=; path=/; max-age=0; ${domainAttr}`;\n document.cookie = `performer_refresh_token=; path=/; max-age=0; ${domainAttr}`;\n}\n","<script lang=\"ts\">\r\n import type { Snippet } from 'svelte';\r\n\r\n interface Props {\r\n checked?: boolean;\r\n indeterminate?: boolean;\r\n value?: string;\r\n name?: string;\r\n disabled?: boolean;\r\n color?: 'blue' | 'red' | 'green' | 'purple' | 'orange' | 'yellow';\r\n class?: string;\r\n id?: string;\r\n 'aria-label'?: string;\r\n 'aria-labelledby'?: string;\r\n 'aria-describedby'?: string;\r\n // eslint-disable-next-line no-unused-vars -- intentionally retained (prop/slot/forward-compat or Svelte-5 reactive binding eslint cannot see)\n onchange?: (detail: { checked: boolean; value: string }) => void;\r\n onclick?: () => void;\r\n onfocus?: () => void;\r\n onblur?: () => void;\r\n // eslint-disable-next-line no-unused-vars -- intentionally retained (prop/slot/forward-compat or Svelte-5 reactive binding eslint cannot see)\n onkeydown?: (e: KeyboardEvent) => void;\r\n // eslint-disable-next-line no-unused-vars -- intentionally retained (prop/slot/forward-compat or Svelte-5 reactive binding eslint cannot see)\n onkeyup?: (e: KeyboardEvent) => void;\r\n children?: Snippet;\r\n }\r\n\r\n let {\r\n checked = $bindable(false),\r\n indeterminate = false,\r\n value = \"\",\r\n name = \"\",\r\n disabled = false,\r\n color = \"blue\",\r\n class: className = \"\",\r\n id,\r\n 'aria-label': ariaLabel,\r\n 'aria-labelledby': ariaLabelledBy,\r\n 'aria-describedby': ariaDescribedBy,\r\n onchange,\r\n onclick,\r\n onfocus,\r\n onblur,\r\n onkeydown,\r\n onkeyup,\r\n children,\r\n }: Props = $props();\r\n\r\n let inputEl = $state<HTMLInputElement>();\r\n // `indeterminate` is a DOM property, not a reflected HTML attribute,\r\n // so it has to be set imperatively after mount/update.\r\n $effect(() => {\r\n if (inputEl) inputEl.indeterminate = indeterminate;\r\n });\r\n\r\n function handleChange() {\r\n // Note: checked is already updated by bind:checked before this handler runs\r\n // We just need to call the onchange callback with the current values\r\n onchange?.({ checked, value });\r\n }\r\n\r\n const colorClasses = {\r\n blue: \"text-brand-primary\",\r\n red: \"text-accent-danger\",\r\n green: \"text-accent-success\",\r\n purple: \"text-brand-primary\",\r\n orange: \"text-accent-warning\",\r\n yellow: \"text-accent-warning\"\r\n };\r\n\r\n let colorClass = $derived(colorClasses[color] || colorClasses.blue);\r\n\r\n let inputClasses = $derived([\r\n \"w-4 h-4 shrink-0 rounded\",\r\n \"border-stroke-primary\",\r\n \"focus:outline-none focus:ring-0 focus:ring-offset-0 focus:shadow-none focus-visible:outline-none\",\r\n colorClass,\r\n disabled ? \"cursor-not-allowed opacity-50\" : \"cursor-pointer\"\r\n ].join(\" \"));\r\n</script>\r\n\r\n<label\r\n class=\"inline-flex items-start gap-2 {disabled ? 'cursor-not-allowed' : 'cursor-pointer'} select-none {className}\"\r\n {id}\r\n>\r\n <input\r\n bind:this={inputEl}\r\n type=\"checkbox\"\r\n {name}\r\n {value}\r\n {disabled}\r\n bind:checked\r\n onchange={handleChange}\r\n {onclick}\r\n {onfocus}\r\n {onblur}\r\n {onkeydown}\r\n {onkeyup}\r\n aria-label={ariaLabel}\r\n aria-labelledby={ariaLabelledBy}\r\n aria-describedby={ariaDescribedBy}\r\n class={inputClasses}\r\n />\r\n {#if children}\r\n <span class=\"text-sm font-medium text-text-primary\">\r\n {@render children()}\r\n </span>\r\n {/if}\r\n</label>\r\n","/**\n * Storage canonical — SSR-safe, JSON-coded, quota-guarded localStorage /\n * sessionStorage primitives. The ONE sanctioned path for storage access across\n * all four Micdrop apps (R-STORAGE-04/05/06).\n *\n * Consumers NEVER touch `localStorage` / `sessionStorage` / `JSON.parse` /\n * `JSON.stringify` directly — the `no-direct-localstorage` lint rule enforces\n * this and points here. Use, in order of preference:\n * - `createPersistedState` (reactive rune store) for state that drives UI\n * - `getLocalStorage` / `setLocalStorage` / `removeLocalStorage` (imperative kv)\n * - `getSessionStorage` / `setSessionStorage` / `removeSessionStorage`\n * - `clearAppStorageOnLogout` to wipe `clearOnLogout`-flagged keys\n *\n * Every read returns a fallback (never throws on corrupt JSON, R-STORAGE-03);\n * every write is quota-guarded and logged, never crashing (R-STORAGE-06);\n * every access is SSR-safe (R-STORAGE-05).\n */\nimport { logger } from '../logger.js';\n/** Resolve the backend, or null on SSR / private-mode property-access throw. */\nfunction getBackend(kind) {\n if (typeof window === 'undefined')\n return null; // SSR — R-STORAGE-05\n try {\n return kind === 'local' ? window.localStorage : window.sessionStorage;\n }\n catch {\n // Some browsers throw merely accessing the property in private mode.\n return null;\n }\n}\n/**\n * Whether the given storage backend can actually be written to. False on SSR\n * and in Safari Private Mode / quota-exhausted contexts (R-STORAGE-06). Use to\n * gate a degradation banner on high-impact features.\n */\nexport function isStorageAvailable(kind = 'local') {\n const backend = getBackend(kind);\n if (!backend)\n return false;\n const probe = '__micdrop_probe__';\n try {\n backend.setItem(probe, '1');\n backend.removeItem(probe);\n return true;\n }\n catch {\n return false;\n }\n}\n/**\n * Parse a JSON string defensively. Returns `fallback` on null / empty / corrupt\n * input instead of throwing (R-STORAGE-03). The single safe decode entry point\n * for any persisted string — storage values AND cookie values.\n */\nexport function safeJsonParse(raw, fallback) {\n if (raw === null || raw === undefined || raw === '')\n return fallback;\n try {\n return JSON.parse(raw);\n }\n catch {\n return fallback;\n }\n}\nfunction read(kind, key, fallback) {\n const backend = getBackend(kind);\n if (!backend)\n return fallback;\n let raw;\n try {\n raw = backend.getItem(key);\n }\n catch {\n return fallback;\n }\n return safeJsonParse(raw, fallback);\n}\nfunction write(kind, key, value) {\n const backend = getBackend(kind);\n if (!backend)\n return false;\n try {\n backend.setItem(key, JSON.stringify(value));\n return true;\n }\n catch (error) {\n // QuotaExceededError / private-mode — log + continue, never crash the page.\n logger.warn(`storage: failed to write \"${key}\"`, error);\n return false;\n }\n}\nfunction remove(kind, key) {\n const backend = getBackend(kind);\n if (!backend)\n return;\n try {\n backend.removeItem(key);\n }\n catch {\n // ignore — removal is best-effort\n }\n}\n/**\n * Read a JSON-coded value from localStorage. Returns `fallback` (default `null`)\n * on miss / SSR / corrupt JSON.\n */\nexport function getLocalStorage(key, fallback = null) {\n return read('local', key, fallback);\n}\n/** Write a JSON-coded value to localStorage. Returns false if unavailable (never throws). */\nexport function setLocalStorage(key, value) {\n return write('local', key, value);\n}\n/** Remove a key from localStorage. No-op on SSR. */\nexport function removeLocalStorage(key) {\n remove('local', key);\n}\n/**\n * Read a JSON-coded value from sessionStorage. Returns `fallback` (default\n * `null`) on miss / SSR / corrupt JSON.\n */\nexport function getSessionStorage(key, fallback = null) {\n return read('session', key, fallback);\n}\n/** Write a JSON-coded value to sessionStorage. Returns false if unavailable (never throws). */\nexport function setSessionStorage(key, value) {\n return write('session', key, value);\n}\n/** Remove a key from sessionStorage. No-op on SSR. */\nexport function removeSessionStorage(key) {\n remove('session', key);\n}\n","/**\n * Storage key registry + canonical namespace (R-STORAGE-01/02/08).\n *\n * Every persisted key is colon-namespaced `micdrop:<feature>:<detail>` and\n * registered with its lifecycle flags so a single `clearAppStorageOnLogout()`\n * can walk the registry instead of every logout flow hand-clearing keys.\n */\nimport { removeLocalStorage, removeSessionStorage, } from './storage.js';\nconst NAMESPACE_RE = /^micdrop:[a-z0-9]+(:[a-z0-9-]+)+$/;\n/**\n * Build a canonical colon-namespaced storage key `micdrop:<feature>:<detail...>`\n * (R-STORAGE-02). Lowercases tokens and throws on empty / malformed segments so\n * a bad key fails at the builder, not silently at runtime.\n *\n * @example buildStorageKey('checkout', 'state', eventId) // micdrop:checkout:state:<id>\n */\nexport function buildStorageKey(feature, ...detail) {\n const segments = [feature, ...detail.map(String)]\n .map((s) => s.trim().toLowerCase())\n .filter((s) => s.length > 0);\n if (segments.length < 2) {\n throw new Error(`buildStorageKey requires a feature and at least one detail segment (got: ${JSON.stringify([feature, ...detail])})`);\n }\n const key = `micdrop:${segments.join(':')}`;\n if (!NAMESPACE_RE.test(key)) {\n throw new Error(`buildStorageKey produced a non-canonical key: \"${key}\"`);\n }\n return key;\n}\n/** Whether a key matches the canonical `micdrop:<feature>:<detail>` namespace. */\nexport function isCanonicalStorageKey(key) {\n return NAMESPACE_RE.test(key);\n}\nconst registry = new Map();\n/**\n * Register a storage key + its lifecycle flags. Idempotent per (kind, key).\n * Returns the key so a caller can `const K = registerStorageKey({...})`.\n */\nexport function registerStorageKey(descriptor) {\n registry.set(`${descriptor.kind}:${descriptor.key}`, descriptor);\n return descriptor.key;\n}\n/** Read-only snapshot of all registered descriptors. */\nexport function getStorageRegistry() {\n return [...registry.values()];\n}\n/**\n * Clear every registered key flagged `clearOnLogout` (R-STORAGE-08). The single\n * canonical logout cleanup — logout flows call this instead of inline clears,\n * so a new persisted key is wiped on logout the moment it declares the flag.\n */\nexport function clearAppStorageOnLogout(options = {}) {\n for (const descriptor of registry.values()) {\n if (!descriptor.clearOnLogout)\n continue;\n if (options.onlyKey && descriptor.key !== options.onlyKey)\n continue;\n if (descriptor.kind === 'local')\n removeLocalStorage(descriptor.key);\n else\n removeSessionStorage(descriptor.key);\n }\n}\n","/**\n * Theme persistence onto the storage canonical (`micdrop:ui:theme`).\n *\n * The theme override is TRINARY: an explicit `'light'` / `'dark'` choice, or its\n * ABSENCE meaning \"auto — follow the system preference\". Absence is load-bearing:\n * removing the key is how `DarkModeToggle` expresses auto mode, so this layer\n * stores ONLY explicit choices and treats a missing key as auto. It never\n * writes a sentinel for auto.\n *\n * Encoding: the storage canonical JSON-codes every value, so the explicit\n * override is persisted JSON-coded (`\"dark\"`) under the canonical key and read\n * back through `getLocalStorage` like any other canonical value.\n *\n * Legacy migration (the codex trap): before the canonical existed, all three\n * theme consumers read/wrote a bare un-namespaced `\"theme\"` key holding a PLAIN\n * string (`dark` / `light`), NOT JSON. That legacy value is unreadable through\n * `getLocalStorage` because `JSON.parse('dark')` throws and falls back to null.\n * `readThemeOverride()` therefore one-time-adopts a legacy bare value into the\n * canonical key (JSON-coded) and deletes the legacy key, so existing users keep\n * their exact theme with zero visual change. The read goes through the shared\n * helper, so whichever of the three consumers mounts first performs the adoption.\n */\nimport { getLocalStorage, setLocalStorage, removeLocalStorage, } from './storage.js';\nimport { registerStorageKey } from './registry.js';\n/**\n * Canonical key for the persisted theme override. Registered (not\n * clearOnLogout — a theme preference outlives the session, matching the\n * pre-canonical behavior where the bare `theme` key was never logout-cleared).\n */\nexport const THEME_STORAGE_KEY = registerStorageKey({\n key: 'micdrop:ui:theme',\n kind: 'local',\n});\n/** Pre-canonical bare key. Held a PLAIN string (`dark` / `light`), not JSON. */\nconst LEGACY_THEME_KEY = 'theme';\n/** Read the legacy bare `theme` value (plain string), SSR/private-mode safe. */\nfunction readLegacyTheme() {\n if (typeof window === 'undefined')\n return null;\n let raw;\n try {\n raw = window.localStorage.getItem(LEGACY_THEME_KEY);\n }\n catch {\n return null;\n }\n return raw === 'dark' || raw === 'light' ? raw : null;\n}\n/** Best-effort removal of the legacy bare key after adoption. */\nfunction removeLegacyTheme() {\n if (typeof window === 'undefined')\n return;\n try {\n window.localStorage.removeItem(LEGACY_THEME_KEY);\n }\n catch {\n // best-effort — removal must never throw\n }\n}\n/**\n * Read the explicit theme override, or `null` for \"auto / follow system\".\n *\n * One-time-migrates a legacy bare `theme` value into the canonical key and\n * deletes the legacy key. Idempotent: once migrated, subsequent calls just read\n * the canonical key. Canonical value always wins over a stale legacy key.\n */\nexport function readThemeOverride() {\n const canonical = getLocalStorage(THEME_STORAGE_KEY);\n const legacy = readLegacyTheme();\n if (canonical === 'dark' || canonical === 'light') {\n // Canonical wins; retire any stale legacy bare key still hanging around.\n if (legacy)\n removeLegacyTheme();\n return canonical;\n }\n if (legacy) {\n // Adopt the legacy choice into the canonical key, then drop the bare key.\n setLocalStorage(THEME_STORAGE_KEY, legacy);\n removeLegacyTheme();\n return legacy;\n }\n return null;\n}\n/** Persist an explicit theme override (`'light'` / `'dark'`). */\nexport function writeThemeOverride(theme) {\n setLocalStorage(THEME_STORAGE_KEY, theme);\n}\n/** Clear the override — back to auto / follow system. Also drops the legacy key. */\nexport function clearThemeOverride() {\n removeLocalStorage(THEME_STORAGE_KEY);\n removeLegacyTheme();\n}\n","\n export const browser = typeof window !== 'undefined';\n export const dev = false;\n export const building = false;\n export const version = '';\n ","/**\n * Stripe Theme Utility\n *\n * Provides reactive dark mode detection for Stripe Elements.\n * Automatically watches for theme changes via MutationObserver.\n *\n * @example\n * ```svelte\n * <script>\n * import { useStripeTheme } from '@getmicdrop/svelte-components/stripe';\n *\n * const stripeTheme = useStripeTheme();\n * // stripeTheme.current is 'stripe' | 'night'\n * // Automatically updates when theme changes\n * </script>\n *\n * <Elements theme={stripeTheme.current} {stripe} clientSecret={paymentIntent}>\n * <PaymentElement />\n * </Elements>\n * ```\n */\nimport { onMount, onDestroy } from 'svelte';\nimport { readThemeOverride } from '../utils/storage/index.js';\n/**\n * Detects if dark mode is active.\n *\n * Checks in order:\n * 1. localStorage 'theme' value\n * 2. Container element classes (.dark, [data-theme=\"dark\"])\n * 3. System preference (prefers-color-scheme)\n */\nexport function detectDarkMode(containerSelector) {\n if (typeof window === 'undefined')\n return false;\n try {\n // Check the persisted theme override first (auto/system → null).\n const saved = readThemeOverride();\n if (saved === 'dark')\n return true;\n if (saved === 'light')\n return false;\n // Check for data-theme on common containers\n const selector = containerSelector || '[data-theme=\"dark\"], [data-theme=\"light\"], .micdrop, [data-theme]';\n const containers = document.querySelectorAll(selector);\n for (const container of containers) {\n if (container.getAttribute('data-theme') === 'dark')\n return true;\n }\n // Check document root\n if (document.documentElement.getAttribute('data-theme') === 'dark')\n return true;\n // Fallback to system preference\n return window.matchMedia?.('(prefers-color-scheme: dark)')?.matches ?? false;\n }\n catch {\n return false;\n }\n}\n/**\n * Maps dark mode state to Stripe theme.\n */\nexport function getStripeTheme(isDark) {\n return isDark ? 'night' : 'stripe';\n}\n/**\n * Creates a reactive Stripe theme that watches for dark mode changes.\n *\n * Returns an object with a `current` property that is reactive and\n * automatically updates when the theme changes.\n *\n * @example\n * ```svelte\n * <script>\n * import { useStripeTheme } from '@getmicdrop/svelte-components/stripe';\n * const stripeTheme = useStripeTheme();\n * </script>\n *\n * <Elements theme={stripeTheme.current} ...>\n * ```\n */\nexport function useStripeTheme(options = {}) {\n const { containerSelector = '[data-theme=\"dark\"], [data-theme=\"light\"], .micdrop, [data-theme]', watchChanges = true, fallback = 'stripe', } = options;\n let current = $state(fallback);\n let observer = null;\n function updateTheme() {\n current = getStripeTheme(detectDarkMode(containerSelector));\n }\n onMount(() => {\n // Initial detection\n updateTheme();\n if (watchChanges && typeof MutationObserver !== 'undefined') {\n // Watch for class changes on potential theme containers\n const containers = document.querySelectorAll(containerSelector);\n const root = document.documentElement;\n observer = new MutationObserver(() => {\n updateTheme();\n });\n // Observe document root\n observer.observe(root, {\n attributes: true,\n attributeFilter: ['class', 'data-theme'],\n });\n // Observe other containers\n containers.forEach(container => {\n observer?.observe(container, {\n attributes: true,\n attributeFilter: ['class', 'data-theme'],\n });\n });\n // Also listen for system preference changes\n const mediaQuery = window.matchMedia('(prefers-color-scheme: dark)');\n mediaQuery.addEventListener('change', updateTheme);\n }\n });\n onDestroy(() => {\n if (observer) {\n observer.disconnect();\n observer = null;\n }\n });\n return {\n get current() {\n return current;\n },\n /** Force refresh the theme detection */\n refresh: updateTheme,\n };\n}\n/**\n * Simple synchronous dark mode detection without reactivity.\n * Use this when you only need a one-time check.\n */\nexport function getInitialStripeTheme(containerSelector) {\n return getStripeTheme(detectDarkMode(containerSelector));\n}\n","<script lang=\"ts\">\r\n import { onMount, onDestroy } from 'svelte';\r\n import { browser } from '$app/environment';\r\n import { ChevronLeft, Warning } from 'carbon-icons-svelte';\r\n import {\r\n Button,\r\n Heading,\r\n Text,\r\n Input,\r\n Checkbox,\r\n isEmailAddress,\r\n } from '@getmicdrop/svelte-components';\r\n import { useStripeTheme } from '@getmicdrop/svelte-components/stripe';\r\n import type {\r\n EventData,\r\n OrderTotals,\r\n CheckoutFormData,\r\n } from '$lib/public-calendar-flow/types';\r\n import CheckoutTimer from './CheckoutTimer.svelte';\r\n\r\n interface Props {\r\n event: EventData;\r\n orderTotals: OrderTotals;\r\n timerSeconds: number;\r\n isExpired: boolean;\r\n isRegistration: boolean;\r\n isFreeOrder: boolean;\r\n promoApplied: boolean;\r\n promoDiscount: number;\r\n onBack: () => void;\r\n onPlaceOrder: (formData: CheckoutFormData) => void;\r\n processing: boolean;\r\n /** Incremented by parent to trigger form validation + submit from external buttons (sidebar, mobile footer) */\r\n submitRequested: number;\r\n /** Stripe instance for production mode (absence = showcase mock inputs) */\r\n stripe?: any;\r\n /** Stripe client secret for PaymentElement */\r\n paymentIntent?: string;\r\n /** Bindable Stripe elements reference for the wrapper to use during payment confirmation */\r\n elements?: any;\r\n /** Payment error message from the wrapper */\r\n paymentError?: string;\r\n labels?: Record<string, string>;\r\n }\r\n\r\n let {\r\n event,\r\n orderTotals,\r\n timerSeconds,\r\n isExpired,\r\n isRegistration,\r\n isFreeOrder,\r\n promoApplied,\r\n promoDiscount,\r\n onBack,\r\n onPlaceOrder,\r\n processing,\r\n submitRequested,\r\n stripe = undefined,\r\n paymentIntent = undefined,\r\n elements = $bindable(undefined),\r\n paymentError = '',\r\n labels = {},\r\n }: Props = $props();\r\n\r\n // Use direct Stripe.js API instead of svelte-stripe (Svelte 5 compatibility bug).\r\n const stripeTheme = browser\r\n ? useStripeTheme()\r\n : { current: 'stripe' as const };\r\n let paymentMountDiv = $state<HTMLElement | null>(null);\r\n let paymentElement: any = null;\r\n // Track which client secret the mounted element is bound to. Re-mounting only\r\n // the Stripe element (not the whole CheckoutForm) when the PaymentIntent changes\r\n // lets a partial gift card swap in a reduced-amount PI WITHOUT wiping the\r\n // buyer's typed name/email/attendee fields (the {#key} approach remounted the\r\n // entire form and blanked them).\r\n let mountedSecret = $state<string | null>(null);\r\n\r\n function mountStripePayment() {\r\n if (!browser || !stripe || !paymentIntent || !paymentMountDiv) return;\r\n // Tear down any element bound to a stale secret before re-creating.\r\n if (paymentElement) {\r\n try {\r\n paymentElement.destroy();\r\n } catch (_) {\r\n // ignore\r\n }\r\n paymentElement = null;\r\n }\r\n elements = stripe.elements({\r\n clientSecret: paymentIntent,\r\n appearance: { theme: stripeTheme.current },\r\n });\r\n paymentElement = elements.create('payment');\r\n paymentElement.mount(paymentMountDiv);\r\n mountedSecret = paymentIntent;\r\n }\r\n\r\n function destroyStripePayment() {\r\n if (paymentElement) {\r\n try {\r\n paymentElement.destroy();\r\n } catch (_) {\r\n // ignore\r\n }\r\n paymentElement = null;\r\n }\r\n mountedSecret = null;\r\n }\r\n\r\n $effect(() => {\r\n // (Re)mount whenever the PaymentIntent secret changes — initial mount, or a\r\n // new reduced-amount PI after a partial gift card is applied.\r\n if (\r\n stripe &&\r\n paymentIntent &&\r\n paymentMountDiv &&\r\n paymentIntent !== mountedSecret\r\n ) {\r\n mountStripePayment();\r\n } else if ((!stripe || !paymentIntent) && mountedSecret) {\r\n // No longer needs Stripe (e.g. a gift card now fully covers the order, so\r\n // the wrapper cleared stripe/paymentIntent and the payment section is\r\n // hidden) — tear down the mounted element instead of orphaning the iframe.\r\n destroyStripePayment();\r\n }\r\n });\r\n\r\n // Re-skin Stripe Elements when the theme changes (data-theme toggle).\r\n $effect(() => {\r\n if (mountedSecret && elements && stripeTheme.current) {\r\n try {\r\n elements.update({ appearance: { theme: stripeTheme.current } });\r\n } catch (_) {\r\n // Stripe Elements is finicky about post-mount updates; ignore.\r\n }\r\n }\r\n });\r\n\r\n onDestroy(() => {\r\n if (paymentElement) {\r\n paymentElement.destroy();\r\n paymentElement = null;\r\n }\r\n mountedSecret = null;\r\n });\r\n\r\n // --- Form state ---\r\n let firstName = $state('');\r\n let lastName = $state('');\r\n let email = $state('');\r\n let keepUpdated = $state(false);\r\n let agreeTerms = $state(false);\r\n\r\n // Payment fields (mock)\r\n let cardNumber = $state('');\r\n let cardExpiry = $state('');\r\n let cardCvc = $state('');\r\n\r\n // Per-ticket attendee fields (for registration events)\r\n let attendees = $state<\r\n Array<{\r\n firstName: string;\r\n lastName: string;\r\n email: string;\r\n sameAsPurchaser: boolean;\r\n }>\r\n >([]);\r\n\r\n // No Stripe charge is needed when the order is free OR fully covered by a gift\r\n // card (total 0 but subtotal > 0, so isFreeOrder alone is false). In both cases\r\n // we must NOT render/validate the showcase mock-card inputs — otherwise a\r\n // fully-gift-card-covered order can't be submitted (the wrapper routes it to\r\n // executeGiftCardOnlyPurchase once onPlaceOrder fires).\r\n let noPaymentRequired = $derived(isFreeOrder || orderTotals.total === 0);\r\n\r\n // Validation state\r\n let submitted = $state(false);\r\n let errors = $derived(\r\n (() => {\r\n if (!submitted) return {};\r\n const e: Record<string, string> = {};\r\n if (!firstName.trim())\r\n e.firstName = labels.firstNameRequired || 'First name is required';\r\n if (!lastName.trim())\r\n e.lastName = labels.lastNameRequired || 'Last name is required';\r\n if (!email.trim()) e.email = labels.emailRequired || 'Email is required';\r\n else if (!isEmailAddress(email))\r\n e.email = labels.emailInvalid || 'Enter a valid email address';\r\n if (!noPaymentRequired && !stripe) {\r\n // Only validate mock card fields in showcase mode (Stripe absent AND a\r\n // real balance is due — not free / not fully gift-card covered)\r\n if (!cardNumber.trim())\r\n e.cardNumber = labels.cardNumberRequired || 'Card number is required';\r\n if (!cardExpiry.trim())\r\n e.cardExpiry = labels.expiryRequired || 'Expiry date is required';\r\n if (!cardCvc.trim())\r\n e.cardCvc = labels.cvcRequired || 'CVC is required';\r\n }\r\n if (!agreeTerms)\r\n e.agreeTerms = labels.mustAgreeTerms || 'You must agree to the terms';\r\n\r\n // Validate attendee fields for registration events\r\n if (isRegistration) {\r\n for (let i = 0; i < attendees.length; i++) {\r\n const att = attendees[i];\r\n if (!att.sameAsPurchaser) {\r\n if (!att.firstName.trim())\r\n e[`attendee_${i}_firstName`] = 'First name is required';\r\n if (!att.lastName.trim())\r\n e[`attendee_${i}_lastName`] = 'Last name is required';\r\n if (!att.email.trim())\r\n e[`attendee_${i}_email`] =\r\n labels.emailRequired || 'Email is required';\r\n else if (!isEmailAddress(att.email))\r\n e[`attendee_${i}_email`] =\r\n labels.emailInvalidShort || 'Enter a valid email';\r\n }\r\n }\r\n }\r\n\r\n return e;\r\n })()\r\n );\r\n\r\n let hasErrors = $derived(Object.keys(errors).length > 0);\r\n\r\n let isTimerUrgent = $derived(timerSeconds > 0 && timerSeconds <= 120);\r\n\r\n // Initialize attendee list based on order lines\r\n $effect(() => {\r\n if (isRegistration) {\r\n const totalTickets = orderTotals.lines.reduce(\r\n (sum, l) => sum + l.quantity,\r\n 0\r\n );\r\n if (attendees.length !== totalTickets) {\r\n attendees = Array.from({ length: totalTickets }, (_, i) => ({\r\n firstName: '',\r\n lastName: '',\r\n email: '',\r\n sameAsPurchaser: i === 0,\r\n }));\r\n }\r\n }\r\n });\r\n\r\n // Watch for external submit requests (from sidebar/mobile footer buttons)\r\n let lastSubmitRequest = $state(0);\r\n $effect(() => {\r\n if (submitRequested > lastSubmitRequest) {\r\n lastSubmitRequest = submitRequested;\r\n handleSubmit();\r\n }\r\n });\r\n\r\n function handleSameAsPurchaser(index: number, checked: boolean) {\r\n attendees = attendees.map((att, i) => {\r\n if (i !== index) return att;\r\n return {\r\n ...att,\r\n sameAsPurchaser: checked,\r\n firstName: checked ? firstName : att.firstName,\r\n lastName: checked ? lastName : att.lastName,\r\n email: checked ? email : att.email,\r\n };\r\n });\r\n }\r\n\r\n function handleSubmit() {\r\n submitted = true;\r\n if (hasErrors) return;\r\n onPlaceOrder({\r\n firstName,\r\n lastName,\r\n email,\r\n keepMeUpdated: keepUpdated,\r\n attendees: isRegistration\r\n ? attendees.map(a => ({\r\n firstName: a.sameAsPurchaser ? firstName : a.firstName,\r\n lastName: a.sameAsPurchaser ? lastName : a.lastName,\r\n email: a.sameAsPurchaser ? email : a.email,\r\n sameAsPurchaser: a.sameAsPurchaser,\r\n }))\r\n : undefined,\r\n });\r\n }\r\n</script>\r\n\r\n<!-- Form sits on a Boxes (surface-secondary) panel, not the page Background, so\r\n form controls — especially the bare Checkbox boxes — keep a consistent,\r\n readable surface on any themed scheme (matches the order-summary sidebar\r\n Card). The timer-header was already built to be this panel's flush header\r\n (it bleeds to the edges and rounds only its top). -->\r\n<div class=\"checkout-form-panel space-y-6 bg-surface-secondary border-default\">\r\n <!-- Timer header bar -->\r\n <div class=\"timer-header border-default bg-surface-secondary\">\r\n <Button\r\n variant=\"icon\"\r\n size=\"icon-sm\"\r\n onclick={onBack}\r\n aria-label={labels.goBack || 'Go back'}\r\n >\r\n <ChevronLeft size={20} />\r\n </Button>\r\n <CheckoutTimer seconds={timerSeconds} isUrgent={isTimerUrgent} />\r\n </div>\r\n\r\n <!-- Personal details -->\r\n <section class=\"space-y-4\">\r\n <Heading level={2} size=\"lg\" weight=\"semibold\">\r\n {labels.yourDetails || 'Your details'}\r\n </Heading>\r\n\r\n <div class=\"grid grid-cols-1 sm:grid-cols-2 gap-4\">\r\n <Input\r\n label={labels.firstName || 'First name'}\r\n placeholder={labels.firstName || 'First name'}\r\n required\r\n bind:value={firstName}\r\n errorText={errors.firstName ?? ''}\r\n color={errors.firstName ? 'red' : 'base'}\r\n />\r\n <Input\r\n label={labels.lastName || 'Last name'}\r\n placeholder={labels.lastName || 'Last name'}\r\n required\r\n bind:value={lastName}\r\n errorText={errors.lastName ?? ''}\r\n color={errors.lastName ? 'red' : 'base'}\r\n />\r\n </div>\r\n\r\n <Input\r\n label={labels.emailAddress || 'Email address'}\r\n type=\"email\"\r\n placeholder={labels.emailPlaceholder || 'you@example.com'}\r\n required\r\n bind:value={email}\r\n errorText={errors.email ?? ''}\r\n color={errors.email ? 'red' : 'base'}\r\n />\r\n\r\n <Checkbox bind:checked={keepUpdated}>\r\n {labels.keepMeUpdated ||\r\n 'Keep me updated about this event and similar events'}\r\n </Checkbox>\r\n </section>\r\n\r\n <!-- Per-ticket attendee details (registration events) -->\r\n {#if isRegistration && attendees.length > 0}\r\n <section class=\"space-y-5\">\r\n <Heading level={2} size=\"lg\" weight=\"semibold\">\r\n {labels.attendeeDetails || 'Attendee details'}\r\n </Heading>\r\n\r\n {#each attendees as attendee, i}\r\n <div class=\"attendee-card border-default bg-surface-secondary\">\r\n <div class=\"flex items-center justify-between mb-3\">\r\n <Text size=\"sm\" class=\"font-semibold\">\r\n {labels.attendee || 'Attendee'}\r\n {i + 1}\r\n </Text>\r\n <Checkbox\r\n checked={attendee.sameAsPurchaser}\r\n onchange={({ checked }) => handleSameAsPurchaser(i, checked)}\r\n >\r\n {labels.sameAsPurchaser || 'Same as purchaser'}\r\n </Checkbox>\r\n </div>\r\n\r\n <div class=\"space-y-3\">\r\n <div class=\"grid grid-cols-1 sm:grid-cols-2 gap-3\">\r\n <Input\r\n label={labels.firstName || 'First name'}\r\n placeholder={labels.firstName || 'First name'}\r\n required\r\n disabled={attendee.sameAsPurchaser}\r\n bind:value={attendee.firstName}\r\n errorText={!attendee.sameAsPurchaser\r\n ? (errors[`attendee_${i}_firstName`] ?? '')\r\n : ''}\r\n color={!attendee.sameAsPurchaser &&\r\n errors[`attendee_${i}_firstName`]\r\n ? 'red'\r\n : 'base'}\r\n />\r\n <Input\r\n label={labels.lastName || 'Last name'}\r\n placeholder={labels.lastName || 'Last name'}\r\n required\r\n disabled={attendee.sameAsPurchaser}\r\n bind:value={attendee.lastName}\r\n errorText={!attendee.sameAsPurchaser\r\n ? (errors[`attendee_${i}_lastName`] ?? '')\r\n : ''}\r\n color={!attendee.sameAsPurchaser &&\r\n errors[`attendee_${i}_lastName`]\r\n ? 'red'\r\n : 'base'}\r\n />\r\n </div>\r\n <Input\r\n label={labels.email || 'Email'}\r\n type=\"email\"\r\n placeholder={labels.attendeeEmailPlaceholder ||\r\n 'attendee@example.com'}\r\n required\r\n disabled={attendee.sameAsPurchaser}\r\n bind:value={attendee.email}\r\n errorText={!attendee.sameAsPurchaser\r\n ? (errors[`attendee_${i}_email`] ?? '')\r\n : ''}\r\n color={!attendee.sameAsPurchaser && errors[`attendee_${i}_email`]\r\n ? 'red'\r\n : 'base'}\r\n />\r\n </div>\r\n </div>\r\n {/each}\r\n </section>\r\n {/if}\r\n\r\n <!-- Payment section -->\r\n <section class=\"space-y-4\">\r\n {#if noPaymentRequired}\r\n <div class=\"free-order-notice border-default bg-surface-secondary\">\r\n <Text size=\"sm\" color=\"secondary\" class=\"font-medium block\">\r\n {labels.noPaymentRequired || 'No payment required'}\r\n </Text>\r\n <Text size=\"xs\" color=\"muted\" class=\"block mt-1\">\r\n {labels.freeEventMessage ||\r\n 'This is a free event. Complete the form above to reserve your spot.'}\r\n </Text>\r\n </div>\r\n {:else if stripe && paymentIntent}\r\n <!-- Production: Real Stripe PaymentElement -->\r\n <Heading level={2} size=\"lg\" weight=\"semibold\">Payment</Heading>\r\n {#if browser}\r\n <div bind:this={paymentMountDiv}></div>\r\n {#if !mountedSecret}\r\n <p class=\"text-sm text-color-muted py-2\">\r\n {labels.loadingPaymentForm || 'Loading secure payment form...'}\r\n </p>\r\n {/if}\r\n {:else}\r\n <p class=\"text-sm text-color-muted py-2\">\r\n {labels.loadingPaymentForm || 'Loading secure payment form...'}\r\n </p>\r\n {/if}\r\n {#if paymentError}\r\n <p class=\"text-sm text-accent-danger mt-2\">{paymentError}</p>\r\n {/if}\r\n {:else}\r\n <!-- Showcase: Mock card inputs -->\r\n <Heading level={2} size=\"lg\" weight=\"semibold\">Payment</Heading>\r\n\r\n <Input\r\n label={labels.cardNumber || 'Card number'}\r\n placeholder={labels.cardPlaceholder || '4242 4242 4242 4242'}\r\n type=\"creditCardNumber\"\r\n required\r\n bind:value={cardNumber}\r\n errorText={errors.cardNumber ?? ''}\r\n color={errors.cardNumber ? 'red' : 'base'}\r\n />\r\n\r\n <div class=\"grid grid-cols-2 gap-4\">\r\n <Input\r\n label={labels.expiryDate || 'Expiry date'}\r\n placeholder={labels.expiryPlaceholder || 'MM / YY'}\r\n required\r\n bind:value={cardExpiry}\r\n errorText={errors.cardExpiry ?? ''}\r\n color={errors.cardExpiry ? 'red' : 'base'}\r\n />\r\n <Input\r\n label={labels.cvc || 'CVC'}\r\n placeholder={labels.cvcPlaceholder || '123'}\r\n required\r\n maxlength={4}\r\n bind:value={cardCvc}\r\n errorText={errors.cardCvc ?? ''}\r\n color={errors.cardCvc ? 'red' : 'base'}\r\n />\r\n </div>\r\n {#if paymentError}\r\n <p class=\"text-sm text-accent-danger mt-2\">{paymentError}</p>\r\n {/if}\r\n {/if}\r\n </section>\r\n\r\n <!-- Terms -->\r\n <section class=\"space-y-2\">\r\n <Checkbox bind:checked={agreeTerms}>\r\n {labels.agreeToTerms || 'I agree to the terms and conditions'}\r\n </Checkbox>\r\n {#if errors.agreeTerms}\r\n <div class=\"flex items-center gap-1.5\">\r\n <Warning size={16} class=\"text-accent-danger shrink-0\" />\r\n <Text size=\"xs\" class=\"text-accent-danger\">{errors.agreeTerms}</Text>\r\n </div>\r\n {/if}\r\n </section>\r\n\r\n <!-- Mobile submit is handled by CheckoutMobileFooter -->\r\n</div>\r\n\r\n<style>\r\n /* Boxes-coloured panel the whole form sits on. 1rem padding pairs with the\r\n timer-header's -1rem bleed so the header spans the panel's full width and\r\n caps its top. rounded/border match the order-summary sidebar Card. */\r\n .checkout-form-panel {\r\n padding: 1rem;\r\n border-radius: 0.5rem;\r\n border-width: 1px;\r\n border-style: solid;\r\n overflow: hidden;\r\n }\r\n\r\n .timer-header {\r\n display: flex;\r\n align-items: center;\r\n gap: 0.75rem;\r\n padding: 0.75rem 1rem;\r\n margin-top: -1rem;\r\n margin-left: -1rem;\r\n margin-right: -1rem;\r\n border-bottom-width: 1px;\r\n border-bottom-style: solid;\r\n border-radius: 0.5rem 0.5rem 0 0;\r\n }\r\n\r\n .attendee-card {\r\n padding: 1rem;\r\n border-radius: 0.5rem;\r\n border-width: 1px;\r\n border-style: solid;\r\n }\r\n\r\n .free-order-notice {\r\n padding: 1rem;\r\n border-radius: 0.5rem;\r\n border-width: 1px;\r\n border-style: solid;\r\n }\r\n</style>\r\n","<script lang=\"ts\">\r\n\timport { Calendar } from 'carbon-icons-svelte';\r\n\timport type { USD } from '@getmicdrop/svelte-components';\r\n\timport { Button, Card, Image, Text, formatCurrency } from '@getmicdrop/svelte-components';\r\n\timport type { EventData, OrderTotals } from '$lib/public-calendar-flow/types';\r\n\timport { formatBrowseDate, formatEventTime } from '$lib/public-calendar-flow/defaults';\r\n\timport { PLACEHOLDER_IMAGE } from '$lib/utils/constants.js';\r\n\timport OrderSummary from './OrderSummary.svelte';\r\n\r\n\tinterface Props {\r\n\t\tevent: EventData;\r\n\t\torderTotals: OrderTotals;\r\n\t\tisFreeOrder: boolean;\r\n\t\tonPlaceOrder: () => void;\r\n\t\tprocessing: boolean;\r\n\t\t\tlabels?: Record<string, string>;\r\n\t}\r\n\r\n\tlet { event, orderTotals, isFreeOrder, onPlaceOrder, processing, labels = {}, }: Props = $props();\r\n</script>\r\n\r\n<!--\r\n\tCap the sidebar card to the viewport so the Place Order button never\r\n\tfalls below the fold when many ticket types + fees + taxes + gift-card\r\n\tlines stack up. Three sections, top-to-bottom:\r\n\t 1. Event info — pinned at top so the buyer always sees what they're\r\n\t paying for.\r\n\t 2. Ticket-line list — scrolls when capacity exceeds viewport.\r\n\t 3. Totals + Place Order button — pinned at bottom so summary + CTA\r\n\t are always visible.\r\n-->\r\n<Card border padding={false} class=\"flex flex-col max-h-[calc(100vh-1.5rem)] rounded-xl shadow-lg overflow-hidden bg-surface-secondary\">\r\n\t<!-- Pinned: event info -->\r\n\t<div class=\"p-5 shrink-0\">\r\n\t\t<div class=\"flex gap-3\">\r\n\t\t\t<div class=\"shrink-0 w-20 h-14 rounded-lg overflow-hidden bg-surface-tertiary\">\r\n\t\t\t\t<Image\r\n\t\t\t\t\tsrc={event.imageUrl}\r\n\t\t\t\t\talt={event.title}\r\n\t\t\t\t\tfallback={PLACEHOLDER_IMAGE}\r\n\t\t\t\t\tfit=\"cover\"\r\n\t\t\t\t\tclass=\"w-full h-full\"\r\n\t\t\t\t/>\r\n\t\t\t</div>\r\n\t\t\t<div class=\"flex-1 min-w-0\">\r\n\t\t\t\t<Text size=\"sm\" class=\"font-semibold block leading-tight line-clamp-2\">\r\n\t\t\t\t\t{event.title}\r\n\t\t\t\t</Text>\r\n\t\t\t\t<div class=\"flex items-center gap-1.5 mt-1\">\r\n\t\t\t\t\t<Calendar size={16} class=\"shrink-0 icon-muted\" />\r\n\t\t\t\t\t<Text size=\"xs\" color=\"muted\">\r\n\t\t\t\t\t\t{formatBrowseDate(event.startDateTime, event.timezone)}\r\n\t\t\t\t\t\t{#if event.displayStartTime !== false}\r\n\t\t\t\t\t\t\t&middot; {formatEventTime(event.startDateTime, event.timezone)}\r\n\t\t\t\t\t\t{/if}\r\n\t\t\t\t\t</Text>\r\n\t\t\t\t</div>\r\n\t\t\t\t<Text size=\"xs\" color=\"muted\" class=\"block mt-0.5\">\r\n\t\t\t\t\t{event.venue.name}\r\n\t\t\t\t</Text>\r\n\t\t\t</div>\r\n\t\t</div>\r\n\t</div>\r\n\r\n\t<!-- Scrollable: ticket-line items only -->\r\n\t<div class=\"px-5 pb-3 flex-1 overflow-y-auto min-h-0 border-t border-default pt-3\">\r\n\t\t<OrderSummary {orderTotals} mode=\"lines\" {labels} />\r\n\t</div>\r\n\r\n\t<!-- Pinned: totals + Place Order button -->\r\n\t<div class=\"px-5 pt-3 pb-5 shrink-0 border-t border-default space-y-3\">\r\n\t\t<OrderSummary {orderTotals} mode=\"totals\" {labels} />\r\n\t\t<Button\r\n\t\t\tvariant=\"default\"\r\n\t\t\tsize=\"lg\"\r\n\t\t\tclass=\"w-full\"\r\n\t\t\tdisabled={processing}\r\n\t\t\tloading={processing}\r\n\t\t\tonclick={onPlaceOrder}\r\n\t\t>\r\n\t\t\t{#if processing}\r\n\t\t\t\t{labels.placingOrder || 'Placing order...'}\r\n\t\t\t{:else if isFreeOrder}\r\n\t\t\t\t{labels.completeOrder || 'Complete order'} &middot; {labels.free || 'Free'}\r\n\t\t\t{:else}\r\n\t\t\t\t{labels.placeOrder || 'Place order'} &middot; {formatCurrency(orderTotals.total as USD)}\r\n\t\t\t{/if}\r\n\t\t</Button>\r\n\t</div>\r\n</Card>\r\n\r\n","<!-- @style-escape: widget-intentional. -->\r\n<script lang=\"ts\">\r\n\timport { Button, Text, formatCurrency } from '@getmicdrop/svelte-components';\r\n\timport type { USD } from '@getmicdrop/svelte-components';\r\n\timport type { OrderTotals } from '$lib/public-calendar-flow/types';\r\n\r\n\tinterface Props {\r\n\t\torderTotals: OrderTotals;\r\n\t\tisFreeOrder: boolean;\r\n\t\tonPlaceOrder: () => void;\r\n\t\tprocessing: boolean;\r\n\t\t\tlabels?: Record<string, string>;\r\n\t}\r\n\r\n\tlet { orderTotals, isFreeOrder, onPlaceOrder, processing, labels = {}, }: Props = $props();\r\n\r\n\tlet totalTicketCount = $derived(\r\n\t\torderTotals.lines.reduce((sum, l) => sum + l.quantity, 0)\r\n\t);\r\n</script>\r\n\r\n<div\r\n\tclass=\"fixed bottom-0 left-0 right-0 z-40 md:hidden bg-surface-primary border-t border-default shadow-lg\"\r\n\tstyle=\"padding-bottom: env(safe-area-inset-bottom, 0);\"\r\n>\r\n\t<div class=\"flex items-center justify-between gap-3 px-4 py-3\">\r\n\t\t<!-- Left: total info -->\r\n\t\t<div class=\"flex-1 min-w-0\">\r\n\t\t\t<Text size=\"sm\" class=\"font-semibold tabular-nums block\">\r\n\t\t\t\t{#if isFreeOrder}\r\n\t\t\t\t\t{totalTicketCount} {totalTicketCount === 1 ? (labels.ticket || 'ticket') : (labels.tickets || 'tickets')} &middot; Free\r\n\t\t\t\t{:else}\r\n\t\t\t\t\t{formatCurrency(orderTotals.total as USD /* FIXME(canonical-cleanup:toFixed): this cast bypasses the brand boundary; replace with toCents/toUSD at the API-response-transform layer. */)}\r\n\t\t\t\t{/if}\r\n\t\t\t</Text>\r\n\t\t</div>\r\n\r\n\t\t<!-- Right: Place order button -->\r\n\t\t<Button\r\n\t\t\tvariant=\"default\"\r\n\t\t\tsize=\"lg\"\r\n\t\t\tdisabled={processing}\r\n\t\t\tloading={processing}\r\n\t\t\tonclick={onPlaceOrder}\r\n\t\t\tclass=\"shrink-0 min-w-36\"\r\n\t\t>\r\n\t\t\t{#if processing}\r\n\t\t\t\t{labels.placingOrder || 'Placing order...'}\r\n\t\t\t{:else if isFreeOrder}\r\n\t\t\t\t{labels.completeOrder || 'Complete order'}\r\n\t\t\t{:else}\r\n\t\t\t\t{labels.placeOrder || 'Place order'}\r\n\t\t\t{/if}\r\n\t\t</Button>\r\n\t</div>\r\n</div>","<!--\r\n @form-handling-escape: `processing` is a Stripe payment-confirmation\r\n reentrancy guard (`if (processing) return; processing = true` around\r\n confirmPayment), NOT form-submission/validation state. createFormStore\r\n cannot express a payment-intent confirmation flow, and forcing it here\r\n would risk double-charge regressions for zero user-visible benefit.\r\n Same call as PP#193. Per Atlas behavior-form-handling convergence record.\r\n-->\r\n<script module>\r\n import {\r\n type USD,\r\n createLogger,\r\n formatCurrency,\r\n } from '@getmicdrop/svelte-components';\r\n const logger = createLogger('CheckoutMain');\r\n</script>\r\n\r\n<script lang=\"ts\">\r\n import { Warning, Time, Calendar } from 'carbon-icons-svelte';\r\n import {\r\n AppError,\r\n Button,\r\n Heading,\r\n Image,\r\n Modal,\r\n Text,\r\n toTicketId,\r\n roundCurrency,\r\n getCookie,\r\n COOKIE_NAMES,\r\n } from '@getmicdrop/svelte-components';\r\n import { toast } from '@getmicdrop/svelte-components/toast';\r\n import { getSessionStatus } from '$lib/api';\r\n import {\r\n formatBrowseDate,\r\n formatEventTime,\r\n } from '$lib/public-calendar-flow/defaults';\r\n import type {\r\n EventData,\r\n OrderTotals,\r\n OrderLine,\r\n CheckoutConfig,\r\n CheckoutFormData,\r\n } from '$lib/public-calendar-flow/types';\r\n import {\r\n mockEvent,\r\n mockFreeEvent,\r\n mockDonationEvent,\r\n mockRegistrationEvent,\r\n } from '$lib/public-calendar-flow/mock-data';\r\n import { calculateTax } from '$lib/public-calendar-flow/defaults';\r\n import {\r\n dataToggleStore,\r\n type ToggleOption,\r\n } from '$lib/public-calendar-flow/data-toggle-store.svelte';\r\n\r\n import CheckoutForm from './CheckoutForm.svelte';\r\n import CheckoutSidebar from './CheckoutSidebar.svelte';\r\n import CheckoutMobileFooter from './CheckoutMobileFooter.svelte';\r\n import OrderSummary from './OrderSummary.svelte';\r\n import { mergeLabels } from '$lib/public-calendar-flow/i18n/labels.js';\r\n\r\n // mergedLabels initialized after props destructuring below\r\n\r\n // --- DataToggle types ---\r\n type CheckoutVariant =\r\n | 'default'\r\n | 'free'\r\n | 'donation'\r\n | 'registration'\r\n | 'timer_warning'\r\n | 'expired'\r\n | 'promo';\r\n\r\n const checkoutStateOptions: ToggleOption[] = [\r\n { value: 'default', label: 'Standard', color: 'bg-accent-success' },\r\n { value: 'free', label: 'Free', color: 'bg-sky-500' },\r\n { value: 'donation', label: 'Donation', color: 'bg-accent-danger' },\r\n { value: 'registration', label: 'Registration', color: 'bg-brand-primary' },\r\n {\r\n value: 'timer_warning',\r\n label: 'Timer warning',\r\n color: 'bg-status-warning-bg0',\r\n },\r\n { value: 'expired', label: 'Expired', color: 'bg-accent-danger' },\r\n { value: 'promo', label: 'Promo discount', color: 'bg-accent-success' },\r\n ];\r\n\r\n let dataVariant = $derived(dataToggleStore.current as CheckoutVariant);\r\n\r\n // --- Props (dual-mode: showcase vs production) ---\r\n interface Props {\r\n /** Pre-built checkout config for production mode. Absence = showcase. */\r\n config?: CheckoutConfig;\r\n /** Production order submission handler. Receives form data for the wrapper to submit. */\r\n onPlaceOrder?: (formData: CheckoutFormData) => Promise<void>;\r\n /** Production back navigation handler. */\r\n onBack?: () => void;\r\n /** Production retry handler for expired sessions. */\r\n onRetryExpired?: () => void;\r\n /** Production session extension handler. Returns extension result with newExpiryTime and remainingExtensions. */\r\n onExtendSession?: () => Promise<{\r\n newExpiryTime?: string;\r\n remainingExtensions?: number;\r\n } | null | void>;\r\n /** Stripe instance passed through to CheckoutForm */\r\n stripe?: any;\r\n /** Stripe client secret passed through to CheckoutForm */\r\n paymentIntent?: string;\r\n /** Bindable Stripe elements for wrapper to access */\r\n elements?: any;\r\n /** Payment error message to display */\r\n paymentError?: string;\r\n labels?: Record<string, string>;\r\n }\r\n\r\n let {\r\n config: configProp = undefined,\r\n onPlaceOrder: onPlaceOrderProp = undefined,\r\n onBack: onBackProp = undefined,\r\n onRetryExpired: onRetryExpiredProp = undefined,\r\n onExtendSession: onExtendSessionProp = undefined,\r\n stripe = undefined,\r\n paymentIntent = undefined,\r\n elements = $bindable(undefined),\r\n paymentError = '',\r\n labels = {},\r\n }: Props = $props();\r\n\r\n const mergedLabels = $derived(mergeLabels(labels));\r\n let isShowcaseMode = $derived(!configProp);\r\n\r\n $effect(() => {\r\n if (!isShowcaseMode) return;\r\n dataToggleStore.options = checkoutStateOptions;\r\n dataToggleStore.current = 'default';\r\n dataToggleStore.isActive = true;\r\n return () => {\r\n dataToggleStore.isActive = false;\r\n dataToggleStore.options = [];\r\n };\r\n });\r\n\r\n // --- Mock order configurations per variant ---\r\n function buildConfig(variant: CheckoutVariant): CheckoutConfig {\r\n switch (variant) {\r\n case 'free':\r\n return {\r\n event: mockFreeEvent,\r\n lines: [\r\n {\r\n ticketId: toTicketId(201),\r\n ticketName: 'Free Entry',\r\n quantity: 2,\r\n unitPrice: 0,\r\n fee: 0,\r\n subtotal: 0,\r\n },\r\n ],\r\n timerStart: 900,\r\n promoApplied: false,\r\n promoDiscount: 0,\r\n forceExpired: false,\r\n };\r\n\r\n case 'donation':\r\n return {\r\n event: mockDonationEvent,\r\n lines: [\r\n {\r\n ticketId: toTicketId(401),\r\n ticketName: 'Donation',\r\n quantity: 1,\r\n unitPrice: 25,\r\n fee: 0,\r\n subtotal: 25,\r\n },\r\n {\r\n ticketId: toTicketId(402),\r\n ticketName: 'Standard Admission',\r\n quantity: 1,\r\n unitPrice: 30,\r\n fee: 3,\r\n subtotal: 30,\r\n },\r\n ],\r\n timerStart: 900,\r\n promoApplied: false,\r\n promoDiscount: 0,\r\n forceExpired: false,\r\n };\r\n\r\n case 'registration':\r\n return {\r\n event: mockRegistrationEvent,\r\n lines: [\r\n {\r\n ticketId: toTicketId(301),\r\n ticketName: 'Workshop Seat',\r\n quantity: 1,\r\n unitPrice: 75,\r\n fee: 7.5,\r\n subtotal: 75,\r\n },\r\n ],\r\n timerStart: 900,\r\n promoApplied: false,\r\n promoDiscount: 0,\r\n forceExpired: false,\r\n };\r\n\r\n case 'timer_warning':\r\n return {\r\n event: mockEvent,\r\n lines: [\r\n {\r\n ticketId: toTicketId(101),\r\n ticketName: 'General Admission',\r\n quantity: 2,\r\n unitPrice: 25,\r\n fee: 2.5,\r\n subtotal: 50,\r\n },\r\n {\r\n ticketId: toTicketId(102),\r\n ticketName: 'VIP Front Row',\r\n quantity: 1,\r\n unitPrice: 55,\r\n fee: 5.5,\r\n subtotal: 55,\r\n },\r\n ],\r\n timerStart: 33, // Just over 30s — warning modal triggers at 30s\r\n promoApplied: false,\r\n promoDiscount: 0,\r\n forceExpired: false,\r\n };\r\n\r\n case 'expired':\r\n return {\r\n event: mockEvent,\r\n lines: [\r\n {\r\n ticketId: toTicketId(101),\r\n ticketName: 'General Admission',\r\n quantity: 2,\r\n unitPrice: 25,\r\n fee: 2.5,\r\n subtotal: 50,\r\n },\r\n {\r\n ticketId: toTicketId(102),\r\n ticketName: 'VIP Front Row',\r\n quantity: 1,\r\n unitPrice: 55,\r\n fee: 5.5,\r\n subtotal: 55,\r\n },\r\n ],\r\n timerStart: 0,\r\n promoApplied: false,\r\n promoDiscount: 0,\r\n forceExpired: true,\r\n };\r\n\r\n case 'promo':\r\n return {\r\n event: mockEvent,\r\n lines: [\r\n {\r\n ticketId: toTicketId(101),\r\n ticketName: 'General Admission',\r\n quantity: 2,\r\n unitPrice: 25,\r\n fee: 2.5,\r\n subtotal: 50,\r\n },\r\n {\r\n ticketId: toTicketId(102),\r\n ticketName: 'VIP Front Row',\r\n quantity: 1,\r\n unitPrice: 55,\r\n fee: 5.5,\r\n subtotal: 55,\r\n },\r\n ],\r\n timerStart: 900,\r\n promoApplied: true,\r\n promoDiscount: 10,\r\n forceExpired: false,\r\n };\r\n\r\n default:\r\n // Standard checkout\r\n return {\r\n event: mockEvent,\r\n lines: [\r\n {\r\n ticketId: toTicketId(101),\r\n ticketName: 'General Admission',\r\n quantity: 2,\r\n unitPrice: 25,\r\n fee: 2.5,\r\n subtotal: 50,\r\n },\r\n {\r\n ticketId: toTicketId(102),\r\n ticketName: 'VIP Front Row',\r\n quantity: 1,\r\n unitPrice: 55,\r\n fee: 5.5,\r\n subtotal: 55,\r\n },\r\n ],\r\n timerStart: 900,\r\n promoApplied: false,\r\n promoDiscount: 0,\r\n forceExpired: false,\r\n };\r\n }\r\n }\r\n\r\n // --- State ---\r\n let config: CheckoutConfig = $derived(\r\n isShowcaseMode ? buildConfig(dataVariant) : configProp!\r\n );\r\n // Deadline-based timer model.\r\n //\r\n // We track an absolute deadline timestamp (ms since epoch) and compute\r\n // the displayed seconds from `deadlineMs - Date.now()` on each tick.\r\n // This is immune to backgrounded-tab throttling: when the browser\r\n // suspends our setInterval, the displayed value falls behind, but the\r\n // very next tick (or the visibilitychange handler) recomputes from\r\n // wall-clock time and the display jumps to truth. Previously we\r\n // decremented `timerSeconds -= 1`, which drifted arbitrarily far from\r\n // reality whenever the tab was backgrounded (Chrome throttles 1Hz\r\n // intervals to ~1/min after a few minutes, then pauses them entirely).\r\n //\r\n // `timerSeconds` remains an explicit state var (rather than `$derived`)\r\n // because legacy code paths still write to it (legacy demo modes); the\r\n // canonical writer is now the deadline-driven tick effect below.\r\n let deadlineMs = $state<number | null>(null);\r\n let timerSeconds = $state(900);\r\n let isExpired = $state(false);\r\n let processing = $state(false);\r\n let submitRequested = $state(0);\r\n\r\n // Cart expiration warning/expired modal states\r\n let showWarningModal = $state(false);\r\n let showExpiredModal = $state(false);\r\n let warningShown = $state(false);\r\n let isExtendingSession = $state(false);\r\n let remainingExtensions = $state(3);\r\n let canExtend = $state(true);\r\n\r\n const WARNING_THRESHOLD = 30; // Show warning modal at 30 seconds remaining\r\n\r\n // Formatted timer display for the modal\r\n let timerDisplay = $derived(\r\n (() => {\r\n const m = Math.floor(timerSeconds / 60);\r\n const s = timerSeconds % 60;\r\n return `${m.toString().padStart(2, '0')}:${s.toString().padStart(2, '0')}`;\r\n })()\r\n );\r\n\r\n // Order totals — the server's payment-intent response is the source\r\n // of truth (it's what Stripe will actually charge). Fall back to a\r\n // local estimate only when the server hasn't responded yet (showcase\r\n // mode, or while the initial /payment-intent is in flight) so the\r\n // skeleton has something to render.\r\n let orderTotals: OrderTotals = $derived(\r\n (() => {\r\n const lines = config.lines;\r\n if (config.serverTotals) {\r\n return {\r\n lines,\r\n subtotal: config.serverTotals.subtotal,\r\n fees: config.serverTotals.fees,\r\n taxes: config.serverTotals.taxes,\r\n total: config.serverTotals.total,\r\n promoDiscount: config.serverTotals.promoDiscount,\r\n giftCardAmount: config.serverTotals.giftCardAmount,\r\n };\r\n }\r\n const subtotal = roundCurrency(\r\n lines.reduce((sum, l) => sum + l.subtotal, 0)\r\n );\r\n const fees = roundCurrency(\r\n lines.reduce((sum, l) => sum + l.fee * l.quantity, 0)\r\n );\r\n const promoDiscount = config.promoApplied ? config.promoDiscount : 0;\r\n const adjustedSubtotal = roundCurrency(\r\n Math.max(0, subtotal - promoDiscount)\r\n );\r\n const taxes = calculateTax(\r\n adjustedSubtotal + fees,\r\n config.event.venue.taxPercentage\r\n );\r\n return {\r\n lines,\r\n subtotal,\r\n fees,\r\n taxes,\r\n total: roundCurrency(adjustedSubtotal + fees + taxes),\r\n promoDiscount,\r\n giftCardAmount: 0,\r\n };\r\n })()\r\n );\r\n\r\n // Disable Place-order until the server has confirmed the totals.\r\n // Showing \"Place order · $55\" while the actual Stripe charge will be\r\n // $63.55 is worse than briefly disabling the button. Free orders\r\n // never call the payment-intent endpoint, so they're exempt.\r\n // Showcase mode has no server, so its mock totals are final — treat them as\r\n // locked, otherwise the Place-order button is perpetually \"Placing order…\".\r\n let totalsLocked = $derived(isShowcaseMode || !!config.serverTotals);\r\n\r\n let isFreeOrder = $derived(\r\n orderTotals.total === 0 && orderTotals.subtotal === 0\r\n );\r\n let isRegistration = $derived(config.event.isRegistrationEvent);\r\n\r\n // Set timer ONCE on initial mount — not on every config change.\r\n // Config changes frequently (Stripe loads, quantities update, etc.)\r\n // and must not reset the timer.\r\n let timerInitialized = false;\r\n $effect(() => {\r\n const c = config;\r\n if (!timerInitialized) {\r\n timerInitialized = true;\r\n // Convert the incoming \"seconds from now\" into an absolute deadline.\r\n // From this point on, deadlineMs is the source of truth.\r\n deadlineMs = Date.now() + c.timerStart * 1000;\r\n timerSeconds = c.timerStart;\r\n isExpired = c.forceExpired;\r\n }\r\n // These are safe to reset on config change (UI state, not timer)\r\n processing = false;\r\n submitRequested = 0;\r\n });\r\n\r\n function getCartIdFromCookie(): string | null {\r\n if (typeof document === 'undefined') return null;\r\n return getCookie(COOKIE_NAMES.CHECKOUT_CARTID);\r\n }\r\n\r\n function recomputeFromDeadline(): number {\r\n if (deadlineMs == null) return timerSeconds;\r\n return Math.max(0, Math.floor((deadlineMs - Date.now()) / 1000));\r\n }\r\n\r\n /**\r\n * Re-sync the deadline from the orders-service. Called on tab focus —\r\n * a backgrounded tab might have missed an extension granted in another\r\n * tab, or the server may have already expired the cart while we slept.\r\n *\r\n * Silent on any error: the existing deadlineMs continues to drive the\r\n * display, which is no worse than the pre-resync state.\r\n */\r\n async function resyncDeadlineFromServer(): Promise<void> {\r\n const cartId = getCartIdFromCookie();\r\n if (!cartId) return;\r\n try {\r\n const status = await getSessionStatus(cartId);\r\n if (status.notFound) {\r\n // Server has already released the reservation. Surface the\r\n // expired modal so the user isn't staring at a phantom timer.\r\n deadlineMs = Date.now();\r\n timerSeconds = 0;\r\n isExpired = true;\r\n showWarningModal = false;\r\n showExpiredModal = true;\r\n return;\r\n }\r\n if (status.expiresAt) {\r\n deadlineMs = new Date(status.expiresAt).getTime();\r\n timerSeconds = recomputeFromDeadline();\r\n }\r\n } catch {\r\n /* network error — keep current deadlineMs */\r\n }\r\n }\r\n\r\n // Timer countdown with warning and expired modal triggers.\r\n //\r\n // The interval ticks at 1 Hz, but its job is to *recompute* timerSeconds\r\n // from deadlineMs, not to decrement a counter. Backgrounded tabs throttle\r\n // this interval — that's fine; when the tab regains focus the\r\n // visibilitychange handler runs the recompute immediately and the display\r\n // jumps to truth.\r\n $effect(() => {\r\n if (isExpired || timerSeconds <= 0) {\r\n if (timerSeconds <= 0 && !isExpired) {\r\n isExpired = true;\r\n showWarningModal = false;\r\n showExpiredModal = true;\r\n }\r\n return;\r\n }\r\n\r\n const tick = () => {\r\n const next = recomputeFromDeadline();\r\n timerSeconds = next;\r\n\r\n // Persist deadline to localStorage every 10 displayed seconds.\r\n // (Storing the deadline directly — not a recomputed end-time —\r\n // keeps reads consistent across tabs.)\r\n if (\r\n next % 10 === 0 &&\r\n deadlineMs != null &&\r\n typeof localStorage !== 'undefined'\r\n ) {\r\n try {\r\n const cartId = getCartIdFromCookie();\r\n if (cartId) {\r\n localStorage.setItem(\r\n // @storage-escape: checkout-deadline timer persistence (intentional raw localStorage)\r\n `checkout-expiry-${cartId}`,\r\n String(deadlineMs)\r\n );\r\n }\r\n } catch {\r\n /* ignore */\r\n }\r\n }\r\n\r\n // Show warning modal at 30 seconds remaining (only once per session).\r\n // We compare with <= because a backgrounded tab may have skipped over\r\n // the exact === WARNING_THRESHOLD boundary while throttled.\r\n if (next <= WARNING_THRESHOLD && next > 0 && !warningShown) {\r\n showWarningModal = true;\r\n warningShown = true;\r\n }\r\n\r\n if (next <= 0) {\r\n isExpired = true;\r\n showWarningModal = false;\r\n showExpiredModal = true;\r\n clearInterval(interval);\r\n }\r\n };\r\n\r\n const interval = setInterval(tick, 1000);\r\n\r\n // Recompute immediately whenever the tab becomes visible — a\r\n // throttled/paused interval may have left the display many minutes\r\n // behind reality. Also kick a server resync to catch the case where\r\n // the reservation already expired (or was extended in another tab).\r\n const onVisible = () => {\r\n if (\r\n typeof document !== 'undefined' &&\r\n document.visibilityState === 'visible'\r\n ) {\r\n tick();\r\n void resyncDeadlineFromServer();\r\n }\r\n };\r\n if (typeof document !== 'undefined') {\r\n document.addEventListener('visibilitychange', onVisible);\r\n }\r\n if (typeof window !== 'undefined') {\r\n window.addEventListener('pageshow', onVisible);\r\n }\r\n\r\n // Initial server resync — catches the case where the user landed on\r\n // checkout with a stale cookie whose cart has already been released.\r\n void resyncDeadlineFromServer();\r\n\r\n return () => {\r\n clearInterval(interval);\r\n if (typeof document !== 'undefined') {\r\n document.removeEventListener('visibilitychange', onVisible);\r\n }\r\n if (typeof window !== 'undefined') {\r\n window.removeEventListener('pageshow', onVisible);\r\n }\r\n };\r\n });\r\n\r\n // --- Handlers ---\r\n function handleBack() {\r\n if (onBackProp) {\r\n onBackProp();\r\n } else {\r\n toast.info('Back button pressed — would navigate to ticket selection');\r\n }\r\n }\r\n\r\n /** Called from sidebar/mobile footer — signals the form to validate + submit */\r\n function requestPlaceOrder() {\r\n submitRequested += 1;\r\n }\r\n\r\n /** Called by CheckoutForm after successful validation */\r\n function handlePlaceOrder(formData: CheckoutFormData) {\r\n if (processing) return;\r\n processing = true;\r\n\r\n if (onPlaceOrderProp) {\r\n // Production mode: delegate to wrapper with form data\r\n onPlaceOrderProp(formData)\r\n .catch(() => {\r\n // Error handling is done by the wrapper; just reset processing\r\n })\r\n .finally(() => {\r\n processing = false;\r\n });\r\n return;\r\n }\r\n\r\n // Showcase mode: existing mock delay\r\n setTimeout(() => {\r\n processing = false;\r\n const count = orderTotals.lines.reduce((s, l) => s + l.quantity, 0);\r\n toast.success(\r\n `Order placed! ${count} ticket(s) for ${formatCurrency(orderTotals.total as USD /* FIXME(canonical-cleanup:toFixed): this cast bypasses the brand boundary; replace with toCents/toUSD at the API-response-transform layer. */)}`\r\n );\r\n }, 1500);\r\n }\r\n\r\n function handleRetryExpired() {\r\n showExpiredModal = false;\r\n if (onRetryExpiredProp) {\r\n onRetryExpiredProp();\r\n } else {\r\n toast.info('Would navigate back to ticket selection to start over');\r\n }\r\n }\r\n\r\n /** Extend the checkout session via the parent wrapper's onExtendSession callback */\r\n async function handleExtendSession() {\r\n if (!canExtend || isExtendingSession) return;\r\n isExtendingSession = true;\r\n\r\n try {\r\n if (onExtendSessionProp) {\r\n const result = await onExtendSessionProp();\r\n // The parent wrapper (Checkout.svelte) returns `null` when the\r\n // extend-session API call fails — e.g. when the reservation\r\n // has already expired server-side past its grace window, or\r\n // the user has used all 3 extensions. Treat null/undefined as\r\n // an unambiguous failure: don't reset the timer to +15min\r\n // locally (that would falsely show the user they still have a\r\n // valid cart when the server has already released the tickets).\r\n if (result == null) {\r\n throw new AppError(\r\n 'Extension request returned no result',\r\n 'lib/premium-ticket-experience/variants/v3-airbnb-split/CheckoutMain/handleExtendSession'\r\n );\r\n }\r\n if (typeof result === 'object') {\r\n if (result.newExpiryTime) {\r\n // Server gave us an absolute new expiry — use it directly.\r\n deadlineMs = new Date(result.newExpiryTime).getTime();\r\n } else {\r\n // Fallback: add 15 minutes from now\r\n deadlineMs = Date.now() + 15 * 60 * 1000;\r\n }\r\n timerSeconds = recomputeFromDeadline();\r\n if (result.remainingExtensions !== undefined) {\r\n remainingExtensions = result.remainingExtensions;\r\n canExtend = remainingExtensions > 0;\r\n }\r\n } else {\r\n // Truthy non-object: legacy callers that just return `true`\r\n // on success. Treat as a simple 15-minute extension.\r\n deadlineMs = Date.now() + 15 * 60 * 1000;\r\n timerSeconds = recomputeFromDeadline();\r\n }\r\n } else {\r\n // Showcase mode: just add 15 minutes\r\n deadlineMs = Date.now() + 15 * 60 * 1000;\r\n timerSeconds = recomputeFromDeadline();\r\n }\r\n\r\n isExpired = false;\r\n warningShown = false;\r\n showWarningModal = false;\r\n showExpiredModal = false;\r\n } catch (err) {\r\n logger.error('Failed to extend session:', err);\r\n // Extension failed — close the warning but keep the expired\r\n // modal open so the user is offered \"Select tickets again\".\r\n // Disable further extend attempts on this dead cart.\r\n showWarningModal = false;\r\n canExtend = false;\r\n if (timerSeconds <= 0) {\r\n isExpired = true;\r\n showExpiredModal = true;\r\n }\r\n } finally {\r\n isExtendingSession = false;\r\n }\r\n }\r\n\r\n /** Dismiss warning modal without extending (user will hurry) */\r\n function dismissWarning() {\r\n showWarningModal = false;\r\n }\r\n</script>\r\n\r\n<div class=\"w-full min-h-screen bg-surface-primary text-color-primary\">\r\n <div class=\"max-w-6xl mx-auto px-4 py-6 md:px-6 md:py-8\">\r\n <!-- Single CheckoutForm shared between desktop and mobile layouts.\r\n\t\t Previously there were two instances (one hidden md:grid, one md:hidden)\r\n\t\t which caused duplicate Stripe elements and broken form submission. -->\r\n <div class=\"max-w-6xl\">\r\n <!-- Desktop: two-column grid; Mobile: single column -->\r\n <div class=\"md:grid md:grid-cols-booking-split gap-8 items-start\">\r\n <!-- LEFT COLUMN (or full width on mobile): Checkout form -->\r\n <div class=\"pb-24 md:pb-0\">\r\n <!-- CheckoutForm re-mounts ONLY the Stripe element internally when\r\n `paymentIntent` changes (e.g. a partial gift card swaps in a\r\n reduced-amount PI), so Elements always binds to the current secret\r\n WITHOUT remounting the whole form and wiping the buyer's typed\r\n fields. (An earlier {#key paymentIntent} here blanked the form.) -->\r\n <CheckoutForm\r\n event={config.event}\r\n {orderTotals}\r\n {timerSeconds}\r\n {isExpired}\r\n {isRegistration}\r\n {isFreeOrder}\r\n promoApplied={config.promoApplied}\r\n promoDiscount={config.promoDiscount}\r\n onBack={handleBack}\r\n onPlaceOrder={handlePlaceOrder}\r\n {processing}\r\n {submitRequested}\r\n {stripe}\r\n {paymentIntent}\r\n bind:elements\r\n {paymentError}\r\n labels={mergedLabels}\r\n />\r\n\r\n <!-- Inline order summary for mobile (below form). Lead with the\r\n event header (poster + title + date + venue) so the buyer\r\n has visual confirmation of what they're paying for as they\r\n approach the sticky Place order button. Mirrors the desktop\r\n CheckoutSidebar layout. -->\r\n <section\r\n class=\"md:hidden mobile-order-summary border-default bg-surface-secondary mt-6\"\r\n >\r\n <div class=\"flex gap-3 pb-3 mb-3 border-b border-default\">\r\n <div\r\n class=\"shrink-0 w-20 h-14 rounded-lg overflow-hidden bg-surface-tertiary\"\r\n >\r\n <Image\r\n src={config.event.imageUrl}\r\n alt={config.event.title}\r\n class=\"w-full h-full\"\r\n />\r\n </div>\r\n <div class=\"flex-1 min-w-0\">\r\n <Text\r\n size=\"sm\"\r\n class=\"font-semibold block leading-tight line-clamp-2\"\r\n >\r\n {config.event.title}\r\n </Text>\r\n <div class=\"flex items-center gap-1.5 mt-1\">\r\n <Calendar size={16} class=\"shrink-0 icon-muted\" />\r\n <Text size=\"xs\" color=\"muted\">\r\n {formatBrowseDate(\r\n config.event.startDateTime,\r\n config.event.timezone\r\n )}\r\n {#if config.event.displayStartTime !== false}\r\n &middot; {formatEventTime(\r\n config.event.startDateTime,\r\n config.event.timezone\r\n )}\r\n {/if}\r\n </Text>\r\n </div>\r\n <Text size=\"xs\" color=\"muted\" class=\"block mt-0.5\">\r\n {config.event.venue.name}\r\n </Text>\r\n </div>\r\n </div>\r\n <OrderSummary {orderTotals} labels={mergedLabels} />\r\n </section>\r\n </div>\r\n\r\n <!-- RIGHT COLUMN: Sticky order summary sidebar (desktop only) -->\r\n <div class=\"hidden md:block sticky top-6\">\r\n <CheckoutSidebar\r\n event={config.event}\r\n {orderTotals}\r\n {isFreeOrder}\r\n onPlaceOrder={requestPlaceOrder}\r\n processing={processing || (!isFreeOrder && !totalsLocked)}\r\n labels={mergedLabels}\r\n />\r\n </div>\r\n </div>\r\n </div>\r\n\r\n <!-- Mobile: Fixed bottom bar -->\r\n <div class=\"md:hidden\">\r\n <CheckoutMobileFooter\r\n {orderTotals}\r\n {isFreeOrder}\r\n onPlaceOrder={requestPlaceOrder}\r\n processing={processing || (!isFreeOrder && !totalsLocked)}\r\n labels={mergedLabels}\r\n />\r\n </div>\r\n </div>\r\n\r\n <!-- Session Warning Modal (30 seconds remaining).\r\n Wrapped in `{#if}` (instead of relying on `bind:open` alone) so\r\n the Modal fully mounts/unmounts each time showWarningModal\r\n transitions. Surfaced by a wall-clock test where, after the\r\n user clicked \"Continue checkout\" on a first warning, the Modal\r\n component's internal open-state never reset cleanly — on the\r\n NEXT 30s threshold the timer text in DOM updated but the modal\r\n stayed visually hidden, so the user lost the cart with no\r\n chance to re-extend. Forcing remount with #if sidesteps that. -->\r\n {#if showWarningModal}\r\n <Modal\r\n open={true}\r\n oncancel={() => (showWarningModal = false)}\r\n size=\"sm\"\r\n aria-labelledby=\"warning-modal-title\"\r\n >\r\n {#snippet body()}\r\n <div class=\"timer-modal\" id=\"warning-modal-description\">\r\n <div class=\"timer-modal-icon warning\">\r\n <Warning size={32} />\r\n </div>\r\n <h3 id=\"warning-modal-title\" class=\"timer-modal-title\">\r\n {mergedLabels.sessionExpiringSoon || 'Session expiring soon'}\r\n </h3>\r\n <p class=\"timer-modal-text\">\r\n {(\r\n mergedLabels.sessionExpiringSoonMessage ||\r\n 'Your checkout session will expire in {time}. Your selected tickets will be released back to inventory.'\r\n ).replace('{time}', timerDisplay)}\r\n </p>\r\n {#if canExtend && remainingExtensions > 0}\r\n <p class=\"timer-modal-hint\">\r\n {(\r\n mergedLabels.extensionsRemaining ||\r\n '{count} extension{plural} remaining'\r\n )\r\n .replace('{count}', String(remainingExtensions))\r\n .replace('{plural}', remainingExtensions === 1 ? '' : 's')}\r\n </p>\r\n {/if}\r\n </div>\r\n {/snippet}\r\n {#snippet footer()}\r\n <div class=\"timer-modal-actions\">\r\n {#if canExtend}\r\n <Button\r\n variant=\"default\"\r\n size=\"lg\"\r\n class=\"w-full\"\r\n onclick={handleExtendSession}\r\n disabled={isExtendingSession}\r\n loading={isExtendingSession}\r\n >\r\n {isExtendingSession\r\n ? mergedLabels.extendingSession || 'Extending...'\r\n : mergedLabels.continueCheckout || 'Continue checkout'}\r\n </Button>\r\n {:else}\r\n <Button\r\n variant=\"default\"\r\n size=\"lg\"\r\n class=\"w-full\"\r\n onclick={dismissWarning}\r\n >\r\n {mergedLabels.okIllHurry || \"OK, I'll hurry\"}\r\n </Button>\r\n <p class=\"timer-modal-hint\">\r\n {mergedLabels.maxExtensionsReached ||\r\n 'Maximum extensions reached'}\r\n </p>\r\n {/if}\r\n </div>\r\n {/snippet}\r\n </Modal>\r\n {/if}\r\n\r\n <!-- Session Expired Modal (\"Want to keep your tickets?\").\r\n Same `{#if}` wrap as the warning modal — forces a fresh\r\n Modal mount each time so reopening after a previous extend\r\n cycle actually re-displays the dialog. -->\r\n {#if showExpiredModal}\r\n <Modal\r\n open={true}\r\n oncancel={() => (showExpiredModal = false)}\r\n size=\"sm\"\r\n aria-labelledby=\"expired-modal-title\"\r\n >\r\n {#snippet body()}\r\n <div class=\"timer-modal\" id=\"expired-modal-description\">\r\n <div class=\"timer-modal-icon expired\">\r\n <Time size={32} />\r\n </div>\r\n <h3 id=\"expired-modal-title\" class=\"timer-modal-title\">\r\n {canExtend\r\n ? mergedLabels.wantToKeepTickets || 'Want to keep your tickets?'\r\n : mergedLabels.sessionExpired || 'Session expired'}\r\n </h3>\r\n <p class=\"timer-modal-text\">\r\n {canExtend\r\n ? mergedLabels.sessionExpiredMessage ||\r\n 'Your checkout session has expired and your selected tickets have been released.'\r\n : mergedLabels.reservationTimedOut ||\r\n 'Your reservation has timed out. Please select your tickets again to continue.'}\r\n </p>\r\n </div>\r\n {/snippet}\r\n {#snippet footer()}\r\n <div class=\"timer-modal-actions\">\r\n {#if canExtend}\r\n <Button\r\n variant=\"default\"\r\n size=\"lg\"\r\n class=\"w-full\"\r\n onclick={handleExtendSession}\r\n disabled={isExtendingSession}\r\n loading={isExtendingSession}\r\n >\r\n {isExtendingSession\r\n ? mergedLabels.extendingSession || 'Extending...'\r\n : mergedLabels.keepMyTickets || 'Keep my tickets'}\r\n </Button>\r\n <Button\r\n variant=\"alternative\"\r\n size=\"lg\"\r\n class=\"w-full\"\r\n onclick={handleRetryExpired}\r\n >\r\n {mergedLabels.selectTicketsAgain || 'Select tickets again'}\r\n </Button>\r\n {:else}\r\n <Button\r\n variant=\"default\"\r\n size=\"lg\"\r\n class=\"w-full\"\r\n onclick={handleRetryExpired}\r\n >\r\n {mergedLabels.selectTicketsAgain || 'Select tickets again'}\r\n </Button>\r\n {/if}\r\n </div>\r\n {/snippet}\r\n </Modal>\r\n {/if}\r\n</div>\r\n\r\n<style>\r\n /* Mobile order summary */\r\n .mobile-order-summary {\r\n padding: 1rem;\r\n border-radius: 0.5rem;\r\n border-width: 1px;\r\n border-style: solid;\r\n }\r\n\r\n /* Timer modals */\r\n .timer-modal {\r\n text-align: center;\r\n padding: 0.5rem 0;\r\n }\r\n\r\n .timer-modal-icon {\r\n display: inline-flex;\r\n align-items: center;\r\n justify-content: center;\r\n width: 3.5rem;\r\n height: 3.5rem;\r\n border-radius: 9999px;\r\n margin: 0 auto 1rem;\r\n }\r\n\r\n .timer-modal-icon.warning {\r\n background-color: hsl(var(--accent-warning) / 8%);\r\n color: hsl(var(--accent-warning));\r\n }\r\n\r\n .timer-modal-icon.expired {\r\n background-color: hsl(var(--accent-danger) / 5%);\r\n color: hsl(var(--accent-danger));\r\n }\r\n\r\n .timer-modal-title {\r\n font-size: 1.125rem;\r\n font-weight: 600;\r\n margin-bottom: 0.5rem;\r\n }\r\n\r\n .timer-modal-text {\r\n font-size: 0.875rem;\r\n line-height: 1.4;\r\n color: hsl(var(--text-tertiary)); /* gray-500 */\r\n }\r\n\r\n .timer-modal-hint {\r\n font-size: 0.75rem;\r\n color: hsl(var(--text-head)); /* gray-400 */\r\n margin-top: 0.5rem;\r\n }\r\n\r\n .timer-modal-actions {\r\n display: flex;\r\n flex-direction: column;\r\n gap: 0.5rem;\r\n }\r\n /* The Button component's class=\"w-full\" is a Tailwind utility that\r\n isn't always emitted in the venue-calendar bundle (no scanner\r\n pass over node_modules), so force full width at the layout\r\n level. Applies to both <button> and the anchor variant. */\r\n .timer-modal-actions :global(button),\r\n .timer-modal-actions :global(a) {\r\n width: 100%;\r\n }\r\n</style>\r\n"],"names":["createLogger","COOKIE_NAMES","asCookieName","parseCookies","acc","cookie","name","valueParts","getCookie","checked","$","$$props","indeterminate","value","disabled","color","className","inputEl","handleChange","colorClasses","colorClass","inputClasses","label","root_1","input","$$value","span","root","$$args","getBackend","kind","safeJsonParse","raw","fallback","read","key","backend","write","error","logger","getLocalStorage","setLocalStorage","registry","registerStorageKey","descriptor","THEME_STORAGE_KEY","LEGACY_THEME_KEY","readLegacyTheme","removeLegacyTheme","readThemeOverride","canonical","legacy","browser","detectDarkMode","containerSelector","saved","selector","containers","container","getStripeTheme","isDark","useStripeTheme","options","watchChanges","current","observer","updateTheme","onMount","onDestroy","stripe","paymentIntent","elements","paymentError","labels","stripeTheme","paymentMountDiv","paymentElement","mountedSecret","mountStripePayment","destroyStripePayment","firstName","lastName","email","keepUpdated","agreeTerms","cardNumber","cardExpiry","cardCvc","attendees","noPaymentRequired","submitted","errors","e","isEmailAddress","i","att","hasErrors","isTimerUrgent","totalTickets","sum","_","lastSubmitRequest","handleSubmit","handleSameAsPurchaser","index","a","div","root_9","div_1","Button","node","ChevronLeft","CheckoutTimer","node_1","section","Heading","node_2","div_2","$3","Input","node_3","node_4","node_5","Checkbox","node_6","section_1","node_8","node_9","attendee","div_3","div_4","Text","node_10","node_11","div_5","div_6","node_12","node_13","node_14","$$render","consequent","section_2","node_7","div_7","root_2","node_16","node_17","node_18","div_8","fragment_9","p","root_3","consequent_2","p_1","consequent_3","alternate","p_2","root_5","consequent_4","node_22","node_23","div_9","node_24","node_25","p_3","consequent_6","consequent_1","consequent_5","alternate_1","section_3","node_27","div_10","root_8","Warning","node_30","text_14","consequent_7","Card","$$anchor","fragment_1","Image","PLACEHOLDER_IMAGE","Calendar","formatEventTime","formatBrowseDate","text_3","OrderSummary","$0","text_6","formatCurrency","totalTicketCount","l","text_1","checkoutStateOptions","dataVariant","dataToggleStore","configProp","onPlaceOrderProp","onBackProp","onRetryExpiredProp","onExtendSessionProp","mergedLabels","mergeLabels","isShowcaseMode","buildConfig","variant","mockFreeEvent","toTicketId","mockDonationEvent","mockRegistrationEvent","mockEvent","config","deadlineMs","timerSeconds","isExpired","processing","submitRequested","showWarningModal","showExpiredModal","warningShown","isExtendingSession","remainingExtensions","canExtend","WARNING_THRESHOLD","timerDisplay","m","s","orderTotals","lines","subtotal","roundCurrency","fees","promoDiscount","adjustedSubtotal","taxes","calculateTax","totalsLocked","isFreeOrder","isRegistration","timerInitialized","c","getCartIdFromCookie","recomputeFromDeadline","resyncDeadlineFromServer","cartId","status","getSessionStatus","tick","next","interval","onVisible","handleBack","toast","requestPlaceOrder","handlePlaceOrder","formData","count","handleRetryExpired","handleExtendSession","result","AppError","err","dismissWarning","root_7","CheckoutForm","CheckoutSidebar","CheckoutMobileFooter","Modal","div_11","div_12","h3","div_13","root_4","div_14","div_15","Time","h3_1","div_16","node_19"],"mappings":"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","x_google_ignoreList":[0,1,2,3,4,6]}