@getmicdrop/venue-calendar 4.0.82 → 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 (94) hide show
  1. package/dist/CartView-DwAGELwG.js +2365 -0
  2. package/dist/CartView-DwAGELwG.js.map +1 -0
  3. package/dist/{Checkout-B15JygVy.js → Checkout-CHo11MsU.js} +466 -470
  4. package/dist/Checkout-CHo11MsU.js.map +1 -0
  5. package/dist/Checkout-_NuoBAKt.js +789 -0
  6. package/dist/Checkout-_NuoBAKt.js.map +1 -0
  7. package/dist/{CheckoutTimer-qLOtPDt3.js → CheckoutTimer-CAGR7L-7.js} +13 -13
  8. package/dist/{CheckoutTimer-qLOtPDt3.js.map → CheckoutTimer-CAGR7L-7.js.map} +1 -1
  9. package/dist/CollectionView-Dstgu8sR.js +299 -0
  10. package/dist/CollectionView-Dstgu8sR.js.map +1 -0
  11. package/dist/{Event-DZpU-Ec1.js → Event-kfRjkMzQ.js} +1875 -1249
  12. package/dist/Event-kfRjkMzQ.js.map +1 -0
  13. package/dist/EventPage-CnhccVCI.js +529 -0
  14. package/dist/EventPage-CnhccVCI.js.map +1 -0
  15. package/dist/{Heading-5CT9Nk0b.js → Heading-Ielp7dS-.js} +2 -2
  16. package/dist/{Heading-5CT9Nk0b.js.map → Heading-Ielp7dS-.js.map} +1 -1
  17. package/dist/{ModalHeader-CGaXzdcs.js → ModalHeader-Q7levd4l.js} +2 -2
  18. package/dist/{ModalHeader-CGaXzdcs.js.map → ModalHeader-Q7levd4l.js.map} +1 -1
  19. package/dist/ScarcityBadge-CG7Mz5YN.js +90 -0
  20. package/dist/ScarcityBadge-CG7Mz5YN.js.map +1 -0
  21. package/dist/SeriesPage-C8fIvt63.js +33 -0
  22. package/dist/SeriesPage-C8fIvt63.js.map +1 -0
  23. package/dist/Success-SZNXo-Ou.js +729 -0
  24. package/dist/{Success-B4CM_cy1.js.map → Success-SZNXo-Ou.js.map} +1 -1
  25. package/dist/{Text-WY42WIKI.js → Text-DHgQP4rA.js} +16 -16
  26. package/dist/{Text-WY42WIKI.js.map → Text-DHgQP4rA.js.map} +1 -1
  27. package/dist/{VenueCalendar-DMSeDRSN.js → VenueCalendar-DTOyAhDU.js} +3820 -4111
  28. package/dist/VenueCalendar-DTOyAhDU.js.map +1 -0
  29. package/dist/{ViewTicketsEmbed-AlVzNvxE.js → ViewTicketsEmbed-DBEOjijY.js} +31 -31
  30. package/dist/{ViewTicketsEmbed-AlVzNvxE.js.map → ViewTicketsEmbed-DBEOjijY.js.map} +1 -1
  31. package/dist/api/api.cjs.map +1 -1
  32. package/dist/api/api.mjs +3 -6
  33. package/dist/api/api.mjs.map +1 -1
  34. package/dist/{data-toggle-store.svelte-CtW4s3Vw.js → data-toggle-store.svelte-E2JyoGTV.js} +2 -2
  35. package/dist/{data-toggle-store.svelte-CtW4s3Vw.js.map → data-toggle-store.svelte-E2JyoGTV.js.map} +1 -1
  36. package/dist/{labels-BB1JG19g.js → labels-nQIooadc.js} +29 -30
  37. package/dist/{labels-BB1JG19g.js.map → labels-nQIooadc.js.map} +1 -1
  38. package/dist/seo/seo.cjs +1 -1
  39. package/dist/seo/seo.cjs.map +1 -1
  40. package/dist/seo/seo.mjs +79 -58
  41. package/dist/seo/seo.mjs.map +1 -1
  42. package/dist/{transform-BlXf-shc.js → transform-C1Rmuzlt.js} +2 -4
  43. package/dist/{transform-BlXf-shc.js.map → transform-C1Rmuzlt.js.map} +1 -1
  44. package/dist/venue-calendar.css +1 -1
  45. package/dist/venue-calendar.es.js +18 -18
  46. package/dist/venue-calendar.iife.js +31 -41
  47. package/dist/venue-calendar.iife.js.map +1 -1
  48. package/dist/venue-calendar.umd.js +24 -34
  49. package/dist/venue-calendar.umd.js.map +1 -1
  50. package/package.json +170 -170
  51. package/dist/CarouselView.legacy-DAixPl6l.js +0 -65
  52. package/dist/CarouselView.legacy-DAixPl6l.js.map +0 -1
  53. package/dist/CartView-CsdqWB24.js +0 -1744
  54. package/dist/CartView-CsdqWB24.js.map +0 -1
  55. package/dist/Checkout-B15JygVy.js.map +0 -1
  56. package/dist/Checkout-CUwOrErt.js +0 -831
  57. package/dist/Checkout-CUwOrErt.js.map +0 -1
  58. package/dist/Checkout.legacy-CrFgAYYe.js +0 -1156
  59. package/dist/Checkout.legacy-CrFgAYYe.js.map +0 -1
  60. package/dist/CollectionView-D2bRelLR.js +0 -338
  61. package/dist/CollectionView-D2bRelLR.js.map +0 -1
  62. package/dist/CollectionView.legacy-DLb5fzi-.js +0 -241
  63. package/dist/CollectionView.legacy-DLb5fzi-.js.map +0 -1
  64. package/dist/Event-DZpU-Ec1.js.map +0 -1
  65. package/dist/EventPage-CL0yuKWZ.js +0 -577
  66. package/dist/EventPage-CL0yuKWZ.js.map +0 -1
  67. package/dist/EventPage.legacy-EujdNc3A.js +0 -1324
  68. package/dist/EventPage.legacy-EujdNc3A.js.map +0 -1
  69. package/dist/FeaturedView.legacy-CbLBD-tn.js +0 -128
  70. package/dist/FeaturedView.legacy-CbLBD-tn.js.map +0 -1
  71. package/dist/GalleryCard-BcIcYghD.js +0 -93
  72. package/dist/GalleryCard-BcIcYghD.js.map +0 -1
  73. package/dist/GalleryView.legacy-XpQUg-5_.js +0 -52
  74. package/dist/GalleryView.legacy-XpQUg-5_.js.map +0 -1
  75. package/dist/GroupedListView.legacy-qvwIV5Qm.js +0 -145
  76. package/dist/GroupedListView.legacy-qvwIV5Qm.js.map +0 -1
  77. package/dist/OrderSummarySkeleton-pqfhjvKT.js +0 -632
  78. package/dist/OrderSummarySkeleton-pqfhjvKT.js.map +0 -1
  79. package/dist/ScarcityBadge-BkRFHRif.js +0 -77
  80. package/dist/ScarcityBadge-BkRFHRif.js.map +0 -1
  81. package/dist/SeriesPage-NWCA_Aez.js +0 -54
  82. package/dist/SeriesPage-NWCA_Aez.js.map +0 -1
  83. package/dist/SeriesPage.legacy-BXSwM5jV.js +0 -193
  84. package/dist/SeriesPage.legacy-BXSwM5jV.js.map +0 -1
  85. package/dist/Success-B4CM_cy1.js +0 -748
  86. package/dist/Success.legacy-BWAmPmZz.js +0 -192
  87. package/dist/Success.legacy-BWAmPmZz.js.map +0 -1
  88. package/dist/VenueCalendar-DMSeDRSN.js.map +0 -1
  89. package/dist/__SKIP_NAVIGATION__-CmipjatL.js +0 -18
  90. package/dist/__SKIP_NAVIGATION__-CmipjatL.js.map +0 -1
  91. package/dist/api-ChzBkXfo.js +0 -6
  92. package/dist/api-ChzBkXfo.js.map +0 -1
  93. package/dist/colors-CmP-sSZD.js +0 -56
  94. package/dist/colors-CmP-sSZD.js.map +0 -1
@@ -1 +1 @@
1
- {"version":3,"file":"seo.mjs","sources":["../../node_modules/@getmicdrop/svelte-components/dist/utils/timezones.js","../../src/lib/seo/helpers.ts","../../src/lib/seo/buildEventJsonLd.ts","../../src/lib/seo/buildSeriesJsonLd.ts","../../src/lib/seo/buildCollectionJsonLd.ts","../../src/lib/utils/textUtils.ts","../../src/lib/seo/HostSeoController.ts"],"sourcesContent":["/**\n * Timezone utility helpers.\n * Canonical source for timezone defaults and validation.\n */\n/**\n * Returns the default timezone for Micdrop, branded as `IANATimezone`.\n * Used when no explicit timezone is provided to date formatters. Bypass\n * call sites that pass `'America/Los_Angeles'` literally are red-squiggle\n * type errors at canonicals that take `IANATimezone` — the fix is to\n * thread `getDefaultTimezone()` through.\n *\n * @returns Branded IANA timezone (`'America/Los_Angeles'`)\n *\n * @example\n * formatDate(new Date(), {}, 'en-US', getDefaultTimezone())\n *\n * @deprecated Duplicate of the canonical default. Use `getDefaultTimezone`\n * (or the `DEFAULT_TIMEZONE` constant) from `$lib/datetime`\n * (re-exported via `datetime/constants`) — the single source of truth.\n */\nexport function getDefaultTimezone() {\n return 'America/Los_Angeles';\n}\n/**\n * Validates an IANA timezone string.\n * Uses Intl.DateTimeFormat to test if the timezone is recognized.\n *\n * @param tz - Timezone string to validate\n * @returns true if valid IANA timezone, false otherwise\n *\n * @example\n * isValidTimezone('America/Los_Angeles') // true\n * isValidTimezone('Invalid/Zone') // false\n *\n * @deprecated Lenient: `Intl.DateTimeFormat` accepts non-canonical inputs.\n * Use `isValidTimezone` from `$lib/datetime` (stricter — rejects\n * abbreviations like `'PST'`), or `toIANATimezone` from `$lib/types/brands`\n * when you need a branded value.\n */\nexport function isValidTimezone(tz) {\n try {\n new Intl.DateTimeFormat('en-US', { timeZone: tz });\n return true;\n }\n catch {\n return false;\n }\n}\n","/**\r\n * Shared helper utilities for JSON-LD builders.\r\n *\r\n * All functions are pure (no DOM, no Svelte, no side effects).\r\n * Extracted verbatim from EventStructuredData.svelte.\r\n *\r\n * @raw-intl-datetimeformat-escape: SEO/JSON-LD implementation. formatDateTimeWithOffset\r\n * must emit Google-required ISO-8601-with-tz-offset (\"2025-07-21T19:00:00-05:00\"), which\r\n * requires Intl.DateTimeFormat.formatToParts(...) with timeZoneName:'shortOffset' to read\r\n * the per-timezone UTC offset. The SC formatDate/DATE_FORMATS canonical produces\r\n * human-readable display strings, not parts-based offset extraction — it cannot express this.\r\n */\r\n\r\nimport { getDefaultTimezone } from '@getmicdrop/svelte-components/utils/timezones';\r\nimport type { EventTicketInput, EventPerformerInput } from './types';\r\n\r\n/**\r\n * Escape a JSON string for safe embedding inside an inline\r\n * <script type=\"application/ld+json\"> element.\r\n *\r\n * JSON.stringify does NOT escape `<`, `>` or `&`, so a field value containing\r\n * `</script><script>...` would break out of the script element and execute\r\n * (stored XSS). Replacing those characters with their unicode escape sequences\r\n * keeps the output valid JSON (parsers decode `<` back to `<`) while making\r\n * it impossible to terminate the surrounding <script> tag.\r\n *\r\n * This is the standard JSON-LD-in-<script> hardening; the sibling\r\n * HostSeoController avoids the issue entirely by using element.textContent.\r\n */\r\nexport function escapeJsonLdForScript(json: string): string {\r\n return json\r\n .replace(/</g, '\\\\u003c')\r\n .replace(/>/g, '\\\\u003e')\r\n .replace(/&/g, '\\\\u0026');\r\n}\r\n\r\n/**\r\n * Format datetime to ISO-8601 with timezone offset.\r\n * Google requires: \"2025-07-21T19:00:00-05:00\"\r\n *\r\n * Extracted from EventStructuredData.svelte lines 25-68.\r\n */\r\nexport function formatDateTimeWithOffset(\r\n isoString: string | null | undefined,\r\n tz: string\r\n): string | null {\r\n if (!isoString) return null;\r\n\r\n const date = new Date(isoString);\r\n if (isNaN(date.getTime())) return null;\r\n\r\n // Get the offset in the target timezone\r\n const offsetFormatOptions: Intl.DateTimeFormatOptions = {\r\n year: 'numeric',\r\n month: '2-digit',\r\n day: '2-digit',\r\n hour: '2-digit',\r\n minute: '2-digit',\r\n second: '2-digit',\r\n hour12: false,\r\n timeZoneName: 'shortOffset',\r\n };\r\n\r\n let formatter: Intl.DateTimeFormat;\r\n try {\r\n formatter = new Intl.DateTimeFormat('en-US', { ...offsetFormatOptions, timeZone: tz });\r\n } catch {\r\n // An invalid IANA timezone makes Intl.DateTimeFormat throw a RangeError,\r\n // which would otherwise escape the JSON-LD builders unguarded. Fall back\r\n // to the canonical default timezone — the same invalid-tz fallback the\r\n // rest of the repo uses ($lib/utils/datetime.js formatInTz).\r\n formatter = new Intl.DateTimeFormat('en-US', {\r\n ...offsetFormatOptions,\r\n timeZone: getDefaultTimezone(),\r\n });\r\n }\r\n\r\n const parts = formatter.formatToParts(date);\r\n const get = (type: string) => parts.find(p => p.type === type)?.value || '';\r\n\r\n const year = get('year');\r\n const month = get('month');\r\n const day = get('day');\r\n const hour = get('hour');\r\n const minute = get('minute');\r\n const second = get('second');\r\n const tzOffset = get('timeZoneName'); // e.g., \"GMT-5\" or \"GMT+2\"\r\n\r\n // Convert \"GMT-5\" to \"-05:00\" format\r\n let offset = '+00:00';\r\n if (tzOffset) {\r\n const match = tzOffset.match(/GMT([+-])(\\d+)(?::(\\d+))?/);\r\n if (match) {\r\n const sign = match[1];\r\n const hours = match[2].padStart(2, '0');\r\n const mins = match[3] || '00';\r\n offset = `${sign}${hours}:${mins}`;\r\n }\r\n }\r\n\r\n return `${year}-${month}-${day}T${hour}:${minute}:${second}${offset}`;\r\n}\r\n\r\n/**\r\n * Parse address string into PostalAddress components.\r\n * Input: \"7702 Santa Monica Blvd, West Hollywood, CA 90046\"\r\n * Output: { @type, streetAddress, addressLocality, addressRegion, postalCode, addressCountry }\r\n *\r\n * Extracted from EventStructuredData.svelte lines 75-111.\r\n */\r\nexport function parseAddressToPostal(\r\n addressStr: string | null | undefined\r\n): Record<string, string> | null {\r\n if (!addressStr) return null;\r\n\r\n const parts = addressStr.split(',').map(p => p.trim());\r\n\r\n if (parts.length < 2) {\r\n return {\r\n '@type': 'PostalAddress',\r\n streetAddress: addressStr,\r\n addressCountry: 'US',\r\n };\r\n }\r\n\r\n // Check if last part is \"State ZIP\" (e.g., \"CA 90046\")\r\n const lastPart = parts[parts.length - 1];\r\n const stateZipMatch = lastPart.match(/^([A-Z]{2})\\s+(\\d{5}(?:-\\d{4})?)$/);\r\n\r\n if (stateZipMatch && parts.length >= 3) {\r\n // Standard US address format\r\n return {\r\n '@type': 'PostalAddress',\r\n streetAddress: parts.slice(0, -2).join(', '),\r\n addressLocality: parts[parts.length - 2],\r\n addressRegion: stateZipMatch[1],\r\n postalCode: stateZipMatch[2],\r\n addressCountry: 'US',\r\n };\r\n }\r\n\r\n // Fallback for non-standard formats\r\n return {\r\n '@type': 'PostalAddress',\r\n streetAddress: parts[0],\r\n addressLocality: parts.slice(1).join(', '),\r\n addressCountry: 'US',\r\n };\r\n}\r\n\r\n/**\r\n * Structured postal-address parts, as the public resolve/series payloads now\r\n * expose them: `address` is the STREET line, with city/state/postalCode/country\r\n * as discrete fields (PublicVenueInfo / SeriesVenueDTO).\r\n */\r\nexport interface PostalAddressParts {\r\n address?: string | null; // street line\r\n city?: string | null;\r\n state?: string | null;\r\n postalCode?: string | null;\r\n country?: string | null;\r\n}\r\n\r\n/**\r\n * Build a schema.org PostalAddress, preferring discrete structured fields when\r\n * any locality signal (city / state / postalCode) is present, and otherwise\r\n * falling back to parsing a single comma-separated string (the street `address`\r\n * field if it carries a full address, else `googleLocationNameCache`).\r\n *\r\n * Discrete fields are more reliable than string-splitting — Google requires a\r\n * well-formed PostalAddress for Event rich-result eligibility (online-only\r\n * events lost eligibility June 2025).\r\n */\r\nexport function buildPostalAddress(\r\n parts: PostalAddressParts | null | undefined,\r\n fallback?: string | null\r\n): Record<string, string> | null {\r\n if (parts && (parts.city || parts.state || parts.postalCode)) {\r\n const out: Record<string, string> = { '@type': 'PostalAddress' };\r\n if (parts.address) out.streetAddress = parts.address;\r\n if (parts.city) out.addressLocality = parts.city;\r\n if (parts.state) out.addressRegion = parts.state;\r\n if (parts.postalCode) out.postalCode = parts.postalCode;\r\n out.addressCountry = parts.country || 'US';\r\n return out;\r\n }\r\n // No discrete locality fields — parse the best single string we have. The\r\n // street `address` may legacy-hold a full comma-separated address, so prefer\r\n // it over the location-name cache.\r\n return parseAddressToPostal((parts && parts.address) || fallback || null);\r\n}\r\n\r\n/**\r\n * Map event status to schema.org EventStatusType.\r\n *\r\n * Extracted from EventStructuredData.svelte lines 116-132.\r\n */\r\nexport function getEventStatus(rawStatus: string | null | undefined): string {\r\n if (!rawStatus) return 'https://schema.org/EventScheduled';\r\n\r\n const status = rawStatus.toLowerCase();\r\n\r\n if (status === 'cancelled' || status === 'canceled') {\r\n return 'https://schema.org/EventCancelled';\r\n }\r\n if (status === 'postponed') {\r\n return 'https://schema.org/EventPostponed';\r\n }\r\n if (status === 'rescheduled') {\r\n return 'https://schema.org/EventRescheduled';\r\n }\r\n\r\n return 'https://schema.org/EventScheduled';\r\n}\r\n\r\n/**\r\n * Map ticket availability to schema.org ItemAvailability.\r\n *\r\n * Extracted from EventStructuredData.svelte lines 137-157.\r\n */\r\nexport function getTicketAvailability(ticket: EventTicketInput): string {\r\n const remaining =\r\n ticket.remainingCapacity ?? ticket.quantityRemaining ?? ticket.quantity;\r\n\r\n if (remaining === 0 || ticket.soldOut) {\r\n return 'https://schema.org/SoldOut';\r\n }\r\n\r\n const now = new Date();\r\n const salesBegin =\r\n ticket.salesBegin ||\r\n ticket.salesStart ||\r\n ticket.saleBegin ||\r\n ticket.onSaleStart;\r\n const salesEnd = ticket.salesEnd || ticket.saleEnd || ticket.onSaleEnd;\r\n\r\n if (salesBegin && new Date(salesBegin) > now) {\r\n return 'https://schema.org/PreOrder';\r\n }\r\n\r\n if (salesEnd && new Date(salesEnd) < now) {\r\n return 'https://schema.org/SoldOut';\r\n }\r\n\r\n return 'https://schema.org/InStock';\r\n}\r\n\r\n/**\r\n * Build offers from available tickets.\r\n *\r\n * Extracted from EventStructuredData.svelte lines 162-208.\r\n */\r\nexport function buildOffers(\r\n tickets: EventTicketInput[] | null | undefined,\r\n url?: string\r\n): Record<string, unknown> | null {\r\n if (!tickets || tickets.length === 0) return null;\r\n\r\n // Filter to public tickets only (exclude door-only and hidden)\r\n const publicTickets = tickets.filter(t => {\r\n const visibility = t.visibility ?? 0;\r\n const isVisible = visibility === 0 || visibility === 1;\r\n const isPublic = t.salesChannel !== 2;\r\n return isVisible && isPublic;\r\n });\r\n\r\n if (publicTickets.length === 0) return null;\r\n\r\n // Find lowest and highest price\r\n const prices = publicTickets\r\n .map(t => parseFloat(String(t.price ?? 0)))\r\n .filter(p => !isNaN(p));\r\n\r\n if (prices.length === 0) return null;\r\n\r\n const lowPrice = Math.min(...prices);\r\n const highPrice = Math.max(...prices);\r\n\r\n // Use AggregateOffer when there are multiple price points\r\n if (lowPrice !== highPrice) {\r\n return {\r\n '@type': 'AggregateOffer',\r\n lowPrice: lowPrice,\r\n highPrice: highPrice,\r\n priceCurrency: 'USD',\r\n availability: getTicketAvailability(publicTickets[0]),\r\n url: url || undefined,\r\n validFrom:\r\n publicTickets[0]?.salesBegin ||\r\n publicTickets[0]?.salesStart ||\r\n undefined,\r\n };\r\n }\r\n\r\n // Single price point - use Offer\r\n const ticket = publicTickets[0];\r\n return {\r\n '@type': 'Offer',\r\n price: lowPrice,\r\n priceCurrency: 'USD',\r\n availability: getTicketAvailability(ticket),\r\n url: url || undefined,\r\n validFrom: ticket?.salesBegin || ticket?.salesStart || undefined,\r\n };\r\n}\r\n\r\n/**\r\n * Build performer array from event performers.\r\n *\r\n * Extracted from EventStructuredData.svelte lines 213-237.\r\n */\r\nexport function buildPerformers(\r\n performers: EventPerformerInput[] | null | undefined\r\n): Array<Record<string, string>> | null {\r\n if (!performers || performers.length === 0) return null;\r\n\r\n // Filter to confirmed, visible performers\r\n const confirmed = performers.filter(p => {\r\n const isConfirmed = p.acceptedState === 2;\r\n const isNotHidden = !p.shouldBeHidden;\r\n const profile = p.RosterPerformer?.User?.performerProfile;\r\n const hasName =\r\n profile?.firstName || profile?.lastName || profile?.displayName;\r\n return isConfirmed && isNotHidden && hasName;\r\n });\r\n\r\n if (confirmed.length === 0) return null;\r\n\r\n return confirmed.map(p => {\r\n const profile = p.RosterPerformer?.User?.performerProfile;\r\n const name =\r\n profile?.displayName ||\r\n [profile?.firstName, profile?.lastName].filter(Boolean).join(' ');\r\n\r\n return {\r\n '@type': 'PerformingGroup',\r\n name: name,\r\n };\r\n });\r\n}\r\n","import { getDefaultTimezone } from '@getmicdrop/svelte-components/utils/timezones';\n/**\n * Pure function that builds Event JSON-LD structured data for Google Events rich results.\n *\n * Extracted from EventStructuredData.svelte (lines 242-313).\n * Produces identical output to the original $derived.by block.\n */\n\nimport type { EventJsonLdInput, VenueInput, OrganizerInput } from './types';\nimport {\n formatDateTimeWithOffset,\n buildPostalAddress,\n getEventStatus,\n buildOffers,\n buildPerformers,\n} from './helpers';\n\nexport interface BuildEventJsonLdOptions {\n event: EventJsonLdInput;\n venue: VenueInput;\n venueTimeZone?: string;\n eventUrl?: string;\n organizer?: OrganizerInput;\n}\n\nexport function buildEventJsonLd(\n options: BuildEventJsonLdOptions\n): Record<string, unknown> | null {\n const {\n event,\n venue,\n venueTimeZone = getDefaultTimezone(),\n eventUrl,\n organizer,\n } = options;\n\n if (!event) return null;\n\n const startDate = formatDateTimeWithOffset(\n event.startDateTime,\n venueTimeZone\n );\n\n // Skip if we don't have required fields\n if (!event.name || !startDate || !venue?.name) {\n return null;\n }\n\n // Google requires a PostalAddress on a physical event's location (online-only\n // events lost rich-result eligibility June 2025). Prefer the venue's discrete\n // address fields (street/city/state/postalCode), falling back to parsing the\n // googleLocationNameCache string. Only attach `address` when we could build\n // one — never emit `address: null`. A missing address is a data-pipeline gap\n // (the resolve payload must carry the venue address).\n const postalAddress = buildPostalAddress(\n venue,\n venue?.googleLocationNameCache\n );\n const location: Record<string, unknown> = {\n '@type': 'Place',\n name: venue.name,\n };\n if (postalAddress) {\n location.address = postalAddress;\n }\n\n const data: Record<string, unknown> = {\n '@context': 'https://schema.org',\n '@type': 'Event',\n name: event.name,\n startDate: startDate,\n eventStatus: getEventStatus(event.status),\n eventAttendanceMode: 'https://schema.org/OfflineEventAttendanceMode',\n location,\n };\n\n // Add endDate if available\n const endDate = formatDateTimeWithOffset(event.endDateTime, venueTimeZone);\n if (endDate) {\n data.endDate = endDate;\n }\n\n // Add description\n const description = event.description || event.eventSummary;\n if (description) {\n data.description = description;\n }\n\n // Add image - Google recommends multiple aspect ratios\n const imageUrl = event.image || event.coverImage || event.ShowImage;\n if (imageUrl) {\n data.image = [imageUrl];\n }\n\n // Add offers from tickets\n const offers = buildOffers(event.availableTickets, eventUrl);\n if (offers) {\n data.offers = offers;\n }\n\n // Add performers\n const performers = buildPerformers(event.performers);\n if (performers && performers.length > 0) {\n data.performer = performers;\n }\n\n // Add organizer if provided\n if (organizer?.name) {\n const org: Record<string, string> = {\n '@type': 'Organization',\n name: organizer.name,\n };\n if (organizer.url) {\n org.url = organizer.url;\n }\n data.organizer = org;\n }\n\n // Add event URL if provided\n if (eventUrl) {\n data.url = eventUrl;\n }\n\n return data;\n}\n","/**\n * Pure function that builds EventSeries JSON-LD structured data.\n *\n * Series use schema.org EventSeries type with subEvent array containing\n * individual Event blocks for each occurrence. Each subEvent has its own\n * offers built from the simplified series ticket DTOs (different shape\n * from the event builder's tickets).\n *\n * Google Rich Results Test does NOT support EventSeries directly, but\n * the individual subEvent Event blocks can qualify for rich results.\n */\n\nimport type { SeriesJsonLdInput } from './types';\nimport {\n formatDateTimeWithOffset,\n buildPostalAddress,\n getEventStatus,\n} from './helpers';\n\nexport interface BuildSeriesJsonLdOptions {\n series: SeriesJsonLdInput;\n baseEventUrl?: string; // e.g., \"https://get-micdrop.com/e\" -- used to build subEvent @id URLs\n}\n\n/**\n * Map a series ticket status string to schema.org ItemAvailability.\n *\n * Series tickets use a simplified { name, price, status } shape from the\n * backend (SeriesTicketDTO), unlike event tickets which have visibility,\n * salesChannel, remainingCapacity, etc.\n */\nfunction mapSeriesTicketAvailability(status: string): string {\n switch (status) {\n case 'available':\n return 'https://schema.org/InStock';\n case 'sold_out':\n return 'https://schema.org/SoldOut';\n case 'coming_soon':\n return 'https://schema.org/PreOrder';\n case 'ended':\n return 'https://schema.org/SoldOut';\n default:\n return 'https://schema.org/InStock';\n }\n}\n\n/**\n * Build offers from series occurrence tickets.\n *\n * Series tickets have a simpler shape than event tickets: { name, price, status }.\n * Uses AggregateOffer for multiple price points, Offer for single price.\n */\nfunction buildSeriesOccurrenceOffers(\n tickets: Array<{ name: string; price: number; status: string }> | undefined,\n occurrenceStatus: string | undefined\n): Record<string, unknown> | null {\n if (!tickets || tickets.length === 0) return null;\n\n // Filter out sold_out and ended tickets for offer building\n const activeTickets = tickets.filter(\n t => t.status !== 'sold_out' && t.status !== 'ended'\n );\n\n // If all tickets are sold out / ended, report SoldOut with the price range\n if (activeTickets.length === 0) {\n const prices = tickets.map(t => t.price).filter(p => !isNaN(p));\n if (prices.length === 0) return null;\n\n const lowPrice = Math.min(...prices);\n const highPrice = Math.max(...prices);\n\n if (lowPrice !== highPrice) {\n return {\n '@type': 'AggregateOffer',\n lowPrice,\n highPrice,\n priceCurrency: 'USD',\n availability: 'https://schema.org/SoldOut',\n };\n }\n\n return {\n '@type': 'Offer',\n price: lowPrice,\n priceCurrency: 'USD',\n availability: 'https://schema.org/SoldOut',\n };\n }\n\n const prices = activeTickets.map(t => t.price).filter(p => !isNaN(p));\n if (prices.length === 0) return null;\n\n const lowPrice = Math.min(...prices);\n const highPrice = Math.max(...prices);\n\n if (lowPrice !== highPrice) {\n return {\n '@type': 'AggregateOffer',\n lowPrice,\n highPrice,\n priceCurrency: 'USD',\n availability: mapSeriesTicketAvailability(activeTickets[0].status),\n };\n }\n\n return {\n '@type': 'Offer',\n price: lowPrice,\n priceCurrency: 'USD',\n availability: mapSeriesTicketAvailability(activeTickets[0].status),\n };\n}\n\nexport function buildSeriesJsonLd(\n options: BuildSeriesJsonLdOptions\n): Record<string, unknown> | null {\n const { series, baseEventUrl } = options;\n\n // Required fields\n if (!series.title) return null;\n if (!series.occurrences || series.occurrences.length === 0) return null;\n if (!series.venue?.name) return null;\n\n // Build location — prefer the venue's discrete address fields, falling back to\n // parsing googleLocationNameCache (see buildEventJsonLd for the rationale).\n const address = buildPostalAddress(\n series.venue,\n series.venue.googleLocationNameCache\n );\n const location: Record<string, unknown> = {\n '@type': 'Place',\n name: series.venue.name,\n };\n if (address) {\n location.address = address;\n }\n\n // Build parent EventSeries\n const data: Record<string, unknown> = {\n '@context': 'https://schema.org',\n '@type': 'EventSeries',\n name: series.title,\n location,\n };\n\n // Description: prefer description, fall back to eventSummary\n const description = series.description || series.eventSummary;\n if (description) {\n data.description = description;\n }\n\n // Image\n if (series.image) {\n data.image = [series.image];\n }\n\n // Performers: flat displayName array (different from event builder's nested shape)\n const performers = series.performers?.filter(p => p.displayName);\n if (performers && performers.length > 0) {\n data.performer = performers.map(p => ({\n '@type': 'PerformingGroup',\n name: p.displayName,\n }));\n }\n\n // SubEvents: build Event blocks for each occurrence\n const subEvents: Record<string, unknown>[] = [];\n\n for (const occ of series.occurrences) {\n const startDate = formatDateTimeWithOffset(\n occ.startDateTime,\n series.timeZone\n );\n if (!startDate) continue; // Skip occurrences without valid startDate\n\n const subEvent: Record<string, unknown> = {\n '@type': 'Event',\n name: occ.title,\n startDate,\n eventStatus: getEventStatus(occ.status),\n location,\n };\n\n // @id from baseEventUrl\n if (baseEventUrl) {\n subEvent['@id'] = `${baseEventUrl}/${occ.id}-${occ.slug}`;\n }\n\n // endDate\n const endDate = formatDateTimeWithOffset(occ.endDateTime, series.timeZone);\n if (endDate) {\n subEvent.endDate = endDate;\n }\n\n // Offers from simplified ticket DTOs\n const offers = buildSeriesOccurrenceOffers(occ.tickets, occ.status);\n if (offers) {\n subEvent.offers = offers;\n }\n\n subEvents.push(subEvent);\n }\n\n // If all occurrences were skipped, return null\n if (subEvents.length === 0) return null;\n\n data.subEvent = subEvents;\n\n return data;\n}\n","/**\r\n * Pure function that builds CollectionPage JSON-LD for an event collection.\r\n *\r\n * A collection is a curated list of shows — NOT a single event, and NOT a\r\n * Festival (the product can't infer festival intent from a collection named\r\n * \"Detroit Comedy Festival\"; it's still just a collection). We emit a\r\n * schema.org `CollectionPage` whose `mainEntity` is a *summary-page* `ItemList`\r\n * — each `ListItem` is just `{ position, name, url }` pointing at the show's\r\n * canonical `/e/{id}-{slug}` leaf. The full `Event` rich-result markup lives on\r\n * each event's own page (one event = one leaf URL); duplicating full Event\r\n * objects here would create competing entities and trigger missing-field\r\n * (offers/address) warnings on the collection page.\r\n *\r\n * (Previously emitted `Festival` + `subEvent`, which Google's Rich Results\r\n * Test does not support.)\r\n */\r\n\r\nimport type { CollectionJsonLdInput } from './types';\r\n\r\nexport interface BuildCollectionJsonLdOptions {\r\n collection: CollectionJsonLdInput;\r\n baseEventUrl?: string; // e.g., \"https://get-micdrop.com/e\"\r\n /** Accepted for API compatibility; the summary ItemList carries no dates. */\r\n defaultTimeZone?: string;\r\n}\r\n\r\nexport function buildCollectionJsonLd(\r\n options: BuildCollectionJsonLdOptions,\r\n): Record<string, unknown> | null {\r\n const { collection, baseEventUrl } = options;\r\n\r\n // Required fields\r\n if (!collection.collectionTitle) return null;\r\n if (!collection.events || collection.events.length === 0) return null;\r\n\r\n // Build a summary ListItem per collection member: position + name + the\r\n // canonical leaf url. Prefer the slugged form, else bare `/e/{id}` (the app\r\n // self-heals bare ids to the slugged canonical).\r\n const itemListElement: Record<string, unknown>[] = [];\r\n\r\n for (const event of collection.events) {\r\n if (!event.title) continue; // Skip members with no name to list\r\n\r\n const listItem: Record<string, unknown> = {\r\n '@type': 'ListItem',\r\n position: itemListElement.length + 1,\r\n name: event.title,\r\n };\r\n\r\n if (baseEventUrl) {\r\n listItem.url = event.slug\r\n ? `${baseEventUrl}/${event.id}-${event.slug}`\r\n : `${baseEventUrl}/${event.id}`;\r\n }\r\n\r\n itemListElement.push(listItem);\r\n }\r\n\r\n // If every event was skipped, there's nothing to mark up.\r\n if (itemListElement.length === 0) return null;\r\n\r\n const data: Record<string, unknown> = {\r\n '@context': 'https://schema.org',\r\n '@type': 'CollectionPage',\r\n name: collection.collectionTitle,\r\n };\r\n\r\n // Description: prefer description, fall back to summary\r\n const description = collection.description || collection.summary;\r\n if (description) {\r\n data.description = description;\r\n }\r\n\r\n // Cover image\r\n if (collection.coverImage) {\r\n data.image = [collection.coverImage];\r\n }\r\n\r\n data.mainEntity = {\r\n '@type': 'ItemList',\r\n numberOfItems: itemListElement.length,\r\n itemListElement,\r\n };\r\n\r\n return data;\r\n}\r\n","/**\r\n * Text utility functions: generateSlug, validateNegativeNumber\r\n *\r\n * classNames and truncateTitle re-exported from @getmicdrop/svelte-components\r\n * via ./utils.js for backward compatibility with existing consumers.\r\n */\r\n\r\n/**\r\n * Generate a URL-friendly slug from a string\r\n */\r\nexport function generateSlug(text: string): string {\r\n if (!text) return '';\r\n\r\n return text\r\n .toLowerCase()\r\n .trim()\r\n .replace(/[^\\w\\s-]/g, '')\r\n .replace(/\\s+/g, '-')\r\n .replace(/--+/g, '-')\r\n .replace(/^-+|-+$/g, '');\r\n}\r\n\r\n/**\r\n * Validate that a value is a positive number\r\n * Returns an error message string, or empty string if valid\r\n */\r\nexport function validateNegativeNumber(\r\n value: string | number | null | undefined\r\n): string {\r\n if (value === null || value === undefined || value === '') {\r\n return 'Must be greater than zero.';\r\n }\r\n\r\n const stringValue = String(value).trim();\r\n\r\n if (\r\n stringValue.includes('--') ||\r\n stringValue.includes('++') ||\r\n stringValue.match(/^-+$/) ||\r\n stringValue.match(/^\\++$/)\r\n ) {\r\n return 'Please enter a valid number';\r\n }\r\n\r\n if (stringValue.match(/^-?\\d+-$/)) {\r\n return 'Please enter a valid number';\r\n }\r\n\r\n if ((stringValue.match(/\\./g) || []).length > 1) {\r\n return 'Please enter a valid number';\r\n }\r\n\r\n const numValue = Number(value);\r\n if (isNaN(numValue)) {\r\n return 'Please enter a valid number';\r\n }\r\n\r\n if (numValue < 0 || numValue === 0) {\r\n return 'Price must be greater than zero.';\r\n }\r\n\r\n return '';\r\n}\r\n","import { getDefaultTimezone } from '@getmicdrop/svelte-components/utils/timezones';\r\n/**\r\n * HostSeoController -- manages JSON-LD structured data, canonical URLs,\r\n * OG tags, meta descriptions, and page title injection into venue pages'\r\n * <head> when the Micdrop calendar widget is embedded.\r\n *\r\n * Follows a save-replace-restore pattern to be a \"good citizen\" on venue pages:\r\n * idle -> listing -> entity -> listing -> ... -> destroyed\r\n *\r\n * Uses textContent (never innerHTML) for JSON-LD script tags to prevent XSS.\r\n */\r\n\r\nimport type { HostSeoControllerOptions } from './types';\r\nimport { buildEventJsonLd } from './buildEventJsonLd';\r\nimport { buildSeriesJsonLd } from './buildSeriesJsonLd';\r\nimport { buildCollectionJsonLd } from './buildCollectionJsonLd';\r\nimport { generateSlug } from '$lib/utils/textUtils';\r\n\r\ntype ControllerState = 'idle' | 'listing' | 'entity' | 'destroyed';\r\n\r\ninterface SavedOriginals {\r\n title: string;\r\n canonicalHref: string | null;\r\n ogTitle: string | null;\r\n ogDescription: string | null;\r\n ogImage: string | null;\r\n ogUrl: string | null;\r\n ogType: string | null;\r\n metaDescription: string | null;\r\n}\r\n\r\n// Minimal event shape for listing mode\r\ninterface ListingEvent {\r\n id: number | string;\r\n name: string;\r\n slug?: string;\r\n startDateTime?: string;\r\n image?: string;\r\n}\r\n\r\n// Venue data passed at runtime\r\ninterface VenueData {\r\n name: string;\r\n address?: string;\r\n googleLocationNameCache?: string;\r\n // Transformed venues carry lowercase `timezone`; raw/host data may use\r\n // PascalCase `timeZone`. Accept both so the venue tz isn't silently dropped.\r\n timeZone?: string;\r\n timezone?: string;\r\n}\r\n\r\nexport class HostSeoController {\r\n /**\r\n * Single active instance. A venue page that mounts >1 calendar embed\r\n * (e.g. \"upcoming shows\" + \"this weekend\") will see the *later*\r\n * controller take over `<head>` — the earlier one is destroyed.\r\n *\r\n * KNOWN LIMITATION: multi-embed pages cannot run two SEO controllers\r\n * in parallel because they would both fight to own `<title>`/OG/etc.\r\n * Pass `data-inject-seo=\"false\"` on all but one embed to opt out.\r\n */\r\n static _activeInstance: HostSeoController | null = null;\r\n\r\n private _state: ControllerState = 'idle';\r\n private _venueName: string;\r\n private _venueAddress: string | undefined;\r\n private _venueTimeZone: string;\r\n private _baseUrl: string;\r\n private _originals: SavedOriginals | null = null;\r\n private _originalsSaved = false;\r\n private _events: ListingEvent[] = [];\r\n\r\n constructor(options: HostSeoControllerOptions) {\r\n // SSR guard\r\n if (typeof document === 'undefined') return;\r\n\r\n // If another instance exists, destroy it first.\r\n if (\r\n HostSeoController._activeInstance &&\r\n HostSeoController._activeInstance !== this\r\n ) {\r\n HostSeoController._activeInstance.destroy();\r\n }\r\n\r\n this._venueName = options.venueName;\r\n this._venueAddress = options.venueAddress;\r\n this._venueTimeZone = options.venueTimeZone || getDefaultTimezone();\r\n this._baseUrl =\r\n options.baseUrl || window.location.origin + window.location.pathname;\r\n\r\n this._saveOriginals();\r\n\r\n HostSeoController._activeInstance = this;\r\n }\r\n\r\n // ---------------------------------------------------------------------------\r\n // Private helpers\r\n // ---------------------------------------------------------------------------\r\n\r\n private _saveOriginals(): void {\r\n if (this._originalsSaved) return;\r\n\r\n const canonical = document.querySelector('link[rel=\"canonical\"]') as HTMLLinkElement | null;\r\n const ogTitle = document.querySelector('meta[property=\"og:title\"]') as HTMLMetaElement | null;\r\n const ogDescription = document.querySelector('meta[property=\"og:description\"]') as HTMLMetaElement | null;\r\n const ogImage = document.querySelector('meta[property=\"og:image\"]') as HTMLMetaElement | null;\r\n const ogUrl = document.querySelector('meta[property=\"og:url\"]') as HTMLMetaElement | null;\r\n const ogType = document.querySelector('meta[property=\"og:type\"]') as HTMLMetaElement | null;\r\n const metaDescription = document.querySelector('meta[name=\"description\"]') as HTMLMetaElement | null;\r\n\r\n this._originals = {\r\n title: document.title,\r\n canonicalHref: canonical?.getAttribute('href') ?? null,\r\n ogTitle: ogTitle?.getAttribute('content') ?? null,\r\n ogDescription: ogDescription?.getAttribute('content') ?? null,\r\n ogImage: ogImage?.getAttribute('content') ?? null,\r\n ogUrl: ogUrl?.getAttribute('content') ?? null,\r\n ogType: ogType?.getAttribute('content') ?? null,\r\n metaDescription: metaDescription?.getAttribute('content') ?? null,\r\n };\r\n\r\n this._originalsSaved = true;\r\n }\r\n\r\n private _injectJsonLd(data: Record<string, unknown>): void {\r\n if (typeof document === 'undefined') return;\r\n\r\n // Remove any existing micdrop JSON-LD\r\n document.querySelectorAll('script[data-micdrop=\"jsonld\"]').forEach((el) => el.remove());\r\n\r\n const script = document.createElement('script');\r\n script.setAttribute('type', 'application/ld+json');\r\n script.setAttribute('data-micdrop', 'jsonld');\r\n // Use textContent (NOT innerHTML) to prevent XSS\r\n script.textContent = JSON.stringify(data);\r\n document.head.appendChild(script);\r\n }\r\n\r\n private _setMeta(property: string, content: string): void {\r\n if (typeof document === 'undefined') return;\r\n\r\n if (property.startsWith('og:')) {\r\n // OG tags use property attribute\r\n let el = document.querySelector(`meta[property=\"${property}\"]`) as HTMLMetaElement | null;\r\n if (el) {\r\n el.setAttribute('content', content);\r\n } else {\r\n el = document.createElement('meta');\r\n el.setAttribute('property', property);\r\n el.setAttribute('content', content);\r\n el.setAttribute('data-micdrop', 'og');\r\n document.head.appendChild(el);\r\n }\r\n } else {\r\n // Standard meta tags use name attribute (e.g., description)\r\n let el = document.querySelector(`meta[name=\"${property}\"]`) as HTMLMetaElement | null;\r\n if (el) {\r\n el.setAttribute('content', content);\r\n } else {\r\n el = document.createElement('meta');\r\n el.setAttribute('name', property);\r\n el.setAttribute('content', content);\r\n el.setAttribute('data-micdrop', 'meta');\r\n document.head.appendChild(el);\r\n }\r\n }\r\n }\r\n\r\n private _setCanonical(url: string): void {\r\n if (typeof document === 'undefined') return;\r\n\r\n let el = document.querySelector('link[rel=\"canonical\"]') as HTMLLinkElement | null;\r\n if (el) {\r\n el.setAttribute('href', url);\r\n } else {\r\n el = document.createElement('link');\r\n el.setAttribute('rel', 'canonical');\r\n el.setAttribute('href', url);\r\n el.setAttribute('data-micdrop', 'canonical');\r\n document.head.appendChild(el);\r\n }\r\n }\r\n\r\n private _removeAllInjected(): void {\r\n if (typeof document === 'undefined') return;\r\n document.querySelectorAll('[data-micdrop]').forEach((el) => el.remove());\r\n }\r\n\r\n private _restoreOriginals(): void {\r\n if (typeof document === 'undefined' || !this._originals) return;\r\n\r\n // Restore title\r\n document.title = this._originals.title;\r\n\r\n // Restore canonical\r\n if (this._originals.canonicalHref !== null) {\r\n const canonical = document.querySelector('link[rel=\"canonical\"]') as HTMLLinkElement | null;\r\n if (canonical) {\r\n canonical.setAttribute('href', this._originals.canonicalHref);\r\n }\r\n }\r\n\r\n // Restore OG tags\r\n this._restoreMetaProperty('og:title', this._originals.ogTitle);\r\n this._restoreMetaProperty('og:description', this._originals.ogDescription);\r\n this._restoreMetaProperty('og:image', this._originals.ogImage);\r\n this._restoreMetaProperty('og:url', this._originals.ogUrl);\r\n this._restoreMetaProperty('og:type', this._originals.ogType);\r\n\r\n // Restore meta description\r\n this._restoreMetaName('description', this._originals.metaDescription);\r\n\r\n this._originalsSaved = false;\r\n }\r\n\r\n private _restoreMetaProperty(property: string, originalContent: string | null): void {\r\n const el = document.querySelector(`meta[property=\"${property}\"]`) as HTMLMetaElement | null;\r\n if (originalContent !== null && el) {\r\n // Had original value -- restore it\r\n el.setAttribute('content', originalContent);\r\n } else if (originalContent === null && el?.hasAttribute('data-micdrop')) {\r\n // We created it -- remove it\r\n el.remove();\r\n }\r\n }\r\n\r\n private _restoreMetaName(name: string, originalContent: string | null): void {\r\n const el = document.querySelector(`meta[name=\"${name}\"]`) as HTMLMetaElement | null;\r\n if (originalContent !== null && el) {\r\n el.setAttribute('content', originalContent);\r\n } else if (originalContent === null && el?.hasAttribute('data-micdrop')) {\r\n el.remove();\r\n }\r\n }\r\n\r\n private _buildItemList(events: ListingEvent[]): Record<string, unknown> {\r\n return {\r\n '@context': 'https://schema.org',\r\n '@type': 'ItemList',\r\n itemListElement: events.map((event, i) => {\r\n const slug = event.slug || generateSlug(event.name);\r\n return {\r\n '@type': 'ListItem',\r\n position: i + 1,\r\n url: `${this._baseUrl}?e=${event.id}-${slug}`,\r\n };\r\n }),\r\n };\r\n }\r\n\r\n private _buildEntityUrl(type: string, id: string | number, slug: string): string {\r\n const prefix = type === 'event' ? 'e' : type === 'series' ? 's' : 'c';\r\n return `${this._baseUrl}?${prefix}=${id}-${slug}`;\r\n }\r\n\r\n // ---------------------------------------------------------------------------\r\n // Public API\r\n // ---------------------------------------------------------------------------\r\n\r\n /**\r\n * Called when events finish loading from the API.\r\n * Injects an ItemList JSON-LD for the listing. Does not modify OG/canonical/title.\r\n */\r\n onEventsLoaded(\r\n events: ListingEvent[],\r\n venue?: VenueData,\r\n ): void {\r\n if (typeof document === 'undefined') return;\r\n if (this._state === 'destroyed') return;\r\n\r\n // Update venue info from parameter if provided\r\n if (venue) {\r\n this._venueName = venue.name || this._venueName;\r\n this._venueAddress = venue.address || venue.googleLocationNameCache || this._venueAddress;\r\n const venueTz = venue.timeZone || venue.timezone;\r\n if (venueTz) {\r\n this._venueTimeZone = venueTz;\r\n }\r\n }\r\n\r\n // Store events for later use (re-injection on back to listing)\r\n this._events = events;\r\n\r\n // Build and inject ItemList JSON-LD\r\n const itemList = this._buildItemList(events);\r\n this._injectJsonLd(itemList);\r\n\r\n this._state = 'listing';\r\n }\r\n\r\n /**\r\n * Called when the user selects an event, series, or collection.\r\n * Replaces ItemList JSON-LD with entity-specific JSON-LD and updates\r\n * all head tags (canonical, OG, meta description, title).\r\n */\r\n onEntitySelected(\r\n type: 'event' | 'series' | 'collection',\r\n data: any,\r\n venue?: VenueData,\r\n ): void {\r\n if (typeof document === 'undefined') return;\r\n if (this._state === 'destroyed') return;\r\n\r\n // Save originals if not already saved\r\n this._saveOriginals();\r\n\r\n // Update venue info from parameter if provided\r\n if (venue) {\r\n this._venueName = venue.name || this._venueName;\r\n this._venueAddress = venue.address || venue.googleLocationNameCache || this._venueAddress;\r\n const venueTz = venue.timeZone || venue.timezone;\r\n if (venueTz) {\r\n this._venueTimeZone = venueTz;\r\n }\r\n }\r\n\r\n let jsonLd: Record<string, unknown> | null = null;\r\n let entityName = '';\r\n let entityDescription = '';\r\n let entityImage = '';\r\n let entityUrl = '';\r\n\r\n if (type === 'event') {\r\n const slug = data.slug || generateSlug(data.name || '');\r\n entityUrl = this._buildEntityUrl('event', data.id || data.eventID || '', slug);\r\n\r\n jsonLd = buildEventJsonLd({\r\n event: data,\r\n venue: {\r\n name: this._venueName,\r\n googleLocationNameCache: this._venueAddress,\r\n },\r\n venueTimeZone: this._venueTimeZone,\r\n eventUrl: entityUrl,\r\n });\r\n\r\n entityName = data.name || '';\r\n entityDescription = data.description || data.eventSummary || '';\r\n entityImage = data.image || data.coverImage || data.ShowImage || '';\r\n } else if (type === 'series') {\r\n const slug = data.slug || generateSlug(data.title || '');\r\n entityUrl = this._buildEntityUrl('series', data.id || '', slug);\r\n\r\n jsonLd = buildSeriesJsonLd({\r\n series: {\r\n ...data,\r\n venue: {\r\n name: this._venueName,\r\n googleLocationNameCache: this._venueAddress,\r\n },\r\n timeZone: this._venueTimeZone,\r\n },\r\n baseEventUrl: entityUrl,\r\n });\r\n\r\n entityName = data.title || '';\r\n entityDescription = data.description || data.eventSummary || '';\r\n entityImage = data.image || '';\r\n } else if (type === 'collection') {\r\n const slug = data.slug || generateSlug(data.collectionTitle || '');\r\n entityUrl = this._buildEntityUrl('collection', data.id || '', slug);\r\n\r\n jsonLd = buildCollectionJsonLd({\r\n collection: data,\r\n baseEventUrl: entityUrl,\r\n defaultTimeZone: this._venueTimeZone,\r\n });\r\n\r\n entityName = data.collectionTitle || '';\r\n entityDescription = data.description || data.summary || '';\r\n entityImage = data.coverImage || '';\r\n }\r\n\r\n // Inject JSON-LD (replaces existing micdrop JSON-LD)\r\n if (jsonLd) {\r\n this._injectJsonLd(jsonLd);\r\n }\r\n\r\n // Update canonical URL\r\n if (entityUrl) {\r\n this._setCanonical(entityUrl);\r\n }\r\n\r\n // Update OG tags\r\n if (entityName) {\r\n this._setMeta('og:title', entityName);\r\n }\r\n if (entityDescription) {\r\n this._setMeta('og:description', entityDescription);\r\n }\r\n if (entityImage) {\r\n this._setMeta('og:image', entityImage);\r\n }\r\n if (entityUrl) {\r\n this._setMeta('og:url', entityUrl);\r\n }\r\n this._setMeta('og:type', 'website');\r\n\r\n // Twitter Card: large image preview\r\n this._setMeta('twitter:card', 'summary_large_image');\r\n\r\n // Update meta description\r\n if (entityDescription) {\r\n this._setMeta('description', entityDescription);\r\n }\r\n\r\n // Update page title\r\n if (entityName) {\r\n document.title = `${entityName} | ${this._venueName}`;\r\n }\r\n\r\n this._state = 'entity';\r\n }\r\n\r\n /**\r\n * Called when the user navigates back to the event listing.\r\n * Restores original head elements and re-injects ItemList JSON-LD.\r\n */\r\n onBackToListing(): void {\r\n if (typeof document === 'undefined') return;\r\n if (this._state === 'destroyed') return;\r\n\r\n // Restore all originals\r\n this._restoreOriginals();\r\n\r\n // Remove any remaining injected elements\r\n this._removeAllInjected();\r\n\r\n // Re-inject ItemList JSON-LD if events are available\r\n if (this._events.length > 0) {\r\n const itemList = this._buildItemList(this._events);\r\n this._injectJsonLd(itemList);\r\n }\r\n\r\n // Re-save originals since we just restored them\r\n this._saveOriginals();\r\n\r\n this._state = 'listing';\r\n }\r\n\r\n /**\r\n * Called when the widget is destroyed. Removes all injected elements\r\n * and restores originals. Clears the static active instance.\r\n */\r\n destroy(): void {\r\n if (typeof document === 'undefined') return;\r\n if (this._state === 'destroyed') return;\r\n\r\n this._removeAllInjected();\r\n this._restoreOriginals();\r\n\r\n if (HostSeoController._activeInstance === this) {\r\n HostSeoController._activeInstance = null;\r\n }\r\n\r\n this._state = 'destroyed';\r\n }\r\n}\r\n"],"names":["getDefaultTimezone","escapeJsonLdForScript","json","formatDateTimeWithOffset","isoString","tz","date","offsetFormatOptions","formatter","parts","get","type","p","year","month","day","hour","minute","second","tzOffset","offset","match","sign","hours","mins","parseAddressToPostal","addressStr","stateZipMatch","buildPostalAddress","fallback","out","getEventStatus","rawStatus","status","getTicketAvailability","ticket","now","salesBegin","salesEnd","buildOffers","tickets","url","publicTickets","t","visibility","isVisible","isPublic","prices","lowPrice","highPrice","buildPerformers","performers","confirmed","isConfirmed","isNotHidden","profile","hasName","buildEventJsonLd","options","event","venue","venueTimeZone","eventUrl","organizer","startDate","postalAddress","location","data","endDate","description","imageUrl","offers","org","mapSeriesTicketAvailability","buildSeriesOccurrenceOffers","occurrenceStatus","activeTickets","buildSeriesJsonLd","series","baseEventUrl","address","subEvents","occ","subEvent","buildCollectionJsonLd","collection","itemListElement","listItem","generateSlug","text","_HostSeoController","canonical","ogTitle","ogDescription","ogImage","ogUrl","ogType","metaDescription","el","script","property","content","originalContent","name","events","i","slug","id","prefix","venueTz","itemList","jsonLd","entityName","entityDescription","entityImage","entityUrl","HostSeoController"],"mappings":"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","x_google_ignoreList":[0]}
1
+ {"version":3,"file":"seo.mjs","sources":["../../node_modules/@getmicdrop/svelte-components/dist/utils/timezones.js","../../node_modules/@getmicdrop/svelte-components/dist/richtext/index.js","../../src/lib/seo/helpers.ts","../../src/lib/seo/buildEventJsonLd.ts","../../src/lib/seo/buildSeriesJsonLd.ts","../../src/lib/seo/buildCollectionJsonLd.ts","../../src/lib/utils/textUtils.ts","../../src/lib/seo/HostSeoController.ts"],"sourcesContent":["/**\n * Timezone utility helpers.\n * Canonical source for timezone defaults and validation.\n */\n/**\n * Returns the default timezone for Micdrop, branded as `IANATimezone`.\n * Used when no explicit timezone is provided to date formatters. Bypass\n * call sites that pass `'America/Los_Angeles'` literally are red-squiggle\n * type errors at canonicals that take `IANATimezone` — the fix is to\n * thread `getDefaultTimezone()` through.\n *\n * @returns Branded IANA timezone (`'America/Los_Angeles'`)\n *\n * @example\n * formatDate(new Date(), {}, 'en-US', getDefaultTimezone())\n *\n * @deprecated Duplicate of the canonical default. Use `getDefaultTimezone`\n * (or the `DEFAULT_TIMEZONE` constant) from `$lib/datetime`\n * (re-exported via `datetime/constants`) — the single source of truth.\n */\nexport function getDefaultTimezone() {\n return 'America/Los_Angeles';\n}\n/**\n * Validates an IANA timezone string.\n * Uses Intl.DateTimeFormat to test if the timezone is recognized.\n *\n * @param tz - Timezone string to validate\n * @returns true if valid IANA timezone, false otherwise\n *\n * @example\n * isValidTimezone('America/Los_Angeles') // true\n * isValidTimezone('Invalid/Zone') // false\n *\n * @deprecated Lenient: `Intl.DateTimeFormat` accepts non-canonical inputs.\n * Use `isValidTimezone` from `$lib/datetime` (stricter — rejects\n * abbreviations like `'PST'`), or `toIANATimezone` from `$lib/types/brands`\n * when you need a branded value.\n */\nexport function isValidTimezone(tz) {\n try {\n new Intl.DateTimeFormat('en-US', { timeZone: tz });\n return true;\n }\n catch {\n return false;\n }\n}\n","/**\n * Rich-text helpers for event / series descriptions.\n *\n * Pure string utilities — NO DOM access and NO external dependencies — so they\n * run safely during SSR (e.g. a `<svelte:head>` meta-description) as well as in\n * the browser. The companion {@link RichTextEditor} produces HTML that already\n * passes through an Eventbrite-parity allowlist; these helpers:\n *\n * - {@link normalizeEventDescriptionHtml} — promote legacy plain-text\n * descriptions to paragraph HTML so they don't render as a single line.\n * - {@link isEventDescriptionEqual} — compare two descriptions\n * whitespace-insensitively, so cosmetic edits don't trip dirty-checking.\n * - {@link stripEventDescriptionHtml} — flatten HTML to plain text (optionally\n * truncated) for meta tags / previews.\n *\n * @module richtext\n */\n/** Block-level closers that should read as a space when flattened to text. */\nconst BLOCK_CLOSE = /<\\/(?:p|div|h[1-6]|li|ul|ol|blockquote|tr)>/gi;\nconst LINE_BREAK = /<br\\s*\\/?>/gi;\nconst ANY_TAG = /<[^>]*>/g;\n/** A leading `<tag>` is a reliable \"this is already HTML\" signal. */\nconst LOOKS_LIKE_HTML = /<[a-z][\\s\\S]*>/i;\nconst NAMED_ENTITIES = {\n '&amp;': '&',\n '&lt;': '<',\n '&gt;': '>',\n '&quot;': '\"',\n '&#39;': \"'\",\n '&apos;': \"'\",\n '&nbsp;': ' ',\n};\nfunction decodeEntities(input) {\n return input\n .replace(/&#x([0-9a-f]+);/gi, (_, hex) => codePointOrEmpty(parseInt(hex, 16)))\n .replace(/&#(\\d+);/g, (_, dec) => codePointOrEmpty(Number(dec)))\n .replace(/&[a-z0-9]+;/gi, m => NAMED_ENTITIES[m.toLowerCase()] ?? m);\n}\nfunction codePointOrEmpty(code) {\n return Number.isFinite(code) && code > 0 && code <= 0x10ffff\n ? String.fromCodePoint(code)\n : '';\n}\nfunction escapeHtml(input) {\n return input\n .replace(/&/g, '&amp;')\n .replace(/</g, '&lt;')\n .replace(/>/g, '&gt;');\n}\n/**\n * Flatten a (possibly HTML) description to a single line of plain text.\n * When `maxLength` is given the result is truncated on a word boundary and an\n * ellipsis appended. Returns `''` for empty/missing input. SSR-safe.\n */\nexport function stripEventDescriptionHtml(html, maxLength) {\n if (!html)\n return '';\n let text = String(html)\n .replace(LINE_BREAK, ' ')\n .replace(BLOCK_CLOSE, ' ')\n .replace(ANY_TAG, '');\n text = decodeEntities(text)\n .replace(/\\s+/g, ' ')\n .trim();\n if (maxLength && maxLength > 0 && text.length > maxLength) {\n const clipped = text.slice(0, maxLength);\n const lastSpace = clipped.lastIndexOf(' ');\n text = (lastSpace > 40 ? clipped.slice(0, lastSpace) : clipped).trimEnd() + '…';\n }\n return text;\n}\n/**\n * Normalize a stored description for the rich-text editor. Existing HTML is\n * returned as-is (trimmed); legacy plain text is escaped and promoted to\n * paragraphs (blank lines → `<p>`, single newlines → `<br>`) so it renders with\n * its original line structure instead of collapsing to one line. SSR-safe.\n */\nexport function normalizeEventDescriptionHtml(html) {\n if (!html)\n return '';\n const value = String(html);\n if (LOOKS_LIKE_HTML.test(value))\n return value.trim();\n return value\n .replace(/\\r\\n/g, '\\n')\n .split(/\\n{2,}/)\n .map(block => block.trim())\n .filter(Boolean)\n .map(block => `<p>${escapeHtml(block).replace(/\\n/g, '<br>')}</p>`)\n .join('');\n}\n/** Collapse a description to a canonical form for equality comparison. */\nfunction canonicalize(html) {\n if (!html)\n return '';\n const collapsed = String(html)\n .replace(/>\\s+</g, '><')\n .replace(/\\s+/g, ' ')\n .trim();\n // An \"empty\" editor value (stray empty paragraphs / breaks) === no value.\n return /^(?:<p>(?:\\s|&nbsp;|<br\\s*\\/?>)*<\\/p>|<br\\s*\\/?>)*$/i.test(collapsed)\n ? ''\n : collapsed;\n}\n/**\n * True when two descriptions are equal ignoring cosmetic whitespace and\n * empty-paragraph noise — used for Save/dirty checking so a no-op edit doesn't\n * keep the form dirty. SSR-safe.\n */\nexport function isEventDescriptionEqual(a, b) {\n return canonicalize(a) === canonicalize(b);\n}\n// Sanitizer + shared allowlist — the security boundary for rendering stored\n// description HTML (which can arrive via CSV import / Eventbrite sync) on the\n// public event page + embed. `RichTextContent` calls this on render; callers\n// rendering description HTML directly with `{@html}` MUST sanitize here first.\nexport { sanitizeEventDescriptionHtml } from './sanitize.js';\nexport { EVENT_DESCRIPTION_TAGS, EVENT_DESCRIPTION_ATTRS, EVENT_DESCRIPTION_LINK_PROTOCOLS, EVENT_DESCRIPTION_WHITELIST, } from './allowlist.js';\n","/**\r\n * Shared helper utilities for JSON-LD builders.\r\n *\r\n * All functions are pure (no DOM, no Svelte, no side effects).\r\n * Extracted verbatim from EventStructuredData.svelte.\r\n *\r\n * @raw-intl-datetimeformat-escape: SEO/JSON-LD implementation. formatDateTimeWithOffset\r\n * must emit Google-required ISO-8601-with-tz-offset (\"2025-07-21T19:00:00-05:00\"), which\r\n * requires Intl.DateTimeFormat.formatToParts(...) with timeZoneName:'shortOffset' to read\r\n * the per-timezone UTC offset. The SC formatDate/DATE_FORMATS canonical produces\r\n * human-readable display strings, not parts-based offset extraction — it cannot express this.\r\n */\r\n\r\nimport { getDefaultTimezone } from '@getmicdrop/svelte-components/utils/timezones';\r\nimport type { EventTicketInput, EventPerformerInput } from './types';\r\n\r\n/**\r\n * Escape a JSON string for safe embedding inside an inline\r\n * <script type=\"application/ld+json\"> element.\r\n *\r\n * JSON.stringify does NOT escape `<`, `>` or `&`, so a field value containing\r\n * `</script><script>...` would break out of the script element and execute\r\n * (stored XSS). Replacing those characters with their unicode escape sequences\r\n * keeps the output valid JSON (parsers decode `<` back to `<`) while making\r\n * it impossible to terminate the surrounding <script> tag.\r\n *\r\n * This is the standard JSON-LD-in-<script> hardening; the sibling\r\n * HostSeoController avoids the issue entirely by using element.textContent.\r\n */\r\nexport function escapeJsonLdForScript(json: string): string {\r\n return json\r\n .replace(/</g, '\\\\u003c')\r\n .replace(/>/g, '\\\\u003e')\r\n .replace(/&/g, '\\\\u0026');\r\n}\r\n\r\n/**\r\n * Format datetime to ISO-8601 with timezone offset.\r\n * Google requires: \"2025-07-21T19:00:00-05:00\"\r\n *\r\n * Extracted from EventStructuredData.svelte lines 25-68.\r\n */\r\nexport function formatDateTimeWithOffset(\r\n isoString: string | null | undefined,\r\n tz: string\r\n): string | null {\r\n if (!isoString) return null;\r\n\r\n const date = new Date(isoString);\r\n if (isNaN(date.getTime())) return null;\r\n\r\n // Get the offset in the target timezone\r\n const offsetFormatOptions: Intl.DateTimeFormatOptions = {\r\n year: 'numeric',\r\n month: '2-digit',\r\n day: '2-digit',\r\n hour: '2-digit',\r\n minute: '2-digit',\r\n second: '2-digit',\r\n hour12: false,\r\n timeZoneName: 'shortOffset',\r\n };\r\n\r\n let formatter: Intl.DateTimeFormat;\r\n try {\r\n formatter = new Intl.DateTimeFormat('en-US', { ...offsetFormatOptions, timeZone: tz });\r\n } catch {\r\n // An invalid IANA timezone makes Intl.DateTimeFormat throw a RangeError,\r\n // which would otherwise escape the JSON-LD builders unguarded. Fall back\r\n // to the canonical default timezone — the same invalid-tz fallback the\r\n // rest of the repo uses ($lib/utils/datetime.js formatInTz).\r\n formatter = new Intl.DateTimeFormat('en-US', {\r\n ...offsetFormatOptions,\r\n timeZone: getDefaultTimezone(),\r\n });\r\n }\r\n\r\n const parts = formatter.formatToParts(date);\r\n const get = (type: string) => parts.find(p => p.type === type)?.value || '';\r\n\r\n const year = get('year');\r\n const month = get('month');\r\n const day = get('day');\r\n const hour = get('hour');\r\n const minute = get('minute');\r\n const second = get('second');\r\n const tzOffset = get('timeZoneName'); // e.g., \"GMT-5\" or \"GMT+2\"\r\n\r\n // Convert \"GMT-5\" to \"-05:00\" format\r\n let offset = '+00:00';\r\n if (tzOffset) {\r\n const match = tzOffset.match(/GMT([+-])(\\d+)(?::(\\d+))?/);\r\n if (match) {\r\n const sign = match[1];\r\n const hours = match[2].padStart(2, '0');\r\n const mins = match[3] || '00';\r\n offset = `${sign}${hours}:${mins}`;\r\n }\r\n }\r\n\r\n return `${year}-${month}-${day}T${hour}:${minute}:${second}${offset}`;\r\n}\r\n\r\n/**\r\n * Parse address string into PostalAddress components.\r\n * Input: \"7702 Santa Monica Blvd, West Hollywood, CA 90046\"\r\n * Output: { @type, streetAddress, addressLocality, addressRegion, postalCode, addressCountry }\r\n *\r\n * Extracted from EventStructuredData.svelte lines 75-111.\r\n */\r\nexport function parseAddressToPostal(\r\n addressStr: string | null | undefined\r\n): Record<string, string> | null {\r\n if (!addressStr) return null;\r\n\r\n const parts = addressStr.split(',').map(p => p.trim());\r\n\r\n if (parts.length < 2) {\r\n return {\r\n '@type': 'PostalAddress',\r\n streetAddress: addressStr,\r\n addressCountry: 'US',\r\n };\r\n }\r\n\r\n // Check if last part is \"State ZIP\" (e.g., \"CA 90046\")\r\n const lastPart = parts[parts.length - 1];\r\n const stateZipMatch = lastPart.match(/^([A-Z]{2})\\s+(\\d{5}(?:-\\d{4})?)$/);\r\n\r\n if (stateZipMatch && parts.length >= 3) {\r\n // Standard US address format\r\n return {\r\n '@type': 'PostalAddress',\r\n streetAddress: parts.slice(0, -2).join(', '),\r\n addressLocality: parts[parts.length - 2],\r\n addressRegion: stateZipMatch[1],\r\n postalCode: stateZipMatch[2],\r\n addressCountry: 'US',\r\n };\r\n }\r\n\r\n // Fallback for non-standard formats\r\n return {\r\n '@type': 'PostalAddress',\r\n streetAddress: parts[0],\r\n addressLocality: parts.slice(1).join(', '),\r\n addressCountry: 'US',\r\n };\r\n}\r\n\r\n/**\r\n * Structured postal-address parts, as the public resolve/series payloads now\r\n * expose them: `address` is the STREET line, with city/state/postalCode/country\r\n * as discrete fields (PublicVenueInfo / SeriesVenueDTO).\r\n */\r\nexport interface PostalAddressParts {\r\n address?: string | null; // street line\r\n city?: string | null;\r\n state?: string | null;\r\n postalCode?: string | null;\r\n country?: string | null;\r\n}\r\n\r\n/**\r\n * Build a schema.org PostalAddress, preferring discrete structured fields when\r\n * any locality signal (city / state / postalCode) is present, and otherwise\r\n * falling back to parsing a single comma-separated string (the street `address`\r\n * field if it carries a full address, else `googleLocationNameCache`).\r\n *\r\n * Discrete fields are more reliable than string-splitting — Google requires a\r\n * well-formed PostalAddress for Event rich-result eligibility (online-only\r\n * events lost eligibility June 2025).\r\n */\r\nexport function buildPostalAddress(\r\n parts: PostalAddressParts | null | undefined,\r\n fallback?: string | null\r\n): Record<string, string> | null {\r\n if (parts && (parts.city || parts.state || parts.postalCode)) {\r\n const out: Record<string, string> = { '@type': 'PostalAddress' };\r\n if (parts.address) out.streetAddress = parts.address;\r\n if (parts.city) out.addressLocality = parts.city;\r\n if (parts.state) out.addressRegion = parts.state;\r\n if (parts.postalCode) out.postalCode = parts.postalCode;\r\n out.addressCountry = parts.country || 'US';\r\n return out;\r\n }\r\n // No discrete locality fields — parse the best single string we have. The\r\n // street `address` may legacy-hold a full comma-separated address, so prefer\r\n // it over the location-name cache.\r\n return parseAddressToPostal((parts && parts.address) || fallback || null);\r\n}\r\n\r\n/**\r\n * Map event status to schema.org EventStatusType.\r\n *\r\n * Extracted from EventStructuredData.svelte lines 116-132.\r\n */\r\nexport function getEventStatus(rawStatus: string | null | undefined): string {\r\n if (!rawStatus) return 'https://schema.org/EventScheduled';\r\n\r\n const status = rawStatus.toLowerCase();\r\n\r\n if (status === 'cancelled' || status === 'canceled') {\r\n return 'https://schema.org/EventCancelled';\r\n }\r\n if (status === 'postponed') {\r\n return 'https://schema.org/EventPostponed';\r\n }\r\n if (status === 'rescheduled') {\r\n return 'https://schema.org/EventRescheduled';\r\n }\r\n\r\n return 'https://schema.org/EventScheduled';\r\n}\r\n\r\n/**\r\n * Map ticket availability to schema.org ItemAvailability.\r\n *\r\n * Extracted from EventStructuredData.svelte lines 137-157.\r\n */\r\nexport function getTicketAvailability(ticket: EventTicketInput): string {\r\n const remaining =\r\n ticket.remainingCapacity ?? ticket.quantityRemaining ?? ticket.quantity;\r\n\r\n if (remaining === 0 || ticket.soldOut) {\r\n return 'https://schema.org/SoldOut';\r\n }\r\n\r\n const now = new Date();\r\n const salesBegin =\r\n ticket.salesBegin ||\r\n ticket.salesStart ||\r\n ticket.saleBegin ||\r\n ticket.onSaleStart;\r\n const salesEnd = ticket.salesEnd || ticket.saleEnd || ticket.onSaleEnd;\r\n\r\n if (salesBegin && new Date(salesBegin) > now) {\r\n return 'https://schema.org/PreOrder';\r\n }\r\n\r\n if (salesEnd && new Date(salesEnd) < now) {\r\n return 'https://schema.org/SoldOut';\r\n }\r\n\r\n return 'https://schema.org/InStock';\r\n}\r\n\r\n/**\r\n * Build offers from available tickets.\r\n *\r\n * Extracted from EventStructuredData.svelte lines 162-208.\r\n */\r\nexport function buildOffers(\r\n tickets: EventTicketInput[] | null | undefined,\r\n url?: string\r\n): Record<string, unknown> | null {\r\n if (!tickets || tickets.length === 0) return null;\r\n\r\n // Filter to public tickets only (exclude door-only and hidden)\r\n const publicTickets = tickets.filter(t => {\r\n const visibility = t.visibility ?? 0;\r\n const isVisible = visibility === 0 || visibility === 1;\r\n const isPublic = t.salesChannel !== 2;\r\n return isVisible && isPublic;\r\n });\r\n\r\n if (publicTickets.length === 0) return null;\r\n\r\n // Find lowest and highest price\r\n const prices = publicTickets\r\n .map(t => parseFloat(String(t.price ?? 0)))\r\n .filter(p => !isNaN(p));\r\n\r\n if (prices.length === 0) return null;\r\n\r\n const lowPrice = Math.min(...prices);\r\n const highPrice = Math.max(...prices);\r\n\r\n // Use AggregateOffer when there are multiple price points\r\n if (lowPrice !== highPrice) {\r\n return {\r\n '@type': 'AggregateOffer',\r\n lowPrice: lowPrice,\r\n highPrice: highPrice,\r\n priceCurrency: 'USD',\r\n availability: getTicketAvailability(publicTickets[0]),\r\n url: url || undefined,\r\n validFrom:\r\n publicTickets[0]?.salesBegin ||\r\n publicTickets[0]?.salesStart ||\r\n undefined,\r\n };\r\n }\r\n\r\n // Single price point - use Offer\r\n const ticket = publicTickets[0];\r\n return {\r\n '@type': 'Offer',\r\n price: lowPrice,\r\n priceCurrency: 'USD',\r\n availability: getTicketAvailability(ticket),\r\n url: url || undefined,\r\n validFrom: ticket?.salesBegin || ticket?.salesStart || undefined,\r\n };\r\n}\r\n\r\n/**\r\n * Build performer array from event performers.\r\n *\r\n * Extracted from EventStructuredData.svelte lines 213-237.\r\n */\r\nexport function buildPerformers(\r\n performers: EventPerformerInput[] | null | undefined\r\n): Array<Record<string, string>> | null {\r\n if (!performers || performers.length === 0) return null;\r\n\r\n // Filter to confirmed, visible performers\r\n const confirmed = performers.filter(p => {\r\n const isConfirmed = p.acceptedState === 2;\r\n const isNotHidden = !p.shouldBeHidden;\r\n const profile = p.RosterPerformer?.User?.performerProfile;\r\n const hasName =\r\n profile?.firstName || profile?.lastName || profile?.displayName;\r\n return isConfirmed && isNotHidden && hasName;\r\n });\r\n\r\n if (confirmed.length === 0) return null;\r\n\r\n return confirmed.map(p => {\r\n const profile = p.RosterPerformer?.User?.performerProfile;\r\n const name =\r\n profile?.displayName ||\r\n [profile?.firstName, profile?.lastName].filter(Boolean).join(' ');\r\n\r\n return {\r\n '@type': 'PerformingGroup',\r\n name: name,\r\n };\r\n });\r\n}\r\n","import { getDefaultTimezone } from '@getmicdrop/svelte-components/utils/timezones';\r\nimport { stripEventDescriptionHtml } from '@getmicdrop/svelte-components/richtext';\r\n/**\r\n * Pure function that builds Event JSON-LD structured data for Google Events rich results.\r\n *\r\n * Extracted from EventStructuredData.svelte (lines 242-313).\r\n * Produces identical output to the original $derived.by block.\r\n */\r\n\r\nimport type { EventJsonLdInput, VenueInput, OrganizerInput } from './types';\r\nimport {\r\n formatDateTimeWithOffset,\r\n buildPostalAddress,\r\n getEventStatus,\r\n buildOffers,\r\n buildPerformers,\r\n} from './helpers';\r\n\r\nexport interface BuildEventJsonLdOptions {\r\n event: EventJsonLdInput;\r\n venue: VenueInput;\r\n venueTimeZone?: string;\r\n eventUrl?: string;\r\n organizer?: OrganizerInput;\r\n}\r\n\r\nexport function buildEventJsonLd(\r\n options: BuildEventJsonLdOptions\r\n): Record<string, unknown> | null {\r\n const {\r\n event,\r\n venue,\r\n venueTimeZone = getDefaultTimezone(),\r\n eventUrl,\r\n organizer,\r\n } = options;\r\n\r\n if (!event) return null;\r\n\r\n const startDate = formatDateTimeWithOffset(\r\n event.startDateTime,\r\n venueTimeZone\r\n );\r\n\r\n // Skip if we don't have required fields\r\n if (!event.name || !startDate || !venue?.name) {\r\n return null;\r\n }\r\n\r\n // Google requires a PostalAddress on a physical event's location (online-only\r\n // events lost rich-result eligibility June 2025). Prefer the venue's discrete\r\n // address fields (street/city/state/postalCode), falling back to parsing the\r\n // googleLocationNameCache string. Only attach `address` when we could build\r\n // one — never emit `address: null`. A missing address is a data-pipeline gap\r\n // (the resolve payload must carry the venue address).\r\n const postalAddress = buildPostalAddress(\r\n venue,\r\n venue?.googleLocationNameCache\r\n );\r\n const location: Record<string, unknown> = {\r\n '@type': 'Place',\r\n name: venue.name,\r\n };\r\n if (postalAddress) {\r\n location.address = postalAddress;\r\n }\r\n\r\n const data: Record<string, unknown> = {\r\n '@context': 'https://schema.org',\r\n '@type': 'Event',\r\n name: event.name,\r\n startDate: startDate,\r\n eventStatus: getEventStatus(event.status),\r\n eventAttendanceMode: 'https://schema.org/OfflineEventAttendanceMode',\r\n location,\r\n };\r\n\r\n // Add endDate if available\r\n const endDate = formatDateTimeWithOffset(event.endDateTime, venueTimeZone);\r\n if (endDate) {\r\n data.endDate = endDate;\r\n }\r\n\r\n // Add description\r\n const description = stripEventDescriptionHtml(event.description) || event.eventSummary;\r\n if (description) {\r\n data.description = description;\r\n }\r\n\r\n // Add image - Google recommends multiple aspect ratios\r\n const imageUrl = event.image || event.coverImage || event.ShowImage;\r\n if (imageUrl) {\r\n data.image = [imageUrl];\r\n }\r\n\r\n // Add offers from tickets\r\n const offers = buildOffers(event.availableTickets, eventUrl);\r\n if (offers) {\r\n data.offers = offers;\r\n }\r\n\r\n // Add performers\r\n const performers = buildPerformers(event.performers);\r\n if (performers && performers.length > 0) {\r\n data.performer = performers;\r\n }\r\n\r\n // Add organizer if provided\r\n if (organizer?.name) {\r\n const org: Record<string, string> = {\r\n '@type': 'Organization',\r\n name: organizer.name,\r\n };\r\n if (organizer.url) {\r\n org.url = organizer.url;\r\n }\r\n data.organizer = org;\r\n }\r\n\r\n // Add event URL if provided\r\n if (eventUrl) {\r\n data.url = eventUrl;\r\n }\r\n\r\n return data;\r\n}\r\n","/**\r\n * Pure function that builds EventSeries JSON-LD structured data.\r\n *\r\n * Series use schema.org EventSeries type with subEvent array containing\r\n * individual Event blocks for each occurrence. Each subEvent has its own\r\n * offers built from the simplified series ticket DTOs (different shape\r\n * from the event builder's tickets).\r\n *\r\n * Google Rich Results Test does NOT support EventSeries directly, but\r\n * the individual subEvent Event blocks can qualify for rich results.\r\n */\r\n\r\nimport type { SeriesJsonLdInput } from './types';\r\nimport { stripEventDescriptionHtml } from '@getmicdrop/svelte-components/richtext';\r\nimport {\r\n formatDateTimeWithOffset,\r\n buildPostalAddress,\r\n getEventStatus,\r\n} from './helpers';\r\n\r\nexport interface BuildSeriesJsonLdOptions {\r\n series: SeriesJsonLdInput;\r\n baseEventUrl?: string; // e.g., \"https://get-micdrop.com/e\" -- used to build subEvent @id URLs\r\n}\r\n\r\n/**\r\n * Map a series ticket status string to schema.org ItemAvailability.\r\n *\r\n * Series tickets use a simplified { name, price, status } shape from the\r\n * backend (SeriesTicketDTO), unlike event tickets which have visibility,\r\n * salesChannel, remainingCapacity, etc.\r\n */\r\nfunction mapSeriesTicketAvailability(status: string): string {\r\n switch (status) {\r\n case 'available':\r\n return 'https://schema.org/InStock';\r\n case 'sold_out':\r\n return 'https://schema.org/SoldOut';\r\n case 'coming_soon':\r\n return 'https://schema.org/PreOrder';\r\n case 'ended':\r\n return 'https://schema.org/SoldOut';\r\n default:\r\n return 'https://schema.org/InStock';\r\n }\r\n}\r\n\r\n/**\r\n * Build offers from series occurrence tickets.\r\n *\r\n * Series tickets have a simpler shape than event tickets: { name, price, status }.\r\n * Uses AggregateOffer for multiple price points, Offer for single price.\r\n */\r\nfunction buildSeriesOccurrenceOffers(\r\n tickets: Array<{ name: string; price: number; status: string }> | undefined,\r\n occurrenceStatus: string | undefined\r\n): Record<string, unknown> | null {\r\n if (!tickets || tickets.length === 0) return null;\r\n\r\n // Filter out sold_out and ended tickets for offer building\r\n const activeTickets = tickets.filter(\r\n t => t.status !== 'sold_out' && t.status !== 'ended'\r\n );\r\n\r\n // If all tickets are sold out / ended, report SoldOut with the price range\r\n if (activeTickets.length === 0) {\r\n const prices = tickets.map(t => t.price).filter(p => !isNaN(p));\r\n if (prices.length === 0) return null;\r\n\r\n const lowPrice = Math.min(...prices);\r\n const highPrice = Math.max(...prices);\r\n\r\n if (lowPrice !== highPrice) {\r\n return {\r\n '@type': 'AggregateOffer',\r\n lowPrice,\r\n highPrice,\r\n priceCurrency: 'USD',\r\n availability: 'https://schema.org/SoldOut',\r\n };\r\n }\r\n\r\n return {\r\n '@type': 'Offer',\r\n price: lowPrice,\r\n priceCurrency: 'USD',\r\n availability: 'https://schema.org/SoldOut',\r\n };\r\n }\r\n\r\n const prices = activeTickets.map(t => t.price).filter(p => !isNaN(p));\r\n if (prices.length === 0) return null;\r\n\r\n const lowPrice = Math.min(...prices);\r\n const highPrice = Math.max(...prices);\r\n\r\n if (lowPrice !== highPrice) {\r\n return {\r\n '@type': 'AggregateOffer',\r\n lowPrice,\r\n highPrice,\r\n priceCurrency: 'USD',\r\n availability: mapSeriesTicketAvailability(activeTickets[0].status),\r\n };\r\n }\r\n\r\n return {\r\n '@type': 'Offer',\r\n price: lowPrice,\r\n priceCurrency: 'USD',\r\n availability: mapSeriesTicketAvailability(activeTickets[0].status),\r\n };\r\n}\r\n\r\nexport function buildSeriesJsonLd(\r\n options: BuildSeriesJsonLdOptions\r\n): Record<string, unknown> | null {\r\n const { series, baseEventUrl } = options;\r\n\r\n // Required fields\r\n if (!series.title) return null;\r\n if (!series.occurrences || series.occurrences.length === 0) return null;\r\n if (!series.venue?.name) return null;\r\n\r\n // Build location — prefer the venue's discrete address fields, falling back to\r\n // parsing googleLocationNameCache (see buildEventJsonLd for the rationale).\r\n const address = buildPostalAddress(\r\n series.venue,\r\n series.venue.googleLocationNameCache\r\n );\r\n const location: Record<string, unknown> = {\r\n '@type': 'Place',\r\n name: series.venue.name,\r\n };\r\n if (address) {\r\n location.address = address;\r\n }\r\n\r\n // Build parent EventSeries\r\n const data: Record<string, unknown> = {\r\n '@context': 'https://schema.org',\r\n '@type': 'EventSeries',\r\n name: series.title,\r\n location,\r\n };\r\n\r\n // Description: prefer description, fall back to eventSummary\r\n const description = stripEventDescriptionHtml(series.description) || series.eventSummary;\r\n if (description) {\r\n data.description = description;\r\n }\r\n\r\n // Image\r\n if (series.image) {\r\n data.image = [series.image];\r\n }\r\n\r\n // Performers: flat displayName array (different from event builder's nested shape)\r\n const performers = series.performers?.filter(p => p.displayName);\r\n if (performers && performers.length > 0) {\r\n data.performer = performers.map(p => ({\r\n '@type': 'PerformingGroup',\r\n name: p.displayName,\r\n }));\r\n }\r\n\r\n // SubEvents: build Event blocks for each occurrence\r\n const subEvents: Record<string, unknown>[] = [];\r\n\r\n for (const occ of series.occurrences) {\r\n const startDate = formatDateTimeWithOffset(\r\n occ.startDateTime,\r\n series.timeZone\r\n );\r\n if (!startDate) continue; // Skip occurrences without valid startDate\r\n\r\n const subEvent: Record<string, unknown> = {\r\n '@type': 'Event',\r\n name: occ.title,\r\n startDate,\r\n eventStatus: getEventStatus(occ.status),\r\n location,\r\n };\r\n\r\n // @id from baseEventUrl\r\n if (baseEventUrl) {\r\n subEvent['@id'] = `${baseEventUrl}/${occ.id}-${occ.slug}`;\r\n }\r\n\r\n // endDate\r\n const endDate = formatDateTimeWithOffset(occ.endDateTime, series.timeZone);\r\n if (endDate) {\r\n subEvent.endDate = endDate;\r\n }\r\n\r\n // Offers from simplified ticket DTOs\r\n const offers = buildSeriesOccurrenceOffers(occ.tickets, occ.status);\r\n if (offers) {\r\n subEvent.offers = offers;\r\n }\r\n\r\n subEvents.push(subEvent);\r\n }\r\n\r\n // If all occurrences were skipped, return null\r\n if (subEvents.length === 0) return null;\r\n\r\n data.subEvent = subEvents;\r\n\r\n return data;\r\n}\r\n","/**\r\n * Pure function that builds CollectionPage JSON-LD for an event collection.\r\n *\r\n * A collection is a curated list of shows — NOT a single event, and NOT a\r\n * Festival (the product can't infer festival intent from a collection named\r\n * \"Detroit Comedy Festival\"; it's still just a collection). We emit a\r\n * schema.org `CollectionPage` whose `mainEntity` is a *summary-page* `ItemList`\r\n * — each `ListItem` is just `{ position, name, url }` pointing at the show's\r\n * canonical `/e/{id}-{slug}` leaf. The full `Event` rich-result markup lives on\r\n * each event's own page (one event = one leaf URL); duplicating full Event\r\n * objects here would create competing entities and trigger missing-field\r\n * (offers/address) warnings on the collection page.\r\n *\r\n * (Previously emitted `Festival` + `subEvent`, which Google's Rich Results\r\n * Test does not support.)\r\n */\r\n\r\nimport type { CollectionJsonLdInput } from './types';\r\nimport { stripEventDescriptionHtml } from '@getmicdrop/svelte-components/richtext';\r\n\r\nexport interface BuildCollectionJsonLdOptions {\r\n collection: CollectionJsonLdInput;\r\n baseEventUrl?: string; // e.g., \"https://get-micdrop.com/e\"\r\n /** Accepted for API compatibility; the summary ItemList carries no dates. */\r\n defaultTimeZone?: string;\r\n}\r\n\r\nexport function buildCollectionJsonLd(\r\n options: BuildCollectionJsonLdOptions,\r\n): Record<string, unknown> | null {\r\n const { collection, baseEventUrl } = options;\r\n\r\n // Required fields\r\n if (!collection.collectionTitle) return null;\r\n if (!collection.events || collection.events.length === 0) return null;\r\n\r\n // Build a summary ListItem per collection member: position + name + the\r\n // canonical leaf url. Prefer the slugged form, else bare `/e/{id}` (the app\r\n // self-heals bare ids to the slugged canonical).\r\n const itemListElement: Record<string, unknown>[] = [];\r\n\r\n for (const event of collection.events) {\r\n if (!event.title) continue; // Skip members with no name to list\r\n\r\n const listItem: Record<string, unknown> = {\r\n '@type': 'ListItem',\r\n position: itemListElement.length + 1,\r\n name: event.title,\r\n };\r\n\r\n if (baseEventUrl) {\r\n listItem.url = event.slug\r\n ? `${baseEventUrl}/${event.id}-${event.slug}`\r\n : `${baseEventUrl}/${event.id}`;\r\n }\r\n\r\n itemListElement.push(listItem);\r\n }\r\n\r\n // If every event was skipped, there's nothing to mark up.\r\n if (itemListElement.length === 0) return null;\r\n\r\n const data: Record<string, unknown> = {\r\n '@context': 'https://schema.org',\r\n '@type': 'CollectionPage',\r\n name: collection.collectionTitle,\r\n };\r\n\r\n // Description: prefer description, fall back to summary\r\n const description = stripEventDescriptionHtml(collection.description) || collection.summary;\r\n if (description) {\r\n data.description = description;\r\n }\r\n\r\n // Cover image\r\n if (collection.coverImage) {\r\n data.image = [collection.coverImage];\r\n }\r\n\r\n data.mainEntity = {\r\n '@type': 'ItemList',\r\n numberOfItems: itemListElement.length,\r\n itemListElement,\r\n };\r\n\r\n return data;\r\n}\r\n","/**\r\n * Text utility functions: generateSlug, validateNegativeNumber\r\n *\r\n * classNames and truncateTitle re-exported from @getmicdrop/svelte-components\r\n * via ./utils.js for backward compatibility with existing consumers.\r\n */\r\n\r\n/**\r\n * Generate a URL-friendly slug from a string\r\n */\r\nexport function generateSlug(text: string): string {\r\n if (!text) return '';\r\n\r\n return text\r\n .toLowerCase()\r\n .trim()\r\n .replace(/[^\\w\\s-]/g, '')\r\n .replace(/\\s+/g, '-')\r\n .replace(/--+/g, '-')\r\n .replace(/^-+|-+$/g, '');\r\n}\r\n\r\n/**\r\n * Validate that a value is a positive number\r\n * Returns an error message string, or empty string if valid\r\n */\r\nexport function validateNegativeNumber(\r\n value: string | number | null | undefined\r\n): string {\r\n if (value === null || value === undefined || value === '') {\r\n return 'Must be greater than zero.';\r\n }\r\n\r\n const stringValue = String(value).trim();\r\n\r\n if (\r\n stringValue.includes('--') ||\r\n stringValue.includes('++') ||\r\n stringValue.match(/^-+$/) ||\r\n stringValue.match(/^\\++$/)\r\n ) {\r\n return 'Please enter a valid number';\r\n }\r\n\r\n if (stringValue.match(/^-?\\d+-$/)) {\r\n return 'Please enter a valid number';\r\n }\r\n\r\n if ((stringValue.match(/\\./g) || []).length > 1) {\r\n return 'Please enter a valid number';\r\n }\r\n\r\n const numValue = Number(value);\r\n if (isNaN(numValue)) {\r\n return 'Please enter a valid number';\r\n }\r\n\r\n if (numValue < 0 || numValue === 0) {\r\n return 'Price must be greater than zero.';\r\n }\r\n\r\n return '';\r\n}\r\n","import { getDefaultTimezone } from '@getmicdrop/svelte-components/utils/timezones';\r\n/**\r\n * HostSeoController -- manages JSON-LD structured data, canonical URLs,\r\n * OG tags, meta descriptions, and page title injection into venue pages'\r\n * <head> when the Micdrop calendar widget is embedded.\r\n *\r\n * Follows a save-replace-restore pattern to be a \"good citizen\" on venue pages:\r\n * idle -> listing -> entity -> listing -> ... -> destroyed\r\n *\r\n * Uses textContent (never innerHTML) for JSON-LD script tags to prevent XSS.\r\n */\r\n\r\nimport type { HostSeoControllerOptions } from './types';\r\nimport { buildEventJsonLd } from './buildEventJsonLd';\r\nimport { buildSeriesJsonLd } from './buildSeriesJsonLd';\r\nimport { buildCollectionJsonLd } from './buildCollectionJsonLd';\r\nimport { generateSlug } from '$lib/utils/textUtils';\r\n\r\ntype ControllerState = 'idle' | 'listing' | 'entity' | 'destroyed';\r\n\r\ninterface SavedOriginals {\r\n title: string;\r\n canonicalHref: string | null;\r\n ogTitle: string | null;\r\n ogDescription: string | null;\r\n ogImage: string | null;\r\n ogUrl: string | null;\r\n ogType: string | null;\r\n metaDescription: string | null;\r\n}\r\n\r\n// Minimal event shape for listing mode\r\ninterface ListingEvent {\r\n id: number | string;\r\n name: string;\r\n slug?: string;\r\n startDateTime?: string;\r\n image?: string;\r\n}\r\n\r\n// Venue data passed at runtime\r\ninterface VenueData {\r\n name: string;\r\n address?: string;\r\n googleLocationNameCache?: string;\r\n // Transformed venues carry lowercase `timezone`; raw/host data may use\r\n // PascalCase `timeZone`. Accept both so the venue tz isn't silently dropped.\r\n timeZone?: string;\r\n timezone?: string;\r\n}\r\n\r\nexport class HostSeoController {\r\n /**\r\n * Single active instance. A venue page that mounts >1 calendar embed\r\n * (e.g. \"upcoming shows\" + \"this weekend\") will see the *later*\r\n * controller take over `<head>` — the earlier one is destroyed.\r\n *\r\n * KNOWN LIMITATION: multi-embed pages cannot run two SEO controllers\r\n * in parallel because they would both fight to own `<title>`/OG/etc.\r\n * Pass `data-inject-seo=\"false\"` on all but one embed to opt out.\r\n */\r\n static _activeInstance: HostSeoController | null = null;\r\n\r\n private _state: ControllerState = 'idle';\r\n private _venueName: string;\r\n private _venueAddress: string | undefined;\r\n private _venueTimeZone: string;\r\n private _baseUrl: string;\r\n private _originals: SavedOriginals | null = null;\r\n private _originalsSaved = false;\r\n private _events: ListingEvent[] = [];\r\n\r\n constructor(options: HostSeoControllerOptions) {\r\n // SSR guard\r\n if (typeof document === 'undefined') return;\r\n\r\n // If another instance exists, destroy it first.\r\n if (\r\n HostSeoController._activeInstance &&\r\n HostSeoController._activeInstance !== this\r\n ) {\r\n HostSeoController._activeInstance.destroy();\r\n }\r\n\r\n this._venueName = options.venueName;\r\n this._venueAddress = options.venueAddress;\r\n this._venueTimeZone = options.venueTimeZone || getDefaultTimezone();\r\n this._baseUrl =\r\n options.baseUrl || window.location.origin + window.location.pathname;\r\n\r\n this._saveOriginals();\r\n\r\n HostSeoController._activeInstance = this;\r\n }\r\n\r\n // ---------------------------------------------------------------------------\r\n // Private helpers\r\n // ---------------------------------------------------------------------------\r\n\r\n private _saveOriginals(): void {\r\n if (this._originalsSaved) return;\r\n\r\n const canonical = document.querySelector('link[rel=\"canonical\"]') as HTMLLinkElement | null;\r\n const ogTitle = document.querySelector('meta[property=\"og:title\"]') as HTMLMetaElement | null;\r\n const ogDescription = document.querySelector('meta[property=\"og:description\"]') as HTMLMetaElement | null;\r\n const ogImage = document.querySelector('meta[property=\"og:image\"]') as HTMLMetaElement | null;\r\n const ogUrl = document.querySelector('meta[property=\"og:url\"]') as HTMLMetaElement | null;\r\n const ogType = document.querySelector('meta[property=\"og:type\"]') as HTMLMetaElement | null;\r\n const metaDescription = document.querySelector('meta[name=\"description\"]') as HTMLMetaElement | null;\r\n\r\n this._originals = {\r\n title: document.title,\r\n canonicalHref: canonical?.getAttribute('href') ?? null,\r\n ogTitle: ogTitle?.getAttribute('content') ?? null,\r\n ogDescription: ogDescription?.getAttribute('content') ?? null,\r\n ogImage: ogImage?.getAttribute('content') ?? null,\r\n ogUrl: ogUrl?.getAttribute('content') ?? null,\r\n ogType: ogType?.getAttribute('content') ?? null,\r\n metaDescription: metaDescription?.getAttribute('content') ?? null,\r\n };\r\n\r\n this._originalsSaved = true;\r\n }\r\n\r\n private _injectJsonLd(data: Record<string, unknown>): void {\r\n if (typeof document === 'undefined') return;\r\n\r\n // Remove any existing micdrop JSON-LD\r\n document.querySelectorAll('script[data-micdrop=\"jsonld\"]').forEach((el) => el.remove());\r\n\r\n const script = document.createElement('script');\r\n script.setAttribute('type', 'application/ld+json');\r\n script.setAttribute('data-micdrop', 'jsonld');\r\n // Use textContent (NOT innerHTML) to prevent XSS\r\n script.textContent = JSON.stringify(data);\r\n document.head.appendChild(script);\r\n }\r\n\r\n private _setMeta(property: string, content: string): void {\r\n if (typeof document === 'undefined') return;\r\n\r\n if (property.startsWith('og:')) {\r\n // OG tags use property attribute\r\n let el = document.querySelector(`meta[property=\"${property}\"]`) as HTMLMetaElement | null;\r\n if (el) {\r\n el.setAttribute('content', content);\r\n } else {\r\n el = document.createElement('meta');\r\n el.setAttribute('property', property);\r\n el.setAttribute('content', content);\r\n el.setAttribute('data-micdrop', 'og');\r\n document.head.appendChild(el);\r\n }\r\n } else {\r\n // Standard meta tags use name attribute (e.g., description)\r\n let el = document.querySelector(`meta[name=\"${property}\"]`) as HTMLMetaElement | null;\r\n if (el) {\r\n el.setAttribute('content', content);\r\n } else {\r\n el = document.createElement('meta');\r\n el.setAttribute('name', property);\r\n el.setAttribute('content', content);\r\n el.setAttribute('data-micdrop', 'meta');\r\n document.head.appendChild(el);\r\n }\r\n }\r\n }\r\n\r\n private _setCanonical(url: string): void {\r\n if (typeof document === 'undefined') return;\r\n\r\n let el = document.querySelector('link[rel=\"canonical\"]') as HTMLLinkElement | null;\r\n if (el) {\r\n el.setAttribute('href', url);\r\n } else {\r\n el = document.createElement('link');\r\n el.setAttribute('rel', 'canonical');\r\n el.setAttribute('href', url);\r\n el.setAttribute('data-micdrop', 'canonical');\r\n document.head.appendChild(el);\r\n }\r\n }\r\n\r\n private _removeAllInjected(): void {\r\n if (typeof document === 'undefined') return;\r\n document.querySelectorAll('[data-micdrop]').forEach((el) => el.remove());\r\n }\r\n\r\n private _restoreOriginals(): void {\r\n if (typeof document === 'undefined' || !this._originals) return;\r\n\r\n // Restore title\r\n document.title = this._originals.title;\r\n\r\n // Restore canonical\r\n if (this._originals.canonicalHref !== null) {\r\n const canonical = document.querySelector('link[rel=\"canonical\"]') as HTMLLinkElement | null;\r\n if (canonical) {\r\n canonical.setAttribute('href', this._originals.canonicalHref);\r\n }\r\n }\r\n\r\n // Restore OG tags\r\n this._restoreMetaProperty('og:title', this._originals.ogTitle);\r\n this._restoreMetaProperty('og:description', this._originals.ogDescription);\r\n this._restoreMetaProperty('og:image', this._originals.ogImage);\r\n this._restoreMetaProperty('og:url', this._originals.ogUrl);\r\n this._restoreMetaProperty('og:type', this._originals.ogType);\r\n\r\n // Restore meta description\r\n this._restoreMetaName('description', this._originals.metaDescription);\r\n\r\n this._originalsSaved = false;\r\n }\r\n\r\n private _restoreMetaProperty(property: string, originalContent: string | null): void {\r\n const el = document.querySelector(`meta[property=\"${property}\"]`) as HTMLMetaElement | null;\r\n if (originalContent !== null && el) {\r\n // Had original value -- restore it\r\n el.setAttribute('content', originalContent);\r\n } else if (originalContent === null && el?.hasAttribute('data-micdrop')) {\r\n // We created it -- remove it\r\n el.remove();\r\n }\r\n }\r\n\r\n private _restoreMetaName(name: string, originalContent: string | null): void {\r\n const el = document.querySelector(`meta[name=\"${name}\"]`) as HTMLMetaElement | null;\r\n if (originalContent !== null && el) {\r\n el.setAttribute('content', originalContent);\r\n } else if (originalContent === null && el?.hasAttribute('data-micdrop')) {\r\n el.remove();\r\n }\r\n }\r\n\r\n private _buildItemList(events: ListingEvent[]): Record<string, unknown> {\r\n return {\r\n '@context': 'https://schema.org',\r\n '@type': 'ItemList',\r\n itemListElement: events.map((event, i) => {\r\n const slug = event.slug || generateSlug(event.name);\r\n return {\r\n '@type': 'ListItem',\r\n position: i + 1,\r\n url: `${this._baseUrl}?e=${event.id}-${slug}`,\r\n };\r\n }),\r\n };\r\n }\r\n\r\n private _buildEntityUrl(type: string, id: string | number, slug: string): string {\r\n const prefix = type === 'event' ? 'e' : type === 'series' ? 's' : 'c';\r\n return `${this._baseUrl}?${prefix}=${id}-${slug}`;\r\n }\r\n\r\n // ---------------------------------------------------------------------------\r\n // Public API\r\n // ---------------------------------------------------------------------------\r\n\r\n /**\r\n * Called when events finish loading from the API.\r\n * Injects an ItemList JSON-LD for the listing. Does not modify OG/canonical/title.\r\n */\r\n onEventsLoaded(\r\n events: ListingEvent[],\r\n venue?: VenueData,\r\n ): void {\r\n if (typeof document === 'undefined') return;\r\n if (this._state === 'destroyed') return;\r\n\r\n // Update venue info from parameter if provided\r\n if (venue) {\r\n this._venueName = venue.name || this._venueName;\r\n this._venueAddress = venue.address || venue.googleLocationNameCache || this._venueAddress;\r\n const venueTz = venue.timeZone || venue.timezone;\r\n if (venueTz) {\r\n this._venueTimeZone = venueTz;\r\n }\r\n }\r\n\r\n // Store events for later use (re-injection on back to listing)\r\n this._events = events;\r\n\r\n // Build and inject ItemList JSON-LD\r\n const itemList = this._buildItemList(events);\r\n this._injectJsonLd(itemList);\r\n\r\n this._state = 'listing';\r\n }\r\n\r\n /**\r\n * Called when the user selects an event, series, or collection.\r\n * Replaces ItemList JSON-LD with entity-specific JSON-LD and updates\r\n * all head tags (canonical, OG, meta description, title).\r\n */\r\n onEntitySelected(\r\n type: 'event' | 'series' | 'collection',\r\n data: any,\r\n venue?: VenueData,\r\n ): void {\r\n if (typeof document === 'undefined') return;\r\n if (this._state === 'destroyed') return;\r\n\r\n // Save originals if not already saved\r\n this._saveOriginals();\r\n\r\n // Update venue info from parameter if provided\r\n if (venue) {\r\n this._venueName = venue.name || this._venueName;\r\n this._venueAddress = venue.address || venue.googleLocationNameCache || this._venueAddress;\r\n const venueTz = venue.timeZone || venue.timezone;\r\n if (venueTz) {\r\n this._venueTimeZone = venueTz;\r\n }\r\n }\r\n\r\n let jsonLd: Record<string, unknown> | null = null;\r\n let entityName = '';\r\n let entityDescription = '';\r\n let entityImage = '';\r\n let entityUrl = '';\r\n\r\n if (type === 'event') {\r\n const slug = data.slug || generateSlug(data.name || '');\r\n entityUrl = this._buildEntityUrl('event', data.id || data.eventID || '', slug);\r\n\r\n jsonLd = buildEventJsonLd({\r\n event: data,\r\n venue: {\r\n name: this._venueName,\r\n googleLocationNameCache: this._venueAddress,\r\n },\r\n venueTimeZone: this._venueTimeZone,\r\n eventUrl: entityUrl,\r\n });\r\n\r\n entityName = data.name || '';\r\n entityDescription = data.description || data.eventSummary || '';\r\n entityImage = data.image || data.coverImage || data.ShowImage || '';\r\n } else if (type === 'series') {\r\n const slug = data.slug || generateSlug(data.title || '');\r\n entityUrl = this._buildEntityUrl('series', data.id || '', slug);\r\n\r\n jsonLd = buildSeriesJsonLd({\r\n series: {\r\n ...data,\r\n venue: {\r\n name: this._venueName,\r\n googleLocationNameCache: this._venueAddress,\r\n },\r\n timeZone: this._venueTimeZone,\r\n },\r\n baseEventUrl: entityUrl,\r\n });\r\n\r\n entityName = data.title || '';\r\n entityDescription = data.description || data.eventSummary || '';\r\n entityImage = data.image || '';\r\n } else if (type === 'collection') {\r\n const slug = data.slug || generateSlug(data.collectionTitle || '');\r\n entityUrl = this._buildEntityUrl('collection', data.id || '', slug);\r\n\r\n jsonLd = buildCollectionJsonLd({\r\n collection: data,\r\n baseEventUrl: entityUrl,\r\n defaultTimeZone: this._venueTimeZone,\r\n });\r\n\r\n entityName = data.collectionTitle || '';\r\n entityDescription = data.description || data.summary || '';\r\n entityImage = data.coverImage || '';\r\n }\r\n\r\n // Inject JSON-LD (replaces existing micdrop JSON-LD)\r\n if (jsonLd) {\r\n this._injectJsonLd(jsonLd);\r\n }\r\n\r\n // Update canonical URL\r\n if (entityUrl) {\r\n this._setCanonical(entityUrl);\r\n }\r\n\r\n // Update OG tags\r\n if (entityName) {\r\n this._setMeta('og:title', entityName);\r\n }\r\n if (entityDescription) {\r\n this._setMeta('og:description', entityDescription);\r\n }\r\n if (entityImage) {\r\n this._setMeta('og:image', entityImage);\r\n }\r\n if (entityUrl) {\r\n this._setMeta('og:url', entityUrl);\r\n }\r\n this._setMeta('og:type', 'website');\r\n\r\n // Twitter Card: large image preview\r\n this._setMeta('twitter:card', 'summary_large_image');\r\n\r\n // Update meta description\r\n if (entityDescription) {\r\n this._setMeta('description', entityDescription);\r\n }\r\n\r\n // Update page title\r\n if (entityName) {\r\n document.title = `${entityName} | ${this._venueName}`;\r\n }\r\n\r\n this._state = 'entity';\r\n }\r\n\r\n /**\r\n * Called when the user navigates back to the event listing.\r\n * Restores original head elements and re-injects ItemList JSON-LD.\r\n */\r\n onBackToListing(): void {\r\n if (typeof document === 'undefined') return;\r\n if (this._state === 'destroyed') return;\r\n\r\n // Restore all originals\r\n this._restoreOriginals();\r\n\r\n // Remove any remaining injected elements\r\n this._removeAllInjected();\r\n\r\n // Re-inject ItemList JSON-LD if events are available\r\n if (this._events.length > 0) {\r\n const itemList = this._buildItemList(this._events);\r\n this._injectJsonLd(itemList);\r\n }\r\n\r\n // Re-save originals since we just restored them\r\n this._saveOriginals();\r\n\r\n this._state = 'listing';\r\n }\r\n\r\n /**\r\n * Called when the widget is destroyed. Removes all injected elements\r\n * and restores originals. Clears the static active instance.\r\n */\r\n destroy(): void {\r\n if (typeof document === 'undefined') return;\r\n if (this._state === 'destroyed') return;\r\n\r\n this._removeAllInjected();\r\n this._restoreOriginals();\r\n\r\n if (HostSeoController._activeInstance === this) {\r\n HostSeoController._activeInstance = null;\r\n }\r\n\r\n this._state = 'destroyed';\r\n }\r\n}\r\n"],"names":["getDefaultTimezone","BLOCK_CLOSE","LINE_BREAK","ANY_TAG","NAMED_ENTITIES","decodeEntities","input","_","hex","codePointOrEmpty","dec","m","code","stripEventDescriptionHtml","html","maxLength","text","escapeJsonLdForScript","json","formatDateTimeWithOffset","isoString","tz","date","offsetFormatOptions","formatter","parts","get","type","p","year","month","day","hour","minute","second","tzOffset","offset","match","sign","hours","mins","parseAddressToPostal","addressStr","stateZipMatch","buildPostalAddress","fallback","out","getEventStatus","rawStatus","status","getTicketAvailability","ticket","now","salesBegin","salesEnd","buildOffers","tickets","url","publicTickets","t","visibility","isVisible","isPublic","prices","lowPrice","highPrice","buildPerformers","performers","confirmed","isConfirmed","isNotHidden","profile","hasName","buildEventJsonLd","options","event","venue","venueTimeZone","eventUrl","organizer","startDate","postalAddress","location","data","endDate","description","imageUrl","offers","org","mapSeriesTicketAvailability","buildSeriesOccurrenceOffers","occurrenceStatus","activeTickets","buildSeriesJsonLd","series","baseEventUrl","address","subEvents","occ","subEvent","buildCollectionJsonLd","collection","itemListElement","listItem","generateSlug","_HostSeoController","canonical","ogTitle","ogDescription","ogImage","ogUrl","ogType","metaDescription","el","script","property","content","originalContent","name","events","i","slug","id","prefix","venueTz","itemList","jsonLd","entityName","entityDescription","entityImage","entityUrl","HostSeoController"],"mappings":"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","x_google_ignoreList":[0,1]}
@@ -1,4 +1,4 @@
1
- import { aF as D, ai as _, aG as P } from "./VenueCalendar-DMSeDRSN.js";
1
+ import { aK as D, ao as _, aL as P } from "./VenueCalendar-DTOyAhDU.js";
2
2
  const q = "https://moxy.sfo3.cdn.digitaloceanspaces.com", A = q;
3
3
  function f(t, a = A) {
4
4
  return t ? t.startsWith("http://") || t.startsWith("https://") ? t : `${a}${t}` : "";
@@ -267,10 +267,8 @@ function x(t) {
267
267
  return E(t);
268
268
  }
269
269
  export {
270
- q as C,
271
270
  k as a,
272
271
  x as b,
273
- f as g,
274
272
  b as t
275
273
  };
276
- //# sourceMappingURL=transform-BlXf-shc.js.map
274
+ //# sourceMappingURL=transform-C1Rmuzlt.js.map
@@ -1 +1 @@
1
- {"version":3,"file":"transform-BlXf-shc.js","sources":["../src/lib/utils/imageUrl.ts","../src/lib/api/transformers/address.ts","../src/lib/api/transformers/series.ts","../src/lib/api/transformers/collection.ts","../src/lib/public-calendar-flow/transform.ts"],"sourcesContent":["/**\r\n * Image URL resolution utilities — npm-calendar's local canonical for\r\n * the moxy.sfo3.*.digitaloceanspaces.com CDN base URL. Other modules in\r\n * this package (event-transform, performerDisplay, core/api-client)\r\n * import {@link CDN_BASE_URL} from here rather than re-declaring the\r\n * literal — Phase G ratchet eliminated three duplicate constant\r\n * definitions.\r\n */\r\n\r\n/** Origin (no CDN) — admin / preview surfaces. */\r\nexport const CDN_BASE_URL_ORIGIN = \"https://moxy.sfo3.digitaloceanspaces.com\";\r\n/** CloudFront-fronted CDN — public-facing surfaces. */\r\nexport const CDN_BASE_URL = \"https://moxy.sfo3.cdn.digitaloceanspaces.com\";\r\n/** @deprecated Use {@link CDN_BASE_URL}. Kept as alias during migration. */\r\nconst DEFAULT_CDN_BASE_URL = CDN_BASE_URL;\r\n\r\n/**\r\n * Resolve an image path to a full URL.\r\n * If the path is already a full URL, returns it as-is.\r\n * Otherwise prepends the CDN base URL.\r\n */\r\nexport function getImageUrl(\r\n imagePath: string | null | undefined,\r\n baseUrl: string = DEFAULT_CDN_BASE_URL\r\n): string {\r\n if (!imagePath) return \"\";\r\n\r\n if (imagePath.startsWith(\"http://\") || imagePath.startsWith(\"https://\")) {\r\n return imagePath;\r\n }\r\n\r\n return `${baseUrl}${imagePath}`;\r\n}\r\n","/**\r\n * Address helpers — shared between the event, series and collection page\r\n * transformers.\r\n *\r\n * Lifted verbatim from public-calendar-flow/transform.ts so the series and\r\n * collection canonicals (and the in-place transformApiEvent) read a single\r\n * source instead of each carrying a private copy. Behavior is byte-identical\r\n * to the previous module-private versions.\r\n */\r\n\r\n/**\r\n * Parse a single-line address string into display lines.\r\n * E.g. \"123 Main St, Los Angeles, CA 90012\" -> [\"123 Main St\", \"Los Angeles, CA 90012\"]\r\n */\r\nexport function parseAddress(addressStr: string): string[] {\r\n if (!addressStr) return [];\r\n const parts = addressStr.split(',').map(p => p.trim());\r\n if (parts.length < 2) return [addressStr];\r\n const lastPart = parts[parts.length - 1];\r\n const stateZipMatch = lastPart.match(/^([A-Z]{2})\\s+(\\d{5}(-\\d{4})?)$/);\r\n if (stateZipMatch && parts.length >= 3) {\r\n const state = stateZipMatch[1];\r\n const zip = stateZipMatch[2];\r\n const city = parts[parts.length - 2];\r\n const lines = parts.slice(0, -2);\r\n lines.push(`${city}, ${state} ${zip}`);\r\n return lines;\r\n }\r\n return [parts[0], parts.slice(1).join(', ')];\r\n}\r\n\r\n/**\r\n * Build a Google Maps search URL from an address string.\r\n */\r\nexport function buildGoogleMapsUrl(address: string): string {\r\n if (!address) return '';\r\n return `https://www.google.com/maps/search/?api=1&query=${encodeURIComponent(address)}`;\r\n}\r\n","/**\r\n * Series Page Transformer\r\n *\r\n * Wire->domain boundary for the public series page. Maps a SeriesPageData\r\n * payload (from the series API) into the clean EventData shape consumed by\r\n * SeriesPage.svelte / EventExperience (full-width series mode, set via\r\n * eventSeriesId).\r\n *\r\n * Lifted byte-identically from public-calendar-flow/transform.ts's\r\n * transformSeriesPageData. Every ||/?? fallthrough is preserved exactly;\r\n * characterization tests pin the output. transformSeriesPageData now delegates\r\n * here.\r\n *\r\n * COMPOSITION NOTE — why this does NOT reuse parseVenue:\r\n * The api `parseVenue` returns the api-domain `Venue` (raw timezone string,\r\n * fee fields read from the payload). The series page needs the pcf `VenueData`\r\n * shape: timezone is the IANA-RESOLVED zone (getIANATimezone) and the fee\r\n * fields are hard-zeroed (the series payload carries no venue fees). Those\r\n * shapes are not byte-compatible, so the venue sub-shape stays local.\r\n */\r\n\r\nimport { formatDateRaw, getIANATimezone } from '@getmicdrop/svelte-components';\r\nimport { getImageUrl } from '$lib/utils/utils.js';\r\nimport { parseAddress, buildGoogleMapsUrl } from './address.js';\r\nimport type { SeriesPageData } from '../types.js';\r\nimport type {\r\n EventData,\r\n EventStatus,\r\n PerformerData,\r\n FAQItem,\r\n SeriesOccurrence,\r\n} from '$lib/public-calendar-flow/types';\r\n\r\n/**\r\n * Transform SeriesPageData (from the series API) into our clean EventData interface.\r\n * Sets eventSeriesId so EventExperience renders in full-width series mode.\r\n */\r\nexport function parseSeries(series: SeriesPageData): EventData {\r\n // Mirror the event transform: raw series carry lowercase `timezone`\r\n // (some PascalCase `timeZone` / snake `time_zone`). Read all casings so a\r\n // non-UTC series doesn't silently fall back to UTC.\r\n const seriesAny = series as SeriesPageData & {\r\n timezone?: string;\r\n time_zone?: string;\r\n };\r\n const timezone = getIANATimezone(\r\n series.timeZone || seriesAny.timezone || seriesAny.time_zone || 'UTC'\r\n );\r\n\r\n // Build image URL\r\n const rawImage = series.image || '';\r\n const imageUrl = rawImage.startsWith('http')\r\n ? rawImage\r\n : rawImage\r\n ? getImageUrl(rawImage)\r\n : '';\r\n\r\n // Map occurrences to SeriesOccurrence[]\r\n const seriesOccurrences: SeriesOccurrence[] = (series.occurrences || []).map(\r\n occ => ({\r\n id: occ.id,\r\n slug: occ.slug || '',\r\n startDateTime: occ.startDateTime,\r\n endDateTime: occ.endDateTime || '',\r\n status: (occ.ctaState?.reason === 'sold_out'\r\n ? 'sold_out'\r\n : 'on_sale') as EventStatus,\r\n ctaText: occ.ctaState?.text || 'Get Tickets',\r\n ctaDisabled: occ.ctaState?.disabled || false,\r\n })\r\n );\r\n\r\n // Use first occurrence for date fields\r\n const firstOcc = series.occurrences?.[0];\r\n let date = '';\r\n if (firstOcc?.startDateTime) {\r\n const dt = new Date(firstOcc.startDateTime);\r\n date = formatDateRaw(\r\n dt,\r\n { timeZone: timezone, year: 'numeric', month: '2-digit', day: '2-digit' },\r\n 'en-CA'\r\n );\r\n }\r\n\r\n // Map FAQs\r\n const rawFaq = series.venue?.faq;\r\n let faqs: FAQItem[] = [];\r\n if (Array.isArray(rawFaq)) {\r\n faqs = rawFaq.map(f => ({\r\n question: f.question || '',\r\n answer: f.answer || '',\r\n }));\r\n } else if (typeof rawFaq === 'string' && rawFaq) {\r\n faqs = [{ question: 'FAQ', answer: rawFaq }];\r\n }\r\n\r\n // Map performers\r\n const performers: PerformerData[] = (series.performers || []).map(\r\n (p, index) => ({\r\n id: index + 1,\r\n displayName:\r\n p.displayName || [p.firstName, p.lastName].filter(Boolean).join(' '),\r\n profileImage: p.profileImage || undefined,\r\n })\r\n );\r\n\r\n // Venue address\r\n const venueAddress =\r\n series.venue?.location || series.venue?.googleLocationNameCache || '';\r\n\r\n return {\r\n id: series.id,\r\n title: series.title || '',\r\n slug: '',\r\n description: series.description || series.eventSummary || '',\r\n date,\r\n startDateTime: firstOcc?.startDateTime || '',\r\n endDateTime: firstOcc?.endDateTime || '',\r\n doorsOpenTime: '',\r\n timezone,\r\n imageUrl,\r\n venue: {\r\n id: series.venue?.id || 0,\r\n name: series.venue?.name || '',\r\n address: venueAddress,\r\n timezone,\r\n serviceFeePercentage: 0,\r\n serviceFeeCents: 0,\r\n taxPercentage: 0,\r\n },\r\n performers,\r\n tickets: [],\r\n faqs,\r\n disclaimer: series.venue?.disclaimer || undefined,\r\n status: 'on_sale',\r\n eventSeriesId: series.id,\r\n seriesOccurrences,\r\n isPasswordProtected: false,\r\n isRegistrationEvent: false,\r\n totalTicketsRemaining: 0,\r\n totalTicketsCapacity: 0,\r\n venueAddressLines: parseAddress(venueAddress),\r\n googleMapsUrl: buildGoogleMapsUrl(venueAddress),\r\n };\r\n}\r\n\r\n/**\r\n * Back-compat alias — the public-calendar-flow shaper name. Existing call\r\n * sites importing transformSeriesPageData continue to work unchanged.\r\n */\r\nexport const transformSeriesPageData = parseSeries;\r\n","/**\r\n * Collection Page Transformer\r\n *\r\n * Wire->domain boundary for the public collection page. Maps a\r\n * PublicCollectionData payload (from the collection API) into the clean\r\n * EventData shape consumed by CollectionView.svelte / EventExperience\r\n * (full-width collection mode, set via collectionId).\r\n *\r\n * Lifted byte-identically from public-calendar-flow/transform.ts's\r\n * transformCollectionData. Every ||/?? fallthrough is preserved exactly;\r\n * characterization tests pin the output. transformCollectionData now delegates\r\n * here.\r\n *\r\n * COMPOSITION NOTE — why this does NOT reuse parseVenue/parseEvent:\r\n * The collection payload has no venue and no timezone; the EventData it\r\n * produces hard-codes an empty venue and 'UTC'. The per-event mapping is a\r\n * tiny CollectionEvent card shape (id/name/slug/image/startDateTime/status),\r\n * not the full Event domain — reusing parseEvent would change the output.\r\n * Both stay local.\r\n */\r\n\r\nimport { getImageUrl } from '$lib/utils/utils.js';\r\nimport type { PublicCollectionData } from '../types.js';\r\nimport type {\r\n EventData,\r\n EventStatus,\r\n CollectionEvent,\r\n} from '$lib/public-calendar-flow/types';\r\n\r\n/**\r\n * Transform PublicCollectionData (from the collection API) into our clean EventData interface.\r\n * Sets collectionId so EventExperience renders in full-width collection mode.\r\n */\r\nexport function parseCollection(collection: PublicCollectionData): EventData {\r\n // Map collection events to CollectionEvent[]\r\n const collectionEvents: CollectionEvent[] = (collection.events || []).map(\r\n ev => {\r\n const rawImg = ev.image || '';\r\n return {\r\n id: ev.id,\r\n name: ev.title || '',\r\n slug: ev.slug || '',\r\n image: rawImg.startsWith('http')\r\n ? rawImg\r\n : rawImg\r\n ? getImageUrl(rawImg)\r\n : '',\r\n startDateTime: ev.startDateTime || '',\r\n status: (ev.status || 'on_sale') as EventStatus,\r\n };\r\n }\r\n );\r\n\r\n // Cover image\r\n const rawCover = collection.coverImage || '';\r\n const imageUrl = rawCover.startsWith('http')\r\n ? rawCover\r\n : rawCover\r\n ? getImageUrl(rawCover)\r\n : '';\r\n\r\n return {\r\n id: collection.id,\r\n title: collection.collectionTitle || '',\r\n slug: '',\r\n description: collection.description || collection.summary || '',\r\n date: '',\r\n startDateTime: '',\r\n endDateTime: '',\r\n doorsOpenTime: '',\r\n timezone: 'UTC',\r\n imageUrl,\r\n venue: {\r\n id: 0,\r\n name: '',\r\n address: '',\r\n timezone: 'UTC',\r\n serviceFeePercentage: 0,\r\n serviceFeeCents: 0,\r\n taxPercentage: 0,\r\n },\r\n performers: [],\r\n tickets: [],\r\n faqs: [],\r\n status: 'on_sale',\r\n collectionId: collection.id,\r\n collectionEvents,\r\n isPasswordProtected: false,\r\n isRegistrationEvent: false,\r\n totalTicketsRemaining: 0,\r\n totalTicketsCapacity: 0,\r\n };\r\n}\r\n\r\n/**\r\n * Back-compat alias — the public-calendar-flow shaper name. Existing call\r\n * sites importing transformCollectionData continue to work unchanged.\r\n */\r\nexport const transformCollectionData = parseCollection;\r\n","import { formatDateRaw, getIANATimezone } from '@getmicdrop/svelte-components';\r\n// Premium Ticket Experience — API → EventData Transformer\r\n// Maps raw API event responses to our clean EventData interface.\r\n// Used when plugging in real API data. The showcase continues using mock-data.ts directly.\r\n\r\nimport { getImageUrl } from '$lib/utils/utils.js';\r\nimport {\r\n parseAddress,\r\n buildGoogleMapsUrl,\r\n} from '$lib/api/transformers/address';\r\n// Canonical wire→domain boundary — all event-level scalar reads go through it.\r\nimport { parseEvent } from '$lib/api/transformers/event';\r\nimport { parseSeries } from '$lib/api/transformers/series';\r\nimport { parseCollection } from '$lib/api/transformers/collection';\r\nimport type {\r\n EventData,\r\n EventStatus,\r\n TicketTypeData,\r\n TicketStatus,\r\n VenueData,\r\n PerformerData,\r\n FAQItem,\r\n SeriesOccurrence,\r\n} from './types';\r\nimport type { SeriesPageData, PublicCollectionData } from '$lib/api/types';\r\n\r\n/**\r\n * Map a raw API ticket status/state to our TicketStatus union.\r\n */\r\nfunction mapTicketStatus(ticket: any): TicketStatus {\r\n const now = new Date();\r\n const remaining =\r\n ticket.remainingCapacity ??\r\n ticket.quantityRemaining ??\r\n ticket.quantity ??\r\n 0;\r\n const salesBegin =\r\n ticket.salesBegin ||\r\n ticket.salesStart ||\r\n ticket.saleBegin ||\r\n ticket.onSaleStart;\r\n const salesEnd = ticket.salesEnd || ticket.saleEnd || ticket.onSaleEnd;\r\n\r\n if (remaining <= 0) return 'sold_out';\r\n if (salesEnd && new Date(salesEnd) < now) return 'sales_ended';\r\n if (salesBegin && new Date(salesBegin) > now) return 'coming_soon';\r\n return 'on_sale';\r\n}\r\n\r\n/**\r\n * Map a raw API event status string to our EventStatus union.\r\n */\r\nfunction mapEventStatus(\r\n rawStatus: string,\r\n tickets: TicketTypeData[]\r\n): EventStatus {\r\n const s = (rawStatus || '').toLowerCase().replace(/\\s+/g, '_');\r\n\r\n if (s === 'cancelled' || s === 'canceled') return 'cancelled';\r\n\r\n // Check if event is past based on end time (handled by caller if needed)\r\n if (s === 'past' || s === 'ended') return 'past';\r\n\r\n // MIC-1039: if any hidden ticket is still on sale, the event is \"on_sale\"\r\n // regardless of visible-ticket states — the user just needs a promo code.\r\n // NOTE: this operates on the pcf DOMAIN ticket shape (post-transform, with a\r\n // computed `status` string), NOT the wire shape SC's\r\n // `isHiddenTicketPurchasable` reads (raw visibility/salesChannel/sale-window\r\n // fields). The sale-window math already happened in `mapTicketStatus`, so the\r\n // SC predicate can't honestly serve this site — kept deliberately local.\r\n const hasPurchasableHiddenTickets = tickets.some(\r\n t => t.isHidden && t.status === 'on_sale'\r\n );\r\n if (hasPurchasableHiddenTickets) return 'on_sale';\r\n\r\n // Derive from ticket states\r\n const publicTickets = tickets.filter(t => !t.isHidden);\r\n if (publicTickets.length > 0) {\r\n if (publicTickets.every(t => t.status === 'sold_out')) return 'sold_out';\r\n if (publicTickets.every(t => t.status === 'coming_soon'))\r\n return 'coming_soon';\r\n if (\r\n publicTickets.every(\r\n t => t.status === 'sales_ended' || t.status === 'sold_out'\r\n )\r\n )\r\n return 'sales_ended';\r\n }\r\n\r\n if (s === 'sold_out' || s === 'soldout') return 'sold_out';\r\n if (s === 'coming_soon' || s === 'comingsoon') return 'coming_soon';\r\n if (s === 'sales_ended' || s === 'salesended') return 'sales_ended';\r\n\r\n return 'on_sale';\r\n}\r\n\r\n/**\r\n * Transform a raw API event response into our clean EventData interface.\r\n * This is the bridge between the API and our premium ticket experience components.\r\n *\r\n * COMPOSITION NOTE (events convergence): the wire is parsed ONCE through the\r\n * canonical parseEvent ($lib/api/transformers/event); every event-level\r\n * scalar below projects off the domain Event. Three deliberate wire seams\r\n * remain, characterization-pinned in\r\n * utils/event-shapers.characterization.test.js:\r\n * 1. TICKETS — the TicketTypeData map uses `??` (nullish) semantics over\r\n * raw ticket fields (remainingCapacity/visibility/fee/…) that\r\n * parseTicket's `||`-folding AvailableTicket neither carries nor\r\n * preserves; routing through parseTicket would change rendered output.\r\n * 2. IMAGE — image→imageUrl→image_url precedence intentionally differs\r\n * from the canonical resolver's (imageUrl-first); see the note in\r\n * api/transformers/event.ts.\r\n * 3. VENUE/PERFORMERS — mapped from the raw passthrough objects\r\n * (domain.venue / domain.performers) with B-specific `??` fee/tax\r\n * semantics; the performer canonical lands in PR #260 and should\r\n * absorb the performer map then.\r\n */\r\nexport function transformApiEvent(rawEvent: any): EventData {\r\n // Wire→domain ONCE (see composition note).\r\n const domain = parseEvent(rawEvent);\r\n const rawVenue = domain.venue || {};\r\n // Event-level timeZone is an OPTIONAL per-event override; when absent/blank it\r\n // must inherit the venue timezone. Falling straight to 'UTC' here rendered\r\n // venue-local times (e.g. a Fri 10pm Central show) in UTC (\"Sat 3:00 AM\"),\r\n // which made already-ended events look like future ones.\r\n // (domain.timeZone is the verbatim PascalCase passthrough; domain.timezone\r\n // folds lowercase + snake_case — same chain as before.)\r\n const timezone = getIANATimezone(\r\n domain.timeZone ||\r\n domain.timezone ||\r\n rawVenue.timeZone ||\r\n rawVenue.timezone ||\r\n 'UTC'\r\n );\r\n\r\n // Transform tickets — TICKET SEAM: raw wire tickets by contract (see note).\r\n const rawTickets = rawEvent.availableTickets || [];\r\n const tickets: TicketTypeData[] = rawTickets.map((t: any) => ({\r\n id: t.id || t.ID,\r\n name: t.name || t.title || '',\r\n description: t.description || undefined,\r\n price: t.price ?? 0,\r\n fee: t.fee ?? 0,\r\n quantityAvailable:\r\n t.remainingCapacity ?? t.quantityRemaining ?? t.quantity ?? 0,\r\n quantityTotal: t.totalCapacity ?? t.quantity ?? 0,\r\n minPerOrder: t.minPerOrder ?? t.minQuantity ?? 1,\r\n maxPerOrder: t.maxPerOrder ?? t.maxQuantity ?? 10,\r\n status: mapTicketStatus(t),\r\n salesStartDate:\r\n t.salesBegin || t.salesStart || t.saleBegin || t.onSaleStart || undefined,\r\n salesEndDate: t.salesEnd || t.saleEnd || t.onSaleEnd || undefined,\r\n isDonation: t.type === 2 || t.ticketType === 2 || t.isDonation === true,\r\n isHidden: t.visibility === 2 || t.isHidden === true,\r\n ticketType: t.type || t.ticketType || undefined,\r\n sectionName: t.sectionName || t.section || undefined,\r\n }));\r\n\r\n // Calculate totals from public tickets\r\n const publicTickets = tickets.filter(t => !t.isHidden && !t.isDonation);\r\n const totalRemaining = publicTickets.reduce(\r\n (sum, t) => sum + t.quantityAvailable,\r\n 0\r\n );\r\n const totalCapacity = publicTickets.reduce(\r\n (sum, t) => sum + t.quantityTotal,\r\n 0\r\n );\r\n\r\n // Determine event status\r\n const status = mapEventStatus(domain.status as string, tickets);\r\n\r\n // Build image URL — IMAGE SEAM: image-first precedence (see note).\r\n const rawImage =\r\n rawEvent.image || rawEvent.imageUrl || rawEvent.image_url || '';\r\n const imageUrl = rawImage.startsWith('http')\r\n ? rawImage\r\n : getImageUrl(rawImage);\r\n\r\n // Venue\r\n const venue: VenueData = {\r\n id: rawVenue.id || rawVenue.ID || 0,\r\n name: rawVenue.name || rawVenue.googleLocationNameCache || '',\r\n address: rawVenue.address || rawVenue.googleLocationNameCache || '',\r\n timezone,\r\n serviceFeePercentage: rawVenue.serviceFeePercentage ?? rawVenue.fee ?? 0,\r\n serviceFeeCents: rawVenue.serviceFeeCents ?? 0,\r\n taxPercentage: rawVenue.taxPercentage ?? rawVenue.tax ?? 0,\r\n contactEmail: rawVenue.contactEmail || rawVenue.email || undefined,\r\n };\r\n\r\n // Performers — VENUE/PERFORMERS SEAM: raw passthrough map (see note).\r\n const rawPerformers = domain.performers || [];\r\n const performers: PerformerData[] = rawPerformers.map((p: any) => ({\r\n id: p.id || p.ID,\r\n displayName: p.displayName || p.name || '',\r\n profileImage: p.avatar || p.profileImage || undefined,\r\n role: p.role || undefined,\r\n }));\r\n\r\n // FAQs (may not be in API response)\r\n const faqs: FAQItem[] = (domain.faqs || []).map((f: any) => ({\r\n question: f.question || '',\r\n answer: f.answer || '',\r\n }));\r\n\r\n // Series showtimes → occurrences. A single series-event detail carries a\r\n // `showtimes` array; the v2 <ShowtimeDesigns> renders event.seriesOccurrences\r\n // under \"Upcoming Shows\". Without this map the section is always empty.\r\n const seriesOccurrences: SeriesOccurrence[] = (domain.showtimes || [])\r\n .filter((s: any) => s.published !== false)\r\n .map((s: any, i: number) => ({\r\n id: s.id ?? s.eventId ?? i + 1,\r\n slug: s.slug || '',\r\n startDateTime: s.startDateTime || s.startTime || '',\r\n endDateTime: s.endDateTime || s.endTime || '',\r\n status: (s.ctaState?.reason === 'sold_out'\r\n ? 'sold_out'\r\n : 'on_sale') as EventStatus,\r\n ctaText: s.ctaState?.text || 'Get tickets',\r\n ctaDisabled: s.ctaState?.disabled || false,\r\n }));\r\n\r\n // Date in venue timezone\r\n let date = '';\r\n if (domain.startDateTime) {\r\n const dt = new Date(domain.startDateTime);\r\n date = formatDateRaw(\r\n dt,\r\n { timeZone: timezone, year: 'numeric', month: '2-digit', day: '2-digit' },\r\n 'en-CA'\r\n );\r\n }\r\n\r\n return {\r\n id: domain.id,\r\n title: domain.title || '',\r\n slug: domain.slug || '',\r\n description: domain.description || domain.eventSummary || '',\r\n date,\r\n startDateTime: domain.startDateTime || '',\r\n endDateTime: domain.endDateTime || '',\r\n doorsOpenTime: domain.doorsOpenTime || '',\r\n timezone,\r\n imageUrl,\r\n venue,\r\n performers,\r\n tickets,\r\n faqs,\r\n disclaimer: domain.disclaimer || undefined,\r\n status,\r\n eventSeriesId: domain.eventSeriesId || undefined,\r\n seriesOccurrences: seriesOccurrences.length ? seriesOccurrences : undefined,\r\n // Public API hides the actual password (json:\"-\") and exposes a boolean\r\n // `hasPassword` flag instead; fall back to a populated `password` for\r\n // admin-shaped payloads.\r\n isPasswordProtected: domain.hasPassword === true || !!domain.password,\r\n isRegistrationEvent:\r\n domain.ticketType === 1 || domain.eventTicketingType === 1,\r\n totalTicketsRemaining: totalRemaining,\r\n totalTicketsCapacity: totalCapacity,\r\n stageName: domain.stage?.name || domain.stageName || undefined,\r\n ageRestriction: domain.ageRestriction || undefined,\r\n displayAgeRestriction: domain.displayAgeRestriction ?? false,\r\n displayStartTime: domain.displayStartTime ?? true,\r\n displayEndTime: domain.displayEndTime ?? true,\r\n displayDoorsTime: domain.displayDoorsTime ?? true,\r\n collectionId: domain.collectionId || undefined, // domain folds collection_id\r\n venueAddressLines: parseAddress(\r\n rawVenue.address || rawVenue.googleLocationNameCache || ''\r\n ),\r\n googleMapsUrl: buildGoogleMapsUrl(\r\n rawVenue.address || rawVenue.googleLocationNameCache || ''\r\n ),\r\n };\r\n}\r\n\r\n/**\r\n * Transform SeriesPageData (from the series API) into our clean EventData interface.\r\n * Sets eventSeriesId so EventExperience renders in full-width series mode.\r\n */\r\nexport function transformSeriesPageData(series: SeriesPageData): EventData {\r\n // Implementation moved to the canonical transformer home\r\n // (`$lib/api/transformers/series.ts`). This export stays as a thin delegate\r\n // so SeriesPage.svelte and other consumers don't churn.\r\n return parseSeries(series);\r\n}\r\n\r\n/**\r\n * Transform PublicCollectionData (from the collection API) into our clean EventData interface.\r\n * Sets collectionId so EventExperience renders in full-width collection mode.\r\n */\r\nexport function transformCollectionData(\r\n collection: PublicCollectionData\r\n): EventData {\r\n // Implementation moved to the canonical transformer home\r\n // (`$lib/api/transformers/collection.ts`). This export stays as a thin\r\n // delegate so CollectionView.svelte and other consumers don't churn.\r\n return parseCollection(collection);\r\n}\r\n"],"names":["CDN_BASE_URL","DEFAULT_CDN_BASE_URL","getImageUrl","imagePath","baseUrl","parseAddress","addressStr","parts","p","stateZipMatch","state","zip","city","lines","buildGoogleMapsUrl","address","parseSeries","series","seriesAny","timezone","getIANATimezone","rawImage","imageUrl","seriesOccurrences","occ","firstOcc","date","dt","formatDateRaw","rawFaq","faqs","f","performers","index","venueAddress","parseCollection","collection","collectionEvents","ev","rawImg","rawCover","mapTicketStatus","ticket","now","remaining","salesBegin","salesEnd","mapEventStatus","rawStatus","tickets","t","publicTickets","transformApiEvent","rawEvent","domain","parseEvent","rawVenue","totalRemaining","sum","totalCapacity","status","venue","s","i","transformSeriesPageData","transformCollectionData"],"mappings":"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"}
1
+ {"version":3,"file":"transform-C1Rmuzlt.js","sources":["../src/lib/utils/imageUrl.ts","../src/lib/api/transformers/address.ts","../src/lib/api/transformers/series.ts","../src/lib/api/transformers/collection.ts","../src/lib/public-calendar-flow/transform.ts"],"sourcesContent":["/**\r\n * Image URL resolution utilities — npm-calendar's local canonical for\r\n * the moxy.sfo3.*.digitaloceanspaces.com CDN base URL. Other modules in\r\n * this package (event-transform, performerDisplay, core/api-client)\r\n * import {@link CDN_BASE_URL} from here rather than re-declaring the\r\n * literal — Phase G ratchet eliminated three duplicate constant\r\n * definitions.\r\n */\r\n\r\n/** Origin (no CDN) — admin / preview surfaces. */\r\nexport const CDN_BASE_URL_ORIGIN = \"https://moxy.sfo3.digitaloceanspaces.com\";\r\n/** CloudFront-fronted CDN — public-facing surfaces. */\r\nexport const CDN_BASE_URL = \"https://moxy.sfo3.cdn.digitaloceanspaces.com\";\r\n/** @deprecated Use {@link CDN_BASE_URL}. Kept as alias during migration. */\r\nconst DEFAULT_CDN_BASE_URL = CDN_BASE_URL;\r\n\r\n/**\r\n * Resolve an image path to a full URL.\r\n * If the path is already a full URL, returns it as-is.\r\n * Otherwise prepends the CDN base URL.\r\n */\r\nexport function getImageUrl(\r\n imagePath: string | null | undefined,\r\n baseUrl: string = DEFAULT_CDN_BASE_URL\r\n): string {\r\n if (!imagePath) return \"\";\r\n\r\n if (imagePath.startsWith(\"http://\") || imagePath.startsWith(\"https://\")) {\r\n return imagePath;\r\n }\r\n\r\n return `${baseUrl}${imagePath}`;\r\n}\r\n","/**\r\n * Address helpers — shared between the event, series and collection page\r\n * transformers.\r\n *\r\n * Lifted verbatim from public-calendar-flow/transform.ts so the series and\r\n * collection canonicals (and the in-place transformApiEvent) read a single\r\n * source instead of each carrying a private copy. Behavior is byte-identical\r\n * to the previous module-private versions.\r\n */\r\n\r\n/**\r\n * Parse a single-line address string into display lines.\r\n * E.g. \"123 Main St, Los Angeles, CA 90012\" -> [\"123 Main St\", \"Los Angeles, CA 90012\"]\r\n */\r\nexport function parseAddress(addressStr: string): string[] {\r\n if (!addressStr) return [];\r\n const parts = addressStr.split(',').map(p => p.trim());\r\n if (parts.length < 2) return [addressStr];\r\n const lastPart = parts[parts.length - 1];\r\n const stateZipMatch = lastPart.match(/^([A-Z]{2})\\s+(\\d{5}(-\\d{4})?)$/);\r\n if (stateZipMatch && parts.length >= 3) {\r\n const state = stateZipMatch[1];\r\n const zip = stateZipMatch[2];\r\n const city = parts[parts.length - 2];\r\n const lines = parts.slice(0, -2);\r\n lines.push(`${city}, ${state} ${zip}`);\r\n return lines;\r\n }\r\n return [parts[0], parts.slice(1).join(', ')];\r\n}\r\n\r\n/**\r\n * Build a Google Maps search URL from an address string.\r\n */\r\nexport function buildGoogleMapsUrl(address: string): string {\r\n if (!address) return '';\r\n return `https://www.google.com/maps/search/?api=1&query=${encodeURIComponent(address)}`;\r\n}\r\n","/**\r\n * Series Page Transformer\r\n *\r\n * Wire->domain boundary for the public series page. Maps a SeriesPageData\r\n * payload (from the series API) into the clean EventData shape consumed by\r\n * SeriesPage.svelte / EventExperience (full-width series mode, set via\r\n * eventSeriesId).\r\n *\r\n * Lifted byte-identically from public-calendar-flow/transform.ts's\r\n * transformSeriesPageData. Every ||/?? fallthrough is preserved exactly;\r\n * characterization tests pin the output. transformSeriesPageData now delegates\r\n * here.\r\n *\r\n * COMPOSITION NOTE — why this does NOT reuse parseVenue:\r\n * The api `parseVenue` returns the api-domain `Venue` (raw timezone string,\r\n * fee fields read from the payload). The series page needs the pcf `VenueData`\r\n * shape: timezone is the IANA-RESOLVED zone (getIANATimezone) and the fee\r\n * fields are hard-zeroed (the series payload carries no venue fees). Those\r\n * shapes are not byte-compatible, so the venue sub-shape stays local.\r\n */\r\n\r\nimport { formatDateRaw, getIANATimezone } from '@getmicdrop/svelte-components';\r\nimport { getImageUrl } from '$lib/utils/utils.js';\r\nimport { parseAddress, buildGoogleMapsUrl } from './address.js';\r\nimport type { SeriesPageData } from '../types.js';\r\nimport type {\r\n EventData,\r\n EventStatus,\r\n PerformerData,\r\n FAQItem,\r\n SeriesOccurrence,\r\n} from '$lib/public-calendar-flow/types';\r\n\r\n/**\r\n * Transform SeriesPageData (from the series API) into our clean EventData interface.\r\n * Sets eventSeriesId so EventExperience renders in full-width series mode.\r\n */\r\nexport function parseSeries(series: SeriesPageData): EventData {\r\n // Mirror the event transform: raw series carry lowercase `timezone`\r\n // (some PascalCase `timeZone` / snake `time_zone`). Read all casings so a\r\n // non-UTC series doesn't silently fall back to UTC.\r\n const seriesAny = series as SeriesPageData & {\r\n timezone?: string;\r\n time_zone?: string;\r\n };\r\n const timezone = getIANATimezone(\r\n series.timeZone || seriesAny.timezone || seriesAny.time_zone || 'UTC'\r\n );\r\n\r\n // Build image URL\r\n const rawImage = series.image || '';\r\n const imageUrl = rawImage.startsWith('http')\r\n ? rawImage\r\n : rawImage\r\n ? getImageUrl(rawImage)\r\n : '';\r\n\r\n // Map occurrences to SeriesOccurrence[]\r\n const seriesOccurrences: SeriesOccurrence[] = (series.occurrences || []).map(\r\n occ => ({\r\n id: occ.id,\r\n slug: occ.slug || '',\r\n startDateTime: occ.startDateTime,\r\n endDateTime: occ.endDateTime || '',\r\n status: (occ.ctaState?.reason === 'sold_out'\r\n ? 'sold_out'\r\n : 'on_sale') as EventStatus,\r\n ctaText: occ.ctaState?.text || 'Get Tickets',\r\n ctaDisabled: occ.ctaState?.disabled || false,\r\n })\r\n );\r\n\r\n // Use first occurrence for date fields\r\n const firstOcc = series.occurrences?.[0];\r\n let date = '';\r\n if (firstOcc?.startDateTime) {\r\n const dt = new Date(firstOcc.startDateTime);\r\n date = formatDateRaw(\r\n dt,\r\n { timeZone: timezone, year: 'numeric', month: '2-digit', day: '2-digit' },\r\n 'en-CA'\r\n );\r\n }\r\n\r\n // Map FAQs\r\n const rawFaq = series.venue?.faq;\r\n let faqs: FAQItem[] = [];\r\n if (Array.isArray(rawFaq)) {\r\n faqs = rawFaq.map(f => ({\r\n question: f.question || '',\r\n answer: f.answer || '',\r\n }));\r\n } else if (typeof rawFaq === 'string' && rawFaq) {\r\n faqs = [{ question: 'FAQ', answer: rawFaq }];\r\n }\r\n\r\n // Map performers\r\n const performers: PerformerData[] = (series.performers || []).map(\r\n (p, index) => ({\r\n id: index + 1,\r\n displayName:\r\n p.displayName || [p.firstName, p.lastName].filter(Boolean).join(' '),\r\n profileImage: p.profileImage || undefined,\r\n })\r\n );\r\n\r\n // Venue address\r\n const venueAddress =\r\n series.venue?.location || series.venue?.googleLocationNameCache || '';\r\n\r\n return {\r\n id: series.id,\r\n title: series.title || '',\r\n slug: '',\r\n description: series.description || series.eventSummary || '',\r\n date,\r\n startDateTime: firstOcc?.startDateTime || '',\r\n endDateTime: firstOcc?.endDateTime || '',\r\n doorsOpenTime: '',\r\n timezone,\r\n imageUrl,\r\n venue: {\r\n id: series.venue?.id || 0,\r\n name: series.venue?.name || '',\r\n address: venueAddress,\r\n timezone,\r\n serviceFeePercentage: 0,\r\n serviceFeeCents: 0,\r\n taxPercentage: 0,\r\n },\r\n performers,\r\n tickets: [],\r\n faqs,\r\n disclaimer: series.venue?.disclaimer || undefined,\r\n status: 'on_sale',\r\n eventSeriesId: series.id,\r\n seriesOccurrences,\r\n isPasswordProtected: false,\r\n isRegistrationEvent: false,\r\n totalTicketsRemaining: 0,\r\n totalTicketsCapacity: 0,\r\n venueAddressLines: parseAddress(venueAddress),\r\n googleMapsUrl: buildGoogleMapsUrl(venueAddress),\r\n };\r\n}\r\n\r\n/**\r\n * Back-compat alias — the public-calendar-flow shaper name. Existing call\r\n * sites importing transformSeriesPageData continue to work unchanged.\r\n */\r\nexport const transformSeriesPageData = parseSeries;\r\n","/**\r\n * Collection Page Transformer\r\n *\r\n * Wire->domain boundary for the public collection page. Maps a\r\n * PublicCollectionData payload (from the collection API) into the clean\r\n * EventData shape consumed by CollectionView.svelte / EventExperience\r\n * (full-width collection mode, set via collectionId).\r\n *\r\n * Lifted byte-identically from public-calendar-flow/transform.ts's\r\n * transformCollectionData. Every ||/?? fallthrough is preserved exactly;\r\n * characterization tests pin the output. transformCollectionData now delegates\r\n * here.\r\n *\r\n * COMPOSITION NOTE — why this does NOT reuse parseVenue/parseEvent:\r\n * The collection payload has no venue and no timezone; the EventData it\r\n * produces hard-codes an empty venue and 'UTC'. The per-event mapping is a\r\n * tiny CollectionEvent card shape (id/name/slug/image/startDateTime/status),\r\n * not the full Event domain — reusing parseEvent would change the output.\r\n * Both stay local.\r\n */\r\n\r\nimport { getImageUrl } from '$lib/utils/utils.js';\r\nimport type { PublicCollectionData } from '../types.js';\r\nimport type {\r\n EventData,\r\n EventStatus,\r\n CollectionEvent,\r\n} from '$lib/public-calendar-flow/types';\r\n\r\n/**\r\n * Transform PublicCollectionData (from the collection API) into our clean EventData interface.\r\n * Sets collectionId so EventExperience renders in full-width collection mode.\r\n */\r\nexport function parseCollection(collection: PublicCollectionData): EventData {\r\n // Map collection events to CollectionEvent[]\r\n const collectionEvents: CollectionEvent[] = (collection.events || []).map(\r\n ev => {\r\n const rawImg = ev.image || '';\r\n return {\r\n id: ev.id,\r\n name: ev.title || '',\r\n slug: ev.slug || '',\r\n image: rawImg.startsWith('http')\r\n ? rawImg\r\n : rawImg\r\n ? getImageUrl(rawImg)\r\n : '',\r\n startDateTime: ev.startDateTime || '',\r\n status: (ev.status || 'on_sale') as EventStatus,\r\n };\r\n }\r\n );\r\n\r\n // Cover image\r\n const rawCover = collection.coverImage || '';\r\n const imageUrl = rawCover.startsWith('http')\r\n ? rawCover\r\n : rawCover\r\n ? getImageUrl(rawCover)\r\n : '';\r\n\r\n return {\r\n id: collection.id,\r\n title: collection.collectionTitle || '',\r\n slug: '',\r\n description: collection.description || collection.summary || '',\r\n date: '',\r\n startDateTime: '',\r\n endDateTime: '',\r\n doorsOpenTime: '',\r\n timezone: 'UTC',\r\n imageUrl,\r\n venue: {\r\n id: 0,\r\n name: '',\r\n address: '',\r\n timezone: 'UTC',\r\n serviceFeePercentage: 0,\r\n serviceFeeCents: 0,\r\n taxPercentage: 0,\r\n },\r\n performers: [],\r\n tickets: [],\r\n faqs: [],\r\n status: 'on_sale',\r\n collectionId: collection.id,\r\n collectionEvents,\r\n isPasswordProtected: false,\r\n isRegistrationEvent: false,\r\n totalTicketsRemaining: 0,\r\n totalTicketsCapacity: 0,\r\n };\r\n}\r\n\r\n/**\r\n * Back-compat alias — the public-calendar-flow shaper name. Existing call\r\n * sites importing transformCollectionData continue to work unchanged.\r\n */\r\nexport const transformCollectionData = parseCollection;\r\n","import { formatDateRaw, getIANATimezone } from '@getmicdrop/svelte-components';\r\n// Premium Ticket Experience — API → EventData Transformer\r\n// Maps raw API event responses to our clean EventData interface.\r\n// Used when plugging in real API data. The showcase continues using mock-data.ts directly.\r\n\r\nimport { getImageUrl } from '$lib/utils/utils.js';\r\nimport {\r\n parseAddress,\r\n buildGoogleMapsUrl,\r\n} from '$lib/api/transformers/address';\r\n// Canonical wire→domain boundary — all event-level scalar reads go through it.\r\nimport { parseEvent } from '$lib/api/transformers/event';\r\nimport { parseSeries } from '$lib/api/transformers/series';\r\nimport { parseCollection } from '$lib/api/transformers/collection';\r\nimport type {\r\n EventData,\r\n EventStatus,\r\n TicketTypeData,\r\n TicketStatus,\r\n VenueData,\r\n PerformerData,\r\n FAQItem,\r\n SeriesOccurrence,\r\n} from './types';\r\nimport type { SeriesPageData, PublicCollectionData } from '$lib/api/types';\r\n\r\n/**\r\n * Map a raw API ticket status/state to our TicketStatus union.\r\n */\r\nfunction mapTicketStatus(ticket: any): TicketStatus {\r\n const now = new Date();\r\n const remaining =\r\n ticket.remainingCapacity ??\r\n ticket.quantityRemaining ??\r\n ticket.quantity ??\r\n 0;\r\n const salesBegin =\r\n ticket.salesBegin ||\r\n ticket.salesStart ||\r\n ticket.saleBegin ||\r\n ticket.onSaleStart;\r\n const salesEnd = ticket.salesEnd || ticket.saleEnd || ticket.onSaleEnd;\r\n\r\n if (remaining <= 0) return 'sold_out';\r\n if (salesEnd && new Date(salesEnd) < now) return 'sales_ended';\r\n if (salesBegin && new Date(salesBegin) > now) return 'coming_soon';\r\n return 'on_sale';\r\n}\r\n\r\n/**\r\n * Map a raw API event status string to our EventStatus union.\r\n */\r\nfunction mapEventStatus(\r\n rawStatus: string,\r\n tickets: TicketTypeData[]\r\n): EventStatus {\r\n const s = (rawStatus || '').toLowerCase().replace(/\\s+/g, '_');\r\n\r\n if (s === 'cancelled' || s === 'canceled') return 'cancelled';\r\n\r\n // Check if event is past based on end time (handled by caller if needed)\r\n if (s === 'past' || s === 'ended') return 'past';\r\n\r\n // MIC-1039: if any hidden ticket is still on sale, the event is \"on_sale\"\r\n // regardless of visible-ticket states — the user just needs a promo code.\r\n // NOTE: this operates on the pcf DOMAIN ticket shape (post-transform, with a\r\n // computed `status` string), NOT the wire shape SC's\r\n // `isHiddenTicketPurchasable` reads (raw visibility/salesChannel/sale-window\r\n // fields). The sale-window math already happened in `mapTicketStatus`, so the\r\n // SC predicate can't honestly serve this site — kept deliberately local.\r\n const hasPurchasableHiddenTickets = tickets.some(\r\n t => t.isHidden && t.status === 'on_sale'\r\n );\r\n if (hasPurchasableHiddenTickets) return 'on_sale';\r\n\r\n // Derive from ticket states\r\n const publicTickets = tickets.filter(t => !t.isHidden);\r\n if (publicTickets.length > 0) {\r\n if (publicTickets.every(t => t.status === 'sold_out')) return 'sold_out';\r\n if (publicTickets.every(t => t.status === 'coming_soon'))\r\n return 'coming_soon';\r\n if (\r\n publicTickets.every(\r\n t => t.status === 'sales_ended' || t.status === 'sold_out'\r\n )\r\n )\r\n return 'sales_ended';\r\n }\r\n\r\n if (s === 'sold_out' || s === 'soldout') return 'sold_out';\r\n if (s === 'coming_soon' || s === 'comingsoon') return 'coming_soon';\r\n if (s === 'sales_ended' || s === 'salesended') return 'sales_ended';\r\n\r\n return 'on_sale';\r\n}\r\n\r\n/**\r\n * Transform a raw API event response into our clean EventData interface.\r\n * This is the bridge between the API and our premium ticket experience components.\r\n *\r\n * COMPOSITION NOTE (events convergence): the wire is parsed ONCE through the\r\n * canonical parseEvent ($lib/api/transformers/event); every event-level\r\n * scalar below projects off the domain Event. Three deliberate wire seams\r\n * remain, characterization-pinned in\r\n * utils/event-shapers.characterization.test.js:\r\n * 1. TICKETS — the TicketTypeData map uses `??` (nullish) semantics over\r\n * raw ticket fields (remainingCapacity/visibility/fee/…) that\r\n * parseTicket's `||`-folding AvailableTicket neither carries nor\r\n * preserves; routing through parseTicket would change rendered output.\r\n * 2. IMAGE — image→imageUrl→image_url precedence intentionally differs\r\n * from the canonical resolver's (imageUrl-first); see the note in\r\n * api/transformers/event.ts.\r\n * 3. VENUE/PERFORMERS — mapped from the raw passthrough objects\r\n * (domain.venue / domain.performers) with B-specific `??` fee/tax\r\n * semantics; the performer canonical lands in PR #260 and should\r\n * absorb the performer map then.\r\n */\r\nexport function transformApiEvent(rawEvent: any): EventData {\r\n // Wire→domain ONCE (see composition note).\r\n const domain = parseEvent(rawEvent);\r\n const rawVenue = domain.venue || {};\r\n // Event-level timeZone is an OPTIONAL per-event override; when absent/blank it\r\n // must inherit the venue timezone. Falling straight to 'UTC' here rendered\r\n // venue-local times (e.g. a Fri 10pm Central show) in UTC (\"Sat 3:00 AM\"),\r\n // which made already-ended events look like future ones.\r\n // (domain.timeZone is the verbatim PascalCase passthrough; domain.timezone\r\n // folds lowercase + snake_case — same chain as before.)\r\n const timezone = getIANATimezone(\r\n domain.timeZone ||\r\n domain.timezone ||\r\n rawVenue.timeZone ||\r\n rawVenue.timezone ||\r\n 'UTC'\r\n );\r\n\r\n // Transform tickets — TICKET SEAM: raw wire tickets by contract (see note).\r\n const rawTickets = rawEvent.availableTickets || [];\r\n const tickets: TicketTypeData[] = rawTickets.map((t: any) => ({\r\n id: t.id || t.ID,\r\n name: t.name || t.title || '',\r\n description: t.description || undefined,\r\n price: t.price ?? 0,\r\n fee: t.fee ?? 0,\r\n quantityAvailable:\r\n t.remainingCapacity ?? t.quantityRemaining ?? t.quantity ?? 0,\r\n quantityTotal: t.totalCapacity ?? t.quantity ?? 0,\r\n minPerOrder: t.minPerOrder ?? t.minQuantity ?? 1,\r\n maxPerOrder: t.maxPerOrder ?? t.maxQuantity ?? 10,\r\n status: mapTicketStatus(t),\r\n salesStartDate:\r\n t.salesBegin || t.salesStart || t.saleBegin || t.onSaleStart || undefined,\r\n salesEndDate: t.salesEnd || t.saleEnd || t.onSaleEnd || undefined,\r\n isDonation: t.type === 2 || t.ticketType === 2 || t.isDonation === true,\r\n isHidden: t.visibility === 2 || t.isHidden === true,\r\n ticketType: t.type || t.ticketType || undefined,\r\n sectionName: t.sectionName || t.section || undefined,\r\n }));\r\n\r\n // Calculate totals from public tickets\r\n const publicTickets = tickets.filter(t => !t.isHidden && !t.isDonation);\r\n const totalRemaining = publicTickets.reduce(\r\n (sum, t) => sum + t.quantityAvailable,\r\n 0\r\n );\r\n const totalCapacity = publicTickets.reduce(\r\n (sum, t) => sum + t.quantityTotal,\r\n 0\r\n );\r\n\r\n // Determine event status\r\n const status = mapEventStatus(domain.status as string, tickets);\r\n\r\n // Build image URL — IMAGE SEAM: image-first precedence (see note).\r\n const rawImage =\r\n rawEvent.image || rawEvent.imageUrl || rawEvent.image_url || '';\r\n const imageUrl = rawImage.startsWith('http')\r\n ? rawImage\r\n : getImageUrl(rawImage);\r\n\r\n // Venue\r\n const venue: VenueData = {\r\n id: rawVenue.id || rawVenue.ID || 0,\r\n name: rawVenue.name || rawVenue.googleLocationNameCache || '',\r\n address: rawVenue.address || rawVenue.googleLocationNameCache || '',\r\n timezone,\r\n serviceFeePercentage: rawVenue.serviceFeePercentage ?? rawVenue.fee ?? 0,\r\n serviceFeeCents: rawVenue.serviceFeeCents ?? 0,\r\n taxPercentage: rawVenue.taxPercentage ?? rawVenue.tax ?? 0,\r\n contactEmail: rawVenue.contactEmail || rawVenue.email || undefined,\r\n };\r\n\r\n // Performers — VENUE/PERFORMERS SEAM: raw passthrough map (see note).\r\n const rawPerformers = domain.performers || [];\r\n const performers: PerformerData[] = rawPerformers.map((p: any) => ({\r\n id: p.id || p.ID,\r\n displayName: p.displayName || p.name || '',\r\n profileImage: p.avatar || p.profileImage || undefined,\r\n role: p.role || undefined,\r\n }));\r\n\r\n // FAQs (may not be in API response)\r\n const faqs: FAQItem[] = (domain.faqs || []).map((f: any) => ({\r\n question: f.question || '',\r\n answer: f.answer || '',\r\n }));\r\n\r\n // Series showtimes → occurrences. A single series-event detail carries a\r\n // `showtimes` array; the v2 <ShowtimeDesigns> renders event.seriesOccurrences\r\n // under \"Upcoming Shows\". Without this map the section is always empty.\r\n const seriesOccurrences: SeriesOccurrence[] = (domain.showtimes || [])\r\n .filter((s: any) => s.published !== false)\r\n .map((s: any, i: number) => ({\r\n id: s.id ?? s.eventId ?? i + 1,\r\n slug: s.slug || '',\r\n startDateTime: s.startDateTime || s.startTime || '',\r\n endDateTime: s.endDateTime || s.endTime || '',\r\n status: (s.ctaState?.reason === 'sold_out'\r\n ? 'sold_out'\r\n : 'on_sale') as EventStatus,\r\n ctaText: s.ctaState?.text || 'Get tickets',\r\n ctaDisabled: s.ctaState?.disabled || false,\r\n }));\r\n\r\n // Date in venue timezone\r\n let date = '';\r\n if (domain.startDateTime) {\r\n const dt = new Date(domain.startDateTime);\r\n date = formatDateRaw(\r\n dt,\r\n { timeZone: timezone, year: 'numeric', month: '2-digit', day: '2-digit' },\r\n 'en-CA'\r\n );\r\n }\r\n\r\n return {\r\n id: domain.id,\r\n title: domain.title || '',\r\n slug: domain.slug || '',\r\n description: domain.description || domain.eventSummary || '',\r\n date,\r\n startDateTime: domain.startDateTime || '',\r\n endDateTime: domain.endDateTime || '',\r\n doorsOpenTime: domain.doorsOpenTime || '',\r\n timezone,\r\n imageUrl,\r\n venue,\r\n performers,\r\n tickets,\r\n faqs,\r\n disclaimer: domain.disclaimer || undefined,\r\n status,\r\n eventSeriesId: domain.eventSeriesId || undefined,\r\n seriesOccurrences: seriesOccurrences.length ? seriesOccurrences : undefined,\r\n // Public API hides the actual password (json:\"-\") and exposes a boolean\r\n // `hasPassword` flag instead; fall back to a populated `password` for\r\n // admin-shaped payloads.\r\n isPasswordProtected: domain.hasPassword === true || !!domain.password,\r\n isRegistrationEvent:\r\n domain.ticketType === 1 || domain.eventTicketingType === 1,\r\n totalTicketsRemaining: totalRemaining,\r\n totalTicketsCapacity: totalCapacity,\r\n stageName: domain.stage?.name || domain.stageName || undefined,\r\n ageRestriction: domain.ageRestriction || undefined,\r\n displayAgeRestriction: domain.displayAgeRestriction ?? false,\r\n displayStartTime: domain.displayStartTime ?? true,\r\n displayEndTime: domain.displayEndTime ?? true,\r\n displayDoorsTime: domain.displayDoorsTime ?? true,\r\n collectionId: domain.collectionId || undefined, // domain folds collection_id\r\n venueAddressLines: parseAddress(\r\n rawVenue.address || rawVenue.googleLocationNameCache || ''\r\n ),\r\n googleMapsUrl: buildGoogleMapsUrl(\r\n rawVenue.address || rawVenue.googleLocationNameCache || ''\r\n ),\r\n };\r\n}\r\n\r\n/**\r\n * Transform SeriesPageData (from the series API) into our clean EventData interface.\r\n * Sets eventSeriesId so EventExperience renders in full-width series mode.\r\n */\r\nexport function transformSeriesPageData(series: SeriesPageData): EventData {\r\n // Implementation moved to the canonical transformer home\r\n // (`$lib/api/transformers/series.ts`). This export stays as a thin delegate\r\n // so SeriesPage.svelte and other consumers don't churn.\r\n return parseSeries(series);\r\n}\r\n\r\n/**\r\n * Transform PublicCollectionData (from the collection API) into our clean EventData interface.\r\n * Sets collectionId so EventExperience renders in full-width collection mode.\r\n */\r\nexport function transformCollectionData(\r\n collection: PublicCollectionData\r\n): EventData {\r\n // Implementation moved to the canonical transformer home\r\n // (`$lib/api/transformers/collection.ts`). This export stays as a thin\r\n // delegate so CollectionView.svelte and other consumers don't churn.\r\n return parseCollection(collection);\r\n}\r\n"],"names":["CDN_BASE_URL","DEFAULT_CDN_BASE_URL","getImageUrl","imagePath","baseUrl","parseAddress","addressStr","parts","p","stateZipMatch","state","zip","city","lines","buildGoogleMapsUrl","address","parseSeries","series","seriesAny","timezone","getIANATimezone","rawImage","imageUrl","seriesOccurrences","occ","firstOcc","date","dt","formatDateRaw","rawFaq","faqs","f","performers","index","venueAddress","parseCollection","collection","collectionEvents","ev","rawImg","rawCover","mapTicketStatus","ticket","now","remaining","salesBegin","salesEnd","mapEventStatus","rawStatus","tickets","t","publicTickets","transformApiEvent","rawEvent","domain","parseEvent","rawVenue","totalRemaining","sum","totalCapacity","status","venue","s","i","transformSeriesPageData","transformCollectionData"],"mappings":";AAYO,MAAMA,IAAe,gDAEtBC,IAAuBD;AAOtB,SAASE,EACdC,GACAC,IAAkBH,GACV;AACR,SAAKE,IAEDA,EAAU,WAAW,SAAS,KAAKA,EAAU,WAAW,UAAU,IAC7DA,IAGF,GAAGC,CAAO,GAAGD,CAAS,KANN;AAOzB;AClBO,SAASE,EAAaC,GAA8B;AACzD,MAAI,CAACA,EAAY,QAAO,CAAA;AACxB,QAAMC,IAAQD,EAAW,MAAM,GAAG,EAAE,IAAI,CAAAE,MAAKA,EAAE,MAAM;AACrD,MAAID,EAAM,SAAS,EAAG,QAAO,CAACD,CAAU;AAExC,QAAMG,IADWF,EAAMA,EAAM,SAAS,CAAC,EACR,MAAM,iCAAiC;AACtE,MAAIE,KAAiBF,EAAM,UAAU,GAAG;AACtC,UAAMG,IAAQD,EAAc,CAAC,GACvBE,IAAMF,EAAc,CAAC,GACrBG,IAAOL,EAAMA,EAAM,SAAS,CAAC,GAC7BM,IAAQN,EAAM,MAAM,GAAG,EAAE;AAC/B,WAAAM,EAAM,KAAK,GAAGD,CAAI,KAAKF,CAAK,IAAIC,CAAG,EAAE,GAC9BE;AAAA,EACT;AACA,SAAO,CAACN,EAAM,CAAC,GAAGA,EAAM,MAAM,CAAC,EAAE,KAAK,IAAI,CAAC;AAC7C;AAKO,SAASO,EAAmBC,GAAyB;AAC1D,SAAKA,IACE,mDAAmD,mBAAmBA,CAAO,CAAC,KADhE;AAEvB;ACAO,SAASC,EAAYC,GAAmC;AAI7D,QAAMC,IAAYD,GAIZE,IAAWC;AAAA,IACfH,EAAO,YAAYC,EAAU,YAAYA,EAAU,aAAa;AAAA,EAAA,GAI5DG,IAAWJ,EAAO,SAAS,IAC3BK,IAAWD,EAAS,WAAW,MAAM,IACvCA,IACAA,IACEnB,EAAYmB,CAAQ,IACpB,IAGAE,KAAyCN,EAAO,eAAe,CAAA,GAAI;AAAA,IACvE,CAAAO,OAAQ;AAAA,MACN,IAAIA,EAAI;AAAA,MACR,MAAMA,EAAI,QAAQ;AAAA,MAClB,eAAeA,EAAI;AAAA,MACnB,aAAaA,EAAI,eAAe;AAAA,MAChC,QAASA,EAAI,UAAU,WAAW,aAC9B,aACA;AAAA,MACJ,SAASA,EAAI,UAAU,QAAQ;AAAA,MAC/B,aAAaA,EAAI,UAAU,YAAY;AAAA,IAAA;AAAA,EACzC,GAIIC,IAAWR,EAAO,cAAc,CAAC;AACvC,MAAIS,IAAO;AACX,MAAID,GAAU,eAAe;AAC3B,UAAME,IAAK,IAAI,KAAKF,EAAS,aAAa;AAC1C,IAAAC,IAAOE;AAAA,MACLD;AAAA,MACA,EAAE,UAAUR,GAAU,MAAM,WAAW,OAAO,WAAW,KAAK,UAAA;AAAA,MAC9D;AAAA,IAAA;AAAA,EAEJ;AAGA,QAAMU,IAASZ,EAAO,OAAO;AAC7B,MAAIa,IAAkB,CAAA;AACtB,EAAI,MAAM,QAAQD,CAAM,IACtBC,IAAOD,EAAO,IAAI,CAAAE,OAAM;AAAA,IACtB,UAAUA,EAAE,YAAY;AAAA,IACxB,QAAQA,EAAE,UAAU;AAAA,EAAA,EACpB,IACO,OAAOF,KAAW,YAAYA,MACvCC,IAAO,CAAC,EAAE,UAAU,OAAO,QAAQD,GAAQ;AAI7C,QAAMG,KAA+Bf,EAAO,cAAc,CAAA,GAAI;AAAA,IAC5D,CAACT,GAAGyB,OAAW;AAAA,MACb,IAAIA,IAAQ;AAAA,MACZ,aACEzB,EAAE,eAAe,CAACA,EAAE,WAAWA,EAAE,QAAQ,EAAE,OAAO,OAAO,EAAE,KAAK,GAAG;AAAA,MACrE,cAAcA,EAAE,gBAAgB;AAAA,IAAA;AAAA,EAClC,GAII0B,IACJjB,EAAO,OAAO,YAAYA,EAAO,OAAO,2BAA2B;AAErE,SAAO;AAAA,IACL,IAAIA,EAAO;AAAA,IACX,OAAOA,EAAO,SAAS;AAAA,IACvB,MAAM;AAAA,IACN,aAAaA,EAAO,eAAeA,EAAO,gBAAgB;AAAA,IAC1D,MAAAS;AAAA,IACA,eAAeD,GAAU,iBAAiB;AAAA,IAC1C,aAAaA,GAAU,eAAe;AAAA,IACtC,eAAe;AAAA,IACf,UAAAN;AAAA,IACA,UAAAG;AAAA,IACA,OAAO;AAAA,MACL,IAAIL,EAAO,OAAO,MAAM;AAAA,MACxB,MAAMA,EAAO,OAAO,QAAQ;AAAA,MAC5B,SAASiB;AAAA,MACT,UAAAf;AAAA,MACA,sBAAsB;AAAA,MACtB,iBAAiB;AAAA,MACjB,eAAe;AAAA,IAAA;AAAA,IAEjB,YAAAa;AAAA,IACA,SAAS,CAAA;AAAA,IACT,MAAAF;AAAA,IACA,YAAYb,EAAO,OAAO,cAAc;AAAA,IACxC,QAAQ;AAAA,IACR,eAAeA,EAAO;AAAA,IACtB,mBAAAM;AAAA,IACA,qBAAqB;AAAA,IACrB,qBAAqB;AAAA,IACrB,uBAAuB;AAAA,IACvB,sBAAsB;AAAA,IACtB,mBAAmBlB,EAAa6B,CAAY;AAAA,IAC5C,eAAepB,EAAmBoB,CAAY;AAAA,EAAA;AAElD;AC/GO,SAASC,EAAgBC,GAA6C;AAE3E,QAAMC,KAAuCD,EAAW,UAAU,CAAA,GAAI;AAAA,IACpE,CAAAE,MAAM;AACJ,YAAMC,IAASD,EAAG,SAAS;AAC3B,aAAO;AAAA,QACL,IAAIA,EAAG;AAAA,QACP,MAAMA,EAAG,SAAS;AAAA,QAClB,MAAMA,EAAG,QAAQ;AAAA,QACjB,OAAOC,EAAO,WAAW,MAAM,IAC3BA,IACAA,IACErC,EAAYqC,CAAM,IAClB;AAAA,QACN,eAAeD,EAAG,iBAAiB;AAAA,QACnC,QAASA,EAAG,UAAU;AAAA,MAAA;AAAA,IAE1B;AAAA,EAAA,GAIIE,IAAWJ,EAAW,cAAc,IACpCd,IAAWkB,EAAS,WAAW,MAAM,IACvCA,IACAA,IACEtC,EAAYsC,CAAQ,IACpB;AAEN,SAAO;AAAA,IACL,IAAIJ,EAAW;AAAA,IACf,OAAOA,EAAW,mBAAmB;AAAA,IACrC,MAAM;AAAA,IACN,aAAaA,EAAW,eAAeA,EAAW,WAAW;AAAA,IAC7D,MAAM;AAAA,IACN,eAAe;AAAA,IACf,aAAa;AAAA,IACb,eAAe;AAAA,IACf,UAAU;AAAA,IACV,UAAAd;AAAA,IACA,OAAO;AAAA,MACL,IAAI;AAAA,MACJ,MAAM;AAAA,MACN,SAAS;AAAA,MACT,UAAU;AAAA,MACV,sBAAsB;AAAA,MACtB,iBAAiB;AAAA,MACjB,eAAe;AAAA,IAAA;AAAA,IAEjB,YAAY,CAAA;AAAA,IACZ,SAAS,CAAA;AAAA,IACT,MAAM,CAAA;AAAA,IACN,QAAQ;AAAA,IACR,cAAcc,EAAW;AAAA,IACzB,kBAAAC;AAAA,IACA,qBAAqB;AAAA,IACrB,qBAAqB;AAAA,IACrB,uBAAuB;AAAA,IACvB,sBAAsB;AAAA,EAAA;AAE1B;AC/DA,SAASI,EAAgBC,GAA2B;AAClD,QAAMC,wBAAU,KAAA,GACVC,IACJF,EAAO,qBACPA,EAAO,qBACPA,EAAO,YACP,GACIG,IACJH,EAAO,cACPA,EAAO,cACPA,EAAO,aACPA,EAAO,aACHI,IAAWJ,EAAO,YAAYA,EAAO,WAAWA,EAAO;AAE7D,SAAIE,KAAa,IAAU,aACvBE,KAAY,IAAI,KAAKA,CAAQ,IAAIH,IAAY,gBAC7CE,KAAc,IAAI,KAAKA,CAAU,IAAIF,IAAY,gBAC9C;AACT;AAKA,SAASI,EACPC,GACAC,GACa;AACb,QAAM,KAAKD,KAAa,IAAI,cAAc,QAAQ,QAAQ,GAAG;AAE7D,MAAI,MAAM,eAAe,MAAM,WAAY,QAAO;AAGlD,MAAI,MAAM,UAAU,MAAM,QAAS,QAAO;AAY1C,MAHoCC,EAAQ;AAAA,IAC1C,CAAAC,MAAKA,EAAE,YAAYA,EAAE,WAAW;AAAA,EAAA,EAED,QAAO;AAGxC,QAAMC,IAAgBF,EAAQ,OAAO,CAAAC,MAAK,CAACA,EAAE,QAAQ;AACrD,MAAIC,EAAc,SAAS,GAAG;AAC5B,QAAIA,EAAc,MAAM,CAAAD,MAAKA,EAAE,WAAW,UAAU,EAAG,QAAO;AAC9D,QAAIC,EAAc,MAAM,CAAAD,MAAKA,EAAE,WAAW,aAAa;AACrD,aAAO;AACT,QACEC,EAAc;AAAA,MACZ,CAAAD,MAAKA,EAAE,WAAW,iBAAiBA,EAAE,WAAW;AAAA,IAAA;AAGlD,aAAO;AAAA,EACX;AAEA,SAAI,MAAM,cAAc,MAAM,YAAkB,aAC5C,MAAM,iBAAiB,MAAM,eAAqB,gBAClD,MAAM,iBAAiB,MAAM,eAAqB,gBAE/C;AACT;AAuBO,SAASE,EAAkBC,GAA0B;AAE1D,QAAMC,IAASC,EAAWF,CAAQ,GAC5BG,IAAWF,EAAO,SAAS,CAAA,GAO3BnC,IAAWC;AAAA,IACfkC,EAAO,YACLA,EAAO,YACPE,EAAS,YACTA,EAAS,YACT;AAAA,EAAA,GAKEP,KADaI,EAAS,oBAAoB,CAAA,GACH,IAAI,CAACH,OAAY;AAAA,IAC5D,IAAIA,EAAE,MAAMA,EAAE;AAAA,IACd,MAAMA,EAAE,QAAQA,EAAE,SAAS;AAAA,IAC3B,aAAaA,EAAE,eAAe;AAAA,IAC9B,OAAOA,EAAE,SAAS;AAAA,IAClB,KAAKA,EAAE,OAAO;AAAA,IACd,mBACEA,EAAE,qBAAqBA,EAAE,qBAAqBA,EAAE,YAAY;AAAA,IAC9D,eAAeA,EAAE,iBAAiBA,EAAE,YAAY;AAAA,IAChD,aAAaA,EAAE,eAAeA,EAAE,eAAe;AAAA,IAC/C,aAAaA,EAAE,eAAeA,EAAE,eAAe;AAAA,IAC/C,QAAQT,EAAgBS,CAAC;AAAA,IACzB,gBACEA,EAAE,cAAcA,EAAE,cAAcA,EAAE,aAAaA,EAAE,eAAe;AAAA,IAClE,cAAcA,EAAE,YAAYA,EAAE,WAAWA,EAAE,aAAa;AAAA,IACxD,YAAYA,EAAE,SAAS,KAAKA,EAAE,eAAe,KAAKA,EAAE,eAAe;AAAA,IACnE,UAAUA,EAAE,eAAe,KAAKA,EAAE,aAAa;AAAA,IAC/C,YAAYA,EAAE,QAAQA,EAAE,cAAc;AAAA,IACtC,aAAaA,EAAE,eAAeA,EAAE,WAAW;AAAA,EAAA,EAC3C,GAGIC,IAAgBF,EAAQ,OAAO,CAAAC,MAAK,CAACA,EAAE,YAAY,CAACA,EAAE,UAAU,GAChEO,IAAiBN,EAAc;AAAA,IACnC,CAACO,GAAKR,MAAMQ,IAAMR,EAAE;AAAA,IACpB;AAAA,EAAA,GAEIS,IAAgBR,EAAc;AAAA,IAClC,CAACO,GAAKR,MAAMQ,IAAMR,EAAE;AAAA,IACpB;AAAA,EAAA,GAIIU,IAASb,EAAeO,EAAO,QAAkBL,CAAO,GAGxD5B,IACJgC,EAAS,SAASA,EAAS,YAAYA,EAAS,aAAa,IACzD/B,IAAWD,EAAS,WAAW,MAAM,IACvCA,IACAnB,EAAYmB,CAAQ,GAGlBwC,IAAmB;AAAA,IACvB,IAAIL,EAAS,MAAMA,EAAS,MAAM;AAAA,IAClC,MAAMA,EAAS,QAAQA,EAAS,2BAA2B;AAAA,IAC3D,SAASA,EAAS,WAAWA,EAAS,2BAA2B;AAAA,IACjE,UAAArC;AAAA,IACA,sBAAsBqC,EAAS,wBAAwBA,EAAS,OAAO;AAAA,IACvE,iBAAiBA,EAAS,mBAAmB;AAAA,IAC7C,eAAeA,EAAS,iBAAiBA,EAAS,OAAO;AAAA,IACzD,cAAcA,EAAS,gBAAgBA,EAAS,SAAS;AAAA,EAAA,GAKrDxB,KADgBsB,EAAO,cAAc,CAAA,GACO,IAAI,CAAC9C,OAAY;AAAA,IACjE,IAAIA,EAAE,MAAMA,EAAE;AAAA,IACd,aAAaA,EAAE,eAAeA,EAAE,QAAQ;AAAA,IACxC,cAAcA,EAAE,UAAUA,EAAE,gBAAgB;AAAA,IAC5C,MAAMA,EAAE,QAAQ;AAAA,EAAA,EAChB,GAGIsB,KAAmBwB,EAAO,QAAQ,CAAA,GAAI,IAAI,CAACvB,OAAY;AAAA,IAC3D,UAAUA,EAAE,YAAY;AAAA,IACxB,QAAQA,EAAE,UAAU;AAAA,EAAA,EACpB,GAKIR,KAAyC+B,EAAO,aAAa,CAAA,GAChE,OAAO,CAACQ,MAAWA,EAAE,cAAc,EAAK,EACxC,IAAI,CAACA,GAAQC,OAAe;AAAA,IAC3B,IAAID,EAAE,MAAMA,EAAE,WAAWC,IAAI;AAAA,IAC7B,MAAMD,EAAE,QAAQ;AAAA,IAChB,eAAeA,EAAE,iBAAiBA,EAAE,aAAa;AAAA,IACjD,aAAaA,EAAE,eAAeA,EAAE,WAAW;AAAA,IAC3C,QAASA,EAAE,UAAU,WAAW,aAC5B,aACA;AAAA,IACJ,SAASA,EAAE,UAAU,QAAQ;AAAA,IAC7B,aAAaA,EAAE,UAAU,YAAY;AAAA,EAAA,EACrC;AAGJ,MAAIpC,IAAO;AACX,MAAI4B,EAAO,eAAe;AACxB,UAAM3B,IAAK,IAAI,KAAK2B,EAAO,aAAa;AACxC,IAAA5B,IAAOE;AAAA,MACLD;AAAA,MACA,EAAE,UAAUR,GAAU,MAAM,WAAW,OAAO,WAAW,KAAK,UAAA;AAAA,MAC9D;AAAA,IAAA;AAAA,EAEJ;AAEA,SAAO;AAAA,IACL,IAAImC,EAAO;AAAA,IACX,OAAOA,EAAO,SAAS;AAAA,IACvB,MAAMA,EAAO,QAAQ;AAAA,IACrB,aAAaA,EAAO,eAAeA,EAAO,gBAAgB;AAAA,IAC1D,MAAA5B;AAAA,IACA,eAAe4B,EAAO,iBAAiB;AAAA,IACvC,aAAaA,EAAO,eAAe;AAAA,IACnC,eAAeA,EAAO,iBAAiB;AAAA,IACvC,UAAAnC;AAAA,IACA,UAAAG;AAAA,IACA,OAAAuC;AAAA,IACA,YAAA7B;AAAA,IACA,SAAAiB;AAAA,IACA,MAAAnB;AAAA,IACA,YAAYwB,EAAO,cAAc;AAAA,IACjC,QAAAM;AAAA,IACA,eAAeN,EAAO,iBAAiB;AAAA,IACvC,mBAAmB/B,EAAkB,SAASA,IAAoB;AAAA;AAAA;AAAA;AAAA,IAIlE,qBAAqB+B,EAAO,gBAAgB,MAAQ,CAAC,CAACA,EAAO;AAAA,IAC7D,qBACEA,EAAO,eAAe,KAAKA,EAAO,uBAAuB;AAAA,IAC3D,uBAAuBG;AAAA,IACvB,sBAAsBE;AAAA,IACtB,WAAWL,EAAO,OAAO,QAAQA,EAAO,aAAa;AAAA,IACrD,gBAAgBA,EAAO,kBAAkB;AAAA,IACzC,uBAAuBA,EAAO,yBAAyB;AAAA,IACvD,kBAAkBA,EAAO,oBAAoB;AAAA,IAC7C,gBAAgBA,EAAO,kBAAkB;AAAA,IACzC,kBAAkBA,EAAO,oBAAoB;AAAA,IAC7C,cAAcA,EAAO,gBAAgB;AAAA;AAAA,IACrC,mBAAmBjD;AAAA,MACjBmD,EAAS,WAAWA,EAAS,2BAA2B;AAAA,IAAA;AAAA,IAE1D,eAAe1C;AAAA,MACb0C,EAAS,WAAWA,EAAS,2BAA2B;AAAA,IAAA;AAAA,EAC1D;AAEJ;AAMO,SAASQ,EAAwB/C,GAAmC;AAIzE,SAAOD,EAAYC,CAAM;AAC3B;AAMO,SAASgD,EACd7B,GACW;AAIX,SAAOD,EAAgBC,CAAU;AACnC;"}