stormcloud-video-player 0.7.43 → 0.7.44
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.
- package/dist/stormcloud-vp.min.js +1 -1
- package/lib/index.cjs +74 -2
- package/lib/index.cjs.map +1 -1
- package/lib/index.js +74 -2
- package/lib/index.js.map +1 -1
- package/lib/ui/OverlayRenderer.cjs.map +1 -1
- package/lib/ui/StormcloudVideoPlayer.cjs +74 -2
- package/lib/ui/StormcloudVideoPlayer.cjs.map +1 -1
- package/lib/utils/overlays.cjs +74 -2
- package/lib/utils/overlays.cjs.map +1 -1
- package/package.json +1 -1
|
@@ -1 +1 @@
|
|
|
1
|
-
{"version":3,"sources":["/home/ubuntu24-new/Dev/stormcloud-vp/lib/ui/OverlayRenderer.cjs","../../src/ui/OverlayRenderer.tsx","../../src/utils/overlays.ts"],"names":["__getOwnPropNames","Object","getOwnPropertyNames","__getProtoOf","getPrototypeOf","__hasOwnProp","hasOwnProperty","prototype","__export","target","all","name","__defProp","get","enumerable","__copyProps","to","from","except","desc","key","call","__getOwnPropDesc","__toESM","mod","isNodeMode","__create","__esModule","value","__toCommonJS","OverlayRenderer_exports","OverlayRenderer","module","exports","import_react","require","OVERLAY_API_BASE","timeStringToSeconds","timeStr","parts","split","length","hours","parseInt","minutes","secStr","dotIdx","indexOf","seconds","substring","msFrag","ms","padEnd","num","parseFloat","isFinite","Math","max","isOverlayActive","overlay","currentTime","visible","startSec","start_time","durationSec","duration","SWIRL_HD_AUTHORING_WIDTH","NAB_DEMO_NAME_PREFIX","overlayAuthoringDimensions","decodeWidth","decodeHeight","startsWith","width","height","SWIRL_HD_AUTHORING_HEIGHT","resolveImageUrl","imageUrl","apiBaseUrl","url","URL","origin","import_jsx_runtime","computeVideoDimensions","video","nativeWidth","videoWidth","nativeHeight","videoHeight","displayWidth","offsetWidth","displayHeight","offsetHeight","videoAspect","displayAspect","renderWidth","renderHeight","offsetX","offsetY","image_url","jsx","src","content","alignItems","justifyContent","color","fontSize","fontFamily","fontWeight","textAlign","boxSizing","wordBreak","textShadow","pointerEvents","userSelect","text","xmlText","maxItems","replace","DOMParser","parseFromString","stripped","doc","querySelectorAll","map","item","querySelector","description","textContent","trim","i","items","cachedFetchRSSItems","Map","rssUrl","expiresAt","now","cached","promise","fetchRSSItems","catch","err","encoded","resp","data","window","location","fetch","ok","includes","parseRSSXml","json","contents","Error","ScrollerOverlay","cfg","scroller_config","uid","useId","useState","rssLoading","setRssLoading","rssError","setRssError","rss_url","max_items","autoRefresh","auto_refresh","updateInterval","update_interval","useEffect","use_custom_text","custom_text","cancelled","then","setRssItems","finally","interval","setInterval","rssCache","delete","clearInterval","sep","segments","separator_char","rssItems","show_title","title","push","show_description","show_timestamp","pubDate","Date","toLocaleDateString","show_author","author","join","show_category","category","scrollSpeed","scroll_speed","direction","font_size","font_family","font_weight","textColor","text_color","bgColor","background_color","bgOpacity","background_opacity","borderRadius","border_radius","itemSpacing","item_spacing","pad","padding","label","labelLine2","label_line2","labelColor","label_color","labelTextColor","label_text_color","accentColor","accent_color","showAccentLine","show_accent_line","isEquitiesStrip","preset","isHorizontal","isReverse","fullText","animId","id","keyframes","jsxs","Fragment","children","style","display","flexDirection","overflow","backgroundColor","hexToRgb","background","flexShrink","flex","minHeight","gap","round","boxShadow","alignSelf","maxWidth","letterSpacing","lineHeight","textTransform","whiteSpace","opacity","position","minWidth","animation","willChange","copy","paddingRight","seg","React","margin","paddingBottom","parseConfig","JSON","parse","ScoreBugOverlay","size","f","w","textOverflow","homeTeam","marginTop","homeScore","period","clock","awayTeam","awayScore","sponsorText","sponsorImageUrl","borderTop","alt","objectFit","LowerThirdOverlay","accentH","h","headline","subtitle"],"mappings":"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","sourcesContent":["\"use strict\";\nvar __create = Object.create;\nvar __defProp = Object.defineProperty;\nvar __getOwnPropDesc = Object.getOwnPropertyDescriptor;\nvar __getOwnPropNames = Object.getOwnPropertyNames;\nvar __getProtoOf = Object.getPrototypeOf;\nvar __hasOwnProp = Object.prototype.hasOwnProperty;\nvar __export = (target, all) => {\n for (var name in all)\n __defProp(target, name, { get: all[name], enumerable: true });\n};\nvar __copyProps = (to, from, except, desc) => {\n if (from && typeof from === \"object\" || typeof from === \"function\") {\n for (let key of __getOwnPropNames(from))\n if (!__hasOwnProp.call(to, key) && key !== except)\n __defProp(to, key, { get: () => from[key], enumerable: !(desc = __getOwnPropDesc(from, key)) || desc.enumerable });\n }\n return to;\n};\nvar __toESM = (mod, isNodeMode, target) => (target = mod != null ? __create(__getProtoOf(mod)) : {}, __copyProps(\n // If the importer is in node compatibility mode or this is not an ESM\n // file that has been converted to a CommonJS file using a Babel-\n // compatible transform (i.e. \"__esModule\" has not been set), then set\n // \"default\" to the CommonJS \"module.exports\" for node compatibility.\n isNodeMode || !mod || !mod.__esModule ? __defProp(target, \"default\", { value: mod, enumerable: true }) : target,\n mod\n));\nvar __toCommonJS = (mod) => __copyProps(__defProp({}, \"__esModule\", { value: true }), mod);\n\n// src/ui/OverlayRenderer.tsx\nvar OverlayRenderer_exports = {};\n__export(OverlayRenderer_exports, {\n OverlayRenderer: () => OverlayRenderer\n});\nmodule.exports = __toCommonJS(OverlayRenderer_exports);\nvar import_react = __toESM(require(\"react\"), 1);\n\n// src/utils/overlays.ts\nvar OVERLAY_API_BASE = \"https://adstorm.co/api-adstorm-dev\";\nfunction timeStringToSeconds(timeStr) {\n if (!timeStr) return 0;\n const parts = timeStr.split(\":\");\n if (parts.length >= 3) {\n const hours = parseInt(parts[0] ?? \"0\", 10) || 0;\n const minutes = parseInt(parts[1] ?? \"0\", 10) || 0;\n const secStr = parts[2] ?? \"0\";\n const dotIdx = secStr.indexOf(\".\");\n const seconds = parseInt(dotIdx >= 0 ? secStr.substring(0, dotIdx) : secStr, 10) || 0;\n const msFrag = dotIdx >= 0 ? secStr.substring(dotIdx + 1) : \"\";\n const ms = msFrag ? parseInt(msFrag.padEnd(3, \"0\").substring(0, 3), 10) || 0 : 0;\n return hours * 3600 + minutes * 60 + seconds + ms / 1e3;\n }\n if (parts.length === 2) {\n const minutes = parseInt(parts[0] ?? \"0\", 10) || 0;\n const secStr = parts[1] ?? \"0\";\n const dotIdx = secStr.indexOf(\".\");\n const seconds = parseInt(dotIdx >= 0 ? secStr.substring(0, dotIdx) : secStr, 10) || 0;\n const msFrag = dotIdx >= 0 ? secStr.substring(dotIdx + 1) : \"\";\n const ms = msFrag ? parseInt(msFrag.padEnd(3, \"0\").substring(0, 3), 10) || 0 : 0;\n return minutes * 60 + seconds + ms / 1e3;\n }\n const num = parseFloat(timeStr);\n return isFinite(num) ? Math.max(0, num) : 0;\n}\nfunction isOverlayActive(overlay, currentTime) {\n if (!overlay.visible) return false;\n const startSec = timeStringToSeconds(overlay.start_time);\n const durationSec = timeStringToSeconds(overlay.duration);\n if (durationSec <= 0) return false;\n return currentTime >= startSec && currentTime < startSec + durationSec;\n}\nvar SWIRL_HD_AUTHORING_WIDTH = 1920;\nvar SWIRL_HD_AUTHORING_HEIGHT = 1080;\nvar NAB_DEMO_NAME_PREFIX = \"NAB Demo \\u2014 \";\nfunction overlayAuthoringDimensions(overlay, decodeWidth, decodeHeight) {\n if (overlay.name.startsWith(NAB_DEMO_NAME_PREFIX)) {\n return {\n width: SWIRL_HD_AUTHORING_WIDTH,\n height: SWIRL_HD_AUTHORING_HEIGHT\n };\n }\n if (!decodeWidth || !decodeHeight) {\n return {\n width: decodeWidth || SWIRL_HD_AUTHORING_WIDTH,\n height: decodeHeight || SWIRL_HD_AUTHORING_HEIGHT\n };\n }\n return { width: decodeWidth, height: decodeHeight };\n}\nfunction resolveImageUrl(imageUrl, apiBaseUrl = OVERLAY_API_BASE) {\n if (!imageUrl) return \"\";\n if (imageUrl.startsWith(\"http://\") || imageUrl.startsWith(\"https://\")) {\n return imageUrl;\n }\n if (imageUrl.startsWith(\"/\")) {\n try {\n const url = new URL(apiBaseUrl);\n return `${url.origin}${imageUrl}`;\n } catch {\n return imageUrl;\n }\n }\n return `${apiBaseUrl}/${imageUrl}`;\n}\n\n// src/ui/OverlayRenderer.tsx\nvar import_jsx_runtime = require(\"react/jsx-runtime\");\nfunction computeVideoDimensions(video) {\n const nativeWidth = video.videoWidth;\n const nativeHeight = video.videoHeight;\n if (!nativeWidth || !nativeHeight) return null;\n const displayWidth = video.offsetWidth;\n const displayHeight = video.offsetHeight;\n if (!displayWidth || !displayHeight) return null;\n const videoAspect = nativeWidth / nativeHeight;\n const displayAspect = displayWidth / displayHeight;\n let renderWidth;\n let renderHeight;\n let offsetX;\n let offsetY;\n if (videoAspect > displayAspect) {\n renderWidth = displayWidth;\n renderHeight = displayWidth / videoAspect;\n offsetX = 0;\n offsetY = (displayHeight - renderHeight) / 2;\n } else {\n renderHeight = displayHeight;\n renderWidth = displayHeight * videoAspect;\n offsetX = (displayWidth - renderWidth) / 2;\n offsetY = 0;\n }\n return {\n nativeWidth,\n nativeHeight,\n displayWidth: renderWidth,\n displayHeight: renderHeight,\n offsetX,\n offsetY,\n scaleX: renderWidth / nativeWidth,\n scaleY: renderHeight / nativeHeight\n };\n}\nfunction ImageOverlay({ overlay }) {\n const src = resolveImageUrl(overlay.image_url || \"\");\n if (!src) return null;\n return /* @__PURE__ */ (0, import_jsx_runtime.jsx)(\n \"img\",\n {\n src,\n alt: overlay.name,\n draggable: false,\n style: {\n width: \"100%\",\n height: \"100%\",\n objectFit: \"contain\",\n display: \"block\",\n pointerEvents: \"none\",\n userSelect: \"none\"\n }\n }\n );\n}\nfunction TextOverlay({ overlay }) {\n const text = overlay.content || \"\";\n return /* @__PURE__ */ (0, import_jsx_runtime.jsx)(\n \"div\",\n {\n style: {\n width: \"100%\",\n height: \"100%\",\n display: \"flex\",\n alignItems: \"center\",\n justifyContent: \"center\",\n color: \"#ffffff\",\n fontSize: \"clamp(10px, 1.4vw, 20px)\",\n fontFamily: \"Roboto, 'Segoe UI', Arial, sans-serif\",\n fontWeight: 600,\n textAlign: \"center\",\n padding: \"4px 8px\",\n boxSizing: \"border-box\",\n wordBreak: \"break-word\",\n textShadow: \"0 1px 4px rgba(0,0,0,0.7)\",\n pointerEvents: \"none\",\n userSelect: \"none\",\n lineHeight: 1.3\n },\n children: text\n }\n );\n}\nfunction parseRSSXml(xmlText, maxItems) {\n const stripped = xmlText.replace(/^<\\?xml[^?]*\\?>\\s*/, \"\");\n const parser = new DOMParser();\n const doc = parser.parseFromString(stripped, \"text/xml\");\n const items = Array.from(doc.querySelectorAll(\"item\")).map((item) => ({\n title: (item.querySelector(\"title\")?.textContent || \"\").replace(/<[^>]*>/g, \"\").trim(),\n description: (item.querySelector(\"description\")?.textContent || \"\").replace(/<[^>]*>/g, \"\").trim(),\n pubDate: item.querySelector(\"pubDate\")?.textContent || \"\",\n author: item.querySelector(\"author, dc\\\\:creator\")?.textContent || \"\",\n category: item.querySelector(\"category\")?.textContent || \"\"\n })).filter((i) => i.title).slice(0, maxItems);\n if (items.length === 0 && doc.querySelector(\"parsererror\")) {\n throw new Error(\"Invalid RSS XML\");\n }\n return items;\n}\nvar RSS_CACHE_TTL_MS = 6e4;\nvar rssCache = /* @__PURE__ */ new Map();\nfunction cachedFetchRSSItems(rssUrl, maxItems) {\n const now = Date.now();\n const cached = rssCache.get(rssUrl);\n if (cached && cached.expiresAt > now) return cached.promise;\n const promise = fetchRSSItems(rssUrl, maxItems).catch((err) => {\n rssCache.delete(rssUrl);\n throw err;\n });\n rssCache.set(rssUrl, { promise, expiresAt: now + RSS_CACHE_TTL_MS });\n return promise;\n}\nasync function fetchRSSItems(rssUrl, maxItems) {\n const encoded = encodeURIComponent(rssUrl);\n try {\n const origin = typeof window !== \"undefined\" ? window.location.origin : \"\";\n const resp = await fetch(`${origin}/api/rss-proxy?url=${encoded}`);\n if (resp.ok) {\n const text = await resp.text();\n if (text.includes(\"<item\")) return parseRSSXml(text, maxItems);\n }\n } catch {\n }\n try {\n const resp = await fetch(`https://api.allorigins.win/get?url=${encoded}`);\n if (resp.ok) {\n const data = await resp.json();\n if (data.contents) return parseRSSXml(data.contents, maxItems);\n }\n } catch {\n }\n try {\n const resp = await fetch(`https://corsproxy.io/?url=${encoded}`);\n if (resp.ok) {\n const text = await resp.text();\n if (text) return parseRSSXml(text, maxItems);\n }\n } catch {\n }\n throw new Error(\"All RSS proxies failed\");\n}\nfunction ScrollerOverlay({ overlay }) {\n const cfg = overlay.scroller_config;\n const uid = (0, import_react.useId)().replace(/:/g, \"\");\n const [rssItems, setRssItems] = (0, import_react.useState)([]);\n const [rssLoading, setRssLoading] = (0, import_react.useState)(true);\n const [rssError, setRssError] = (0, import_react.useState)(false);\n const rssUrl = cfg?.rss_url || \"\";\n const maxItems = cfg?.max_items ?? 10;\n const autoRefresh = cfg?.auto_refresh !== false;\n const updateInterval = cfg?.update_interval ?? 5;\n (0, import_react.useEffect)(() => {\n if (!rssUrl || cfg?.use_custom_text && cfg?.custom_text) {\n setRssLoading(false);\n return;\n }\n let cancelled = false;\n setRssLoading(true);\n setRssError(false);\n cachedFetchRSSItems(rssUrl, maxItems).then((items) => {\n if (!cancelled) {\n setRssItems(items);\n setRssError(false);\n }\n }).catch(() => {\n if (!cancelled) setRssError(true);\n }).finally(() => {\n if (!cancelled) setRssLoading(false);\n });\n return () => {\n cancelled = true;\n };\n }, [rssUrl, maxItems, cfg?.use_custom_text, cfg?.custom_text]);\n (0, import_react.useEffect)(() => {\n if (!rssUrl || !autoRefresh || cfg?.use_custom_text && cfg?.custom_text) return;\n const interval = setInterval(() => {\n rssCache.delete(rssUrl);\n cachedFetchRSSItems(rssUrl, maxItems).then((items) => {\n setRssItems(items);\n setRssError(false);\n }).catch(() => {\n });\n }, updateInterval * 60 * 1e3);\n return () => clearInterval(interval);\n }, [rssUrl, autoRefresh, updateInterval, maxItems, cfg?.use_custom_text, cfg?.custom_text]);\n const sep = cfg?.separator_char ?? \"\\u25C6\";\n let segments;\n if (cfg?.use_custom_text && cfg?.custom_text) {\n segments = [cfg.custom_text];\n } else if (rssItems.length > 0) {\n segments = rssItems.map((item) => {\n const parts = [];\n if (cfg?.show_title !== false && item.title) parts.push(item.title);\n if (cfg?.show_description && item.description) parts.push(item.description);\n if (cfg?.show_timestamp && item.pubDate) {\n try {\n parts.push(new Date(item.pubDate).toLocaleDateString());\n } catch {\n }\n }\n if (cfg?.show_author && item.author) parts.push(`\\u2014 ${item.author}`);\n if (cfg?.show_category && item.category) parts.push(`[${item.category}]`);\n return parts.join(\" \");\n });\n } else if (rssLoading) {\n segments = [\"Loading feed\\u2026\"];\n } else if (rssError) {\n segments = overlay.content ? [overlay.content] : [\"RSS feed unavailable\"];\n } else if (overlay.content) {\n segments = [overlay.content];\n } else {\n segments = rssUrl ? [\"Loading feed\\u2026\"] : [\"RSS Ticker\"];\n }\n const scrollSpeed = cfg?.scroll_speed ?? 40;\n const direction = cfg?.direction ?? \"left\";\n const fontSize = cfg?.font_size ?? 15;\n const fontFamily = cfg?.font_family || \"Roboto, 'Segoe UI', Arial, sans-serif\";\n const fontWeight = cfg?.font_weight || \"700\";\n const textColor = cfg?.text_color || \"#ffffff\";\n const bgColor = cfg?.background_color || \"#0d0d1a\";\n const bgOpacity = cfg?.background_opacity !== void 0 ? cfg.background_opacity / 100 : 0.95;\n const borderRadius = cfg?.border_radius ?? 0;\n const itemSpacing = cfg?.item_spacing ?? 60;\n const pad = cfg?.padding !== void 0 && cfg.padding >= 0 ? cfg.padding : 8;\n const label = cfg?.label ?? \"NEWS\";\n const labelLine2 = cfg?.label_line2 ?? \"\";\n const labelColor = cfg?.label_color ?? \"#f97316\";\n const labelTextColor = cfg?.label_text_color ?? \"#ffffff\";\n const accentColor = cfg?.accent_color ?? labelColor;\n const showAccentLine = cfg?.show_accent_line !== false;\n const isEquitiesStrip = cfg?.preset === \"equities_strip\";\n const isHorizontal = direction === \"left\" || direction === \"right\";\n const isReverse = direction === \"right\" || direction === \"down\";\n const fullText = segments.join(` ${sep} `);\n const durationSec = Math.max(6, fullText.length * 9 / scrollSpeed);\n const animId = `sc-ticker-${overlay.id}-${uid}`;\n const keyframes = isHorizontal ? `@keyframes ${animId} {\n ${isReverse ? \"0% { transform: translateX(-50%); } 100% { transform: translateX(0%); }\" : \"0% { transform: translateX(0); } 100% { transform: translateX(-50%); }\"}\n }` : `@keyframes ${animId} {\n ${isReverse ? \"0% { transform: translateY(-50%); } 100% { transform: translateY(0%); }\" : \"0% { transform: translateY(0); } 100% { transform: translateY(-50%); }\"}\n }`;\n return /* @__PURE__ */ (0, import_jsx_runtime.jsxs)(import_jsx_runtime.Fragment, { children: [\n /* @__PURE__ */ (0, import_jsx_runtime.jsx)(\"style\", { children: keyframes }),\n /* @__PURE__ */ (0, import_jsx_runtime.jsxs)(\n \"div\",\n {\n style: {\n width: \"100%\",\n height: \"100%\",\n display: \"flex\",\n flexDirection: \"column\",\n overflow: \"hidden\",\n borderRadius: borderRadius > 0 ? `${borderRadius}px` : void 0,\n backgroundColor: `rgba(${hexToRgb(bgColor)}, ${bgOpacity})`,\n fontFamily,\n fontSize: `${fontSize}px`,\n fontWeight,\n color: textColor,\n pointerEvents: \"none\",\n userSelect: \"none\"\n },\n children: [\n showAccentLine && /* @__PURE__ */ (0, import_jsx_runtime.jsx)(\n \"div\",\n {\n style: {\n height: isEquitiesStrip ? 2 : 3,\n background: accentColor,\n flexShrink: 0,\n width: \"100%\"\n }\n }\n ),\n /* @__PURE__ */ (0, import_jsx_runtime.jsxs)(\n \"div\",\n {\n style: {\n display: \"flex\",\n flex: 1,\n overflow: \"hidden\",\n minHeight: 0,\n alignItems: \"center\",\n padding: isEquitiesStrip ? `${Math.max(2, pad * 0.5)}px ${pad}px` : `${Math.max(4, pad * 0.75)}px ${pad}px`,\n gap: isEquitiesStrip ? Math.max(4, Math.round(pad * 0.75)) : Math.max(6, pad),\n boxSizing: \"border-box\"\n },\n children: [\n label && /* @__PURE__ */ (0, import_jsx_runtime.jsxs)(\n \"div\",\n {\n style: {\n background: labelColor,\n color: labelTextColor,\n padding: isEquitiesStrip ? \"6px 12px\" : \"10px 14px\",\n display: \"flex\",\n flexDirection: isEquitiesStrip ? \"row\" : \"column\",\n alignItems: \"center\",\n justifyContent: \"center\",\n flexShrink: 0,\n textAlign: \"center\",\n gap: isEquitiesStrip ? 0 : 2,\n borderRadius: Math.max(2, borderRadius > 0 ? borderRadius : 6),\n boxShadow: isEquitiesStrip ? \"inset 0 1px 0 rgba(255,255,255,0.08)\" : \"0 2px 8px rgba(0,0,0,0.35)\",\n alignSelf: \"stretch\",\n maxWidth: isEquitiesStrip ? \"min(200px, 28%)\" : void 0\n },\n children: [\n /* @__PURE__ */ (0, import_jsx_runtime.jsx)(\n \"span\",\n {\n style: {\n fontWeight: isEquitiesStrip ? 700 : 800,\n fontSize: isEquitiesStrip ? \"0.82em\" : \"0.78em\",\n letterSpacing: isEquitiesStrip ? \"0.04em\" : \"0.08em\",\n lineHeight: 1.2,\n textTransform: isEquitiesStrip ? \"none\" : \"uppercase\",\n whiteSpace: isEquitiesStrip ? \"normal\" : \"nowrap\",\n textAlign: isEquitiesStrip ? \"left\" : \"center\"\n },\n children: label\n }\n ),\n labelLine2 && !isEquitiesStrip && /* @__PURE__ */ (0, import_jsx_runtime.jsx)(\n \"span\",\n {\n style: {\n fontWeight: 600,\n fontSize: \"0.58em\",\n letterSpacing: \"0.06em\",\n lineHeight: 1.15,\n opacity: 0.92,\n whiteSpace: \"nowrap\"\n },\n children: labelLine2\n }\n )\n ]\n }\n ),\n label && /* @__PURE__ */ (0, import_jsx_runtime.jsx)(\n \"div\",\n {\n style: {\n width: isEquitiesStrip ? 1 : 2,\n alignSelf: \"stretch\",\n minHeight: isEquitiesStrip ? 20 : 24,\n background: accentColor,\n flexShrink: 0,\n borderRadius: 1,\n opacity: 0.85\n }\n }\n ),\n /* @__PURE__ */ (0, import_jsx_runtime.jsx)(\n \"div\",\n {\n style: {\n flex: 1,\n overflow: \"hidden\",\n position: \"relative\",\n display: \"flex\",\n alignItems: \"center\",\n minWidth: 0\n },\n children: isHorizontal ? /* @__PURE__ */ (0, import_jsx_runtime.jsx)(\n \"div\",\n {\n style: {\n display: \"inline-flex\",\n whiteSpace: \"nowrap\",\n animation: `${animId} ${durationSec}s linear infinite`,\n willChange: \"transform\"\n },\n children: [0, 1].map((copy) => /* @__PURE__ */ (0, import_jsx_runtime.jsx)(\"span\", { style: { paddingRight: `${itemSpacing}px` }, children: segments.map((seg, i) => /* @__PURE__ */ (0, import_jsx_runtime.jsxs)(import_react.default.Fragment, { children: [\n i > 0 && /* @__PURE__ */ (0, import_jsx_runtime.jsx)(\"span\", { style: { opacity: 0.5, margin: \"0 8px\" }, children: sep }),\n /* @__PURE__ */ (0, import_jsx_runtime.jsx)(\"span\", { style: { textShadow: \"0 1px 3px rgba(0,0,0,0.6)\" }, children: seg })\n ] }, i)) }, copy))\n }\n ) : /* @__PURE__ */ (0, import_jsx_runtime.jsx)(\n \"div\",\n {\n style: {\n display: \"flex\",\n flexDirection: \"column\",\n whiteSpace: \"nowrap\",\n animation: `${animId} ${durationSec}s linear infinite`,\n willChange: \"transform\"\n },\n children: [0, 1].map(\n (copy) => segments.map((seg, i) => /* @__PURE__ */ (0, import_jsx_runtime.jsx)(\"div\", { style: { paddingBottom: `${itemSpacing / 4}px` }, children: seg }, `${copy}-${i}`))\n )\n }\n )\n }\n )\n ]\n }\n )\n ]\n }\n )\n ] });\n}\nfunction parseConfig(content) {\n if (!content) return null;\n try {\n return JSON.parse(content);\n } catch {\n return null;\n }\n}\nfunction ScoreBugOverlay({ overlay, size }) {\n const cfg = parseConfig(overlay.content);\n if (!cfg) return null;\n const f = Math.max(6, size.w * 0.052);\n return /* @__PURE__ */ (0, import_jsx_runtime.jsxs)(\n \"div\",\n {\n style: {\n width: \"100%\",\n height: \"100%\",\n borderRadius: Math.max(2, size.w * 0.028),\n display: \"flex\",\n flexDirection: \"column\",\n background: cfg.backgroundColor,\n color: cfg.textColor,\n fontFamily: \"'Inter', 'Roboto', 'Segoe UI', system-ui, sans-serif\",\n overflow: \"hidden\",\n pointerEvents: \"none\",\n userSelect: \"none\",\n fontSize: `${f}px`,\n boxShadow: \"0 8px 28px rgba(0,0,0,0.45)\"\n },\n children: [\n /* @__PURE__ */ (0, import_jsx_runtime.jsxs)(\n \"div\",\n {\n style: {\n flex: 1,\n display: \"flex\",\n alignItems: \"center\",\n padding: `${f * 0.42}px ${f * 0.85}px`,\n gap: f * 0.45,\n minHeight: 0\n },\n children: [\n /* @__PURE__ */ (0, import_jsx_runtime.jsxs)(\"div\", { style: { flex: 1, textAlign: \"center\", minWidth: 0, padding: `0 ${f * 0.15}px` }, children: [\n /* @__PURE__ */ (0, import_jsx_runtime.jsx)(\n \"div\",\n {\n style: {\n fontSize: \"1.05em\",\n fontWeight: 800,\n letterSpacing: \"0.02em\",\n overflow: \"hidden\",\n textOverflow: \"ellipsis\",\n whiteSpace: \"nowrap\"\n },\n children: cfg.homeTeam\n }\n ),\n /* @__PURE__ */ (0, import_jsx_runtime.jsx)(\"div\", { style: { fontSize: \"1.75em\", fontWeight: 900, lineHeight: 1.05, marginTop: f * 0.08 }, children: cfg.homeScore })\n ] }),\n /* @__PURE__ */ (0, import_jsx_runtime.jsxs)(\n \"div\",\n {\n style: {\n fontSize: \"0.88em\",\n textAlign: \"center\",\n fontWeight: 600,\n opacity: 0.92,\n color: cfg.textColor,\n padding: `0 ${f * 0.5}px`,\n flexShrink: 0,\n lineHeight: 1.25,\n textTransform: \"uppercase\",\n letterSpacing: \"0.04em\"\n },\n children: [\n /* @__PURE__ */ (0, import_jsx_runtime.jsx)(\"div\", { children: cfg.period }),\n /* @__PURE__ */ (0, import_jsx_runtime.jsx)(\"div\", { style: { fontWeight: 700, opacity: 1 }, children: cfg.clock })\n ]\n }\n ),\n /* @__PURE__ */ (0, import_jsx_runtime.jsxs)(\"div\", { style: { flex: 1, textAlign: \"center\", minWidth: 0, padding: `0 ${f * 0.15}px` }, children: [\n /* @__PURE__ */ (0, import_jsx_runtime.jsx)(\n \"div\",\n {\n style: {\n fontSize: \"1.05em\",\n fontWeight: 800,\n letterSpacing: \"0.02em\",\n overflow: \"hidden\",\n textOverflow: \"ellipsis\",\n whiteSpace: \"nowrap\"\n },\n children: cfg.awayTeam\n }\n ),\n /* @__PURE__ */ (0, import_jsx_runtime.jsx)(\"div\", { style: { fontSize: \"1.75em\", fontWeight: 900, lineHeight: 1.05, marginTop: f * 0.08 }, children: cfg.awayScore })\n ] })\n ]\n }\n ),\n (cfg.sponsorText || cfg.sponsorImageUrl) && /* @__PURE__ */ (0, import_jsx_runtime.jsxs)(\n \"div\",\n {\n style: {\n fontSize: \"0.72em\",\n textAlign: \"center\",\n fontWeight: 600,\n opacity: 0.88,\n padding: `${f * 0.34}px ${f * 0.55}px`,\n borderTop: `1px solid ${cfg.accentColor}55`,\n display: \"flex\",\n alignItems: \"center\",\n justifyContent: \"center\",\n gap: f * 0.45,\n overflow: \"hidden\",\n letterSpacing: \"0.04em\",\n textTransform: \"uppercase\",\n background: `linear-gradient(180deg, rgba(0,0,0,0.12) 0%, rgba(0,0,0,0.22) 100%)`\n },\n children: [\n cfg.sponsorImageUrl && /* @__PURE__ */ (0, import_jsx_runtime.jsx)(\"img\", { src: cfg.sponsorImageUrl, alt: \"sponsor\", style: { height: `${f * 1.35}px`, objectFit: \"contain\", flexShrink: 0 } }),\n cfg.sponsorText && /* @__PURE__ */ (0, import_jsx_runtime.jsx)(\"span\", { style: { overflow: \"hidden\", textOverflow: \"ellipsis\", whiteSpace: \"nowrap\" }, children: cfg.sponsorText })\n ]\n }\n )\n ]\n }\n );\n}\nfunction LowerThirdOverlay({ overlay, size }) {\n const cfg = parseConfig(overlay.content);\n if (!cfg) return null;\n const f = Math.max(6, size.w * 0.05);\n const accentH = Math.max(3, size.h * 0.038);\n return /* @__PURE__ */ (0, import_jsx_runtime.jsxs)(\n \"div\",\n {\n style: {\n width: \"100%\",\n height: \"100%\",\n borderRadius: Math.max(2, size.w * 0.018),\n display: \"flex\",\n flexDirection: \"column\",\n justifyContent: \"flex-end\",\n background: cfg.backgroundColor,\n color: cfg.textColor,\n fontFamily: \"'Inter', 'Roboto', 'Segoe UI', system-ui, sans-serif\",\n overflow: \"hidden\",\n pointerEvents: \"none\",\n userSelect: \"none\",\n fontSize: `${f}px`,\n boxShadow: \"0 10px 32px rgba(0,0,0,0.4)\"\n },\n children: [\n /* @__PURE__ */ (0, import_jsx_runtime.jsx)(\"div\", { style: { width: \"100%\", height: accentH, backgroundColor: cfg.accentColor, flexShrink: 0 } }),\n /* @__PURE__ */ (0, import_jsx_runtime.jsxs)(\n \"div\",\n {\n style: {\n flex: 1,\n display: \"flex\",\n flexDirection: \"column\",\n justifyContent: \"center\",\n padding: `${f * 0.62}px ${f * 1.05}px`,\n minHeight: 0,\n gap: f * 0.18\n },\n children: [\n /* @__PURE__ */ (0, import_jsx_runtime.jsx)(\n \"div\",\n {\n style: {\n fontSize: \"1.28em\",\n fontWeight: 800,\n lineHeight: 1.22,\n letterSpacing: \"0.01em\",\n textShadow: \"0 2px 6px rgba(0,0,0,0.55)\"\n },\n children: cfg.headline\n }\n ),\n /* @__PURE__ */ (0, import_jsx_runtime.jsx)(\n \"div\",\n {\n style: {\n fontSize: \"0.94em\",\n fontWeight: 500,\n opacity: 0.92,\n lineHeight: 1.35,\n color: cfg.textColor\n },\n children: cfg.subtitle\n }\n )\n ]\n }\n ),\n (cfg.sponsorText || cfg.sponsorImageUrl) && /* @__PURE__ */ (0, import_jsx_runtime.jsxs)(\n \"div\",\n {\n style: {\n fontSize: \"0.68em\",\n fontWeight: 600,\n opacity: 0.72,\n padding: `${f * 0.28}px ${f * 1.05}px ${f * 0.48}px`,\n display: \"flex\",\n alignItems: \"center\",\n gap: f * 0.4,\n overflow: \"hidden\",\n letterSpacing: \"0.06em\",\n textTransform: \"uppercase\",\n borderTop: `1px solid rgba(255,255,255,0.08)`\n },\n children: [\n cfg.sponsorImageUrl && /* @__PURE__ */ (0, import_jsx_runtime.jsx)(\"img\", { src: cfg.sponsorImageUrl, alt: \"sponsor\", style: { height: `${f * 1.25}px`, objectFit: \"contain\", flexShrink: 0 } }),\n cfg.sponsorText && /* @__PURE__ */ (0, import_jsx_runtime.jsx)(\"span\", { style: { overflow: \"hidden\", textOverflow: \"ellipsis\", whiteSpace: \"nowrap\" }, children: cfg.sponsorText })\n ]\n }\n )\n ]\n }\n );\n}\nfunction QrCodeOverlay({ overlay, size }) {\n const cfg = parseConfig(overlay.content);\n if (!cfg) return null;\n const qrSide = Math.max(32, Math.min(size.w, size.h) * 0.55);\n const qrUrl = `https://api.qrserver.com/v1/create-qr-code/?size=${Math.round(qrSide * 2)}x${Math.round(qrSide * 2)}&data=${encodeURIComponent(cfg.url || \"https://example.com\")}`;\n const f = Math.max(6, size.w * 0.06);\n return /* @__PURE__ */ (0, import_jsx_runtime.jsxs)(\"div\", { style: { width: \"100%\", height: \"100%\", borderRadius: Math.max(2, size.w * 0.035), display: \"flex\", flexDirection: \"column\", alignItems: \"center\", justifyContent: \"center\", gap: f * 0.4, background: cfg.backgroundColor, color: cfg.textColor, fontFamily: \"Roboto, 'Segoe UI', Arial, sans-serif\", padding: f * 0.6, boxSizing: \"border-box\", pointerEvents: \"none\", userSelect: \"none\", overflow: \"hidden\" }, children: [\n /* @__PURE__ */ (0, import_jsx_runtime.jsx)(\"div\", { style: { flexShrink: 0, background: \"#fff\", borderRadius: Math.max(2, qrSide * 0.06), padding: Math.max(2, qrSide * 0.06), lineHeight: 0 }, children: /* @__PURE__ */ (0, import_jsx_runtime.jsx)(\"img\", { src: qrUrl, alt: \"QR Code\", style: { width: `${qrSide}px`, height: `${qrSide}px`, display: \"block\" } }) }),\n /* @__PURE__ */ (0, import_jsx_runtime.jsx)(\"div\", { style: { fontSize: `${f * 1.1}px`, fontWeight: 700, textAlign: \"center\", color: cfg.accentColor, overflow: \"hidden\", textOverflow: \"ellipsis\", whiteSpace: \"nowrap\", width: \"100%\" }, children: cfg.ctaText }),\n cfg.description && /* @__PURE__ */ (0, import_jsx_runtime.jsx)(\"div\", { style: { fontSize: `${f * 0.75}px`, opacity: 0.6, textAlign: \"center\", overflow: \"hidden\", textOverflow: \"ellipsis\", whiteSpace: \"nowrap\", width: \"100%\" }, children: cfg.description })\n ] });\n}\nfunction ComingUpNextOverlay({ overlay, size }) {\n const cfg = parseConfig(overlay.content);\n if (!cfg) return null;\n const f = Math.max(8, Math.min(size.h * 0.13, size.w * 0.048));\n const showSubtitle = size.h >= 60;\n const showThumbnail = false;\n const thumbW = 0;\n return /* @__PURE__ */ (0, import_jsx_runtime.jsxs)(\"div\", { style: {\n width: \"100%\",\n height: \"100%\",\n borderRadius: Math.max(2, size.w * 0.025),\n display: \"flex\",\n background: cfg.backgroundColor,\n color: cfg.textColor,\n fontFamily: \"'Arial', 'Helvetica Neue', Helvetica, sans-serif\",\n overflow: \"hidden\",\n pointerEvents: \"none\",\n userSelect: \"none\",\n fontSize: `${f}px`,\n WebkitFontSmoothing: \"antialiased\"\n }, children: [\n /* @__PURE__ */ (0, import_jsx_runtime.jsx)(\"div\", { style: { width: Math.max(3, size.w * 0.018), flexShrink: 0, backgroundColor: cfg.accentColor } }),\n /* @__PURE__ */ (0, import_jsx_runtime.jsxs)(\"div\", { style: {\n flex: 1,\n display: \"flex\",\n flexDirection: \"column\",\n justifyContent: \"center\",\n padding: `${f * 0.35}px ${f * 0.75}px`,\n minWidth: 0,\n gap: `${f * 0.08}px`\n }, children: [\n /* @__PURE__ */ (0, import_jsx_runtime.jsx)(\"div\", { style: {\n fontSize: \"0.7em\",\n fontWeight: 700,\n textTransform: \"uppercase\",\n letterSpacing: \"0.09em\",\n color: cfg.accentColor,\n lineHeight: 1\n }, children: \"Coming Up Next\" }),\n /* @__PURE__ */ (0, import_jsx_runtime.jsx)(\"div\", { style: {\n fontSize: \"1.2em\",\n fontWeight: 700,\n lineHeight: 1.2,\n overflow: \"hidden\",\n textOverflow: \"ellipsis\",\n whiteSpace: \"nowrap\"\n }, children: cfg.title }),\n showSubtitle && /* @__PURE__ */ (0, import_jsx_runtime.jsx)(\"div\", { style: {\n fontSize: \"0.82em\",\n opacity: 0.65,\n lineHeight: 1.1,\n overflow: \"hidden\",\n textOverflow: \"ellipsis\",\n whiteSpace: \"nowrap\"\n }, children: cfg.subtitle }),\n cfg.scheduledTime && /* @__PURE__ */ (0, import_jsx_runtime.jsx)(\"div\", { style: {\n fontSize: \"0.9em\",\n fontWeight: 700,\n color: cfg.accentColor,\n lineHeight: 1\n }, children: cfg.scheduledTime })\n ] }),\n showThumbnail && /* @__PURE__ */ (0, import_jsx_runtime.jsx)(\"div\", { style: { flexShrink: 0, width: `${thumbW}px`, overflow: \"hidden\" }, children: /* @__PURE__ */ (0, import_jsx_runtime.jsx)(\n \"img\",\n {\n src: cfg.thumbnailUrl,\n alt: \"\",\n style: { width: \"100%\", height: \"100%\", objectFit: \"cover\", display: \"block\" }\n }\n ) })\n ] });\n}\nfunction ContextualTriggerOverlay({ overlay, size }) {\n const cfg = parseConfig(overlay.content);\n if (!cfg) return null;\n const icons = { alert: \"\\u26A0\\uFE0F\", celebration: \"\\u{1F389}\", info: \"\\u2139\\uFE0F\", warning: \"\\u{1F514}\" };\n const f = Math.max(6, size.w * 0.05);\n return /* @__PURE__ */ (0, import_jsx_runtime.jsxs)(\"div\", { style: { width: \"100%\", height: \"100%\", borderRadius: Math.max(2, size.w * 0.035), display: \"flex\", alignItems: \"center\", gap: f * 0.8, padding: `0 ${f * 1.2}px`, background: cfg.backgroundColor, color: cfg.textColor, fontFamily: \"Roboto, 'Segoe UI', Arial, sans-serif\", borderLeft: `${Math.max(2, size.w * 0.02)}px solid ${cfg.accentColor}`, boxSizing: \"border-box\", pointerEvents: \"none\", userSelect: \"none\", fontSize: `${f}px` }, children: [\n /* @__PURE__ */ (0, import_jsx_runtime.jsx)(\"span\", { style: { fontSize: \"2em\", flexShrink: 0 }, children: icons[cfg.iconType] || \"\\u26A1\" }),\n /* @__PURE__ */ (0, import_jsx_runtime.jsxs)(\"div\", { style: { flex: 1, minWidth: 0 }, children: [\n /* @__PURE__ */ (0, import_jsx_runtime.jsx)(\"div\", { style: { fontSize: \"1.3em\", fontWeight: 700, overflow: \"hidden\", textOverflow: \"ellipsis\", whiteSpace: \"nowrap\" }, children: cfg.headline }),\n /* @__PURE__ */ (0, import_jsx_runtime.jsx)(\"div\", { style: { fontSize: \"0.9em\", opacity: 0.7, overflow: \"hidden\", textOverflow: \"ellipsis\", whiteSpace: \"nowrap\" }, children: cfg.message })\n ] })\n ] });\n}\nfunction OddsBettingOverlay({ overlay, size }) {\n const cfg = parseConfig(overlay.content);\n if (!cfg) return null;\n const f = Math.max(6, size.w * 0.052);\n return /* @__PURE__ */ (0, import_jsx_runtime.jsxs)(\"div\", { style: { width: \"100%\", height: \"100%\", borderRadius: Math.max(2, size.w * 0.035), display: \"flex\", flexDirection: \"column\", padding: f * 0.8, background: cfg.backgroundColor, color: cfg.textColor, fontFamily: \"Roboto, 'Segoe UI', Arial, sans-serif\", boxSizing: \"border-box\", pointerEvents: \"none\", userSelect: \"none\", fontSize: `${f}px` }, children: [\n /* @__PURE__ */ (0, import_jsx_runtime.jsx)(\"div\", { style: { fontSize: \"0.9em\", fontWeight: 700, textTransform: \"uppercase\", letterSpacing: \"0.05em\", color: cfg.accentColor, marginBottom: f * 0.4 }, children: cfg.eventTitle }),\n /* @__PURE__ */ (0, import_jsx_runtime.jsx)(\"div\", { style: { flex: 1, display: \"flex\", flexDirection: \"column\", gap: f * 0.2, justifyContent: \"center\" }, children: (cfg.options || []).slice(0, 5).map((opt, i) => /* @__PURE__ */ (0, import_jsx_runtime.jsxs)(\"div\", { style: { display: \"flex\", justifyContent: \"space-between\", alignItems: \"center\", padding: `${f * 0.2}px ${f * 0.6}px`, borderRadius: Math.max(2, f * 0.3), background: `${cfg.accentColor}15`, fontSize: \"1em\" }, children: [\n /* @__PURE__ */ (0, import_jsx_runtime.jsx)(\"span\", { style: { overflow: \"hidden\", textOverflow: \"ellipsis\", whiteSpace: \"nowrap\", flex: 1 }, children: opt.label }),\n /* @__PURE__ */ (0, import_jsx_runtime.jsx)(\"span\", { style: { fontWeight: 700, marginLeft: f * 0.8, flexShrink: 0, color: cfg.accentColor }, children: opt.odds })\n ] }, i)) }),\n cfg.sponsorText && /* @__PURE__ */ (0, import_jsx_runtime.jsx)(\"div\", { style: { fontSize: \"0.7em\", opacity: 0.4, textAlign: \"center\", marginTop: f * 0.4 }, children: cfg.sponsorText })\n ] });\n}\nfunction BreakingNewsOverlay({ overlay, size }) {\n const cfg = parseConfig(overlay.content);\n if (!cfg) return null;\n const urgencyColors = { breaking: \"#dc2626\", urgent: \"#ea580c\", normal: \"#2563eb\" };\n const labelBg = urgencyColors[cfg.urgency] || urgencyColors.normal;\n const label = cfg.urgency === \"breaking\" ? \"BREAKING\" : cfg.urgency === \"urgent\" ? \"URGENT\" : \"NEWS\";\n const f = Math.max(6, size.w * 0.05);\n return /* @__PURE__ */ (0, import_jsx_runtime.jsxs)(\"div\", { style: { width: \"100%\", height: \"100%\", borderRadius: Math.max(2, size.w * 0.02), display: \"flex\", alignItems: \"center\", background: cfg.backgroundColor, color: cfg.textColor, fontFamily: \"Roboto, 'Segoe UI', Arial, sans-serif\", overflow: \"hidden\", pointerEvents: \"none\", userSelect: \"none\", fontSize: `${f}px` }, children: [\n /* @__PURE__ */ (0, import_jsx_runtime.jsx)(\"div\", { style: { padding: `0 ${f * 0.8}px`, height: \"100%\", display: \"flex\", alignItems: \"center\", background: labelBg, color: \"#fff\", fontSize: \"1em\", fontWeight: 900, textTransform: \"uppercase\", letterSpacing: \"0.05em\", flexShrink: 0 }, children: label }),\n /* @__PURE__ */ (0, import_jsx_runtime.jsxs)(\"div\", { style: { flex: 1, padding: `0 ${f * 1}px`, minWidth: 0 }, children: [\n /* @__PURE__ */ (0, import_jsx_runtime.jsx)(\"div\", { style: { fontSize: \"1.3em\", fontWeight: 700, overflow: \"hidden\", textOverflow: \"ellipsis\", whiteSpace: \"nowrap\" }, children: cfg.headline }),\n cfg.body && /* @__PURE__ */ (0, import_jsx_runtime.jsx)(\"div\", { style: { fontSize: \"0.9em\", opacity: 0.7, overflow: \"hidden\", textOverflow: \"ellipsis\", whiteSpace: \"nowrap\" }, children: cfg.body })\n ] })\n ] });\n}\nfunction calcCountdownRemaining(targetTime) {\n const diff = Math.max(0, new Date(targetTime).getTime() - Date.now());\n return {\n d: Math.floor(diff / 864e5),\n h: Math.floor(diff % 864e5 / 36e5),\n m: Math.floor(diff % 36e5 / 6e4),\n s: Math.floor(diff % 6e4 / 1e3),\n expired: diff === 0\n };\n}\nfunction CountdownOverlay({ overlay, size }) {\n const cfg = parseConfig(overlay.content);\n const targetTime = cfg?.targetTime ?? \"\";\n const [remaining, setRemaining] = (0, import_react.useState)(\n () => targetTime ? calcCountdownRemaining(targetTime) : { d: 0, h: 0, m: 0, s: 0, expired: false }\n );\n (0, import_react.useEffect)(() => {\n if (!targetTime) return;\n setRemaining(calcCountdownRemaining(targetTime));\n const id = setInterval(() => setRemaining(calcCountdownRemaining(targetTime)), 1e3);\n return () => clearInterval(id);\n }, [targetTime]);\n if (!cfg) return null;\n const f = Math.max(6, size.w * 0.055);\n const pad = (n) => String(n).padStart(2, \"0\");\n const units = [\n { show: cfg.showDays, value: pad(remaining.d), label: \"DAYS\" },\n { show: cfg.showHours, value: pad(remaining.h), label: \"HRS\" },\n { show: cfg.showMinutes, value: pad(remaining.m), label: \"MIN\" },\n { show: cfg.showSeconds, value: pad(remaining.s), label: \"SEC\" }\n ];\n return /* @__PURE__ */ (0, import_jsx_runtime.jsxs)(\"div\", { style: { width: \"100%\", height: \"100%\", borderRadius: Math.max(2, size.w * 0.035), display: \"flex\", flexDirection: \"column\", alignItems: \"center\", justifyContent: \"center\", padding: f * 0.8, background: cfg.backgroundColor, color: cfg.textColor, fontFamily: \"Roboto, 'Segoe UI', Arial, sans-serif\", boxSizing: \"border-box\", pointerEvents: \"none\", userSelect: \"none\", fontSize: `${f}px` }, children: [\n /* @__PURE__ */ (0, import_jsx_runtime.jsx)(\"div\", { style: { fontSize: \"0.8em\", fontWeight: 600, textTransform: \"uppercase\", letterSpacing: \"0.05em\", color: cfg.accentColor, marginBottom: f * 0.4 }, children: cfg.eventName }),\n remaining.expired ? /* @__PURE__ */ (0, import_jsx_runtime.jsx)(\"div\", { style: { fontSize: \"1em\", fontWeight: 700, opacity: 0.6 }, children: cfg.message || \"Event ended\" }) : /* @__PURE__ */ (0, import_jsx_runtime.jsx)(\"div\", { style: { display: \"flex\", gap: f * 0.6, alignItems: \"center\" }, children: units.filter((u) => u.show).map((u, i, arr) => /* @__PURE__ */ (0, import_jsx_runtime.jsxs)(import_react.default.Fragment, { children: [\n /* @__PURE__ */ (0, import_jsx_runtime.jsxs)(\"div\", { style: { textAlign: \"center\" }, children: [\n /* @__PURE__ */ (0, import_jsx_runtime.jsx)(\"div\", { style: { fontSize: \"2em\", fontWeight: 900, lineHeight: 1, borderRadius: Math.max(2, f * 0.4), padding: `${f * 0.2}px ${f * 0.4}px`, background: `${cfg.accentColor}20` }, children: u.value }),\n /* @__PURE__ */ (0, import_jsx_runtime.jsx)(\"div\", { style: { fontSize: \"0.5em\", opacity: 0.5, marginTop: f * 0.2 }, children: u.label })\n ] }),\n i < arr.length - 1 && /* @__PURE__ */ (0, import_jsx_runtime.jsx)(\"div\", { style: { fontSize: \"1.8em\", fontWeight: 700, opacity: 0.3 }, children: \":\" })\n ] }, u.label)) }),\n !remaining.expired && cfg.message && /* @__PURE__ */ (0, import_jsx_runtime.jsx)(\"div\", { style: { fontSize: \"0.8em\", opacity: 0.6, marginTop: f * 0.4, textAlign: \"center\" }, children: cfg.message })\n ] });\n}\nfunction ShapeOverlay({ overlay, size }) {\n const f = Math.max(6, size.w * 0.05);\n return /* @__PURE__ */ (0, import_jsx_runtime.jsx)(\"div\", { style: { width: \"100%\", height: \"100%\", borderRadius: Math.max(2, size.w * 0.03), background: \"rgba(99, 102, 241, 0.2)\", border: \"2px solid rgba(99, 102, 241, 0.4)\", display: \"flex\", alignItems: \"center\", justifyContent: \"center\", pointerEvents: \"none\", userSelect: \"none\" }, children: /* @__PURE__ */ (0, import_jsx_runtime.jsx)(\"div\", { style: { fontSize: `${f}px`, fontWeight: 500, color: \"rgba(163, 163, 163, 0.8)\", textTransform: \"uppercase\" }, children: overlay.name }) });\n}\nfunction hexToRgb(hex) {\n if (!hex || !hex.startsWith(\"#\")) return \"0,0,0\";\n const clean = hex.slice(1);\n const num = parseInt(clean.length === 3 ? clean.replace(/./g, \"$&$&\") : clean, 16);\n return `${num >> 16 & 255},${num >> 8 & 255},${num & 255}`;\n}\nvar FADE_DURATION_MS = 1e3;\nvar SHOWCASE_CYCLE_MS_DEFAULT = 36e3;\nvar SHOWCASE_MIN_BEAT_MS = 6e3;\nvar BEAT_POP_IN_MS = 520;\nvar BEAT_POP_OUT_MS = 520;\nvar SHOWCASE_PERSISTENT_TYPES = /* @__PURE__ */ new Set([\n \"scroller\",\n \"breaking_news\",\n \"image\",\n \"text\",\n \"shape\",\n \"countdown\",\n \"qr_code\"\n]);\nfunction easeOutCubic(t) {\n const u = 1 - t;\n return 1 - u * u * u;\n}\nfunction easeInCubic(t) {\n return t * t * t;\n}\nfunction easeOutBack(t) {\n const c1 = 1.70158;\n const c3 = c1 + 1;\n return 1 + c3 * (t - 1) ** 3 + c1 * (t - 1) ** 2;\n}\nfunction beatOpacity(phase, beatMs) {\n if (phase < BEAT_POP_IN_MS) return easeOutCubic(phase / BEAT_POP_IN_MS);\n const holdEnd = beatMs - BEAT_POP_OUT_MS;\n if (phase >= holdEnd) {\n return Math.max(\n 0,\n 1 - easeInCubic((phase - holdEnd) / BEAT_POP_OUT_MS)\n );\n }\n return 1;\n}\nfunction beatScale(phase, beatMs) {\n if (phase < BEAT_POP_IN_MS) {\n return Math.min(1, 0.92 + 0.08 * easeOutBack(phase / BEAT_POP_IN_MS));\n }\n const holdEnd = beatMs - BEAT_POP_OUT_MS;\n if (phase >= holdEnd) {\n return 1 - 0.06 * easeInCubic((phase - holdEnd) / BEAT_POP_OUT_MS);\n }\n return 1;\n}\nfunction partitionShowcase(overlays) {\n const visible = overlays.filter((o) => o.visible);\n const persistent = [];\n const spotlight = [];\n for (const o of visible) {\n if (SHOWCASE_PERSISTENT_TYPES.has(o.type)) persistent.push(o);\n else spotlight.push(o);\n }\n spotlight.sort(\n (a, b) => a.z_index - b.z_index || a.id - b.id\n );\n const beats = [];\n for (const o of spotlight) beats.push([o]);\n return { persistent, beats };\n}\nfunction useShowcaseBeat(enabled, cycleMs, beats) {\n const safeBeats = Math.max(1, beats);\n const beatMs = Math.max(\n SHOWCASE_MIN_BEAT_MS,\n Math.floor(cycleMs / safeBeats)\n );\n const totalMs = beatMs * safeBeats;\n const [now, setNow] = (0, import_react.useState)(() => Date.now());\n (0, import_react.useEffect)(() => {\n if (!enabled) return;\n const id = window.setInterval(() => setNow(Date.now()), 64);\n return () => clearInterval(id);\n }, [enabled]);\n const phase = now % totalMs;\n const beatIndex = Math.min(safeBeats - 1, Math.floor(phase / beatMs));\n const beatPhase = phase - beatIndex * beatMs;\n return { beatIndex, beatPhase, beatMs };\n}\nvar OverlayRenderer = ({\n overlays,\n currentTime,\n videoRef,\n coordinateSpace,\n showcaseMode = false,\n showcaseCycleMs = SHOWCASE_CYCLE_MS_DEFAULT\n}) => {\n const [dims, setDims] = (0, import_react.useState)(null);\n const rafRef = (0, import_react.useRef)(null);\n const [fadeMap, setFadeMap] = (0, import_react.useState)(/* @__PURE__ */ new Map());\n const removeTimers = (0, import_react.useRef)(/* @__PURE__ */ new Map());\n const { persistent: showcasePersistent, beats: showcaseBeats } = (0, import_react.useMemo)(\n () => partitionShowcase(overlays),\n [overlays]\n );\n const {\n beatIndex: showcaseBeatIndex,\n beatPhase: showcaseBeatPhase,\n beatMs: showcaseBeatMs\n } = useShowcaseBeat(showcaseMode, showcaseCycleMs, showcaseBeats.length);\n const showcasePersistentIds = (0, import_react.useMemo)(\n () => new Set(showcasePersistent.map((o) => o.id)),\n [showcasePersistent]\n );\n const showcaseSpotlightIds = (0, import_react.useMemo)(() => {\n const beat = showcaseBeats[showcaseBeatIndex];\n return new Set((beat ?? []).map((o) => o.id));\n }, [showcaseBeats, showcaseBeatIndex]);\n const updateDims = (0, import_react.useCallback)(() => {\n const video = videoRef.current;\n if (video) {\n const computed = computeVideoDimensions(video);\n setDims((prev) => {\n if (!computed || prev && prev.nativeWidth === computed.nativeWidth && prev.nativeHeight === computed.nativeHeight && prev.displayWidth === computed.displayWidth && prev.displayHeight === computed.displayHeight && prev.offsetX === computed.offsetX && prev.offsetY === computed.offsetY) {\n return prev;\n }\n return computed;\n });\n }\n }, [videoRef]);\n (0, import_react.useEffect)(() => {\n updateDims();\n const interval = setInterval(updateDims, 500);\n const handleResize = () => {\n if (rafRef.current) cancelAnimationFrame(rafRef.current);\n rafRef.current = requestAnimationFrame(updateDims);\n };\n window.addEventListener(\"resize\", handleResize);\n return () => {\n clearInterval(interval);\n window.removeEventListener(\"resize\", handleResize);\n if (rafRef.current) cancelAnimationFrame(rafRef.current);\n };\n }, [updateDims]);\n const activeOverlays = (0, import_react.useMemo)(() => {\n if (showcaseMode) {\n const beat = showcaseBeats[showcaseBeatIndex] ?? [];\n return [...showcasePersistent, ...beat];\n }\n return overlays.filter((o) => isOverlayActive(o, currentTime));\n }, [\n overlays,\n currentTime,\n showcaseMode,\n showcasePersistent,\n showcaseBeats,\n showcaseBeatIndex\n ]);\n (0, import_react.useLayoutEffect)(() => {\n const activeIds = new Set(activeOverlays.map((o) => o.id));\n setFadeMap((prev) => {\n const next = new Map(prev);\n for (const overlay of activeOverlays) {\n if (!next.has(overlay.id)) {\n next.set(overlay.id, { overlay, visible: false });\n } else {\n const existing = next.get(overlay.id);\n next.set(overlay.id, { ...existing, overlay });\n }\n }\n for (const [id, state] of next) {\n if (!activeIds.has(id) && state.visible) {\n next.set(id, { ...state, visible: false });\n if (!removeTimers.current.has(id)) {\n const timer = setTimeout(() => {\n setFadeMap((m) => {\n const updated = new Map(m);\n updated.delete(id);\n return updated;\n });\n removeTimers.current.delete(id);\n }, FADE_DURATION_MS);\n removeTimers.current.set(id, timer);\n }\n } else if (!activeIds.has(id) && !state.visible) {\n }\n }\n return next;\n });\n }, [activeOverlays]);\n (0, import_react.useEffect)(() => {\n const toFadeIn = [];\n for (const [id, state] of fadeMap) {\n if (!state.visible) {\n const isActive = activeOverlays.some((o) => o.id === id);\n if (isActive) toFadeIn.push(id);\n }\n }\n if (toFadeIn.length === 0) return;\n const raf = requestAnimationFrame(() => {\n setFadeMap((prev) => {\n const next = new Map(prev);\n for (const id of toFadeIn) {\n const state = next.get(id);\n if (state) next.set(id, { ...state, visible: true });\n }\n return next;\n });\n });\n return () => cancelAnimationFrame(raf);\n }, [fadeMap, activeOverlays]);\n (0, import_react.useEffect)(() => {\n return () => {\n for (const timer of removeTimers.current.values()) clearTimeout(timer);\n };\n }, []);\n if (!dims || fadeMap.size === 0) return null;\n return /* @__PURE__ */ (0, import_jsx_runtime.jsx)(\n \"div\",\n {\n \"aria-hidden\": \"true\",\n style: {\n position: \"absolute\",\n left: `${dims.offsetX}px`,\n top: `${dims.offsetY}px`,\n width: `${dims.displayWidth}px`,\n height: `${dims.displayHeight}px`,\n pointerEvents: \"none\",\n overflow: \"hidden\",\n zIndex: 8\n },\n children: [...fadeMap.values()].map(({ overlay, visible }) => {\n const uniform = coordinateSpace && coordinateSpace.width > 0 && coordinateSpace.height > 0;\n const auth = uniform ? { width: coordinateSpace.width, height: coordinateSpace.height } : overlayAuthoringDimensions(\n overlay,\n dims.nativeWidth,\n dims.nativeHeight\n );\n const scaleX = dims.displayWidth / auth.width;\n const scaleY = dims.displayHeight / auth.height;\n const left = overlay.x * scaleX;\n const top = overlay.y * scaleY;\n const width = overlay.width * scaleX;\n const height = overlay.height * scaleY;\n const baseOpacity = Math.max(0, Math.min(100, overlay.opacity)) / 100;\n const isShowcasePersistent = showcaseMode && showcasePersistentIds.has(overlay.id);\n const isShowcaseSpotlight = showcaseMode && showcaseSpotlightIds.has(overlay.id);\n const isShowcaseDormant = showcaseMode && !isShowcasePersistent && !isShowcaseSpotlight;\n const useShowcasePop = isShowcaseSpotlight;\n const opacity = useShowcasePop ? baseOpacity * beatOpacity(showcaseBeatPhase, showcaseBeatMs) : isShowcaseDormant ? 0 : visible ? baseOpacity : 0;\n const popScale = useShowcasePop ? beatScale(showcaseBeatPhase, showcaseBeatMs) : 1;\n const sz = { w: width, h: height };\n return /* @__PURE__ */ (0, import_jsx_runtime.jsxs)(\n \"div\",\n {\n style: {\n position: \"absolute\",\n left: `${left}px`,\n top: `${top}px`,\n width: `${width}px`,\n height: `${height}px`,\n opacity,\n transition: useShowcasePop ? \"none\" : `opacity ${FADE_DURATION_MS}ms ease`,\n transform: showcaseMode && useShowcasePop ? `scale(${popScale})` : void 0,\n transformOrigin: showcaseMode && useShowcasePop ? \"center center\" : void 0,\n zIndex: overlay.z_index,\n overflow: \"hidden\"\n },\n children: [\n overlay.type === \"image\" && /* @__PURE__ */ (0, import_jsx_runtime.jsx)(ImageOverlay, { overlay }),\n overlay.type === \"text\" && /* @__PURE__ */ (0, import_jsx_runtime.jsx)(TextOverlay, { overlay }),\n overlay.type === \"scroller\" && /* @__PURE__ */ (0, import_jsx_runtime.jsx)(ScrollerOverlay, { overlay }),\n overlay.type === \"shape\" && /* @__PURE__ */ (0, import_jsx_runtime.jsx)(ShapeOverlay, { overlay, size: sz }),\n overlay.type === \"score_bug\" && /* @__PURE__ */ (0, import_jsx_runtime.jsx)(ScoreBugOverlay, { overlay, size: sz }),\n overlay.type === \"lower_third\" && /* @__PURE__ */ (0, import_jsx_runtime.jsx)(LowerThirdOverlay, { overlay, size: sz }),\n overlay.type === \"qr_code\" && /* @__PURE__ */ (0, import_jsx_runtime.jsx)(QrCodeOverlay, { overlay, size: sz }),\n overlay.type === \"coming_up_next\" && /* @__PURE__ */ (0, import_jsx_runtime.jsx)(ComingUpNextOverlay, { overlay, size: sz }),\n overlay.type === \"contextual_trigger\" && /* @__PURE__ */ (0, import_jsx_runtime.jsx)(ContextualTriggerOverlay, { overlay, size: sz }),\n overlay.type === \"odds_betting\" && /* @__PURE__ */ (0, import_jsx_runtime.jsx)(OddsBettingOverlay, { overlay, size: sz }),\n overlay.type === \"breaking_news\" && /* @__PURE__ */ (0, import_jsx_runtime.jsx)(BreakingNewsOverlay, { overlay, size: sz }),\n overlay.type === \"countdown\" && /* @__PURE__ */ (0, import_jsx_runtime.jsx)(CountdownOverlay, { overlay, size: sz })\n ]\n },\n overlay.id\n );\n })\n }\n );\n};\n// Annotate the CommonJS export names for ESM import in node:\n0 && (module.exports = {\n OverlayRenderer\n});\n","import React, {\n useEffect,\n useLayoutEffect,\n useRef,\n useState,\n useCallback,\n useMemo,\n useId,\n} from \"react\";\nimport {\n type SwirlOverlay,\n isOverlayActive,\n resolveImageUrl,\n overlayAuthoringDimensions,\n} from \"../utils/overlays\";\n\ninterface VideoDimensions {\n nativeWidth: number;\n nativeHeight: number;\n displayWidth: number;\n displayHeight: number;\n offsetX: number;\n offsetY: number;\n scaleX: number;\n scaleY: number;\n}\n\ninterface OverlayRendererProps {\n overlays: SwirlOverlay[];\n currentTime: number;\n videoRef: React.RefObject<HTMLVideoElement | null>;\n coordinateSpace?: { width: number; height: number } | null;\n showcaseMode?: boolean;\n showcaseCycleMs?: number;\n}\n\nfunction computeVideoDimensions(\n video: HTMLVideoElement\n): VideoDimensions | null {\n const nativeWidth = video.videoWidth;\n const nativeHeight = video.videoHeight;\n if (!nativeWidth || !nativeHeight) return null;\n\n const displayWidth = video.offsetWidth;\n const displayHeight = video.offsetHeight;\n if (!displayWidth || !displayHeight) return null;\n\n const videoAspect = nativeWidth / nativeHeight;\n const displayAspect = displayWidth / displayHeight;\n\n let renderWidth: number;\n let renderHeight: number;\n let offsetX: number;\n let offsetY: number;\n\n if (videoAspect > displayAspect) {\n renderWidth = displayWidth;\n renderHeight = displayWidth / videoAspect;\n offsetX = 0;\n offsetY = (displayHeight - renderHeight) / 2;\n } else {\n renderHeight = displayHeight;\n renderWidth = displayHeight * videoAspect;\n offsetX = (displayWidth - renderWidth) / 2;\n offsetY = 0;\n }\n\n return {\n nativeWidth,\n nativeHeight,\n displayWidth: renderWidth,\n displayHeight: renderHeight,\n offsetX,\n offsetY,\n scaleX: renderWidth / nativeWidth,\n scaleY: renderHeight / nativeHeight,\n };\n}\n\nfunction ImageOverlay({ overlay }: { overlay: SwirlOverlay }) {\n const src = resolveImageUrl(overlay.image_url || \"\");\n if (!src) return null;\n return (\n <img\n src={src}\n alt={overlay.name}\n draggable={false}\n style={{\n width: \"100%\",\n height: \"100%\",\n objectFit: \"contain\",\n display: \"block\",\n pointerEvents: \"none\",\n userSelect: \"none\",\n }}\n />\n );\n}\n\nfunction TextOverlay({ overlay }: { overlay: SwirlOverlay }) {\n const text = overlay.content || \"\";\n return (\n <div\n style={{\n width: \"100%\",\n height: \"100%\",\n display: \"flex\",\n alignItems: \"center\",\n justifyContent: \"center\",\n color: \"#ffffff\",\n fontSize: \"clamp(10px, 1.4vw, 20px)\",\n fontFamily: \"Roboto, 'Segoe UI', Arial, sans-serif\",\n fontWeight: 600,\n textAlign: \"center\",\n padding: \"4px 8px\",\n boxSizing: \"border-box\",\n wordBreak: \"break-word\",\n textShadow: \"0 1px 4px rgba(0,0,0,0.7)\",\n pointerEvents: \"none\",\n userSelect: \"none\",\n lineHeight: 1.3,\n }}\n >\n {text}\n </div>\n );\n}\n\ninterface RSSItem {\n title: string;\n description: string;\n pubDate: string;\n author: string;\n category: string;\n}\n\nfunction parseRSSXml(xmlText: string, maxItems: number): RSSItem[] {\n const stripped = xmlText.replace(/^<\\?xml[^?]*\\?>\\s*/, \"\");\n const parser = new DOMParser();\n const doc = parser.parseFromString(stripped, \"text/xml\");\n const items = Array.from(doc.querySelectorAll(\"item\"))\n .map((item) => ({\n title: (item.querySelector(\"title\")?.textContent || \"\").replace(/<[^>]*>/g, \"\").trim(),\n description: (item.querySelector(\"description\")?.textContent || \"\").replace(/<[^>]*>/g, \"\").trim(),\n pubDate: item.querySelector(\"pubDate\")?.textContent || \"\",\n author: item.querySelector(\"author, dc\\\\:creator\")?.textContent || \"\",\n category: item.querySelector(\"category\")?.textContent || \"\",\n }))\n .filter((i) => i.title)\n .slice(0, maxItems);\n if (items.length === 0 && doc.querySelector(\"parsererror\")) {\n throw new Error(\"Invalid RSS XML\");\n }\n return items;\n}\n\nconst RSS_CACHE_TTL_MS = 60_000;\nconst rssCache = new Map<string, { promise: Promise<RSSItem[]>; expiresAt: number }>();\n\nfunction cachedFetchRSSItems(rssUrl: string, maxItems: number): Promise<RSSItem[]> {\n const now = Date.now();\n const cached = rssCache.get(rssUrl);\n if (cached && cached.expiresAt > now) return cached.promise;\n const promise = fetchRSSItems(rssUrl, maxItems).catch((err) => {\n rssCache.delete(rssUrl);\n throw err;\n });\n rssCache.set(rssUrl, { promise, expiresAt: now + RSS_CACHE_TTL_MS });\n return promise;\n}\n\nasync function fetchRSSItems(rssUrl: string, maxItems: number): Promise<RSSItem[]> {\n const encoded = encodeURIComponent(rssUrl);\n\n try {\n const origin = typeof window !== \"undefined\" ? window.location.origin : \"\";\n const resp = await fetch(`${origin}/api/rss-proxy?url=${encoded}`);\n if (resp.ok) {\n const text = await resp.text();\n if (text.includes(\"<item\")) return parseRSSXml(text, maxItems);\n }\n } catch { /* fall through */ }\n\n try {\n const resp = await fetch(`https://api.allorigins.win/get?url=${encoded}`);\n if (resp.ok) {\n const data = await resp.json();\n if (data.contents) return parseRSSXml(data.contents, maxItems);\n }\n } catch { /* fall through */ }\n\n try {\n const resp = await fetch(`https://corsproxy.io/?url=${encoded}`);\n if (resp.ok) {\n const text = await resp.text();\n if (text) return parseRSSXml(text, maxItems);\n }\n } catch { /* fall through */ }\n\n throw new Error(\"All RSS proxies failed\");\n}\n\nfunction ScrollerOverlay({ overlay }: { overlay: SwirlOverlay }) {\n const cfg = overlay.scroller_config;\n const uid = useId().replace(/:/g, \"\");\n\n const [rssItems, setRssItems] = useState<RSSItem[]>([]);\n const [rssLoading, setRssLoading] = useState(true);\n const [rssError, setRssError] = useState(false);\n\n const rssUrl = cfg?.rss_url || \"\";\n const maxItems = cfg?.max_items ?? 10;\n const autoRefresh = cfg?.auto_refresh !== false;\n const updateInterval = cfg?.update_interval ?? 5;\n\n useEffect(() => {\n if (!rssUrl || (cfg?.use_custom_text && cfg?.custom_text)) {\n setRssLoading(false);\n return;\n }\n let cancelled = false;\n setRssLoading(true);\n setRssError(false);\n cachedFetchRSSItems(rssUrl, maxItems)\n .then((items) => { if (!cancelled) { setRssItems(items); setRssError(false); } })\n .catch(() => { if (!cancelled) setRssError(true); })\n .finally(() => { if (!cancelled) setRssLoading(false); });\n return () => { cancelled = true; };\n }, [rssUrl, maxItems, cfg?.use_custom_text, cfg?.custom_text]);\n\n useEffect(() => {\n if (!rssUrl || !autoRefresh || (cfg?.use_custom_text && cfg?.custom_text)) return;\n const interval = setInterval(() => {\n rssCache.delete(rssUrl);\n cachedFetchRSSItems(rssUrl, maxItems)\n .then((items) => { setRssItems(items); setRssError(false); })\n .catch(() => { /* keep showing last good items */ });\n }, updateInterval * 60 * 1000);\n return () => clearInterval(interval);\n }, [rssUrl, autoRefresh, updateInterval, maxItems, cfg?.use_custom_text, cfg?.custom_text]);\n\n const sep = cfg?.separator_char ?? \"◆\";\n\n let segments: string[];\n if (cfg?.use_custom_text && cfg?.custom_text) {\n segments = [cfg.custom_text];\n } else if (rssItems.length > 0) {\n segments = rssItems.map((item) => {\n const parts: string[] = [];\n if (cfg?.show_title !== false && item.title) parts.push(item.title);\n if (cfg?.show_description && item.description) parts.push(item.description);\n if (cfg?.show_timestamp && item.pubDate) {\n try { parts.push(new Date(item.pubDate).toLocaleDateString()); } catch { /* ignore */ }\n }\n if (cfg?.show_author && item.author) parts.push(`— ${item.author}`);\n if (cfg?.show_category && item.category) parts.push(`[${item.category}]`);\n return parts.join(\" \");\n });\n } else if (rssLoading) {\n segments = [\"Loading feed…\"];\n } else if (rssError) {\n segments = overlay.content ? [overlay.content] : [\"RSS feed unavailable\"];\n } else if (overlay.content) {\n segments = [overlay.content];\n } else {\n segments = rssUrl ? [\"Loading feed…\"] : [\"RSS Ticker\"];\n }\n\n const scrollSpeed = cfg?.scroll_speed ?? 40;\n const direction = cfg?.direction ?? \"left\";\n const fontSize = cfg?.font_size ?? 15;\n const fontFamily = cfg?.font_family || \"Roboto, 'Segoe UI', Arial, sans-serif\";\n const fontWeight = cfg?.font_weight || \"700\";\n const textColor = cfg?.text_color || \"#ffffff\";\n const bgColor = cfg?.background_color || \"#0d0d1a\";\n const bgOpacity = cfg?.background_opacity !== undefined ? cfg.background_opacity / 100 : 0.95;\n const borderRadius = cfg?.border_radius ?? 0;\n const itemSpacing = cfg?.item_spacing ?? 60;\n const pad = cfg?.padding !== undefined && cfg.padding >= 0 ? cfg.padding : 8;\n\n const label = cfg?.label ?? \"NEWS\";\n const labelLine2 = cfg?.label_line2 ?? \"\";\n const labelColor = cfg?.label_color ?? \"#f97316\";\n const labelTextColor = cfg?.label_text_color ?? \"#ffffff\";\n const accentColor = cfg?.accent_color ?? labelColor;\n const showAccentLine = cfg?.show_accent_line !== false;\n const isEquitiesStrip = cfg?.preset === \"equities_strip\";\n\n const isHorizontal = direction === \"left\" || direction === \"right\";\n const isReverse = direction === \"right\" || direction === \"down\";\n\n const fullText = segments.join(` ${sep} `);\n const durationSec = Math.max(6, (fullText.length * 9) / scrollSpeed);\n\n const animId = `sc-ticker-${overlay.id}-${uid}`;\n const keyframes = isHorizontal\n ? `@keyframes ${animId} {\n ${isReverse\n ? \"0% { transform: translateX(-50%); } 100% { transform: translateX(0%); }\"\n : \"0% { transform: translateX(0); } 100% { transform: translateX(-50%); }\"}\n }`\n : `@keyframes ${animId} {\n ${isReverse\n ? \"0% { transform: translateY(-50%); } 100% { transform: translateY(0%); }\"\n : \"0% { transform: translateY(0); } 100% { transform: translateY(-50%); }\"}\n }`;\n\n return (\n <>\n <style>{keyframes}</style>\n <div\n style={{\n width: \"100%\",\n height: \"100%\",\n display: \"flex\",\n flexDirection: \"column\",\n overflow: \"hidden\",\n borderRadius: borderRadius > 0 ? `${borderRadius}px` : undefined,\n backgroundColor: `rgba(${hexToRgb(bgColor)}, ${bgOpacity})`,\n fontFamily,\n fontSize: `${fontSize}px`,\n fontWeight,\n color: textColor,\n pointerEvents: \"none\",\n userSelect: \"none\",\n }}\n >\n {showAccentLine && (\n <div\n style={{\n height: isEquitiesStrip ? 2 : 3,\n background: accentColor,\n flexShrink: 0,\n width: \"100%\",\n }}\n />\n )}\n\n <div\n style={{\n display: \"flex\",\n flex: 1,\n overflow: \"hidden\",\n minHeight: 0,\n alignItems: \"center\",\n padding: isEquitiesStrip\n ? `${Math.max(2, pad * 0.5)}px ${pad}px`\n : `${Math.max(4, pad * 0.75)}px ${pad}px`,\n gap: isEquitiesStrip ? Math.max(4, Math.round(pad * 0.75)) : Math.max(6, pad),\n boxSizing: \"border-box\",\n }}\n >\n {label && (\n <div\n style={{\n background: labelColor,\n color: labelTextColor,\n padding: isEquitiesStrip ? \"6px 12px\" : \"10px 14px\",\n display: \"flex\",\n flexDirection: isEquitiesStrip ? \"row\" : \"column\",\n alignItems: \"center\",\n justifyContent: \"center\",\n flexShrink: 0,\n textAlign: \"center\",\n gap: isEquitiesStrip ? 0 : 2,\n borderRadius: Math.max(2, borderRadius > 0 ? borderRadius : 6),\n boxShadow: isEquitiesStrip\n ? \"inset 0 1px 0 rgba(255,255,255,0.08)\"\n : \"0 2px 8px rgba(0,0,0,0.35)\",\n alignSelf: \"stretch\",\n maxWidth: isEquitiesStrip ? \"min(200px, 28%)\" : undefined,\n }}\n >\n <span\n style={{\n fontWeight: isEquitiesStrip ? 700 : 800,\n fontSize: isEquitiesStrip ? \"0.82em\" : \"0.78em\",\n letterSpacing: isEquitiesStrip ? \"0.04em\" : \"0.08em\",\n lineHeight: 1.2,\n textTransform: isEquitiesStrip ? \"none\" : \"uppercase\",\n whiteSpace: isEquitiesStrip ? \"normal\" : \"nowrap\",\n textAlign: isEquitiesStrip ? \"left\" : \"center\",\n }}\n >\n {label}\n </span>\n {labelLine2 && !isEquitiesStrip && (\n <span\n style={{\n fontWeight: 600,\n fontSize: \"0.58em\",\n letterSpacing: \"0.06em\",\n lineHeight: 1.15,\n opacity: 0.92,\n whiteSpace: \"nowrap\",\n }}\n >\n {labelLine2}\n </span>\n )}\n </div>\n )}\n\n {label && (\n <div\n style={{\n width: isEquitiesStrip ? 1 : 2,\n alignSelf: \"stretch\",\n minHeight: isEquitiesStrip ? 20 : 24,\n background: accentColor,\n flexShrink: 0,\n borderRadius: 1,\n opacity: 0.85,\n }}\n />\n )}\n\n <div\n style={{\n flex: 1,\n overflow: \"hidden\",\n position: \"relative\",\n display: \"flex\",\n alignItems: \"center\",\n minWidth: 0,\n }}\n >\n {isHorizontal ? (\n <div\n style={{\n display: \"inline-flex\",\n whiteSpace: \"nowrap\",\n animation: `${animId} ${durationSec}s linear infinite`,\n willChange: \"transform\",\n }}\n >\n {[0, 1].map((copy) => (\n <span key={copy} style={{ paddingRight: `${itemSpacing}px` }}>\n {segments.map((seg, i) => (\n <React.Fragment key={i}>\n {i > 0 && (\n <span style={{ opacity: 0.5, margin: \"0 8px\" }}>{sep}</span>\n )}\n <span style={{ textShadow: \"0 1px 3px rgba(0,0,0,0.6)\" }}>{seg}</span>\n </React.Fragment>\n ))}\n </span>\n ))}\n </div>\n ) : (\n <div\n style={{\n display: \"flex\",\n flexDirection: \"column\",\n whiteSpace: \"nowrap\",\n animation: `${animId} ${durationSec}s linear infinite`,\n willChange: \"transform\",\n }}\n >\n {[0, 1].map((copy) =>\n segments.map((seg, i) => (\n <div key={`${copy}-${i}`} style={{ paddingBottom: `${itemSpacing / 4}px` }}>\n {seg}\n </div>\n ))\n )}\n </div>\n )}\n </div>\n </div>\n </div>\n </>\n );\n}\n\nfunction parseConfig<T>(content?: string): T | null {\n if (!content) return null;\n try { return JSON.parse(content) as T; } catch { return null; }\n}\n\ninterface OverlaySize { w: number; h: number; }\n\ninterface ScoreBugCfg { homeTeam: string; awayTeam: string; homeScore: number; awayScore: number; period: string; clock: string; sponsorText: string; sponsorImageUrl: string; backgroundColor: string; textColor: string; accentColor: string; }\ninterface LowerThirdCfg { headline: string; subtitle: string; sponsorText: string; sponsorImageUrl: string; backgroundColor: string; textColor: string; accentColor: string; style: string; }\ninterface QrCodeCfg { url: string; ctaText: string; description: string; size: number; backgroundColor: string; textColor: string; accentColor: string; }\ninterface ComingUpNextCfg { title: string; subtitle: string; scheduledTime: string; thumbnailUrl: string; backgroundColor: string; textColor: string; accentColor: string; }\ninterface ContextualTriggerCfg { triggerType: string; headline: string; message: string; iconType: string; backgroundColor: string; textColor: string; accentColor: string; animationStyle: string; }\ninterface OddsBettingCfg { eventTitle: string; options: Array<{ label: string; odds: string }>; sponsorText: string; backgroundColor: string; textColor: string; accentColor: string; oddsFormat: string; }\ninterface BreakingNewsCfg { headline: string; body: string; urgency: string; backgroundColor: string; textColor: string; accentColor: string; }\ninterface CountdownCfg { eventName: string; targetTime: string; message: string; showDays: boolean; showHours: boolean; showMinutes: boolean; showSeconds: boolean; backgroundColor: string; textColor: string; accentColor: string; }\n\nfunction ScoreBugOverlay({ overlay, size }: { overlay: SwirlOverlay; size: OverlaySize }) {\n const cfg = parseConfig<ScoreBugCfg>(overlay.content);\n if (!cfg) return null;\n const f = Math.max(6, size.w * 0.052);\n return (\n <div\n style={{\n width: \"100%\",\n height: \"100%\",\n borderRadius: Math.max(2, size.w * 0.028),\n display: \"flex\",\n flexDirection: \"column\",\n background: cfg.backgroundColor,\n color: cfg.textColor,\n fontFamily: \"'Inter', 'Roboto', 'Segoe UI', system-ui, sans-serif\",\n overflow: \"hidden\",\n pointerEvents: \"none\",\n userSelect: \"none\",\n fontSize: `${f}px`,\n boxShadow: \"0 8px 28px rgba(0,0,0,0.45)\",\n }}\n >\n <div\n style={{\n flex: 1,\n display: \"flex\",\n alignItems: \"center\",\n padding: `${f * 0.42}px ${f * 0.85}px`,\n gap: f * 0.45,\n minHeight: 0,\n }}\n >\n <div style={{ flex: 1, textAlign: \"center\", minWidth: 0, padding: `0 ${f * 0.15}px` }}>\n <div\n style={{\n fontSize: \"1.05em\",\n fontWeight: 800,\n letterSpacing: \"0.02em\",\n overflow: \"hidden\",\n textOverflow: \"ellipsis\",\n whiteSpace: \"nowrap\",\n }}\n >\n {cfg.homeTeam}\n </div>\n <div style={{ fontSize: \"1.75em\", fontWeight: 900, lineHeight: 1.05, marginTop: f * 0.08 }}>\n {cfg.homeScore}\n </div>\n </div>\n <div\n style={{\n fontSize: \"0.88em\",\n textAlign: \"center\",\n fontWeight: 600,\n opacity: 0.92,\n color: cfg.textColor,\n padding: `0 ${f * 0.5}px`,\n flexShrink: 0,\n lineHeight: 1.25,\n textTransform: \"uppercase\",\n letterSpacing: \"0.04em\",\n }}\n >\n <div>{cfg.period}</div>\n <div style={{ fontWeight: 700, opacity: 1 }}>{cfg.clock}</div>\n </div>\n <div style={{ flex: 1, textAlign: \"center\", minWidth: 0, padding: `0 ${f * 0.15}px` }}>\n <div\n style={{\n fontSize: \"1.05em\",\n fontWeight: 800,\n letterSpacing: \"0.02em\",\n overflow: \"hidden\",\n textOverflow: \"ellipsis\",\n whiteSpace: \"nowrap\",\n }}\n >\n {cfg.awayTeam}\n </div>\n <div style={{ fontSize: \"1.75em\", fontWeight: 900, lineHeight: 1.05, marginTop: f * 0.08 }}>\n {cfg.awayScore}\n </div>\n </div>\n </div>\n {(cfg.sponsorText || cfg.sponsorImageUrl) && (\n <div\n style={{\n fontSize: \"0.72em\",\n textAlign: \"center\",\n fontWeight: 600,\n opacity: 0.88,\n padding: `${f * 0.34}px ${f * 0.55}px`,\n borderTop: `1px solid ${cfg.accentColor}55`,\n display: \"flex\",\n alignItems: \"center\",\n justifyContent: \"center\",\n gap: f * 0.45,\n overflow: \"hidden\",\n letterSpacing: \"0.04em\",\n textTransform: \"uppercase\",\n background: `linear-gradient(180deg, rgba(0,0,0,0.12) 0%, rgba(0,0,0,0.22) 100%)`,\n }}\n >\n {cfg.sponsorImageUrl && (\n <img src={cfg.sponsorImageUrl} alt=\"sponsor\" style={{ height: `${f * 1.35}px`, objectFit: \"contain\", flexShrink: 0 }} />\n )}\n {cfg.sponsorText && (\n <span style={{ overflow: \"hidden\", textOverflow: \"ellipsis\", whiteSpace: \"nowrap\" }}>{cfg.sponsorText}</span>\n )}\n </div>\n )}\n </div>\n );\n}\n\nfunction LowerThirdOverlay({ overlay, size }: { overlay: SwirlOverlay; size: OverlaySize }) {\n const cfg = parseConfig<LowerThirdCfg>(overlay.content);\n if (!cfg) return null;\n const f = Math.max(6, size.w * 0.05);\n const accentH = Math.max(3, size.h * 0.038);\n return (\n <div\n style={{\n width: \"100%\",\n height: \"100%\",\n borderRadius: Math.max(2, size.w * 0.018),\n display: \"flex\",\n flexDirection: \"column\",\n justifyContent: \"flex-end\",\n background: cfg.backgroundColor,\n color: cfg.textColor,\n fontFamily: \"'Inter', 'Roboto', 'Segoe UI', system-ui, sans-serif\",\n overflow: \"hidden\",\n pointerEvents: \"none\",\n userSelect: \"none\",\n fontSize: `${f}px`,\n boxShadow: \"0 10px 32px rgba(0,0,0,0.4)\",\n }}\n >\n <div style={{ width: \"100%\", height: accentH, backgroundColor: cfg.accentColor, flexShrink: 0 }} />\n <div\n style={{\n flex: 1,\n display: \"flex\",\n flexDirection: \"column\",\n justifyContent: \"center\",\n padding: `${f * 0.62}px ${f * 1.05}px`,\n minHeight: 0,\n gap: f * 0.18,\n }}\n >\n <div\n style={{\n fontSize: \"1.28em\",\n fontWeight: 800,\n lineHeight: 1.22,\n letterSpacing: \"0.01em\",\n textShadow: \"0 2px 6px rgba(0,0,0,0.55)\",\n }}\n >\n {cfg.headline}\n </div>\n <div\n style={{\n fontSize: \"0.94em\",\n fontWeight: 500,\n opacity: 0.92,\n lineHeight: 1.35,\n color: cfg.textColor,\n }}\n >\n {cfg.subtitle}\n </div>\n </div>\n {(cfg.sponsorText || cfg.sponsorImageUrl) && (\n <div\n style={{\n fontSize: \"0.68em\",\n fontWeight: 600,\n opacity: 0.72,\n padding: `${f * 0.28}px ${f * 1.05}px ${f * 0.48}px`,\n display: \"flex\",\n alignItems: \"center\",\n gap: f * 0.4,\n overflow: \"hidden\",\n letterSpacing: \"0.06em\",\n textTransform: \"uppercase\",\n borderTop: `1px solid rgba(255,255,255,0.08)`,\n }}\n >\n {cfg.sponsorImageUrl && (\n <img src={cfg.sponsorImageUrl} alt=\"sponsor\" style={{ height: `${f * 1.25}px`, objectFit: \"contain\", flexShrink: 0 }} />\n )}\n {cfg.sponsorText && (\n <span style={{ overflow: \"hidden\", textOverflow: \"ellipsis\", whiteSpace: \"nowrap\" }}>{cfg.sponsorText}</span>\n )}\n </div>\n )}\n </div>\n );\n}\n\nfunction QrCodeOverlay({ overlay, size }: { overlay: SwirlOverlay; size: OverlaySize }) {\n const cfg = parseConfig<QrCodeCfg>(overlay.content);\n if (!cfg) return null;\n const qrSide = Math.max(32, Math.min(size.w, size.h) * 0.55);\n const qrUrl = `https://api.qrserver.com/v1/create-qr-code/?size=${Math.round(qrSide * 2)}x${Math.round(qrSide * 2)}&data=${encodeURIComponent(cfg.url || \"https://example.com\")}`;\n const f = Math.max(6, size.w * 0.06);\n return (\n <div style={{ width: \"100%\", height: \"100%\", borderRadius: Math.max(2, size.w * 0.035), display: \"flex\", flexDirection: \"column\", alignItems: \"center\", justifyContent: \"center\", gap: f * 0.4, background: cfg.backgroundColor, color: cfg.textColor, fontFamily: \"Roboto, 'Segoe UI', Arial, sans-serif\", padding: f * 0.6, boxSizing: \"border-box\", pointerEvents: \"none\", userSelect: \"none\", overflow: \"hidden\" }}>\n <div style={{ flexShrink: 0, background: \"#fff\", borderRadius: Math.max(2, qrSide * 0.06), padding: Math.max(2, qrSide * 0.06), lineHeight: 0 }}>\n <img src={qrUrl} alt=\"QR Code\" style={{ width: `${qrSide}px`, height: `${qrSide}px`, display: \"block\" }} />\n </div>\n <div style={{ fontSize: `${f * 1.1}px`, fontWeight: 700, textAlign: \"center\", color: cfg.accentColor, overflow: \"hidden\", textOverflow: \"ellipsis\", whiteSpace: \"nowrap\", width: \"100%\" }}>{cfg.ctaText}</div>\n {cfg.description && <div style={{ fontSize: `${f * 0.75}px`, opacity: 0.6, textAlign: \"center\", overflow: \"hidden\", textOverflow: \"ellipsis\", whiteSpace: \"nowrap\", width: \"100%\" }}>{cfg.description}</div>}\n </div>\n );\n}\n\nfunction ComingUpNextOverlay({ overlay, size }: { overlay: SwirlOverlay; size: OverlaySize }) {\n const cfg = parseConfig<ComingUpNextCfg>(overlay.content);\n if (!cfg) return null;\n const f = Math.max(8, Math.min(size.h * 0.13, size.w * 0.048));\n const showSubtitle = size.h >= 60;\n const showThumbnail = false;\n const thumbW = 0;\n return (\n <div style={{\n width: \"100%\", height: \"100%\",\n borderRadius: Math.max(2, size.w * 0.025),\n display: \"flex\",\n background: cfg.backgroundColor,\n color: cfg.textColor,\n fontFamily: \"'Arial', 'Helvetica Neue', Helvetica, sans-serif\",\n overflow: \"hidden\",\n pointerEvents: \"none\", userSelect: \"none\",\n fontSize: `${f}px`,\n WebkitFontSmoothing: \"antialiased\",\n } as React.CSSProperties}>\n <div style={{ width: Math.max(3, size.w * 0.018), flexShrink: 0, backgroundColor: cfg.accentColor }} />\n <div style={{\n flex: 1, display: \"flex\", flexDirection: \"column\", justifyContent: \"center\",\n padding: `${f * 0.35}px ${f * 0.75}px`,\n minWidth: 0, gap: `${f * 0.08}px`,\n }}>\n <div style={{\n fontSize: \"0.7em\", fontWeight: 700,\n textTransform: \"uppercase\", letterSpacing: \"0.09em\",\n color: cfg.accentColor, lineHeight: 1,\n }}>\n Coming Up Next\n </div>\n <div style={{\n fontSize: \"1.2em\", fontWeight: 700, lineHeight: 1.2,\n overflow: \"hidden\", textOverflow: \"ellipsis\", whiteSpace: \"nowrap\",\n }}>\n {cfg.title}\n </div>\n {showSubtitle && (\n <div style={{\n fontSize: \"0.82em\", opacity: 0.65, lineHeight: 1.1,\n overflow: \"hidden\", textOverflow: \"ellipsis\", whiteSpace: \"nowrap\",\n }}>\n {cfg.subtitle}\n </div>\n )}\n {cfg.scheduledTime && (\n <div style={{\n fontSize: \"0.9em\", fontWeight: 700,\n color: cfg.accentColor, lineHeight: 1,\n }}>\n {cfg.scheduledTime}\n </div>\n )}\n </div>\n {showThumbnail && (\n <div style={{ flexShrink: 0, width: `${thumbW}px`, overflow: \"hidden\" }}>\n <img\n src={cfg.thumbnailUrl}\n alt=\"\"\n style={{ width: \"100%\", height: \"100%\", objectFit: \"cover\", display: \"block\" }}\n />\n </div>\n )}\n </div>\n );\n}\n\nfunction ContextualTriggerOverlay({ overlay, size }: { overlay: SwirlOverlay; size: OverlaySize }) {\n const cfg = parseConfig<ContextualTriggerCfg>(overlay.content);\n if (!cfg) return null;\n const icons: Record<string, string> = { alert: \"\\u26A0\\uFE0F\", celebration: \"\\uD83C\\uDF89\", info: \"\\u2139\\uFE0F\", warning: \"\\uD83D\\uDD14\" };\n const f = Math.max(6, size.w * 0.05);\n return (\n <div style={{ width: \"100%\", height: \"100%\", borderRadius: Math.max(2, size.w * 0.035), display: \"flex\", alignItems: \"center\", gap: f * 0.8, padding: `0 ${f * 1.2}px`, background: cfg.backgroundColor, color: cfg.textColor, fontFamily: \"Roboto, 'Segoe UI', Arial, sans-serif\", borderLeft: `${Math.max(2, size.w * 0.02)}px solid ${cfg.accentColor}`, boxSizing: \"border-box\", pointerEvents: \"none\", userSelect: \"none\", fontSize: `${f}px` }}>\n <span style={{ fontSize: \"2em\", flexShrink: 0 }}>{icons[cfg.iconType] || \"\\u26A1\"}</span>\n <div style={{ flex: 1, minWidth: 0 }}>\n <div style={{ fontSize: \"1.3em\", fontWeight: 700, overflow: \"hidden\", textOverflow: \"ellipsis\", whiteSpace: \"nowrap\" }}>{cfg.headline}</div>\n <div style={{ fontSize: \"0.9em\", opacity: 0.7, overflow: \"hidden\", textOverflow: \"ellipsis\", whiteSpace: \"nowrap\" }}>{cfg.message}</div>\n </div>\n </div>\n );\n}\n\nfunction OddsBettingOverlay({ overlay, size }: { overlay: SwirlOverlay; size: OverlaySize }) {\n const cfg = parseConfig<OddsBettingCfg>(overlay.content);\n if (!cfg) return null;\n const f = Math.max(6, size.w * 0.052);\n return (\n <div style={{ width: \"100%\", height: \"100%\", borderRadius: Math.max(2, size.w * 0.035), display: \"flex\", flexDirection: \"column\", padding: f * 0.8, background: cfg.backgroundColor, color: cfg.textColor, fontFamily: \"Roboto, 'Segoe UI', Arial, sans-serif\", boxSizing: \"border-box\", pointerEvents: \"none\", userSelect: \"none\", fontSize: `${f}px` }}>\n <div style={{ fontSize: \"0.9em\", fontWeight: 700, textTransform: \"uppercase\", letterSpacing: \"0.05em\", color: cfg.accentColor, marginBottom: f * 0.4 }}>{cfg.eventTitle}</div>\n <div style={{ flex: 1, display: \"flex\", flexDirection: \"column\", gap: f * 0.2, justifyContent: \"center\" }}>\n {(cfg.options || []).slice(0, 5).map((opt, i) => (\n <div key={i} style={{ display: \"flex\", justifyContent: \"space-between\", alignItems: \"center\", padding: `${f * 0.2}px ${f * 0.6}px`, borderRadius: Math.max(2, f * 0.3), background: `${cfg.accentColor}15`, fontSize: \"1em\" }}>\n <span style={{ overflow: \"hidden\", textOverflow: \"ellipsis\", whiteSpace: \"nowrap\", flex: 1 }}>{opt.label}</span>\n <span style={{ fontWeight: 700, marginLeft: f * 0.8, flexShrink: 0, color: cfg.accentColor }}>{opt.odds}</span>\n </div>\n ))}\n </div>\n {cfg.sponsorText && <div style={{ fontSize: \"0.7em\", opacity: 0.4, textAlign: \"center\", marginTop: f * 0.4 }}>{cfg.sponsorText}</div>}\n </div>\n );\n}\n\nfunction BreakingNewsOverlay({ overlay, size }: { overlay: SwirlOverlay; size: OverlaySize }) {\n const cfg = parseConfig<BreakingNewsCfg>(overlay.content);\n if (!cfg) return null;\n const urgencyColors: Record<string, string> = { breaking: \"#dc2626\", urgent: \"#ea580c\", normal: \"#2563eb\" };\n const labelBg = urgencyColors[cfg.urgency] || urgencyColors.normal;\n const label = cfg.urgency === \"breaking\" ? \"BREAKING\" : cfg.urgency === \"urgent\" ? \"URGENT\" : \"NEWS\";\n const f = Math.max(6, size.w * 0.05);\n return (\n <div style={{ width: \"100%\", height: \"100%\", borderRadius: Math.max(2, size.w * 0.02), display: \"flex\", alignItems: \"center\", background: cfg.backgroundColor, color: cfg.textColor, fontFamily: \"Roboto, 'Segoe UI', Arial, sans-serif\", overflow: \"hidden\", pointerEvents: \"none\", userSelect: \"none\", fontSize: `${f}px` }}>\n <div style={{ padding: `0 ${f * 0.8}px`, height: \"100%\", display: \"flex\", alignItems: \"center\", background: labelBg, color: \"#fff\", fontSize: \"1em\", fontWeight: 900, textTransform: \"uppercase\", letterSpacing: \"0.05em\", flexShrink: 0 }}>{label}</div>\n <div style={{ flex: 1, padding: `0 ${f * 1.0}px`, minWidth: 0 }}>\n <div style={{ fontSize: \"1.3em\", fontWeight: 700, overflow: \"hidden\", textOverflow: \"ellipsis\", whiteSpace: \"nowrap\" }}>{cfg.headline}</div>\n {cfg.body && <div style={{ fontSize: \"0.9em\", opacity: 0.7, overflow: \"hidden\", textOverflow: \"ellipsis\", whiteSpace: \"nowrap\" }}>{cfg.body}</div>}\n </div>\n </div>\n );\n}\n\nfunction calcCountdownRemaining(targetTime: string) {\n const diff = Math.max(0, new Date(targetTime).getTime() - Date.now());\n return {\n d: Math.floor(diff / 86400000),\n h: Math.floor((diff % 86400000) / 3600000),\n m: Math.floor((diff % 3600000) / 60000),\n s: Math.floor((diff % 60000) / 1000),\n expired: diff === 0,\n };\n}\n\nfunction CountdownOverlay({ overlay, size }: { overlay: SwirlOverlay; size: OverlaySize }) {\n const cfg = parseConfig<CountdownCfg>(overlay.content);\n const targetTime = cfg?.targetTime ?? \"\";\n const [remaining, setRemaining] = useState(() =>\n targetTime ? calcCountdownRemaining(targetTime) : { d: 0, h: 0, m: 0, s: 0, expired: false }\n );\n\n useEffect(() => {\n if (!targetTime) return;\n setRemaining(calcCountdownRemaining(targetTime));\n const id = setInterval(() => setRemaining(calcCountdownRemaining(targetTime)), 1000);\n return () => clearInterval(id);\n }, [targetTime]);\n\n if (!cfg) return null;\n\n const f = Math.max(6, size.w * 0.055);\n const pad = (n: number) => String(n).padStart(2, \"0\");\n const units: Array<{ show: boolean; value: string; label: string }> = [\n { show: cfg.showDays, value: pad(remaining.d), label: \"DAYS\" },\n { show: cfg.showHours, value: pad(remaining.h), label: \"HRS\" },\n { show: cfg.showMinutes, value: pad(remaining.m), label: \"MIN\" },\n { show: cfg.showSeconds, value: pad(remaining.s), label: \"SEC\" },\n ];\n\n return (\n <div style={{ width: \"100%\", height: \"100%\", borderRadius: Math.max(2, size.w * 0.035), display: \"flex\", flexDirection: \"column\", alignItems: \"center\", justifyContent: \"center\", padding: f * 0.8, background: cfg.backgroundColor, color: cfg.textColor, fontFamily: \"Roboto, 'Segoe UI', Arial, sans-serif\", boxSizing: \"border-box\", pointerEvents: \"none\", userSelect: \"none\", fontSize: `${f}px` }}>\n <div style={{ fontSize: \"0.8em\", fontWeight: 600, textTransform: \"uppercase\", letterSpacing: \"0.05em\", color: cfg.accentColor, marginBottom: f * 0.4 }}>{cfg.eventName}</div>\n {remaining.expired ? (\n <div style={{ fontSize: \"1em\", fontWeight: 700, opacity: 0.6 }}>{cfg.message || \"Event ended\"}</div>\n ) : (\n <div style={{ display: \"flex\", gap: f * 0.6, alignItems: \"center\" }}>\n {units.filter(u => u.show).map((u, i, arr) => (\n <React.Fragment key={u.label}>\n <div style={{ textAlign: \"center\" }}>\n <div style={{ fontSize: \"2em\", fontWeight: 900, lineHeight: 1, borderRadius: Math.max(2, f * 0.4), padding: `${f * 0.2}px ${f * 0.4}px`, background: `${cfg.accentColor}20` }}>{u.value}</div>\n <div style={{ fontSize: \"0.5em\", opacity: 0.5, marginTop: f * 0.2 }}>{u.label}</div>\n </div>\n {i < arr.length - 1 && <div style={{ fontSize: \"1.8em\", fontWeight: 700, opacity: 0.3 }}>:</div>}\n </React.Fragment>\n ))}\n </div>\n )}\n {!remaining.expired && cfg.message && <div style={{ fontSize: \"0.8em\", opacity: 0.6, marginTop: f * 0.4, textAlign: \"center\" }}>{cfg.message}</div>}\n </div>\n );\n}\n\nfunction ShapeOverlay({ overlay, size }: { overlay: SwirlOverlay; size: OverlaySize }) {\n const f = Math.max(6, size.w * 0.05);\n return (\n <div style={{ width: \"100%\", height: \"100%\", borderRadius: Math.max(2, size.w * 0.03), background: \"rgba(99, 102, 241, 0.2)\", border: \"2px solid rgba(99, 102, 241, 0.4)\", display: \"flex\", alignItems: \"center\", justifyContent: \"center\", pointerEvents: \"none\", userSelect: \"none\" }}>\n <div style={{ fontSize: `${f}px`, fontWeight: 500, color: \"rgba(163, 163, 163, 0.8)\", textTransform: \"uppercase\" }}>{overlay.name}</div>\n </div>\n );\n}\n\nfunction hexToRgb(hex: string): string {\n if (!hex || !hex.startsWith(\"#\")) return \"0,0,0\";\n const clean = hex.slice(1);\n const num = parseInt(clean.length === 3 ? clean.replace(/./g, \"$&$&\") : clean, 16);\n return `${(num >> 16) & 255},${(num >> 8) & 255},${num & 255}`;\n}\n\ninterface OverlayFadeState {\n overlay: SwirlOverlay;\n visible: boolean;\n}\n\nconst FADE_DURATION_MS = 1000;\n\nconst SHOWCASE_CYCLE_MS_DEFAULT = 36_000;\nconst SHOWCASE_MIN_BEAT_MS = 6_000;\nconst BEAT_POP_IN_MS = 520;\nconst BEAT_POP_OUT_MS = 520;\n\nconst SHOWCASE_PERSISTENT_TYPES: ReadonlySet<string> = new Set([\n \"scroller\",\n \"breaking_news\",\n \"image\",\n \"text\",\n \"shape\",\n \"countdown\",\n \"qr_code\",\n]);\n\nfunction easeOutCubic(t: number): number {\n const u = 1 - t;\n return 1 - u * u * u;\n}\n\nfunction easeInCubic(t: number): number {\n return t * t * t;\n}\n\nfunction easeOutBack(t: number): number {\n const c1 = 1.70158;\n const c3 = c1 + 1;\n return 1 + c3 * (t - 1) ** 3 + c1 * (t - 1) ** 2;\n}\n\nfunction beatOpacity(phase: number, beatMs: number): number {\n if (phase < BEAT_POP_IN_MS) return easeOutCubic(phase / BEAT_POP_IN_MS);\n const holdEnd = beatMs - BEAT_POP_OUT_MS;\n if (phase >= holdEnd) {\n return Math.max(\n 0,\n 1 - easeInCubic((phase - holdEnd) / BEAT_POP_OUT_MS)\n );\n }\n return 1;\n}\n\nfunction beatScale(phase: number, beatMs: number): number {\n if (phase < BEAT_POP_IN_MS) {\n return Math.min(1, 0.92 + 0.08 * easeOutBack(phase / BEAT_POP_IN_MS));\n }\n const holdEnd = beatMs - BEAT_POP_OUT_MS;\n if (phase >= holdEnd) {\n return 1 - 0.06 * easeInCubic((phase - holdEnd) / BEAT_POP_OUT_MS);\n }\n return 1;\n}\n\nfunction partitionShowcase(overlays: SwirlOverlay[]): {\n persistent: SwirlOverlay[];\n beats: SwirlOverlay[][];\n} {\n const visible = overlays.filter((o) => o.visible);\n const persistent: SwirlOverlay[] = [];\n const spotlight: SwirlOverlay[] = [];\n for (const o of visible) {\n if (SHOWCASE_PERSISTENT_TYPES.has(o.type)) persistent.push(o);\n else spotlight.push(o);\n }\n spotlight.sort(\n (a, b) => (a.z_index - b.z_index) || (a.id - b.id)\n );\n const beats: SwirlOverlay[][] = [];\n for (const o of spotlight) beats.push([o]);\n return { persistent, beats };\n}\n\nfunction useShowcaseBeat(\n enabled: boolean,\n cycleMs: number,\n beats: number\n): { beatIndex: number; beatPhase: number; beatMs: number } {\n const safeBeats = Math.max(1, beats);\n const beatMs = Math.max(\n SHOWCASE_MIN_BEAT_MS,\n Math.floor(cycleMs / safeBeats)\n );\n const totalMs = beatMs * safeBeats;\n const [now, setNow] = useState<number>(() => Date.now());\n useEffect(() => {\n if (!enabled) return;\n const id = window.setInterval(() => setNow(Date.now()), 64);\n return () => clearInterval(id);\n }, [enabled]);\n const phase = now % totalMs;\n const beatIndex = Math.min(safeBeats - 1, Math.floor(phase / beatMs));\n const beatPhase = phase - beatIndex * beatMs;\n return { beatIndex, beatPhase, beatMs };\n}\n\nexport const OverlayRenderer: React.FC<OverlayRendererProps> = ({\n overlays,\n currentTime,\n videoRef,\n coordinateSpace,\n showcaseMode = false,\n showcaseCycleMs = SHOWCASE_CYCLE_MS_DEFAULT,\n}) => {\n const [dims, setDims] = useState<VideoDimensions | null>(null);\n const rafRef = useRef<number | null>(null);\n const [fadeMap, setFadeMap] = useState<Map<number, OverlayFadeState>>(new Map());\n const removeTimers = useRef<Map<number, ReturnType<typeof setTimeout>>>(new Map());\n\n const { persistent: showcasePersistent, beats: showcaseBeats } = useMemo(\n () => partitionShowcase(overlays),\n [overlays]\n );\n const {\n beatIndex: showcaseBeatIndex,\n beatPhase: showcaseBeatPhase,\n beatMs: showcaseBeatMs,\n } = useShowcaseBeat(showcaseMode, showcaseCycleMs, showcaseBeats.length);\n\n const showcasePersistentIds = useMemo(\n () => new Set(showcasePersistent.map((o) => o.id)),\n [showcasePersistent]\n );\n const showcaseSpotlightIds = useMemo(() => {\n const beat = showcaseBeats[showcaseBeatIndex];\n return new Set((beat ?? []).map((o) => o.id));\n }, [showcaseBeats, showcaseBeatIndex]);\n\n const updateDims = useCallback(() => {\n const video = videoRef.current;\n if (video) {\n const computed = computeVideoDimensions(video);\n setDims((prev) => {\n if (\n !computed ||\n (prev &&\n prev.nativeWidth === computed.nativeWidth &&\n prev.nativeHeight === computed.nativeHeight &&\n prev.displayWidth === computed.displayWidth &&\n prev.displayHeight === computed.displayHeight &&\n prev.offsetX === computed.offsetX &&\n prev.offsetY === computed.offsetY)\n ) {\n return prev;\n }\n return computed;\n });\n }\n }, [videoRef]);\n\n useEffect(() => {\n updateDims();\n const interval = setInterval(updateDims, 500);\n\n const handleResize = () => {\n if (rafRef.current) cancelAnimationFrame(rafRef.current);\n rafRef.current = requestAnimationFrame(updateDims);\n };\n window.addEventListener(\"resize\", handleResize);\n\n return () => {\n clearInterval(interval);\n window.removeEventListener(\"resize\", handleResize);\n if (rafRef.current) cancelAnimationFrame(rafRef.current);\n };\n }, [updateDims]);\n\n const activeOverlays = useMemo(() => {\n if (showcaseMode) {\n const beat = showcaseBeats[showcaseBeatIndex] ?? [];\n return [...showcasePersistent, ...beat];\n }\n return overlays.filter((o) => isOverlayActive(o, currentTime));\n }, [\n overlays,\n currentTime,\n showcaseMode,\n showcasePersistent,\n showcaseBeats,\n showcaseBeatIndex,\n ]);\n\n useLayoutEffect(() => {\n const activeIds = new Set(activeOverlays.map((o) => o.id));\n\n setFadeMap((prev) => {\n const next = new Map(prev);\n\n for (const overlay of activeOverlays) {\n if (!next.has(overlay.id)) {\n next.set(overlay.id, { overlay, visible: false });\n } else {\n const existing = next.get(overlay.id)!;\n next.set(overlay.id, { ...existing, overlay });\n }\n }\n\n for (const [id, state] of next) {\n if (!activeIds.has(id) && state.visible) {\n next.set(id, { ...state, visible: false });\n if (!removeTimers.current.has(id)) {\n const timer = setTimeout(() => {\n setFadeMap((m) => {\n const updated = new Map(m);\n updated.delete(id);\n return updated;\n });\n removeTimers.current.delete(id);\n }, FADE_DURATION_MS);\n removeTimers.current.set(id, timer);\n }\n } else if (!activeIds.has(id) && !state.visible) {\n }\n }\n\n return next;\n });\n }, [activeOverlays]);\n\n useEffect(() => {\n const toFadeIn: number[] = [];\n for (const [id, state] of fadeMap) {\n if (!state.visible) {\n const isActive = activeOverlays.some((o) => o.id === id);\n if (isActive) toFadeIn.push(id);\n }\n }\n if (toFadeIn.length === 0) return;\n\n const raf = requestAnimationFrame(() => {\n setFadeMap((prev) => {\n const next = new Map(prev);\n for (const id of toFadeIn) {\n const state = next.get(id);\n if (state) next.set(id, { ...state, visible: true });\n }\n return next;\n });\n });\n return () => cancelAnimationFrame(raf);\n }, [fadeMap, activeOverlays]);\n\n useEffect(() => {\n return () => {\n for (const timer of removeTimers.current.values()) clearTimeout(timer);\n };\n }, []);\n\n if (!dims || fadeMap.size === 0) return null;\n\n return (\n <div\n aria-hidden=\"true\"\n style={{\n position: \"absolute\",\n left: `${dims.offsetX}px`,\n top: `${dims.offsetY}px`,\n width: `${dims.displayWidth}px`,\n height: `${dims.displayHeight}px`,\n pointerEvents: \"none\",\n overflow: \"hidden\",\n zIndex: 8,\n }}\n >\n {[...fadeMap.values()].map(({ overlay, visible }) => {\n const uniform =\n coordinateSpace &&\n coordinateSpace.width > 0 &&\n coordinateSpace.height > 0;\n const auth = uniform\n ? { width: coordinateSpace.width, height: coordinateSpace.height }\n : overlayAuthoringDimensions(\n overlay,\n dims.nativeWidth,\n dims.nativeHeight\n );\n const scaleX = dims.displayWidth / auth.width;\n const scaleY = dims.displayHeight / auth.height;\n const left = overlay.x * scaleX;\n const top = overlay.y * scaleY;\n const width = overlay.width * scaleX;\n const height = overlay.height * scaleY;\n const baseOpacity = Math.max(0, Math.min(100, overlay.opacity)) / 100;\n const isShowcasePersistent =\n showcaseMode && showcasePersistentIds.has(overlay.id);\n const isShowcaseSpotlight =\n showcaseMode && showcaseSpotlightIds.has(overlay.id);\n\n const isShowcaseDormant =\n showcaseMode && !isShowcasePersistent && !isShowcaseSpotlight;\n const useShowcasePop = isShowcaseSpotlight;\n const opacity = useShowcasePop\n ? baseOpacity *\n beatOpacity(showcaseBeatPhase, showcaseBeatMs)\n : isShowcaseDormant\n ? 0\n : visible\n ? baseOpacity\n : 0;\n const popScale = useShowcasePop\n ? beatScale(showcaseBeatPhase, showcaseBeatMs)\n : 1;\n const sz: OverlaySize = { w: width, h: height };\n\n return (\n <div\n key={overlay.id}\n style={{\n position: \"absolute\",\n left: `${left}px`,\n top: `${top}px`,\n width: `${width}px`,\n height: `${height}px`,\n opacity,\n transition: useShowcasePop\n ? \"none\"\n : `opacity ${FADE_DURATION_MS}ms ease`,\n transform: showcaseMode && useShowcasePop ? `scale(${popScale})` : undefined,\n transformOrigin: showcaseMode && useShowcasePop ? \"center center\" : undefined,\n zIndex: overlay.z_index,\n overflow: \"hidden\",\n }}\n >\n {overlay.type === \"image\" && <ImageOverlay overlay={overlay} />}\n {overlay.type === \"text\" && <TextOverlay overlay={overlay} />}\n {overlay.type === \"scroller\" && <ScrollerOverlay overlay={overlay} />}\n {overlay.type === \"shape\" && <ShapeOverlay overlay={overlay} size={sz} />}\n {overlay.type === \"score_bug\" && <ScoreBugOverlay overlay={overlay} size={sz} />}\n {overlay.type === \"lower_third\" && <LowerThirdOverlay overlay={overlay} size={sz} />}\n {overlay.type === \"qr_code\" && <QrCodeOverlay overlay={overlay} size={sz} />}\n {overlay.type === \"coming_up_next\" && <ComingUpNextOverlay overlay={overlay} size={sz} />}\n {overlay.type === \"contextual_trigger\" && <ContextualTriggerOverlay overlay={overlay} size={sz} />}\n {overlay.type === \"odds_betting\" && <OddsBettingOverlay overlay={overlay} size={sz} />}\n {overlay.type === \"breaking_news\" && <BreakingNewsOverlay overlay={overlay} size={sz} />}\n {overlay.type === \"countdown\" && <CountdownOverlay overlay={overlay} size={sz} />}\n </div>\n );\n })}\n </div>\n );\n};\n","export const OVERLAY_API_BASE = \"https://adstorm.co/api-adstorm-dev\";\n\nexport interface OverlayCoordinateSpace {\n width: number;\n height: number;\n}\n\nexport interface SwirlScrollerConfig {\n rss_url?: string;\n update_interval?: number;\n scroll_speed?: number;\n direction?: string;\n font_size?: number;\n font_family?: string;\n font_weight?: string;\n text_color?: string;\n background_color?: string;\n background_opacity?: number;\n border_color?: string;\n border_width?: number;\n border_radius?: number;\n padding?: number;\n margin?: number;\n show_title?: boolean;\n show_description?: boolean;\n show_timestamp?: boolean;\n show_author?: boolean;\n show_category?: boolean;\n max_items?: number;\n item_spacing?: number;\n fade_in_out?: boolean;\n fade_distance?: number;\n auto_refresh?: boolean;\n use_custom_text?: boolean;\n custom_text?: string;\n label?: string;\n label_line2?: string;\n label_color?: string;\n label_text_color?: string;\n accent_color?: string;\n show_accent_line?: boolean;\n separator_char?: string;\n preset?: string;\n}\n\nexport type SwirlOverlayType =\n | \"image\"\n | \"text\"\n | \"scroller\"\n | \"shape\"\n | \"score_bug\"\n | \"lower_third\"\n | \"qr_code\"\n | \"coming_up_next\"\n | \"contextual_trigger\"\n | \"odds_betting\"\n | \"breaking_news\"\n | \"countdown\";\n\nexport interface SwirlOverlay {\n id: number;\n project_id: number;\n name: string;\n type: SwirlOverlayType | string;\n visible: boolean;\n x: number;\n y: number;\n width: number;\n height: number;\n opacity: number;\n start_time: string;\n duration: string;\n content?: string;\n image_url?: string;\n scroller_config?: SwirlScrollerConfig;\n z_index: number;\n created_at?: string;\n updated_at?: string;\n}\n\nexport function timeStringToSeconds(timeStr: string): number {\n if (!timeStr) return 0;\n\n const parts = timeStr.split(\":\");\n\n if (parts.length >= 3) {\n const hours = parseInt(parts[0] ?? \"0\", 10) || 0;\n const minutes = parseInt(parts[1] ?? \"0\", 10) || 0;\n const secStr = parts[2] ?? \"0\";\n const dotIdx = secStr.indexOf(\".\");\n const seconds =\n parseInt(dotIdx >= 0 ? secStr.substring(0, dotIdx) : secStr, 10) || 0;\n const msFrag = dotIdx >= 0 ? secStr.substring(dotIdx + 1) : \"\";\n const ms = msFrag ? parseInt(msFrag.padEnd(3, \"0\").substring(0, 3), 10) || 0 : 0;\n return hours * 3600 + minutes * 60 + seconds + ms / 1000;\n }\n\n if (parts.length === 2) {\n const minutes = parseInt(parts[0] ?? \"0\", 10) || 0;\n const secStr = parts[1] ?? \"0\";\n const dotIdx = secStr.indexOf(\".\");\n const seconds =\n parseInt(dotIdx >= 0 ? secStr.substring(0, dotIdx) : secStr, 10) || 0;\n const msFrag = dotIdx >= 0 ? secStr.substring(dotIdx + 1) : \"\";\n const ms = msFrag ? parseInt(msFrag.padEnd(3, \"0\").substring(0, 3), 10) || 0 : 0;\n return minutes * 60 + seconds + ms / 1000;\n }\n\n const num = parseFloat(timeStr);\n return isFinite(num) ? Math.max(0, num) : 0;\n}\n\nexport function isOverlayActive(\n overlay: SwirlOverlay,\n currentTime: number\n): boolean {\n if (!overlay.visible) return false;\n const startSec = timeStringToSeconds(overlay.start_time);\n const durationSec = timeStringToSeconds(overlay.duration);\n if (durationSec <= 0) return false;\n return currentTime >= startSec && currentTime < startSec + durationSec;\n}\n\nexport const SWIRL_HD_AUTHORING_WIDTH = 1920;\nexport const SWIRL_HD_AUTHORING_HEIGHT = 1080;\n\nconst NAB_DEMO_NAME_PREFIX = \"NAB Demo — \";\n\nexport function overlayAuthoringDimensions(\n overlay: SwirlOverlay,\n decodeWidth: number,\n decodeHeight: number\n): { width: number; height: number } {\n if (overlay.name.startsWith(NAB_DEMO_NAME_PREFIX)) {\n return {\n width: SWIRL_HD_AUTHORING_WIDTH,\n height: SWIRL_HD_AUTHORING_HEIGHT,\n };\n }\n if (!decodeWidth || !decodeHeight) {\n return {\n width: decodeWidth || SWIRL_HD_AUTHORING_WIDTH,\n height: decodeHeight || SWIRL_HD_AUTHORING_HEIGHT,\n };\n }\n return { width: decodeWidth, height: decodeHeight };\n}\n\nfunction overlayExtents(overlays: SwirlOverlay[]): { maxR: number; maxB: number } {\n let maxR = 0;\n let maxB = 0;\n for (const o of overlays) {\n if (!o.visible) continue;\n maxR = Math.max(maxR, o.x + o.width);\n maxB = Math.max(maxB, o.y + o.height);\n }\n return { maxR, maxB };\n}\n\nexport function swirlProjectHasNabDemoMixedWithOther(overlays: SwirlOverlay[]): boolean {\n let hasNab = false;\n let hasOther = false;\n for (const o of overlays) {\n if (!o.visible) continue;\n if (o.name.startsWith(NAB_DEMO_NAME_PREFIX)) hasNab = true;\n else hasOther = true;\n }\n return hasNab && hasOther;\n}\n\nexport function inferSwirlOverlayCoordinateSpace(\n overlays: SwirlOverlay[],\n videoWidth: number,\n videoHeight: number\n): OverlayCoordinateSpace {\n if (!videoWidth || !videoHeight) {\n return {\n width: videoWidth || SWIRL_HD_AUTHORING_WIDTH,\n height: videoHeight || SWIRL_HD_AUTHORING_HEIGHT,\n };\n }\n if (!overlays.length) {\n return { width: videoWidth, height: videoHeight };\n }\n\n const { maxR, maxB } = overlayExtents(overlays);\n const EPS = 1;\n const exceedsDecode = maxR > videoWidth + EPS || maxB > videoHeight + EPS;\n const fitsHdCanvas =\n maxR <= SWIRL_HD_AUTHORING_WIDTH + EPS &&\n maxB <= SWIRL_HD_AUTHORING_HEIGHT + EPS;\n const mixed = swirlProjectHasNabDemoMixedWithOther(overlays);\n const decodeLargerThanHd =\n videoWidth > SWIRL_HD_AUTHORING_WIDTH + EPS ||\n videoHeight > SWIRL_HD_AUTHORING_HEIGHT + EPS;\n\n if (fitsHdCanvas && (decodeLargerThanHd || (exceedsDecode && !mixed))) {\n return {\n width: SWIRL_HD_AUTHORING_WIDTH,\n height: SWIRL_HD_AUTHORING_HEIGHT,\n };\n }\n\n return { width: videoWidth, height: videoHeight };\n}\n\nfunction normalizeScrollerConfig(\n raw: SwirlScrollerConfig | Record<string, unknown> | undefined\n): SwirlScrollerConfig | undefined {\n if (!raw || typeof raw !== \"object\") return undefined;\n const r = raw as Record<string, unknown>;\n const merged: SwirlScrollerConfig = { ...(raw as SwirlScrollerConfig) };\n if (merged.use_custom_text === undefined && typeof r.useCustomText === \"boolean\") {\n merged.use_custom_text = r.useCustomText;\n }\n if ((merged.custom_text === undefined || merged.custom_text === \"\") && typeof r.customText === \"string\") {\n merged.custom_text = r.customText;\n }\n if (!merged.rss_url && typeof r.rssUrl === \"string\") {\n merged.rss_url = r.rssUrl;\n }\n return merged;\n}\n\nexport function normalizeSwirlOverlay(raw: SwirlOverlay & Record<string, unknown>): SwirlOverlay {\n const o = { ...raw };\n if (o.type === \"scroller\") {\n const sc = raw.scroller_config ?? (raw as Record<string, unknown>).scrollerConfig;\n const normalized = normalizeScrollerConfig(sc as SwirlScrollerConfig);\n if (normalized) o.scroller_config = normalized;\n }\n return o;\n}\n\nexport async function fetchProjectOverlays(\n projectId: number,\n apiBaseUrl: string = OVERLAY_API_BASE\n): Promise<SwirlOverlay[]> {\n const base = apiBaseUrl.replace(/\\/$/, \"\");\n const response = await fetch(\n `${base}/adstorm/swirl/projects/${projectId}/overlays`\n );\n if (!response.ok) {\n throw new Error(\n `Failed to fetch overlays: ${response.status} ${response.statusText}`\n );\n }\n const data = await response.json();\n if (!Array.isArray(data)) return [];\n return data.map((row: SwirlOverlay & Record<string, unknown>) =>\n normalizeSwirlOverlay(row)\n );\n}\n\nexport function resolveImageUrl(\n imageUrl: string,\n apiBaseUrl: string = OVERLAY_API_BASE\n): string {\n if (!imageUrl) return \"\";\n if (imageUrl.startsWith(\"http://\") || imageUrl.startsWith(\"https://\")) {\n return imageUrl;\n }\n if (imageUrl.startsWith(\"/\")) {\n try {\n const url = new URL(apiBaseUrl);\n return `${url.origin}${imageUrl}`;\n } catch {\n return imageUrl;\n }\n }\n return `${apiBaseUrl}/${imageUrl}`;\n}\n"]}
|
|
1
|
+
{"version":3,"sources":["/home/ubuntu24-new/Dev/stormcloud-vp/lib/ui/OverlayRenderer.cjs","../../src/ui/OverlayRenderer.tsx","../../src/utils/overlays.ts"],"names":["__getOwnPropNames","Object","getOwnPropertyNames","__getProtoOf","getPrototypeOf","__hasOwnProp","hasOwnProperty","prototype","__export","target","all","name","__defProp","get","enumerable","__copyProps","to","from","except","desc","key","call","__getOwnPropDesc","__toESM","mod","isNodeMode","__create","__esModule","value","__toCommonJS","OverlayRenderer_exports","OverlayRenderer","module","exports","import_react","require","OVERLAY_API_BASE","timeStringToSeconds","timeStr","parts","split","length","hours","parseInt","minutes","secStr","dotIdx","indexOf","seconds","substring","msFrag","ms","padEnd","num","parseFloat","isFinite","Math","max","isOverlayActive","overlay","currentTime","visible","startSec","start_time","durationSec","duration","SWIRL_HD_AUTHORING_WIDTH","NAB_DEMO_NAME_PREFIX","overlayAuthoringDimensions","decodeWidth","decodeHeight","startsWith","width","height","SWIRL_HD_AUTHORING_HEIGHT","resolveImageUrl","imageUrl","apiBaseUrl","url","URL","origin","import_jsx_runtime","computeVideoDimensions","video","nativeWidth","videoWidth","nativeHeight","videoHeight","displayWidth","offsetWidth","displayHeight","offsetHeight","videoAspect","displayAspect","renderWidth","renderHeight","offsetX","offsetY","image_url","jsx","src","content","alignItems","justifyContent","color","fontSize","fontFamily","fontWeight","textAlign","boxSizing","wordBreak","textShadow","pointerEvents","userSelect","text","xmlText","maxItems","replace","DOMParser","parseFromString","stripped","doc","querySelectorAll","map","item","querySelector","description","textContent","trim","i","items","cachedFetchRSSItems","Map","rssUrl","expiresAt","now","cached","promise","fetchRSSItems","catch","err","encoded","resp","data","window","location","fetch","ok","includes","parseRSSXml","json","contents","Error","ScrollerOverlay","cfg","scroller_config","uid","useId","useState","rssLoading","setRssLoading","rssError","setRssError","rss_url","max_items","autoRefresh","auto_refresh","updateInterval","update_interval","useEffect","use_custom_text","custom_text","cancelled","then","setRssItems","finally","interval","setInterval","rssCache","delete","clearInterval","sep","segments","separator_char","rssItems","show_title","title","push","show_description","show_timestamp","pubDate","Date","toLocaleDateString","show_author","author","join","show_category","category","scrollSpeed","scroll_speed","direction","font_size","font_family","font_weight","textColor","text_color","bgColor","background_color","bgOpacity","background_opacity","borderRadius","border_radius","itemSpacing","item_spacing","pad","padding","label","labelLine2","label_line2","labelColor","label_color","labelTextColor","label_text_color","accentColor","accent_color","showAccentLine","show_accent_line","isEquitiesStrip","preset","isHorizontal","isReverse","fullText","animId","id","keyframes","jsxs","Fragment","children","style","display","flexDirection","overflow","backgroundColor","hexToRgb","background","flexShrink","flex","minHeight","gap","round","boxShadow","alignSelf","maxWidth","letterSpacing","lineHeight","textTransform","whiteSpace","opacity","position","minWidth","animation","willChange","copy","paddingRight","seg","React","margin","paddingBottom","parseConfig","JSON","parse","ScoreBugOverlay","size","f","w","textOverflow","homeTeam","marginTop","homeScore","period","clock","awayTeam","awayScore","sponsorText","sponsorImageUrl","borderTop","alt","objectFit","LowerThirdOverlay","accentH","h","headline","subtitle"],"mappings":"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","sourcesContent":["\"use strict\";\nvar __create = Object.create;\nvar __defProp = Object.defineProperty;\nvar __getOwnPropDesc = Object.getOwnPropertyDescriptor;\nvar __getOwnPropNames = Object.getOwnPropertyNames;\nvar __getProtoOf = Object.getPrototypeOf;\nvar __hasOwnProp = Object.prototype.hasOwnProperty;\nvar __export = (target, all) => {\n for (var name in all)\n __defProp(target, name, { get: all[name], enumerable: true });\n};\nvar __copyProps = (to, from, except, desc) => {\n if (from && typeof from === \"object\" || typeof from === \"function\") {\n for (let key of __getOwnPropNames(from))\n if (!__hasOwnProp.call(to, key) && key !== except)\n __defProp(to, key, { get: () => from[key], enumerable: !(desc = __getOwnPropDesc(from, key)) || desc.enumerable });\n }\n return to;\n};\nvar __toESM = (mod, isNodeMode, target) => (target = mod != null ? __create(__getProtoOf(mod)) : {}, __copyProps(\n // If the importer is in node compatibility mode or this is not an ESM\n // file that has been converted to a CommonJS file using a Babel-\n // compatible transform (i.e. \"__esModule\" has not been set), then set\n // \"default\" to the CommonJS \"module.exports\" for node compatibility.\n isNodeMode || !mod || !mod.__esModule ? __defProp(target, \"default\", { value: mod, enumerable: true }) : target,\n mod\n));\nvar __toCommonJS = (mod) => __copyProps(__defProp({}, \"__esModule\", { value: true }), mod);\n\n// src/ui/OverlayRenderer.tsx\nvar OverlayRenderer_exports = {};\n__export(OverlayRenderer_exports, {\n OverlayRenderer: () => OverlayRenderer\n});\nmodule.exports = __toCommonJS(OverlayRenderer_exports);\nvar import_react = __toESM(require(\"react\"), 1);\n\n// src/utils/overlays.ts\nvar OVERLAY_API_BASE = \"https://adstorm.co/api-adstorm-dev\";\nfunction timeStringToSeconds(timeStr) {\n if (!timeStr) return 0;\n const parts = timeStr.split(\":\");\n if (parts.length >= 3) {\n const hours = parseInt(parts[0] ?? \"0\", 10) || 0;\n const minutes = parseInt(parts[1] ?? \"0\", 10) || 0;\n const secStr = parts[2] ?? \"0\";\n const dotIdx = secStr.indexOf(\".\");\n const seconds = parseInt(dotIdx >= 0 ? secStr.substring(0, dotIdx) : secStr, 10) || 0;\n const msFrag = dotIdx >= 0 ? secStr.substring(dotIdx + 1) : \"\";\n const ms = msFrag ? parseInt(msFrag.padEnd(3, \"0\").substring(0, 3), 10) || 0 : 0;\n return hours * 3600 + minutes * 60 + seconds + ms / 1e3;\n }\n if (parts.length === 2) {\n const minutes = parseInt(parts[0] ?? \"0\", 10) || 0;\n const secStr = parts[1] ?? \"0\";\n const dotIdx = secStr.indexOf(\".\");\n const seconds = parseInt(dotIdx >= 0 ? secStr.substring(0, dotIdx) : secStr, 10) || 0;\n const msFrag = dotIdx >= 0 ? secStr.substring(dotIdx + 1) : \"\";\n const ms = msFrag ? parseInt(msFrag.padEnd(3, \"0\").substring(0, 3), 10) || 0 : 0;\n return minutes * 60 + seconds + ms / 1e3;\n }\n const num = parseFloat(timeStr);\n return isFinite(num) ? Math.max(0, num) : 0;\n}\nfunction isOverlayActive(overlay, currentTime) {\n if (!overlay.visible) return false;\n const startSec = timeStringToSeconds(overlay.start_time);\n const durationSec = timeStringToSeconds(overlay.duration);\n if (durationSec <= 0) return false;\n return currentTime >= startSec && currentTime < startSec + durationSec;\n}\nvar SWIRL_HD_AUTHORING_WIDTH = 1920;\nvar SWIRL_HD_AUTHORING_HEIGHT = 1080;\nvar NAB_DEMO_NAME_PREFIX = \"NAB Demo \\u2014 \";\nfunction overlayAuthoringDimensions(overlay, decodeWidth, decodeHeight) {\n if (overlay.name.startsWith(NAB_DEMO_NAME_PREFIX)) {\n return {\n width: SWIRL_HD_AUTHORING_WIDTH,\n height: SWIRL_HD_AUTHORING_HEIGHT\n };\n }\n if (!decodeWidth || !decodeHeight) {\n return {\n width: decodeWidth || SWIRL_HD_AUTHORING_WIDTH,\n height: decodeHeight || SWIRL_HD_AUTHORING_HEIGHT\n };\n }\n return { width: decodeWidth, height: decodeHeight };\n}\nfunction resolveImageUrl(imageUrl, apiBaseUrl = OVERLAY_API_BASE) {\n if (!imageUrl) return \"\";\n if (imageUrl.startsWith(\"http://\") || imageUrl.startsWith(\"https://\")) {\n return imageUrl;\n }\n if (imageUrl.startsWith(\"/\")) {\n try {\n const url = new URL(apiBaseUrl);\n return `${url.origin}${imageUrl}`;\n } catch {\n return imageUrl;\n }\n }\n return `${apiBaseUrl}/${imageUrl}`;\n}\n\n// src/ui/OverlayRenderer.tsx\nvar import_jsx_runtime = require(\"react/jsx-runtime\");\nfunction computeVideoDimensions(video) {\n const nativeWidth = video.videoWidth;\n const nativeHeight = video.videoHeight;\n if (!nativeWidth || !nativeHeight) return null;\n const displayWidth = video.offsetWidth;\n const displayHeight = video.offsetHeight;\n if (!displayWidth || !displayHeight) return null;\n const videoAspect = nativeWidth / nativeHeight;\n const displayAspect = displayWidth / displayHeight;\n let renderWidth;\n let renderHeight;\n let offsetX;\n let offsetY;\n if (videoAspect > displayAspect) {\n renderWidth = displayWidth;\n renderHeight = displayWidth / videoAspect;\n offsetX = 0;\n offsetY = (displayHeight - renderHeight) / 2;\n } else {\n renderHeight = displayHeight;\n renderWidth = displayHeight * videoAspect;\n offsetX = (displayWidth - renderWidth) / 2;\n offsetY = 0;\n }\n return {\n nativeWidth,\n nativeHeight,\n displayWidth: renderWidth,\n displayHeight: renderHeight,\n offsetX,\n offsetY,\n scaleX: renderWidth / nativeWidth,\n scaleY: renderHeight / nativeHeight\n };\n}\nfunction ImageOverlay({ overlay }) {\n const src = resolveImageUrl(overlay.image_url || \"\");\n if (!src) return null;\n return /* @__PURE__ */ (0, import_jsx_runtime.jsx)(\n \"img\",\n {\n src,\n alt: overlay.name,\n draggable: false,\n style: {\n width: \"100%\",\n height: \"100%\",\n objectFit: \"contain\",\n display: \"block\",\n pointerEvents: \"none\",\n userSelect: \"none\"\n }\n }\n );\n}\nfunction TextOverlay({ overlay }) {\n const text = overlay.content || \"\";\n return /* @__PURE__ */ (0, import_jsx_runtime.jsx)(\n \"div\",\n {\n style: {\n width: \"100%\",\n height: \"100%\",\n display: \"flex\",\n alignItems: \"center\",\n justifyContent: \"center\",\n color: \"#ffffff\",\n fontSize: \"clamp(10px, 1.4vw, 20px)\",\n fontFamily: \"Roboto, 'Segoe UI', Arial, sans-serif\",\n fontWeight: 600,\n textAlign: \"center\",\n padding: \"4px 8px\",\n boxSizing: \"border-box\",\n wordBreak: \"break-word\",\n textShadow: \"0 1px 4px rgba(0,0,0,0.7)\",\n pointerEvents: \"none\",\n userSelect: \"none\",\n lineHeight: 1.3\n },\n children: text\n }\n );\n}\nfunction parseRSSXml(xmlText, maxItems) {\n const stripped = xmlText.replace(/^<\\?xml[^?]*\\?>\\s*/, \"\");\n const parser = new DOMParser();\n const doc = parser.parseFromString(stripped, \"text/xml\");\n const items = Array.from(doc.querySelectorAll(\"item\")).map((item) => ({\n title: (item.querySelector(\"title\")?.textContent || \"\").replace(/<[^>]*>/g, \"\").trim(),\n description: (item.querySelector(\"description\")?.textContent || \"\").replace(/<[^>]*>/g, \"\").trim(),\n pubDate: item.querySelector(\"pubDate\")?.textContent || \"\",\n author: item.querySelector(\"author, dc\\\\:creator\")?.textContent || \"\",\n category: item.querySelector(\"category\")?.textContent || \"\"\n })).filter((i) => i.title).slice(0, maxItems);\n if (items.length === 0 && doc.querySelector(\"parsererror\")) {\n throw new Error(\"Invalid RSS XML\");\n }\n return items;\n}\nvar RSS_CACHE_TTL_MS = 6e4;\nvar rssCache = /* @__PURE__ */ new Map();\nfunction cachedFetchRSSItems(rssUrl, maxItems) {\n const now = Date.now();\n const cached = rssCache.get(rssUrl);\n if (cached && cached.expiresAt > now) return cached.promise;\n const promise = fetchRSSItems(rssUrl, maxItems).catch((err) => {\n rssCache.delete(rssUrl);\n throw err;\n });\n rssCache.set(rssUrl, { promise, expiresAt: now + RSS_CACHE_TTL_MS });\n return promise;\n}\nasync function fetchRSSItems(rssUrl, maxItems) {\n const encoded = encodeURIComponent(rssUrl);\n try {\n const origin = typeof window !== \"undefined\" ? window.location.origin : \"\";\n const resp = await fetch(`${origin}/api/rss-proxy?url=${encoded}`);\n if (resp.ok) {\n const text = await resp.text();\n if (text.includes(\"<item\")) return parseRSSXml(text, maxItems);\n }\n } catch {\n }\n try {\n const resp = await fetch(`https://api.allorigins.win/get?url=${encoded}`);\n if (resp.ok) {\n const data = await resp.json();\n if (data.contents) return parseRSSXml(data.contents, maxItems);\n }\n } catch {\n }\n try {\n const resp = await fetch(`https://corsproxy.io/?url=${encoded}`);\n if (resp.ok) {\n const text = await resp.text();\n if (text) return parseRSSXml(text, maxItems);\n }\n } catch {\n }\n throw new Error(\"All RSS proxies failed\");\n}\nfunction ScrollerOverlay({ overlay }) {\n const cfg = overlay.scroller_config;\n const uid = (0, import_react.useId)().replace(/:/g, \"\");\n const [rssItems, setRssItems] = (0, import_react.useState)([]);\n const [rssLoading, setRssLoading] = (0, import_react.useState)(true);\n const [rssError, setRssError] = (0, import_react.useState)(false);\n const rssUrl = cfg?.rss_url || \"\";\n const maxItems = cfg?.max_items ?? 10;\n const autoRefresh = cfg?.auto_refresh !== false;\n const updateInterval = cfg?.update_interval ?? 5;\n (0, import_react.useEffect)(() => {\n if (!rssUrl || cfg?.use_custom_text && cfg?.custom_text) {\n setRssLoading(false);\n return;\n }\n let cancelled = false;\n setRssLoading(true);\n setRssError(false);\n cachedFetchRSSItems(rssUrl, maxItems).then((items) => {\n if (!cancelled) {\n setRssItems(items);\n setRssError(false);\n }\n }).catch(() => {\n if (!cancelled) setRssError(true);\n }).finally(() => {\n if (!cancelled) setRssLoading(false);\n });\n return () => {\n cancelled = true;\n };\n }, [rssUrl, maxItems, cfg?.use_custom_text, cfg?.custom_text]);\n (0, import_react.useEffect)(() => {\n if (!rssUrl || !autoRefresh || cfg?.use_custom_text && cfg?.custom_text) return;\n const interval = setInterval(() => {\n rssCache.delete(rssUrl);\n cachedFetchRSSItems(rssUrl, maxItems).then((items) => {\n setRssItems(items);\n setRssError(false);\n }).catch(() => {\n });\n }, updateInterval * 60 * 1e3);\n return () => clearInterval(interval);\n }, [rssUrl, autoRefresh, updateInterval, maxItems, cfg?.use_custom_text, cfg?.custom_text]);\n const sep = cfg?.separator_char ?? \"\\u25C6\";\n let segments;\n if (cfg?.use_custom_text && cfg?.custom_text) {\n segments = [cfg.custom_text];\n } else if (rssItems.length > 0) {\n segments = rssItems.map((item) => {\n const parts = [];\n if (cfg?.show_title !== false && item.title) parts.push(item.title);\n if (cfg?.show_description && item.description) parts.push(item.description);\n if (cfg?.show_timestamp && item.pubDate) {\n try {\n parts.push(new Date(item.pubDate).toLocaleDateString());\n } catch {\n }\n }\n if (cfg?.show_author && item.author) parts.push(`\\u2014 ${item.author}`);\n if (cfg?.show_category && item.category) parts.push(`[${item.category}]`);\n return parts.join(\" \");\n });\n } else if (rssLoading) {\n segments = [\"Loading feed\\u2026\"];\n } else if (rssError) {\n segments = overlay.content ? [overlay.content] : [\"RSS feed unavailable\"];\n } else if (overlay.content) {\n segments = [overlay.content];\n } else {\n segments = rssUrl ? [\"Loading feed\\u2026\"] : [\"RSS Ticker\"];\n }\n const scrollSpeed = cfg?.scroll_speed ?? 40;\n const direction = cfg?.direction ?? \"left\";\n const fontSize = cfg?.font_size ?? 15;\n const fontFamily = cfg?.font_family || \"Roboto, 'Segoe UI', Arial, sans-serif\";\n const fontWeight = cfg?.font_weight || \"700\";\n const textColor = cfg?.text_color || \"#ffffff\";\n const bgColor = cfg?.background_color || \"#0d0d1a\";\n const bgOpacity = cfg?.background_opacity !== void 0 ? cfg.background_opacity / 100 : 0.95;\n const borderRadius = cfg?.border_radius ?? 0;\n const itemSpacing = cfg?.item_spacing ?? 60;\n const pad = cfg?.padding !== void 0 && cfg.padding >= 0 ? cfg.padding : 8;\n const label = cfg?.label ?? \"NEWS\";\n const labelLine2 = cfg?.label_line2 ?? \"\";\n const labelColor = cfg?.label_color ?? \"#f97316\";\n const labelTextColor = cfg?.label_text_color ?? \"#ffffff\";\n const accentColor = cfg?.accent_color ?? labelColor;\n const showAccentLine = cfg?.show_accent_line !== false;\n const isEquitiesStrip = cfg?.preset === \"equities_strip\";\n const isHorizontal = direction === \"left\" || direction === \"right\";\n const isReverse = direction === \"right\" || direction === \"down\";\n const fullText = segments.join(` ${sep} `);\n const durationSec = Math.max(6, fullText.length * 9 / scrollSpeed);\n const animId = `sc-ticker-${overlay.id}-${uid}`;\n const keyframes = isHorizontal ? `@keyframes ${animId} {\n ${isReverse ? \"0% { transform: translateX(-50%); } 100% { transform: translateX(0%); }\" : \"0% { transform: translateX(0); } 100% { transform: translateX(-50%); }\"}\n }` : `@keyframes ${animId} {\n ${isReverse ? \"0% { transform: translateY(-50%); } 100% { transform: translateY(0%); }\" : \"0% { transform: translateY(0); } 100% { transform: translateY(-50%); }\"}\n }`;\n return /* @__PURE__ */ (0, import_jsx_runtime.jsxs)(import_jsx_runtime.Fragment, { children: [\n /* @__PURE__ */ (0, import_jsx_runtime.jsx)(\"style\", { children: keyframes }),\n /* @__PURE__ */ (0, import_jsx_runtime.jsxs)(\n \"div\",\n {\n style: {\n width: \"100%\",\n height: \"100%\",\n display: \"flex\",\n flexDirection: \"column\",\n overflow: \"hidden\",\n borderRadius: borderRadius > 0 ? `${borderRadius}px` : void 0,\n backgroundColor: `rgba(${hexToRgb(bgColor)}, ${bgOpacity})`,\n fontFamily,\n fontSize: `${fontSize}px`,\n fontWeight,\n color: textColor,\n pointerEvents: \"none\",\n userSelect: \"none\"\n },\n children: [\n showAccentLine && /* @__PURE__ */ (0, import_jsx_runtime.jsx)(\n \"div\",\n {\n style: {\n height: isEquitiesStrip ? 2 : 3,\n background: accentColor,\n flexShrink: 0,\n width: \"100%\"\n }\n }\n ),\n /* @__PURE__ */ (0, import_jsx_runtime.jsxs)(\n \"div\",\n {\n style: {\n display: \"flex\",\n flex: 1,\n overflow: \"hidden\",\n minHeight: 0,\n alignItems: \"center\",\n padding: isEquitiesStrip ? `${Math.max(2, pad * 0.5)}px ${pad}px` : `${Math.max(4, pad * 0.75)}px ${pad}px`,\n gap: isEquitiesStrip ? Math.max(4, Math.round(pad * 0.75)) : Math.max(6, pad),\n boxSizing: \"border-box\"\n },\n children: [\n label && /* @__PURE__ */ (0, import_jsx_runtime.jsxs)(\n \"div\",\n {\n style: {\n background: labelColor,\n color: labelTextColor,\n padding: isEquitiesStrip ? \"6px 12px\" : \"10px 14px\",\n display: \"flex\",\n flexDirection: isEquitiesStrip ? \"row\" : \"column\",\n alignItems: \"center\",\n justifyContent: \"center\",\n flexShrink: 0,\n textAlign: \"center\",\n gap: isEquitiesStrip ? 0 : 2,\n borderRadius: Math.max(2, borderRadius > 0 ? borderRadius : 6),\n boxShadow: isEquitiesStrip ? \"inset 0 1px 0 rgba(255,255,255,0.08)\" : \"0 2px 8px rgba(0,0,0,0.35)\",\n alignSelf: \"stretch\",\n maxWidth: isEquitiesStrip ? \"min(200px, 28%)\" : void 0\n },\n children: [\n /* @__PURE__ */ (0, import_jsx_runtime.jsx)(\n \"span\",\n {\n style: {\n fontWeight: isEquitiesStrip ? 700 : 800,\n fontSize: isEquitiesStrip ? \"0.82em\" : \"0.78em\",\n letterSpacing: isEquitiesStrip ? \"0.04em\" : \"0.08em\",\n lineHeight: 1.2,\n textTransform: isEquitiesStrip ? \"none\" : \"uppercase\",\n whiteSpace: isEquitiesStrip ? \"normal\" : \"nowrap\",\n textAlign: isEquitiesStrip ? \"left\" : \"center\"\n },\n children: label\n }\n ),\n labelLine2 && !isEquitiesStrip && /* @__PURE__ */ (0, import_jsx_runtime.jsx)(\n \"span\",\n {\n style: {\n fontWeight: 600,\n fontSize: \"0.58em\",\n letterSpacing: \"0.06em\",\n lineHeight: 1.15,\n opacity: 0.92,\n whiteSpace: \"nowrap\"\n },\n children: labelLine2\n }\n )\n ]\n }\n ),\n label && /* @__PURE__ */ (0, import_jsx_runtime.jsx)(\n \"div\",\n {\n style: {\n width: isEquitiesStrip ? 1 : 2,\n alignSelf: \"stretch\",\n minHeight: isEquitiesStrip ? 20 : 24,\n background: accentColor,\n flexShrink: 0,\n borderRadius: 1,\n opacity: 0.85\n }\n }\n ),\n /* @__PURE__ */ (0, import_jsx_runtime.jsx)(\n \"div\",\n {\n style: {\n flex: 1,\n overflow: \"hidden\",\n position: \"relative\",\n display: \"flex\",\n alignItems: \"center\",\n minWidth: 0\n },\n children: isHorizontal ? /* @__PURE__ */ (0, import_jsx_runtime.jsx)(\n \"div\",\n {\n style: {\n display: \"inline-flex\",\n whiteSpace: \"nowrap\",\n animation: `${animId} ${durationSec}s linear infinite`,\n willChange: \"transform\"\n },\n children: [0, 1].map((copy) => /* @__PURE__ */ (0, import_jsx_runtime.jsx)(\"span\", { style: { paddingRight: `${itemSpacing}px` }, children: segments.map((seg, i) => /* @__PURE__ */ (0, import_jsx_runtime.jsxs)(import_react.default.Fragment, { children: [\n i > 0 && /* @__PURE__ */ (0, import_jsx_runtime.jsx)(\"span\", { style: { opacity: 0.5, margin: \"0 8px\" }, children: sep }),\n /* @__PURE__ */ (0, import_jsx_runtime.jsx)(\"span\", { style: { textShadow: \"0 1px 3px rgba(0,0,0,0.6)\" }, children: seg })\n ] }, i)) }, copy))\n }\n ) : /* @__PURE__ */ (0, import_jsx_runtime.jsx)(\n \"div\",\n {\n style: {\n display: \"flex\",\n flexDirection: \"column\",\n whiteSpace: \"nowrap\",\n animation: `${animId} ${durationSec}s linear infinite`,\n willChange: \"transform\"\n },\n children: [0, 1].map(\n (copy) => segments.map((seg, i) => /* @__PURE__ */ (0, import_jsx_runtime.jsx)(\"div\", { style: { paddingBottom: `${itemSpacing / 4}px` }, children: seg }, `${copy}-${i}`))\n )\n }\n )\n }\n )\n ]\n }\n )\n ]\n }\n )\n ] });\n}\nfunction parseConfig(content) {\n if (!content) return null;\n try {\n return JSON.parse(content);\n } catch {\n return null;\n }\n}\nfunction ScoreBugOverlay({ overlay, size }) {\n const cfg = parseConfig(overlay.content);\n if (!cfg) return null;\n const f = Math.max(6, size.w * 0.052);\n return /* @__PURE__ */ (0, import_jsx_runtime.jsxs)(\n \"div\",\n {\n style: {\n width: \"100%\",\n height: \"100%\",\n borderRadius: Math.max(2, size.w * 0.028),\n display: \"flex\",\n flexDirection: \"column\",\n background: cfg.backgroundColor,\n color: cfg.textColor,\n fontFamily: \"'Inter', 'Roboto', 'Segoe UI', system-ui, sans-serif\",\n overflow: \"hidden\",\n pointerEvents: \"none\",\n userSelect: \"none\",\n fontSize: `${f}px`,\n boxShadow: \"0 8px 28px rgba(0,0,0,0.45)\"\n },\n children: [\n /* @__PURE__ */ (0, import_jsx_runtime.jsxs)(\n \"div\",\n {\n style: {\n flex: 1,\n display: \"flex\",\n alignItems: \"center\",\n padding: `${f * 0.42}px ${f * 0.85}px`,\n gap: f * 0.45,\n minHeight: 0\n },\n children: [\n /* @__PURE__ */ (0, import_jsx_runtime.jsxs)(\"div\", { style: { flex: 1, textAlign: \"center\", minWidth: 0, padding: `0 ${f * 0.15}px` }, children: [\n /* @__PURE__ */ (0, import_jsx_runtime.jsx)(\n \"div\",\n {\n style: {\n fontSize: \"1.05em\",\n fontWeight: 800,\n letterSpacing: \"0.02em\",\n overflow: \"hidden\",\n textOverflow: \"ellipsis\",\n whiteSpace: \"nowrap\"\n },\n children: cfg.homeTeam\n }\n ),\n /* @__PURE__ */ (0, import_jsx_runtime.jsx)(\"div\", { style: { fontSize: \"1.75em\", fontWeight: 900, lineHeight: 1.05, marginTop: f * 0.08 }, children: cfg.homeScore })\n ] }),\n /* @__PURE__ */ (0, import_jsx_runtime.jsxs)(\n \"div\",\n {\n style: {\n fontSize: \"0.88em\",\n textAlign: \"center\",\n fontWeight: 600,\n opacity: 0.92,\n color: cfg.textColor,\n padding: `0 ${f * 0.5}px`,\n flexShrink: 0,\n lineHeight: 1.25,\n textTransform: \"uppercase\",\n letterSpacing: \"0.04em\"\n },\n children: [\n /* @__PURE__ */ (0, import_jsx_runtime.jsx)(\"div\", { children: cfg.period }),\n /* @__PURE__ */ (0, import_jsx_runtime.jsx)(\"div\", { style: { fontWeight: 700, opacity: 1 }, children: cfg.clock })\n ]\n }\n ),\n /* @__PURE__ */ (0, import_jsx_runtime.jsxs)(\"div\", { style: { flex: 1, textAlign: \"center\", minWidth: 0, padding: `0 ${f * 0.15}px` }, children: [\n /* @__PURE__ */ (0, import_jsx_runtime.jsx)(\n \"div\",\n {\n style: {\n fontSize: \"1.05em\",\n fontWeight: 800,\n letterSpacing: \"0.02em\",\n overflow: \"hidden\",\n textOverflow: \"ellipsis\",\n whiteSpace: \"nowrap\"\n },\n children: cfg.awayTeam\n }\n ),\n /* @__PURE__ */ (0, import_jsx_runtime.jsx)(\"div\", { style: { fontSize: \"1.75em\", fontWeight: 900, lineHeight: 1.05, marginTop: f * 0.08 }, children: cfg.awayScore })\n ] })\n ]\n }\n ),\n (cfg.sponsorText || cfg.sponsorImageUrl) && /* @__PURE__ */ (0, import_jsx_runtime.jsxs)(\n \"div\",\n {\n style: {\n fontSize: \"0.72em\",\n textAlign: \"center\",\n fontWeight: 600,\n opacity: 0.88,\n padding: `${f * 0.34}px ${f * 0.55}px`,\n borderTop: `1px solid ${cfg.accentColor}55`,\n display: \"flex\",\n alignItems: \"center\",\n justifyContent: \"center\",\n gap: f * 0.45,\n overflow: \"hidden\",\n letterSpacing: \"0.04em\",\n textTransform: \"uppercase\",\n background: `linear-gradient(180deg, rgba(0,0,0,0.12) 0%, rgba(0,0,0,0.22) 100%)`\n },\n children: [\n cfg.sponsorImageUrl && /* @__PURE__ */ (0, import_jsx_runtime.jsx)(\"img\", { src: cfg.sponsorImageUrl, alt: \"sponsor\", style: { height: `${f * 1.35}px`, objectFit: \"contain\", flexShrink: 0 } }),\n cfg.sponsorText && /* @__PURE__ */ (0, import_jsx_runtime.jsx)(\"span\", { style: { overflow: \"hidden\", textOverflow: \"ellipsis\", whiteSpace: \"nowrap\" }, children: cfg.sponsorText })\n ]\n }\n )\n ]\n }\n );\n}\nfunction LowerThirdOverlay({ overlay, size }) {\n const cfg = parseConfig(overlay.content);\n if (!cfg) return null;\n const f = Math.max(6, size.w * 0.05);\n const accentH = Math.max(3, size.h * 0.038);\n return /* @__PURE__ */ (0, import_jsx_runtime.jsxs)(\n \"div\",\n {\n style: {\n width: \"100%\",\n height: \"100%\",\n borderRadius: Math.max(2, size.w * 0.018),\n display: \"flex\",\n flexDirection: \"column\",\n justifyContent: \"flex-end\",\n background: cfg.backgroundColor,\n color: cfg.textColor,\n fontFamily: \"'Inter', 'Roboto', 'Segoe UI', system-ui, sans-serif\",\n overflow: \"hidden\",\n pointerEvents: \"none\",\n userSelect: \"none\",\n fontSize: `${f}px`,\n boxShadow: \"0 10px 32px rgba(0,0,0,0.4)\"\n },\n children: [\n /* @__PURE__ */ (0, import_jsx_runtime.jsx)(\"div\", { style: { width: \"100%\", height: accentH, backgroundColor: cfg.accentColor, flexShrink: 0 } }),\n /* @__PURE__ */ (0, import_jsx_runtime.jsxs)(\n \"div\",\n {\n style: {\n flex: 1,\n display: \"flex\",\n flexDirection: \"column\",\n justifyContent: \"center\",\n padding: `${f * 0.62}px ${f * 1.05}px`,\n minHeight: 0,\n gap: f * 0.18\n },\n children: [\n /* @__PURE__ */ (0, import_jsx_runtime.jsx)(\n \"div\",\n {\n style: {\n fontSize: \"1.28em\",\n fontWeight: 800,\n lineHeight: 1.22,\n letterSpacing: \"0.01em\",\n textShadow: \"0 2px 6px rgba(0,0,0,0.55)\"\n },\n children: cfg.headline\n }\n ),\n /* @__PURE__ */ (0, import_jsx_runtime.jsx)(\n \"div\",\n {\n style: {\n fontSize: \"0.94em\",\n fontWeight: 500,\n opacity: 0.92,\n lineHeight: 1.35,\n color: cfg.textColor\n },\n children: cfg.subtitle\n }\n )\n ]\n }\n ),\n (cfg.sponsorText || cfg.sponsorImageUrl) && /* @__PURE__ */ (0, import_jsx_runtime.jsxs)(\n \"div\",\n {\n style: {\n fontSize: \"0.68em\",\n fontWeight: 600,\n opacity: 0.72,\n padding: `${f * 0.28}px ${f * 1.05}px ${f * 0.48}px`,\n display: \"flex\",\n alignItems: \"center\",\n gap: f * 0.4,\n overflow: \"hidden\",\n letterSpacing: \"0.06em\",\n textTransform: \"uppercase\",\n borderTop: `1px solid rgba(255,255,255,0.08)`\n },\n children: [\n cfg.sponsorImageUrl && /* @__PURE__ */ (0, import_jsx_runtime.jsx)(\"img\", { src: cfg.sponsorImageUrl, alt: \"sponsor\", style: { height: `${f * 1.25}px`, objectFit: \"contain\", flexShrink: 0 } }),\n cfg.sponsorText && /* @__PURE__ */ (0, import_jsx_runtime.jsx)(\"span\", { style: { overflow: \"hidden\", textOverflow: \"ellipsis\", whiteSpace: \"nowrap\" }, children: cfg.sponsorText })\n ]\n }\n )\n ]\n }\n );\n}\nfunction QrCodeOverlay({ overlay, size }) {\n const cfg = parseConfig(overlay.content);\n if (!cfg) return null;\n const qrSide = Math.max(32, Math.min(size.w, size.h) * 0.55);\n const qrUrl = `https://api.qrserver.com/v1/create-qr-code/?size=${Math.round(qrSide * 2)}x${Math.round(qrSide * 2)}&data=${encodeURIComponent(cfg.url || \"https://example.com\")}`;\n const f = Math.max(6, size.w * 0.06);\n return /* @__PURE__ */ (0, import_jsx_runtime.jsxs)(\"div\", { style: { width: \"100%\", height: \"100%\", borderRadius: Math.max(2, size.w * 0.035), display: \"flex\", flexDirection: \"column\", alignItems: \"center\", justifyContent: \"center\", gap: f * 0.4, background: cfg.backgroundColor, color: cfg.textColor, fontFamily: \"Roboto, 'Segoe UI', Arial, sans-serif\", padding: f * 0.6, boxSizing: \"border-box\", pointerEvents: \"none\", userSelect: \"none\", overflow: \"hidden\" }, children: [\n /* @__PURE__ */ (0, import_jsx_runtime.jsx)(\"div\", { style: { flexShrink: 0, background: \"#fff\", borderRadius: Math.max(2, qrSide * 0.06), padding: Math.max(2, qrSide * 0.06), lineHeight: 0 }, children: /* @__PURE__ */ (0, import_jsx_runtime.jsx)(\"img\", { src: qrUrl, alt: \"QR Code\", style: { width: `${qrSide}px`, height: `${qrSide}px`, display: \"block\" } }) }),\n /* @__PURE__ */ (0, import_jsx_runtime.jsx)(\"div\", { style: { fontSize: `${f * 1.1}px`, fontWeight: 700, textAlign: \"center\", color: cfg.accentColor, overflow: \"hidden\", textOverflow: \"ellipsis\", whiteSpace: \"nowrap\", width: \"100%\" }, children: cfg.ctaText }),\n cfg.description && /* @__PURE__ */ (0, import_jsx_runtime.jsx)(\"div\", { style: { fontSize: `${f * 0.75}px`, opacity: 0.6, textAlign: \"center\", overflow: \"hidden\", textOverflow: \"ellipsis\", whiteSpace: \"nowrap\", width: \"100%\" }, children: cfg.description })\n ] });\n}\nfunction ComingUpNextOverlay({ overlay, size }) {\n const cfg = parseConfig(overlay.content);\n if (!cfg) return null;\n const f = Math.max(8, Math.min(size.h * 0.13, size.w * 0.048));\n const showSubtitle = size.h >= 60;\n const showThumbnail = false;\n const thumbW = 0;\n return /* @__PURE__ */ (0, import_jsx_runtime.jsxs)(\"div\", { style: {\n width: \"100%\",\n height: \"100%\",\n borderRadius: Math.max(2, size.w * 0.025),\n display: \"flex\",\n background: cfg.backgroundColor,\n color: cfg.textColor,\n fontFamily: \"'Arial', 'Helvetica Neue', Helvetica, sans-serif\",\n overflow: \"hidden\",\n pointerEvents: \"none\",\n userSelect: \"none\",\n fontSize: `${f}px`,\n WebkitFontSmoothing: \"antialiased\"\n }, children: [\n /* @__PURE__ */ (0, import_jsx_runtime.jsx)(\"div\", { style: { width: Math.max(3, size.w * 0.018), flexShrink: 0, backgroundColor: cfg.accentColor } }),\n /* @__PURE__ */ (0, import_jsx_runtime.jsxs)(\"div\", { style: {\n flex: 1,\n display: \"flex\",\n flexDirection: \"column\",\n justifyContent: \"center\",\n padding: `${f * 0.35}px ${f * 0.75}px`,\n minWidth: 0,\n gap: `${f * 0.08}px`\n }, children: [\n /* @__PURE__ */ (0, import_jsx_runtime.jsx)(\"div\", { style: {\n fontSize: \"0.7em\",\n fontWeight: 700,\n textTransform: \"uppercase\",\n letterSpacing: \"0.09em\",\n color: cfg.accentColor,\n lineHeight: 1\n }, children: \"Coming Up Next\" }),\n /* @__PURE__ */ (0, import_jsx_runtime.jsx)(\"div\", { style: {\n fontSize: \"1.2em\",\n fontWeight: 700,\n lineHeight: 1.2,\n overflow: \"hidden\",\n textOverflow: \"ellipsis\",\n whiteSpace: \"nowrap\"\n }, children: cfg.title }),\n showSubtitle && /* @__PURE__ */ (0, import_jsx_runtime.jsx)(\"div\", { style: {\n fontSize: \"0.82em\",\n opacity: 0.65,\n lineHeight: 1.1,\n overflow: \"hidden\",\n textOverflow: \"ellipsis\",\n whiteSpace: \"nowrap\"\n }, children: cfg.subtitle }),\n cfg.scheduledTime && /* @__PURE__ */ (0, import_jsx_runtime.jsx)(\"div\", { style: {\n fontSize: \"0.9em\",\n fontWeight: 700,\n color: cfg.accentColor,\n lineHeight: 1\n }, children: cfg.scheduledTime })\n ] }),\n showThumbnail && /* @__PURE__ */ (0, import_jsx_runtime.jsx)(\"div\", { style: { flexShrink: 0, width: `${thumbW}px`, overflow: \"hidden\" }, children: /* @__PURE__ */ (0, import_jsx_runtime.jsx)(\n \"img\",\n {\n src: cfg.thumbnailUrl,\n alt: \"\",\n style: { width: \"100%\", height: \"100%\", objectFit: \"cover\", display: \"block\" }\n }\n ) })\n ] });\n}\nfunction ContextualTriggerOverlay({ overlay, size }) {\n const cfg = parseConfig(overlay.content);\n if (!cfg) return null;\n const icons = { alert: \"\\u26A0\\uFE0F\", celebration: \"\\u{1F389}\", info: \"\\u2139\\uFE0F\", warning: \"\\u{1F514}\" };\n const f = Math.max(6, size.w * 0.05);\n return /* @__PURE__ */ (0, import_jsx_runtime.jsxs)(\"div\", { style: { width: \"100%\", height: \"100%\", borderRadius: Math.max(2, size.w * 0.035), display: \"flex\", alignItems: \"center\", gap: f * 0.8, padding: `0 ${f * 1.2}px`, background: cfg.backgroundColor, color: cfg.textColor, fontFamily: \"Roboto, 'Segoe UI', Arial, sans-serif\", borderLeft: `${Math.max(2, size.w * 0.02)}px solid ${cfg.accentColor}`, boxSizing: \"border-box\", pointerEvents: \"none\", userSelect: \"none\", fontSize: `${f}px` }, children: [\n /* @__PURE__ */ (0, import_jsx_runtime.jsx)(\"span\", { style: { fontSize: \"2em\", flexShrink: 0 }, children: icons[cfg.iconType] || \"\\u26A1\" }),\n /* @__PURE__ */ (0, import_jsx_runtime.jsxs)(\"div\", { style: { flex: 1, minWidth: 0 }, children: [\n /* @__PURE__ */ (0, import_jsx_runtime.jsx)(\"div\", { style: { fontSize: \"1.3em\", fontWeight: 700, overflow: \"hidden\", textOverflow: \"ellipsis\", whiteSpace: \"nowrap\" }, children: cfg.headline }),\n /* @__PURE__ */ (0, import_jsx_runtime.jsx)(\"div\", { style: { fontSize: \"0.9em\", opacity: 0.7, overflow: \"hidden\", textOverflow: \"ellipsis\", whiteSpace: \"nowrap\" }, children: cfg.message })\n ] })\n ] });\n}\nfunction OddsBettingOverlay({ overlay, size }) {\n const cfg = parseConfig(overlay.content);\n if (!cfg) return null;\n const f = Math.max(6, size.w * 0.052);\n return /* @__PURE__ */ (0, import_jsx_runtime.jsxs)(\"div\", { style: { width: \"100%\", height: \"100%\", borderRadius: Math.max(2, size.w * 0.035), display: \"flex\", flexDirection: \"column\", padding: f * 0.8, background: cfg.backgroundColor, color: cfg.textColor, fontFamily: \"Roboto, 'Segoe UI', Arial, sans-serif\", boxSizing: \"border-box\", pointerEvents: \"none\", userSelect: \"none\", fontSize: `${f}px` }, children: [\n /* @__PURE__ */ (0, import_jsx_runtime.jsx)(\"div\", { style: { fontSize: \"0.9em\", fontWeight: 700, textTransform: \"uppercase\", letterSpacing: \"0.05em\", color: cfg.accentColor, marginBottom: f * 0.4 }, children: cfg.eventTitle }),\n /* @__PURE__ */ (0, import_jsx_runtime.jsx)(\"div\", { style: { flex: 1, display: \"flex\", flexDirection: \"column\", gap: f * 0.2, justifyContent: \"center\" }, children: (cfg.options || []).slice(0, 5).map((opt, i) => /* @__PURE__ */ (0, import_jsx_runtime.jsxs)(\"div\", { style: { display: \"flex\", justifyContent: \"space-between\", alignItems: \"center\", padding: `${f * 0.2}px ${f * 0.6}px`, borderRadius: Math.max(2, f * 0.3), background: `${cfg.accentColor}15`, fontSize: \"1em\" }, children: [\n /* @__PURE__ */ (0, import_jsx_runtime.jsx)(\"span\", { style: { overflow: \"hidden\", textOverflow: \"ellipsis\", whiteSpace: \"nowrap\", flex: 1 }, children: opt.label }),\n /* @__PURE__ */ (0, import_jsx_runtime.jsx)(\"span\", { style: { fontWeight: 700, marginLeft: f * 0.8, flexShrink: 0, color: cfg.accentColor }, children: opt.odds })\n ] }, i)) }),\n cfg.sponsorText && /* @__PURE__ */ (0, import_jsx_runtime.jsx)(\"div\", { style: { fontSize: \"0.7em\", opacity: 0.4, textAlign: \"center\", marginTop: f * 0.4 }, children: cfg.sponsorText })\n ] });\n}\nfunction BreakingNewsOverlay({ overlay, size }) {\n const cfg = parseConfig(overlay.content);\n if (!cfg) return null;\n const urgencyColors = { breaking: \"#dc2626\", urgent: \"#ea580c\", normal: \"#2563eb\" };\n const labelBg = urgencyColors[cfg.urgency] || urgencyColors.normal;\n const label = cfg.urgency === \"breaking\" ? \"BREAKING\" : cfg.urgency === \"urgent\" ? \"URGENT\" : \"NEWS\";\n const f = Math.max(6, size.w * 0.05);\n return /* @__PURE__ */ (0, import_jsx_runtime.jsxs)(\"div\", { style: { width: \"100%\", height: \"100%\", borderRadius: Math.max(2, size.w * 0.02), display: \"flex\", alignItems: \"center\", background: cfg.backgroundColor, color: cfg.textColor, fontFamily: \"Roboto, 'Segoe UI', Arial, sans-serif\", overflow: \"hidden\", pointerEvents: \"none\", userSelect: \"none\", fontSize: `${f}px` }, children: [\n /* @__PURE__ */ (0, import_jsx_runtime.jsx)(\"div\", { style: { padding: `0 ${f * 0.8}px`, height: \"100%\", display: \"flex\", alignItems: \"center\", background: labelBg, color: \"#fff\", fontSize: \"1em\", fontWeight: 900, textTransform: \"uppercase\", letterSpacing: \"0.05em\", flexShrink: 0 }, children: label }),\n /* @__PURE__ */ (0, import_jsx_runtime.jsxs)(\"div\", { style: { flex: 1, padding: `0 ${f * 1}px`, minWidth: 0 }, children: [\n /* @__PURE__ */ (0, import_jsx_runtime.jsx)(\"div\", { style: { fontSize: \"1.3em\", fontWeight: 700, overflow: \"hidden\", textOverflow: \"ellipsis\", whiteSpace: \"nowrap\" }, children: cfg.headline }),\n cfg.body && /* @__PURE__ */ (0, import_jsx_runtime.jsx)(\"div\", { style: { fontSize: \"0.9em\", opacity: 0.7, overflow: \"hidden\", textOverflow: \"ellipsis\", whiteSpace: \"nowrap\" }, children: cfg.body })\n ] })\n ] });\n}\nfunction calcCountdownRemaining(targetTime) {\n const diff = Math.max(0, new Date(targetTime).getTime() - Date.now());\n return {\n d: Math.floor(diff / 864e5),\n h: Math.floor(diff % 864e5 / 36e5),\n m: Math.floor(diff % 36e5 / 6e4),\n s: Math.floor(diff % 6e4 / 1e3),\n expired: diff === 0\n };\n}\nfunction CountdownOverlay({ overlay, size }) {\n const cfg = parseConfig(overlay.content);\n const targetTime = cfg?.targetTime ?? \"\";\n const [remaining, setRemaining] = (0, import_react.useState)(\n () => targetTime ? calcCountdownRemaining(targetTime) : { d: 0, h: 0, m: 0, s: 0, expired: false }\n );\n (0, import_react.useEffect)(() => {\n if (!targetTime) return;\n setRemaining(calcCountdownRemaining(targetTime));\n const id = setInterval(() => setRemaining(calcCountdownRemaining(targetTime)), 1e3);\n return () => clearInterval(id);\n }, [targetTime]);\n if (!cfg) return null;\n const f = Math.max(6, size.w * 0.055);\n const pad = (n) => String(n).padStart(2, \"0\");\n const units = [\n { show: cfg.showDays, value: pad(remaining.d), label: \"DAYS\" },\n { show: cfg.showHours, value: pad(remaining.h), label: \"HRS\" },\n { show: cfg.showMinutes, value: pad(remaining.m), label: \"MIN\" },\n { show: cfg.showSeconds, value: pad(remaining.s), label: \"SEC\" }\n ];\n return /* @__PURE__ */ (0, import_jsx_runtime.jsxs)(\"div\", { style: { width: \"100%\", height: \"100%\", borderRadius: Math.max(2, size.w * 0.035), display: \"flex\", flexDirection: \"column\", alignItems: \"center\", justifyContent: \"center\", padding: f * 0.8, background: cfg.backgroundColor, color: cfg.textColor, fontFamily: \"Roboto, 'Segoe UI', Arial, sans-serif\", boxSizing: \"border-box\", pointerEvents: \"none\", userSelect: \"none\", fontSize: `${f}px` }, children: [\n /* @__PURE__ */ (0, import_jsx_runtime.jsx)(\"div\", { style: { fontSize: \"0.8em\", fontWeight: 600, textTransform: \"uppercase\", letterSpacing: \"0.05em\", color: cfg.accentColor, marginBottom: f * 0.4 }, children: cfg.eventName }),\n remaining.expired ? /* @__PURE__ */ (0, import_jsx_runtime.jsx)(\"div\", { style: { fontSize: \"1em\", fontWeight: 700, opacity: 0.6 }, children: cfg.message || \"Event ended\" }) : /* @__PURE__ */ (0, import_jsx_runtime.jsx)(\"div\", { style: { display: \"flex\", gap: f * 0.6, alignItems: \"center\" }, children: units.filter((u) => u.show).map((u, i, arr) => /* @__PURE__ */ (0, import_jsx_runtime.jsxs)(import_react.default.Fragment, { children: [\n /* @__PURE__ */ (0, import_jsx_runtime.jsxs)(\"div\", { style: { textAlign: \"center\" }, children: [\n /* @__PURE__ */ (0, import_jsx_runtime.jsx)(\"div\", { style: { fontSize: \"2em\", fontWeight: 900, lineHeight: 1, borderRadius: Math.max(2, f * 0.4), padding: `${f * 0.2}px ${f * 0.4}px`, background: `${cfg.accentColor}20` }, children: u.value }),\n /* @__PURE__ */ (0, import_jsx_runtime.jsx)(\"div\", { style: { fontSize: \"0.5em\", opacity: 0.5, marginTop: f * 0.2 }, children: u.label })\n ] }),\n i < arr.length - 1 && /* @__PURE__ */ (0, import_jsx_runtime.jsx)(\"div\", { style: { fontSize: \"1.8em\", fontWeight: 700, opacity: 0.3 }, children: \":\" })\n ] }, u.label)) }),\n !remaining.expired && cfg.message && /* @__PURE__ */ (0, import_jsx_runtime.jsx)(\"div\", { style: { fontSize: \"0.8em\", opacity: 0.6, marginTop: f * 0.4, textAlign: \"center\" }, children: cfg.message })\n ] });\n}\nfunction ShapeOverlay({ overlay, size }) {\n const f = Math.max(6, size.w * 0.05);\n return /* @__PURE__ */ (0, import_jsx_runtime.jsx)(\"div\", { style: { width: \"100%\", height: \"100%\", borderRadius: Math.max(2, size.w * 0.03), background: \"rgba(99, 102, 241, 0.2)\", border: \"2px solid rgba(99, 102, 241, 0.4)\", display: \"flex\", alignItems: \"center\", justifyContent: \"center\", pointerEvents: \"none\", userSelect: \"none\" }, children: /* @__PURE__ */ (0, import_jsx_runtime.jsx)(\"div\", { style: { fontSize: `${f}px`, fontWeight: 500, color: \"rgba(163, 163, 163, 0.8)\", textTransform: \"uppercase\" }, children: overlay.name }) });\n}\nfunction hexToRgb(hex) {\n if (!hex || !hex.startsWith(\"#\")) return \"0,0,0\";\n const clean = hex.slice(1);\n const num = parseInt(clean.length === 3 ? clean.replace(/./g, \"$&$&\") : clean, 16);\n return `${num >> 16 & 255},${num >> 8 & 255},${num & 255}`;\n}\nvar FADE_DURATION_MS = 1e3;\nvar SHOWCASE_CYCLE_MS_DEFAULT = 36e3;\nvar SHOWCASE_MIN_BEAT_MS = 6e3;\nvar BEAT_POP_IN_MS = 520;\nvar BEAT_POP_OUT_MS = 520;\nvar SHOWCASE_PERSISTENT_TYPES = /* @__PURE__ */ new Set([\n \"scroller\",\n \"breaking_news\",\n \"image\",\n \"text\",\n \"shape\",\n \"countdown\",\n \"qr_code\"\n]);\nfunction easeOutCubic(t) {\n const u = 1 - t;\n return 1 - u * u * u;\n}\nfunction easeInCubic(t) {\n return t * t * t;\n}\nfunction easeOutBack(t) {\n const c1 = 1.70158;\n const c3 = c1 + 1;\n return 1 + c3 * (t - 1) ** 3 + c1 * (t - 1) ** 2;\n}\nfunction beatOpacity(phase, beatMs) {\n if (phase < BEAT_POP_IN_MS) return easeOutCubic(phase / BEAT_POP_IN_MS);\n const holdEnd = beatMs - BEAT_POP_OUT_MS;\n if (phase >= holdEnd) {\n return Math.max(\n 0,\n 1 - easeInCubic((phase - holdEnd) / BEAT_POP_OUT_MS)\n );\n }\n return 1;\n}\nfunction beatScale(phase, beatMs) {\n if (phase < BEAT_POP_IN_MS) {\n return Math.min(1, 0.92 + 0.08 * easeOutBack(phase / BEAT_POP_IN_MS));\n }\n const holdEnd = beatMs - BEAT_POP_OUT_MS;\n if (phase >= holdEnd) {\n return 1 - 0.06 * easeInCubic((phase - holdEnd) / BEAT_POP_OUT_MS);\n }\n return 1;\n}\nfunction partitionShowcase(overlays) {\n const visible = overlays.filter((o) => o.visible);\n const persistent = [];\n const spotlight = [];\n for (const o of visible) {\n if (SHOWCASE_PERSISTENT_TYPES.has(o.type)) persistent.push(o);\n else spotlight.push(o);\n }\n spotlight.sort(\n (a, b) => a.z_index - b.z_index || a.id - b.id\n );\n const beats = [];\n for (const o of spotlight) beats.push([o]);\n return { persistent, beats };\n}\nfunction useShowcaseBeat(enabled, cycleMs, beats) {\n const safeBeats = Math.max(1, beats);\n const beatMs = Math.max(\n SHOWCASE_MIN_BEAT_MS,\n Math.floor(cycleMs / safeBeats)\n );\n const totalMs = beatMs * safeBeats;\n const [now, setNow] = (0, import_react.useState)(() => Date.now());\n (0, import_react.useEffect)(() => {\n if (!enabled) return;\n const id = window.setInterval(() => setNow(Date.now()), 64);\n return () => clearInterval(id);\n }, [enabled]);\n const phase = now % totalMs;\n const beatIndex = Math.min(safeBeats - 1, Math.floor(phase / beatMs));\n const beatPhase = phase - beatIndex * beatMs;\n return { beatIndex, beatPhase, beatMs };\n}\nvar OverlayRenderer = ({\n overlays,\n currentTime,\n videoRef,\n coordinateSpace,\n showcaseMode = false,\n showcaseCycleMs = SHOWCASE_CYCLE_MS_DEFAULT\n}) => {\n const [dims, setDims] = (0, import_react.useState)(null);\n const rafRef = (0, import_react.useRef)(null);\n const [fadeMap, setFadeMap] = (0, import_react.useState)(/* @__PURE__ */ new Map());\n const removeTimers = (0, import_react.useRef)(/* @__PURE__ */ new Map());\n const { persistent: showcasePersistent, beats: showcaseBeats } = (0, import_react.useMemo)(\n () => partitionShowcase(overlays),\n [overlays]\n );\n const {\n beatIndex: showcaseBeatIndex,\n beatPhase: showcaseBeatPhase,\n beatMs: showcaseBeatMs\n } = useShowcaseBeat(showcaseMode, showcaseCycleMs, showcaseBeats.length);\n const showcasePersistentIds = (0, import_react.useMemo)(\n () => new Set(showcasePersistent.map((o) => o.id)),\n [showcasePersistent]\n );\n const showcaseSpotlightIds = (0, import_react.useMemo)(() => {\n const beat = showcaseBeats[showcaseBeatIndex];\n return new Set((beat ?? []).map((o) => o.id));\n }, [showcaseBeats, showcaseBeatIndex]);\n const updateDims = (0, import_react.useCallback)(() => {\n const video = videoRef.current;\n if (video) {\n const computed = computeVideoDimensions(video);\n setDims((prev) => {\n if (!computed || prev && prev.nativeWidth === computed.nativeWidth && prev.nativeHeight === computed.nativeHeight && prev.displayWidth === computed.displayWidth && prev.displayHeight === computed.displayHeight && prev.offsetX === computed.offsetX && prev.offsetY === computed.offsetY) {\n return prev;\n }\n return computed;\n });\n }\n }, [videoRef]);\n (0, import_react.useEffect)(() => {\n updateDims();\n const interval = setInterval(updateDims, 500);\n const handleResize = () => {\n if (rafRef.current) cancelAnimationFrame(rafRef.current);\n rafRef.current = requestAnimationFrame(updateDims);\n };\n window.addEventListener(\"resize\", handleResize);\n return () => {\n clearInterval(interval);\n window.removeEventListener(\"resize\", handleResize);\n if (rafRef.current) cancelAnimationFrame(rafRef.current);\n };\n }, [updateDims]);\n const activeOverlays = (0, import_react.useMemo)(() => {\n if (showcaseMode) {\n const beat = showcaseBeats[showcaseBeatIndex] ?? [];\n return [...showcasePersistent, ...beat];\n }\n return overlays.filter((o) => isOverlayActive(o, currentTime));\n }, [\n overlays,\n currentTime,\n showcaseMode,\n showcasePersistent,\n showcaseBeats,\n showcaseBeatIndex\n ]);\n (0, import_react.useLayoutEffect)(() => {\n const activeIds = new Set(activeOverlays.map((o) => o.id));\n setFadeMap((prev) => {\n const next = new Map(prev);\n for (const overlay of activeOverlays) {\n if (!next.has(overlay.id)) {\n next.set(overlay.id, { overlay, visible: false });\n } else {\n const existing = next.get(overlay.id);\n next.set(overlay.id, { ...existing, overlay });\n }\n }\n for (const [id, state] of next) {\n if (!activeIds.has(id) && state.visible) {\n next.set(id, { ...state, visible: false });\n if (!removeTimers.current.has(id)) {\n const timer = setTimeout(() => {\n setFadeMap((m) => {\n const updated = new Map(m);\n updated.delete(id);\n return updated;\n });\n removeTimers.current.delete(id);\n }, FADE_DURATION_MS);\n removeTimers.current.set(id, timer);\n }\n } else if (!activeIds.has(id) && !state.visible) {\n }\n }\n return next;\n });\n }, [activeOverlays]);\n (0, import_react.useEffect)(() => {\n const toFadeIn = [];\n for (const [id, state] of fadeMap) {\n if (!state.visible) {\n const isActive = activeOverlays.some((o) => o.id === id);\n if (isActive) toFadeIn.push(id);\n }\n }\n if (toFadeIn.length === 0) return;\n const raf = requestAnimationFrame(() => {\n setFadeMap((prev) => {\n const next = new Map(prev);\n for (const id of toFadeIn) {\n const state = next.get(id);\n if (state) next.set(id, { ...state, visible: true });\n }\n return next;\n });\n });\n return () => cancelAnimationFrame(raf);\n }, [fadeMap, activeOverlays]);\n (0, import_react.useEffect)(() => {\n return () => {\n for (const timer of removeTimers.current.values()) clearTimeout(timer);\n };\n }, []);\n if (!dims || fadeMap.size === 0) return null;\n return /* @__PURE__ */ (0, import_jsx_runtime.jsx)(\n \"div\",\n {\n \"aria-hidden\": \"true\",\n style: {\n position: \"absolute\",\n left: `${dims.offsetX}px`,\n top: `${dims.offsetY}px`,\n width: `${dims.displayWidth}px`,\n height: `${dims.displayHeight}px`,\n pointerEvents: \"none\",\n overflow: \"hidden\",\n zIndex: 8\n },\n children: [...fadeMap.values()].map(({ overlay, visible }) => {\n const uniform = coordinateSpace && coordinateSpace.width > 0 && coordinateSpace.height > 0;\n const auth = uniform ? { width: coordinateSpace.width, height: coordinateSpace.height } : overlayAuthoringDimensions(\n overlay,\n dims.nativeWidth,\n dims.nativeHeight\n );\n const scaleX = dims.displayWidth / auth.width;\n const scaleY = dims.displayHeight / auth.height;\n const left = overlay.x * scaleX;\n const top = overlay.y * scaleY;\n const width = overlay.width * scaleX;\n const height = overlay.height * scaleY;\n const baseOpacity = Math.max(0, Math.min(100, overlay.opacity)) / 100;\n const isShowcasePersistent = showcaseMode && showcasePersistentIds.has(overlay.id);\n const isShowcaseSpotlight = showcaseMode && showcaseSpotlightIds.has(overlay.id);\n const isShowcaseDormant = showcaseMode && !isShowcasePersistent && !isShowcaseSpotlight;\n const useShowcasePop = isShowcaseSpotlight;\n const opacity = useShowcasePop ? baseOpacity * beatOpacity(showcaseBeatPhase, showcaseBeatMs) : isShowcaseDormant ? 0 : visible ? baseOpacity : 0;\n const popScale = useShowcasePop ? beatScale(showcaseBeatPhase, showcaseBeatMs) : 1;\n const sz = { w: width, h: height };\n return /* @__PURE__ */ (0, import_jsx_runtime.jsxs)(\n \"div\",\n {\n style: {\n position: \"absolute\",\n left: `${left}px`,\n top: `${top}px`,\n width: `${width}px`,\n height: `${height}px`,\n opacity,\n transition: useShowcasePop ? \"none\" : `opacity ${FADE_DURATION_MS}ms ease`,\n transform: showcaseMode && useShowcasePop ? `scale(${popScale})` : void 0,\n transformOrigin: showcaseMode && useShowcasePop ? \"center center\" : void 0,\n zIndex: overlay.z_index,\n overflow: \"hidden\"\n },\n children: [\n overlay.type === \"image\" && /* @__PURE__ */ (0, import_jsx_runtime.jsx)(ImageOverlay, { overlay }),\n overlay.type === \"text\" && /* @__PURE__ */ (0, import_jsx_runtime.jsx)(TextOverlay, { overlay }),\n overlay.type === \"scroller\" && /* @__PURE__ */ (0, import_jsx_runtime.jsx)(ScrollerOverlay, { overlay }),\n overlay.type === \"shape\" && /* @__PURE__ */ (0, import_jsx_runtime.jsx)(ShapeOverlay, { overlay, size: sz }),\n overlay.type === \"score_bug\" && /* @__PURE__ */ (0, import_jsx_runtime.jsx)(ScoreBugOverlay, { overlay, size: sz }),\n overlay.type === \"lower_third\" && /* @__PURE__ */ (0, import_jsx_runtime.jsx)(LowerThirdOverlay, { overlay, size: sz }),\n overlay.type === \"qr_code\" && /* @__PURE__ */ (0, import_jsx_runtime.jsx)(QrCodeOverlay, { overlay, size: sz }),\n overlay.type === \"coming_up_next\" && /* @__PURE__ */ (0, import_jsx_runtime.jsx)(ComingUpNextOverlay, { overlay, size: sz }),\n overlay.type === \"contextual_trigger\" && /* @__PURE__ */ (0, import_jsx_runtime.jsx)(ContextualTriggerOverlay, { overlay, size: sz }),\n overlay.type === \"odds_betting\" && /* @__PURE__ */ (0, import_jsx_runtime.jsx)(OddsBettingOverlay, { overlay, size: sz }),\n overlay.type === \"breaking_news\" && /* @__PURE__ */ (0, import_jsx_runtime.jsx)(BreakingNewsOverlay, { overlay, size: sz }),\n overlay.type === \"countdown\" && /* @__PURE__ */ (0, import_jsx_runtime.jsx)(CountdownOverlay, { overlay, size: sz })\n ]\n },\n overlay.id\n );\n })\n }\n );\n};\n// Annotate the CommonJS export names for ESM import in node:\n0 && (module.exports = {\n OverlayRenderer\n});\n","import React, {\n useEffect,\n useLayoutEffect,\n useRef,\n useState,\n useCallback,\n useMemo,\n useId,\n} from \"react\";\nimport {\n type SwirlOverlay,\n isOverlayActive,\n resolveImageUrl,\n overlayAuthoringDimensions,\n} from \"../utils/overlays\";\n\ninterface VideoDimensions {\n nativeWidth: number;\n nativeHeight: number;\n displayWidth: number;\n displayHeight: number;\n offsetX: number;\n offsetY: number;\n scaleX: number;\n scaleY: number;\n}\n\ninterface OverlayRendererProps {\n overlays: SwirlOverlay[];\n currentTime: number;\n videoRef: React.RefObject<HTMLVideoElement | null>;\n coordinateSpace?: { width: number; height: number } | null;\n showcaseMode?: boolean;\n showcaseCycleMs?: number;\n}\n\nfunction computeVideoDimensions(\n video: HTMLVideoElement\n): VideoDimensions | null {\n const nativeWidth = video.videoWidth;\n const nativeHeight = video.videoHeight;\n if (!nativeWidth || !nativeHeight) return null;\n\n const displayWidth = video.offsetWidth;\n const displayHeight = video.offsetHeight;\n if (!displayWidth || !displayHeight) return null;\n\n const videoAspect = nativeWidth / nativeHeight;\n const displayAspect = displayWidth / displayHeight;\n\n let renderWidth: number;\n let renderHeight: number;\n let offsetX: number;\n let offsetY: number;\n\n if (videoAspect > displayAspect) {\n renderWidth = displayWidth;\n renderHeight = displayWidth / videoAspect;\n offsetX = 0;\n offsetY = (displayHeight - renderHeight) / 2;\n } else {\n renderHeight = displayHeight;\n renderWidth = displayHeight * videoAspect;\n offsetX = (displayWidth - renderWidth) / 2;\n offsetY = 0;\n }\n\n return {\n nativeWidth,\n nativeHeight,\n displayWidth: renderWidth,\n displayHeight: renderHeight,\n offsetX,\n offsetY,\n scaleX: renderWidth / nativeWidth,\n scaleY: renderHeight / nativeHeight,\n };\n}\n\nfunction ImageOverlay({ overlay }: { overlay: SwirlOverlay }) {\n const src = resolveImageUrl(overlay.image_url || \"\");\n if (!src) return null;\n return (\n <img\n src={src}\n alt={overlay.name}\n draggable={false}\n style={{\n width: \"100%\",\n height: \"100%\",\n objectFit: \"contain\",\n display: \"block\",\n pointerEvents: \"none\",\n userSelect: \"none\",\n }}\n />\n );\n}\n\nfunction TextOverlay({ overlay }: { overlay: SwirlOverlay }) {\n const text = overlay.content || \"\";\n return (\n <div\n style={{\n width: \"100%\",\n height: \"100%\",\n display: \"flex\",\n alignItems: \"center\",\n justifyContent: \"center\",\n color: \"#ffffff\",\n fontSize: \"clamp(10px, 1.4vw, 20px)\",\n fontFamily: \"Roboto, 'Segoe UI', Arial, sans-serif\",\n fontWeight: 600,\n textAlign: \"center\",\n padding: \"4px 8px\",\n boxSizing: \"border-box\",\n wordBreak: \"break-word\",\n textShadow: \"0 1px 4px rgba(0,0,0,0.7)\",\n pointerEvents: \"none\",\n userSelect: \"none\",\n lineHeight: 1.3,\n }}\n >\n {text}\n </div>\n );\n}\n\ninterface RSSItem {\n title: string;\n description: string;\n pubDate: string;\n author: string;\n category: string;\n}\n\nfunction parseRSSXml(xmlText: string, maxItems: number): RSSItem[] {\n const stripped = xmlText.replace(/^<\\?xml[^?]*\\?>\\s*/, \"\");\n const parser = new DOMParser();\n const doc = parser.parseFromString(stripped, \"text/xml\");\n const items = Array.from(doc.querySelectorAll(\"item\"))\n .map((item) => ({\n title: (item.querySelector(\"title\")?.textContent || \"\").replace(/<[^>]*>/g, \"\").trim(),\n description: (item.querySelector(\"description\")?.textContent || \"\").replace(/<[^>]*>/g, \"\").trim(),\n pubDate: item.querySelector(\"pubDate\")?.textContent || \"\",\n author: item.querySelector(\"author, dc\\\\:creator\")?.textContent || \"\",\n category: item.querySelector(\"category\")?.textContent || \"\",\n }))\n .filter((i) => i.title)\n .slice(0, maxItems);\n if (items.length === 0 && doc.querySelector(\"parsererror\")) {\n throw new Error(\"Invalid RSS XML\");\n }\n return items;\n}\n\nconst RSS_CACHE_TTL_MS = 60_000;\nconst rssCache = new Map<string, { promise: Promise<RSSItem[]>; expiresAt: number }>();\n\nfunction cachedFetchRSSItems(rssUrl: string, maxItems: number): Promise<RSSItem[]> {\n const now = Date.now();\n const cached = rssCache.get(rssUrl);\n if (cached && cached.expiresAt > now) return cached.promise;\n const promise = fetchRSSItems(rssUrl, maxItems).catch((err) => {\n rssCache.delete(rssUrl);\n throw err;\n });\n rssCache.set(rssUrl, { promise, expiresAt: now + RSS_CACHE_TTL_MS });\n return promise;\n}\n\nasync function fetchRSSItems(rssUrl: string, maxItems: number): Promise<RSSItem[]> {\n const encoded = encodeURIComponent(rssUrl);\n\n try {\n const origin = typeof window !== \"undefined\" ? window.location.origin : \"\";\n const resp = await fetch(`${origin}/api/rss-proxy?url=${encoded}`);\n if (resp.ok) {\n const text = await resp.text();\n if (text.includes(\"<item\")) return parseRSSXml(text, maxItems);\n }\n } catch { /* fall through */ }\n\n try {\n const resp = await fetch(`https://api.allorigins.win/get?url=${encoded}`);\n if (resp.ok) {\n const data = await resp.json();\n if (data.contents) return parseRSSXml(data.contents, maxItems);\n }\n } catch { /* fall through */ }\n\n try {\n const resp = await fetch(`https://corsproxy.io/?url=${encoded}`);\n if (resp.ok) {\n const text = await resp.text();\n if (text) return parseRSSXml(text, maxItems);\n }\n } catch { /* fall through */ }\n\n throw new Error(\"All RSS proxies failed\");\n}\n\nfunction ScrollerOverlay({ overlay }: { overlay: SwirlOverlay }) {\n const cfg = overlay.scroller_config;\n const uid = useId().replace(/:/g, \"\");\n\n const [rssItems, setRssItems] = useState<RSSItem[]>([]);\n const [rssLoading, setRssLoading] = useState(true);\n const [rssError, setRssError] = useState(false);\n\n const rssUrl = cfg?.rss_url || \"\";\n const maxItems = cfg?.max_items ?? 10;\n const autoRefresh = cfg?.auto_refresh !== false;\n const updateInterval = cfg?.update_interval ?? 5;\n\n useEffect(() => {\n if (!rssUrl || (cfg?.use_custom_text && cfg?.custom_text)) {\n setRssLoading(false);\n return;\n }\n let cancelled = false;\n setRssLoading(true);\n setRssError(false);\n cachedFetchRSSItems(rssUrl, maxItems)\n .then((items) => { if (!cancelled) { setRssItems(items); setRssError(false); } })\n .catch(() => { if (!cancelled) setRssError(true); })\n .finally(() => { if (!cancelled) setRssLoading(false); });\n return () => { cancelled = true; };\n }, [rssUrl, maxItems, cfg?.use_custom_text, cfg?.custom_text]);\n\n useEffect(() => {\n if (!rssUrl || !autoRefresh || (cfg?.use_custom_text && cfg?.custom_text)) return;\n const interval = setInterval(() => {\n rssCache.delete(rssUrl);\n cachedFetchRSSItems(rssUrl, maxItems)\n .then((items) => { setRssItems(items); setRssError(false); })\n .catch(() => { /* keep showing last good items */ });\n }, updateInterval * 60 * 1000);\n return () => clearInterval(interval);\n }, [rssUrl, autoRefresh, updateInterval, maxItems, cfg?.use_custom_text, cfg?.custom_text]);\n\n const sep = cfg?.separator_char ?? \"◆\";\n\n let segments: string[];\n if (cfg?.use_custom_text && cfg?.custom_text) {\n segments = [cfg.custom_text];\n } else if (rssItems.length > 0) {\n segments = rssItems.map((item) => {\n const parts: string[] = [];\n if (cfg?.show_title !== false && item.title) parts.push(item.title);\n if (cfg?.show_description && item.description) parts.push(item.description);\n if (cfg?.show_timestamp && item.pubDate) {\n try { parts.push(new Date(item.pubDate).toLocaleDateString()); } catch { /* ignore */ }\n }\n if (cfg?.show_author && item.author) parts.push(`— ${item.author}`);\n if (cfg?.show_category && item.category) parts.push(`[${item.category}]`);\n return parts.join(\" \");\n });\n } else if (rssLoading) {\n segments = [\"Loading feed…\"];\n } else if (rssError) {\n segments = overlay.content ? [overlay.content] : [\"RSS feed unavailable\"];\n } else if (overlay.content) {\n segments = [overlay.content];\n } else {\n segments = rssUrl ? [\"Loading feed…\"] : [\"RSS Ticker\"];\n }\n\n const scrollSpeed = cfg?.scroll_speed ?? 40;\n const direction = cfg?.direction ?? \"left\";\n const fontSize = cfg?.font_size ?? 15;\n const fontFamily = cfg?.font_family || \"Roboto, 'Segoe UI', Arial, sans-serif\";\n const fontWeight = cfg?.font_weight || \"700\";\n const textColor = cfg?.text_color || \"#ffffff\";\n const bgColor = cfg?.background_color || \"#0d0d1a\";\n const bgOpacity = cfg?.background_opacity !== undefined ? cfg.background_opacity / 100 : 0.95;\n const borderRadius = cfg?.border_radius ?? 0;\n const itemSpacing = cfg?.item_spacing ?? 60;\n const pad = cfg?.padding !== undefined && cfg.padding >= 0 ? cfg.padding : 8;\n\n const label = cfg?.label ?? \"NEWS\";\n const labelLine2 = cfg?.label_line2 ?? \"\";\n const labelColor = cfg?.label_color ?? \"#f97316\";\n const labelTextColor = cfg?.label_text_color ?? \"#ffffff\";\n const accentColor = cfg?.accent_color ?? labelColor;\n const showAccentLine = cfg?.show_accent_line !== false;\n const isEquitiesStrip = cfg?.preset === \"equities_strip\";\n\n const isHorizontal = direction === \"left\" || direction === \"right\";\n const isReverse = direction === \"right\" || direction === \"down\";\n\n const fullText = segments.join(` ${sep} `);\n const durationSec = Math.max(6, (fullText.length * 9) / scrollSpeed);\n\n const animId = `sc-ticker-${overlay.id}-${uid}`;\n const keyframes = isHorizontal\n ? `@keyframes ${animId} {\n ${isReverse\n ? \"0% { transform: translateX(-50%); } 100% { transform: translateX(0%); }\"\n : \"0% { transform: translateX(0); } 100% { transform: translateX(-50%); }\"}\n }`\n : `@keyframes ${animId} {\n ${isReverse\n ? \"0% { transform: translateY(-50%); } 100% { transform: translateY(0%); }\"\n : \"0% { transform: translateY(0); } 100% { transform: translateY(-50%); }\"}\n }`;\n\n return (\n <>\n <style>{keyframes}</style>\n <div\n style={{\n width: \"100%\",\n height: \"100%\",\n display: \"flex\",\n flexDirection: \"column\",\n overflow: \"hidden\",\n borderRadius: borderRadius > 0 ? `${borderRadius}px` : undefined,\n backgroundColor: `rgba(${hexToRgb(bgColor)}, ${bgOpacity})`,\n fontFamily,\n fontSize: `${fontSize}px`,\n fontWeight,\n color: textColor,\n pointerEvents: \"none\",\n userSelect: \"none\",\n }}\n >\n {showAccentLine && (\n <div\n style={{\n height: isEquitiesStrip ? 2 : 3,\n background: accentColor,\n flexShrink: 0,\n width: \"100%\",\n }}\n />\n )}\n\n <div\n style={{\n display: \"flex\",\n flex: 1,\n overflow: \"hidden\",\n minHeight: 0,\n alignItems: \"center\",\n padding: isEquitiesStrip\n ? `${Math.max(2, pad * 0.5)}px ${pad}px`\n : `${Math.max(4, pad * 0.75)}px ${pad}px`,\n gap: isEquitiesStrip ? Math.max(4, Math.round(pad * 0.75)) : Math.max(6, pad),\n boxSizing: \"border-box\",\n }}\n >\n {label && (\n <div\n style={{\n background: labelColor,\n color: labelTextColor,\n padding: isEquitiesStrip ? \"6px 12px\" : \"10px 14px\",\n display: \"flex\",\n flexDirection: isEquitiesStrip ? \"row\" : \"column\",\n alignItems: \"center\",\n justifyContent: \"center\",\n flexShrink: 0,\n textAlign: \"center\",\n gap: isEquitiesStrip ? 0 : 2,\n borderRadius: Math.max(2, borderRadius > 0 ? borderRadius : 6),\n boxShadow: isEquitiesStrip\n ? \"inset 0 1px 0 rgba(255,255,255,0.08)\"\n : \"0 2px 8px rgba(0,0,0,0.35)\",\n alignSelf: \"stretch\",\n maxWidth: isEquitiesStrip ? \"min(200px, 28%)\" : undefined,\n }}\n >\n <span\n style={{\n fontWeight: isEquitiesStrip ? 700 : 800,\n fontSize: isEquitiesStrip ? \"0.82em\" : \"0.78em\",\n letterSpacing: isEquitiesStrip ? \"0.04em\" : \"0.08em\",\n lineHeight: 1.2,\n textTransform: isEquitiesStrip ? \"none\" : \"uppercase\",\n whiteSpace: isEquitiesStrip ? \"normal\" : \"nowrap\",\n textAlign: isEquitiesStrip ? \"left\" : \"center\",\n }}\n >\n {label}\n </span>\n {labelLine2 && !isEquitiesStrip && (\n <span\n style={{\n fontWeight: 600,\n fontSize: \"0.58em\",\n letterSpacing: \"0.06em\",\n lineHeight: 1.15,\n opacity: 0.92,\n whiteSpace: \"nowrap\",\n }}\n >\n {labelLine2}\n </span>\n )}\n </div>\n )}\n\n {label && (\n <div\n style={{\n width: isEquitiesStrip ? 1 : 2,\n alignSelf: \"stretch\",\n minHeight: isEquitiesStrip ? 20 : 24,\n background: accentColor,\n flexShrink: 0,\n borderRadius: 1,\n opacity: 0.85,\n }}\n />\n )}\n\n <div\n style={{\n flex: 1,\n overflow: \"hidden\",\n position: \"relative\",\n display: \"flex\",\n alignItems: \"center\",\n minWidth: 0,\n }}\n >\n {isHorizontal ? (\n <div\n style={{\n display: \"inline-flex\",\n whiteSpace: \"nowrap\",\n animation: `${animId} ${durationSec}s linear infinite`,\n willChange: \"transform\",\n }}\n >\n {[0, 1].map((copy) => (\n <span key={copy} style={{ paddingRight: `${itemSpacing}px` }}>\n {segments.map((seg, i) => (\n <React.Fragment key={i}>\n {i > 0 && (\n <span style={{ opacity: 0.5, margin: \"0 8px\" }}>{sep}</span>\n )}\n <span style={{ textShadow: \"0 1px 3px rgba(0,0,0,0.6)\" }}>{seg}</span>\n </React.Fragment>\n ))}\n </span>\n ))}\n </div>\n ) : (\n <div\n style={{\n display: \"flex\",\n flexDirection: \"column\",\n whiteSpace: \"nowrap\",\n animation: `${animId} ${durationSec}s linear infinite`,\n willChange: \"transform\",\n }}\n >\n {[0, 1].map((copy) =>\n segments.map((seg, i) => (\n <div key={`${copy}-${i}`} style={{ paddingBottom: `${itemSpacing / 4}px` }}>\n {seg}\n </div>\n ))\n )}\n </div>\n )}\n </div>\n </div>\n </div>\n </>\n );\n}\n\nfunction parseConfig<T>(content?: string): T | null {\n if (!content) return null;\n try { return JSON.parse(content) as T; } catch { return null; }\n}\n\ninterface OverlaySize { w: number; h: number; }\n\ninterface ScoreBugCfg { homeTeam: string; awayTeam: string; homeScore: number; awayScore: number; period: string; clock: string; sponsorText: string; sponsorImageUrl: string; backgroundColor: string; textColor: string; accentColor: string; }\ninterface LowerThirdCfg { headline: string; subtitle: string; sponsorText: string; sponsorImageUrl: string; backgroundColor: string; textColor: string; accentColor: string; style: string; }\ninterface QrCodeCfg { url: string; ctaText: string; description: string; size: number; backgroundColor: string; textColor: string; accentColor: string; }\ninterface ComingUpNextCfg { title: string; subtitle: string; scheduledTime: string; thumbnailUrl: string; backgroundColor: string; textColor: string; accentColor: string; }\ninterface ContextualTriggerCfg { triggerType: string; headline: string; message: string; iconType: string; backgroundColor: string; textColor: string; accentColor: string; animationStyle: string; }\ninterface OddsBettingCfg { eventTitle: string; options: Array<{ label: string; odds: string }>; sponsorText: string; backgroundColor: string; textColor: string; accentColor: string; oddsFormat: string; }\ninterface BreakingNewsCfg { headline: string; body: string; urgency: string; backgroundColor: string; textColor: string; accentColor: string; }\ninterface CountdownCfg { eventName: string; targetTime: string; message: string; showDays: boolean; showHours: boolean; showMinutes: boolean; showSeconds: boolean; backgroundColor: string; textColor: string; accentColor: string; }\n\nfunction ScoreBugOverlay({ overlay, size }: { overlay: SwirlOverlay; size: OverlaySize }) {\n const cfg = parseConfig<ScoreBugCfg>(overlay.content);\n if (!cfg) return null;\n const f = Math.max(6, size.w * 0.052);\n return (\n <div\n style={{\n width: \"100%\",\n height: \"100%\",\n borderRadius: Math.max(2, size.w * 0.028),\n display: \"flex\",\n flexDirection: \"column\",\n background: cfg.backgroundColor,\n color: cfg.textColor,\n fontFamily: \"'Inter', 'Roboto', 'Segoe UI', system-ui, sans-serif\",\n overflow: \"hidden\",\n pointerEvents: \"none\",\n userSelect: \"none\",\n fontSize: `${f}px`,\n boxShadow: \"0 8px 28px rgba(0,0,0,0.45)\",\n }}\n >\n <div\n style={{\n flex: 1,\n display: \"flex\",\n alignItems: \"center\",\n padding: `${f * 0.42}px ${f * 0.85}px`,\n gap: f * 0.45,\n minHeight: 0,\n }}\n >\n <div style={{ flex: 1, textAlign: \"center\", minWidth: 0, padding: `0 ${f * 0.15}px` }}>\n <div\n style={{\n fontSize: \"1.05em\",\n fontWeight: 800,\n letterSpacing: \"0.02em\",\n overflow: \"hidden\",\n textOverflow: \"ellipsis\",\n whiteSpace: \"nowrap\",\n }}\n >\n {cfg.homeTeam}\n </div>\n <div style={{ fontSize: \"1.75em\", fontWeight: 900, lineHeight: 1.05, marginTop: f * 0.08 }}>\n {cfg.homeScore}\n </div>\n </div>\n <div\n style={{\n fontSize: \"0.88em\",\n textAlign: \"center\",\n fontWeight: 600,\n opacity: 0.92,\n color: cfg.textColor,\n padding: `0 ${f * 0.5}px`,\n flexShrink: 0,\n lineHeight: 1.25,\n textTransform: \"uppercase\",\n letterSpacing: \"0.04em\",\n }}\n >\n <div>{cfg.period}</div>\n <div style={{ fontWeight: 700, opacity: 1 }}>{cfg.clock}</div>\n </div>\n <div style={{ flex: 1, textAlign: \"center\", minWidth: 0, padding: `0 ${f * 0.15}px` }}>\n <div\n style={{\n fontSize: \"1.05em\",\n fontWeight: 800,\n letterSpacing: \"0.02em\",\n overflow: \"hidden\",\n textOverflow: \"ellipsis\",\n whiteSpace: \"nowrap\",\n }}\n >\n {cfg.awayTeam}\n </div>\n <div style={{ fontSize: \"1.75em\", fontWeight: 900, lineHeight: 1.05, marginTop: f * 0.08 }}>\n {cfg.awayScore}\n </div>\n </div>\n </div>\n {(cfg.sponsorText || cfg.sponsorImageUrl) && (\n <div\n style={{\n fontSize: \"0.72em\",\n textAlign: \"center\",\n fontWeight: 600,\n opacity: 0.88,\n padding: `${f * 0.34}px ${f * 0.55}px`,\n borderTop: `1px solid ${cfg.accentColor}55`,\n display: \"flex\",\n alignItems: \"center\",\n justifyContent: \"center\",\n gap: f * 0.45,\n overflow: \"hidden\",\n letterSpacing: \"0.04em\",\n textTransform: \"uppercase\",\n background: `linear-gradient(180deg, rgba(0,0,0,0.12) 0%, rgba(0,0,0,0.22) 100%)`,\n }}\n >\n {cfg.sponsorImageUrl && (\n <img src={cfg.sponsorImageUrl} alt=\"sponsor\" style={{ height: `${f * 1.35}px`, objectFit: \"contain\", flexShrink: 0 }} />\n )}\n {cfg.sponsorText && (\n <span style={{ overflow: \"hidden\", textOverflow: \"ellipsis\", whiteSpace: \"nowrap\" }}>{cfg.sponsorText}</span>\n )}\n </div>\n )}\n </div>\n );\n}\n\nfunction LowerThirdOverlay({ overlay, size }: { overlay: SwirlOverlay; size: OverlaySize }) {\n const cfg = parseConfig<LowerThirdCfg>(overlay.content);\n if (!cfg) return null;\n const f = Math.max(6, size.w * 0.05);\n const accentH = Math.max(3, size.h * 0.038);\n return (\n <div\n style={{\n width: \"100%\",\n height: \"100%\",\n borderRadius: Math.max(2, size.w * 0.018),\n display: \"flex\",\n flexDirection: \"column\",\n justifyContent: \"flex-end\",\n background: cfg.backgroundColor,\n color: cfg.textColor,\n fontFamily: \"'Inter', 'Roboto', 'Segoe UI', system-ui, sans-serif\",\n overflow: \"hidden\",\n pointerEvents: \"none\",\n userSelect: \"none\",\n fontSize: `${f}px`,\n boxShadow: \"0 10px 32px rgba(0,0,0,0.4)\",\n }}\n >\n <div style={{ width: \"100%\", height: accentH, backgroundColor: cfg.accentColor, flexShrink: 0 }} />\n <div\n style={{\n flex: 1,\n display: \"flex\",\n flexDirection: \"column\",\n justifyContent: \"center\",\n padding: `${f * 0.62}px ${f * 1.05}px`,\n minHeight: 0,\n gap: f * 0.18,\n }}\n >\n <div\n style={{\n fontSize: \"1.28em\",\n fontWeight: 800,\n lineHeight: 1.22,\n letterSpacing: \"0.01em\",\n textShadow: \"0 2px 6px rgba(0,0,0,0.55)\",\n }}\n >\n {cfg.headline}\n </div>\n <div\n style={{\n fontSize: \"0.94em\",\n fontWeight: 500,\n opacity: 0.92,\n lineHeight: 1.35,\n color: cfg.textColor,\n }}\n >\n {cfg.subtitle}\n </div>\n </div>\n {(cfg.sponsorText || cfg.sponsorImageUrl) && (\n <div\n style={{\n fontSize: \"0.68em\",\n fontWeight: 600,\n opacity: 0.72,\n padding: `${f * 0.28}px ${f * 1.05}px ${f * 0.48}px`,\n display: \"flex\",\n alignItems: \"center\",\n gap: f * 0.4,\n overflow: \"hidden\",\n letterSpacing: \"0.06em\",\n textTransform: \"uppercase\",\n borderTop: `1px solid rgba(255,255,255,0.08)`,\n }}\n >\n {cfg.sponsorImageUrl && (\n <img src={cfg.sponsorImageUrl} alt=\"sponsor\" style={{ height: `${f * 1.25}px`, objectFit: \"contain\", flexShrink: 0 }} />\n )}\n {cfg.sponsorText && (\n <span style={{ overflow: \"hidden\", textOverflow: \"ellipsis\", whiteSpace: \"nowrap\" }}>{cfg.sponsorText}</span>\n )}\n </div>\n )}\n </div>\n );\n}\n\nfunction QrCodeOverlay({ overlay, size }: { overlay: SwirlOverlay; size: OverlaySize }) {\n const cfg = parseConfig<QrCodeCfg>(overlay.content);\n if (!cfg) return null;\n const qrSide = Math.max(32, Math.min(size.w, size.h) * 0.55);\n const qrUrl = `https://api.qrserver.com/v1/create-qr-code/?size=${Math.round(qrSide * 2)}x${Math.round(qrSide * 2)}&data=${encodeURIComponent(cfg.url || \"https://example.com\")}`;\n const f = Math.max(6, size.w * 0.06);\n return (\n <div style={{ width: \"100%\", height: \"100%\", borderRadius: Math.max(2, size.w * 0.035), display: \"flex\", flexDirection: \"column\", alignItems: \"center\", justifyContent: \"center\", gap: f * 0.4, background: cfg.backgroundColor, color: cfg.textColor, fontFamily: \"Roboto, 'Segoe UI', Arial, sans-serif\", padding: f * 0.6, boxSizing: \"border-box\", pointerEvents: \"none\", userSelect: \"none\", overflow: \"hidden\" }}>\n <div style={{ flexShrink: 0, background: \"#fff\", borderRadius: Math.max(2, qrSide * 0.06), padding: Math.max(2, qrSide * 0.06), lineHeight: 0 }}>\n <img src={qrUrl} alt=\"QR Code\" style={{ width: `${qrSide}px`, height: `${qrSide}px`, display: \"block\" }} />\n </div>\n <div style={{ fontSize: `${f * 1.1}px`, fontWeight: 700, textAlign: \"center\", color: cfg.accentColor, overflow: \"hidden\", textOverflow: \"ellipsis\", whiteSpace: \"nowrap\", width: \"100%\" }}>{cfg.ctaText}</div>\n {cfg.description && <div style={{ fontSize: `${f * 0.75}px`, opacity: 0.6, textAlign: \"center\", overflow: \"hidden\", textOverflow: \"ellipsis\", whiteSpace: \"nowrap\", width: \"100%\" }}>{cfg.description}</div>}\n </div>\n );\n}\n\nfunction ComingUpNextOverlay({ overlay, size }: { overlay: SwirlOverlay; size: OverlaySize }) {\n const cfg = parseConfig<ComingUpNextCfg>(overlay.content);\n if (!cfg) return null;\n const f = Math.max(8, Math.min(size.h * 0.13, size.w * 0.048));\n const showSubtitle = size.h >= 60;\n const showThumbnail = false;\n const thumbW = 0;\n return (\n <div style={{\n width: \"100%\", height: \"100%\",\n borderRadius: Math.max(2, size.w * 0.025),\n display: \"flex\",\n background: cfg.backgroundColor,\n color: cfg.textColor,\n fontFamily: \"'Arial', 'Helvetica Neue', Helvetica, sans-serif\",\n overflow: \"hidden\",\n pointerEvents: \"none\", userSelect: \"none\",\n fontSize: `${f}px`,\n WebkitFontSmoothing: \"antialiased\",\n } as React.CSSProperties}>\n <div style={{ width: Math.max(3, size.w * 0.018), flexShrink: 0, backgroundColor: cfg.accentColor }} />\n <div style={{\n flex: 1, display: \"flex\", flexDirection: \"column\", justifyContent: \"center\",\n padding: `${f * 0.35}px ${f * 0.75}px`,\n minWidth: 0, gap: `${f * 0.08}px`,\n }}>\n <div style={{\n fontSize: \"0.7em\", fontWeight: 700,\n textTransform: \"uppercase\", letterSpacing: \"0.09em\",\n color: cfg.accentColor, lineHeight: 1,\n }}>\n Coming Up Next\n </div>\n <div style={{\n fontSize: \"1.2em\", fontWeight: 700, lineHeight: 1.2,\n overflow: \"hidden\", textOverflow: \"ellipsis\", whiteSpace: \"nowrap\",\n }}>\n {cfg.title}\n </div>\n {showSubtitle && (\n <div style={{\n fontSize: \"0.82em\", opacity: 0.65, lineHeight: 1.1,\n overflow: \"hidden\", textOverflow: \"ellipsis\", whiteSpace: \"nowrap\",\n }}>\n {cfg.subtitle}\n </div>\n )}\n {cfg.scheduledTime && (\n <div style={{\n fontSize: \"0.9em\", fontWeight: 700,\n color: cfg.accentColor, lineHeight: 1,\n }}>\n {cfg.scheduledTime}\n </div>\n )}\n </div>\n {showThumbnail && (\n <div style={{ flexShrink: 0, width: `${thumbW}px`, overflow: \"hidden\" }}>\n <img\n src={cfg.thumbnailUrl}\n alt=\"\"\n style={{ width: \"100%\", height: \"100%\", objectFit: \"cover\", display: \"block\" }}\n />\n </div>\n )}\n </div>\n );\n}\n\nfunction ContextualTriggerOverlay({ overlay, size }: { overlay: SwirlOverlay; size: OverlaySize }) {\n const cfg = parseConfig<ContextualTriggerCfg>(overlay.content);\n if (!cfg) return null;\n const icons: Record<string, string> = { alert: \"\\u26A0\\uFE0F\", celebration: \"\\uD83C\\uDF89\", info: \"\\u2139\\uFE0F\", warning: \"\\uD83D\\uDD14\" };\n const f = Math.max(6, size.w * 0.05);\n return (\n <div style={{ width: \"100%\", height: \"100%\", borderRadius: Math.max(2, size.w * 0.035), display: \"flex\", alignItems: \"center\", gap: f * 0.8, padding: `0 ${f * 1.2}px`, background: cfg.backgroundColor, color: cfg.textColor, fontFamily: \"Roboto, 'Segoe UI', Arial, sans-serif\", borderLeft: `${Math.max(2, size.w * 0.02)}px solid ${cfg.accentColor}`, boxSizing: \"border-box\", pointerEvents: \"none\", userSelect: \"none\", fontSize: `${f}px` }}>\n <span style={{ fontSize: \"2em\", flexShrink: 0 }}>{icons[cfg.iconType] || \"\\u26A1\"}</span>\n <div style={{ flex: 1, minWidth: 0 }}>\n <div style={{ fontSize: \"1.3em\", fontWeight: 700, overflow: \"hidden\", textOverflow: \"ellipsis\", whiteSpace: \"nowrap\" }}>{cfg.headline}</div>\n <div style={{ fontSize: \"0.9em\", opacity: 0.7, overflow: \"hidden\", textOverflow: \"ellipsis\", whiteSpace: \"nowrap\" }}>{cfg.message}</div>\n </div>\n </div>\n );\n}\n\nfunction OddsBettingOverlay({ overlay, size }: { overlay: SwirlOverlay; size: OverlaySize }) {\n const cfg = parseConfig<OddsBettingCfg>(overlay.content);\n if (!cfg) return null;\n const f = Math.max(6, size.w * 0.052);\n return (\n <div style={{ width: \"100%\", height: \"100%\", borderRadius: Math.max(2, size.w * 0.035), display: \"flex\", flexDirection: \"column\", padding: f * 0.8, background: cfg.backgroundColor, color: cfg.textColor, fontFamily: \"Roboto, 'Segoe UI', Arial, sans-serif\", boxSizing: \"border-box\", pointerEvents: \"none\", userSelect: \"none\", fontSize: `${f}px` }}>\n <div style={{ fontSize: \"0.9em\", fontWeight: 700, textTransform: \"uppercase\", letterSpacing: \"0.05em\", color: cfg.accentColor, marginBottom: f * 0.4 }}>{cfg.eventTitle}</div>\n <div style={{ flex: 1, display: \"flex\", flexDirection: \"column\", gap: f * 0.2, justifyContent: \"center\" }}>\n {(cfg.options || []).slice(0, 5).map((opt, i) => (\n <div key={i} style={{ display: \"flex\", justifyContent: \"space-between\", alignItems: \"center\", padding: `${f * 0.2}px ${f * 0.6}px`, borderRadius: Math.max(2, f * 0.3), background: `${cfg.accentColor}15`, fontSize: \"1em\" }}>\n <span style={{ overflow: \"hidden\", textOverflow: \"ellipsis\", whiteSpace: \"nowrap\", flex: 1 }}>{opt.label}</span>\n <span style={{ fontWeight: 700, marginLeft: f * 0.8, flexShrink: 0, color: cfg.accentColor }}>{opt.odds}</span>\n </div>\n ))}\n </div>\n {cfg.sponsorText && <div style={{ fontSize: \"0.7em\", opacity: 0.4, textAlign: \"center\", marginTop: f * 0.4 }}>{cfg.sponsorText}</div>}\n </div>\n );\n}\n\nfunction BreakingNewsOverlay({ overlay, size }: { overlay: SwirlOverlay; size: OverlaySize }) {\n const cfg = parseConfig<BreakingNewsCfg>(overlay.content);\n if (!cfg) return null;\n const urgencyColors: Record<string, string> = { breaking: \"#dc2626\", urgent: \"#ea580c\", normal: \"#2563eb\" };\n const labelBg = urgencyColors[cfg.urgency] || urgencyColors.normal;\n const label = cfg.urgency === \"breaking\" ? \"BREAKING\" : cfg.urgency === \"urgent\" ? \"URGENT\" : \"NEWS\";\n const f = Math.max(6, size.w * 0.05);\n return (\n <div style={{ width: \"100%\", height: \"100%\", borderRadius: Math.max(2, size.w * 0.02), display: \"flex\", alignItems: \"center\", background: cfg.backgroundColor, color: cfg.textColor, fontFamily: \"Roboto, 'Segoe UI', Arial, sans-serif\", overflow: \"hidden\", pointerEvents: \"none\", userSelect: \"none\", fontSize: `${f}px` }}>\n <div style={{ padding: `0 ${f * 0.8}px`, height: \"100%\", display: \"flex\", alignItems: \"center\", background: labelBg, color: \"#fff\", fontSize: \"1em\", fontWeight: 900, textTransform: \"uppercase\", letterSpacing: \"0.05em\", flexShrink: 0 }}>{label}</div>\n <div style={{ flex: 1, padding: `0 ${f * 1.0}px`, minWidth: 0 }}>\n <div style={{ fontSize: \"1.3em\", fontWeight: 700, overflow: \"hidden\", textOverflow: \"ellipsis\", whiteSpace: \"nowrap\" }}>{cfg.headline}</div>\n {cfg.body && <div style={{ fontSize: \"0.9em\", opacity: 0.7, overflow: \"hidden\", textOverflow: \"ellipsis\", whiteSpace: \"nowrap\" }}>{cfg.body}</div>}\n </div>\n </div>\n );\n}\n\nfunction calcCountdownRemaining(targetTime: string) {\n const diff = Math.max(0, new Date(targetTime).getTime() - Date.now());\n return {\n d: Math.floor(diff / 86400000),\n h: Math.floor((diff % 86400000) / 3600000),\n m: Math.floor((diff % 3600000) / 60000),\n s: Math.floor((diff % 60000) / 1000),\n expired: diff === 0,\n };\n}\n\nfunction CountdownOverlay({ overlay, size }: { overlay: SwirlOverlay; size: OverlaySize }) {\n const cfg = parseConfig<CountdownCfg>(overlay.content);\n const targetTime = cfg?.targetTime ?? \"\";\n const [remaining, setRemaining] = useState(() =>\n targetTime ? calcCountdownRemaining(targetTime) : { d: 0, h: 0, m: 0, s: 0, expired: false }\n );\n\n useEffect(() => {\n if (!targetTime) return;\n setRemaining(calcCountdownRemaining(targetTime));\n const id = setInterval(() => setRemaining(calcCountdownRemaining(targetTime)), 1000);\n return () => clearInterval(id);\n }, [targetTime]);\n\n if (!cfg) return null;\n\n const f = Math.max(6, size.w * 0.055);\n const pad = (n: number) => String(n).padStart(2, \"0\");\n const units: Array<{ show: boolean; value: string; label: string }> = [\n { show: cfg.showDays, value: pad(remaining.d), label: \"DAYS\" },\n { show: cfg.showHours, value: pad(remaining.h), label: \"HRS\" },\n { show: cfg.showMinutes, value: pad(remaining.m), label: \"MIN\" },\n { show: cfg.showSeconds, value: pad(remaining.s), label: \"SEC\" },\n ];\n\n return (\n <div style={{ width: \"100%\", height: \"100%\", borderRadius: Math.max(2, size.w * 0.035), display: \"flex\", flexDirection: \"column\", alignItems: \"center\", justifyContent: \"center\", padding: f * 0.8, background: cfg.backgroundColor, color: cfg.textColor, fontFamily: \"Roboto, 'Segoe UI', Arial, sans-serif\", boxSizing: \"border-box\", pointerEvents: \"none\", userSelect: \"none\", fontSize: `${f}px` }}>\n <div style={{ fontSize: \"0.8em\", fontWeight: 600, textTransform: \"uppercase\", letterSpacing: \"0.05em\", color: cfg.accentColor, marginBottom: f * 0.4 }}>{cfg.eventName}</div>\n {remaining.expired ? (\n <div style={{ fontSize: \"1em\", fontWeight: 700, opacity: 0.6 }}>{cfg.message || \"Event ended\"}</div>\n ) : (\n <div style={{ display: \"flex\", gap: f * 0.6, alignItems: \"center\" }}>\n {units.filter(u => u.show).map((u, i, arr) => (\n <React.Fragment key={u.label}>\n <div style={{ textAlign: \"center\" }}>\n <div style={{ fontSize: \"2em\", fontWeight: 900, lineHeight: 1, borderRadius: Math.max(2, f * 0.4), padding: `${f * 0.2}px ${f * 0.4}px`, background: `${cfg.accentColor}20` }}>{u.value}</div>\n <div style={{ fontSize: \"0.5em\", opacity: 0.5, marginTop: f * 0.2 }}>{u.label}</div>\n </div>\n {i < arr.length - 1 && <div style={{ fontSize: \"1.8em\", fontWeight: 700, opacity: 0.3 }}>:</div>}\n </React.Fragment>\n ))}\n </div>\n )}\n {!remaining.expired && cfg.message && <div style={{ fontSize: \"0.8em\", opacity: 0.6, marginTop: f * 0.4, textAlign: \"center\" }}>{cfg.message}</div>}\n </div>\n );\n}\n\nfunction ShapeOverlay({ overlay, size }: { overlay: SwirlOverlay; size: OverlaySize }) {\n const f = Math.max(6, size.w * 0.05);\n return (\n <div style={{ width: \"100%\", height: \"100%\", borderRadius: Math.max(2, size.w * 0.03), background: \"rgba(99, 102, 241, 0.2)\", border: \"2px solid rgba(99, 102, 241, 0.4)\", display: \"flex\", alignItems: \"center\", justifyContent: \"center\", pointerEvents: \"none\", userSelect: \"none\" }}>\n <div style={{ fontSize: `${f}px`, fontWeight: 500, color: \"rgba(163, 163, 163, 0.8)\", textTransform: \"uppercase\" }}>{overlay.name}</div>\n </div>\n );\n}\n\nfunction hexToRgb(hex: string): string {\n if (!hex || !hex.startsWith(\"#\")) return \"0,0,0\";\n const clean = hex.slice(1);\n const num = parseInt(clean.length === 3 ? clean.replace(/./g, \"$&$&\") : clean, 16);\n return `${(num >> 16) & 255},${(num >> 8) & 255},${num & 255}`;\n}\n\ninterface OverlayFadeState {\n overlay: SwirlOverlay;\n visible: boolean;\n}\n\nconst FADE_DURATION_MS = 1000;\n\nconst SHOWCASE_CYCLE_MS_DEFAULT = 36_000;\nconst SHOWCASE_MIN_BEAT_MS = 6_000;\nconst BEAT_POP_IN_MS = 520;\nconst BEAT_POP_OUT_MS = 520;\n\nconst SHOWCASE_PERSISTENT_TYPES: ReadonlySet<string> = new Set([\n \"scroller\",\n \"breaking_news\",\n \"image\",\n \"text\",\n \"shape\",\n \"countdown\",\n \"qr_code\",\n]);\n\nfunction easeOutCubic(t: number): number {\n const u = 1 - t;\n return 1 - u * u * u;\n}\n\nfunction easeInCubic(t: number): number {\n return t * t * t;\n}\n\nfunction easeOutBack(t: number): number {\n const c1 = 1.70158;\n const c3 = c1 + 1;\n return 1 + c3 * (t - 1) ** 3 + c1 * (t - 1) ** 2;\n}\n\nfunction beatOpacity(phase: number, beatMs: number): number {\n if (phase < BEAT_POP_IN_MS) return easeOutCubic(phase / BEAT_POP_IN_MS);\n const holdEnd = beatMs - BEAT_POP_OUT_MS;\n if (phase >= holdEnd) {\n return Math.max(\n 0,\n 1 - easeInCubic((phase - holdEnd) / BEAT_POP_OUT_MS)\n );\n }\n return 1;\n}\n\nfunction beatScale(phase: number, beatMs: number): number {\n if (phase < BEAT_POP_IN_MS) {\n return Math.min(1, 0.92 + 0.08 * easeOutBack(phase / BEAT_POP_IN_MS));\n }\n const holdEnd = beatMs - BEAT_POP_OUT_MS;\n if (phase >= holdEnd) {\n return 1 - 0.06 * easeInCubic((phase - holdEnd) / BEAT_POP_OUT_MS);\n }\n return 1;\n}\n\nfunction partitionShowcase(overlays: SwirlOverlay[]): {\n persistent: SwirlOverlay[];\n beats: SwirlOverlay[][];\n} {\n const visible = overlays.filter((o) => o.visible);\n const persistent: SwirlOverlay[] = [];\n const spotlight: SwirlOverlay[] = [];\n for (const o of visible) {\n if (SHOWCASE_PERSISTENT_TYPES.has(o.type)) persistent.push(o);\n else spotlight.push(o);\n }\n spotlight.sort(\n (a, b) => (a.z_index - b.z_index) || (a.id - b.id)\n );\n const beats: SwirlOverlay[][] = [];\n for (const o of spotlight) beats.push([o]);\n return { persistent, beats };\n}\n\nfunction useShowcaseBeat(\n enabled: boolean,\n cycleMs: number,\n beats: number\n): { beatIndex: number; beatPhase: number; beatMs: number } {\n const safeBeats = Math.max(1, beats);\n const beatMs = Math.max(\n SHOWCASE_MIN_BEAT_MS,\n Math.floor(cycleMs / safeBeats)\n );\n const totalMs = beatMs * safeBeats;\n const [now, setNow] = useState<number>(() => Date.now());\n useEffect(() => {\n if (!enabled) return;\n const id = window.setInterval(() => setNow(Date.now()), 64);\n return () => clearInterval(id);\n }, [enabled]);\n const phase = now % totalMs;\n const beatIndex = Math.min(safeBeats - 1, Math.floor(phase / beatMs));\n const beatPhase = phase - beatIndex * beatMs;\n return { beatIndex, beatPhase, beatMs };\n}\n\nexport const OverlayRenderer: React.FC<OverlayRendererProps> = ({\n overlays,\n currentTime,\n videoRef,\n coordinateSpace,\n showcaseMode = false,\n showcaseCycleMs = SHOWCASE_CYCLE_MS_DEFAULT,\n}) => {\n const [dims, setDims] = useState<VideoDimensions | null>(null);\n const rafRef = useRef<number | null>(null);\n const [fadeMap, setFadeMap] = useState<Map<number, OverlayFadeState>>(new Map());\n const removeTimers = useRef<Map<number, ReturnType<typeof setTimeout>>>(new Map());\n\n const { persistent: showcasePersistent, beats: showcaseBeats } = useMemo(\n () => partitionShowcase(overlays),\n [overlays]\n );\n const {\n beatIndex: showcaseBeatIndex,\n beatPhase: showcaseBeatPhase,\n beatMs: showcaseBeatMs,\n } = useShowcaseBeat(showcaseMode, showcaseCycleMs, showcaseBeats.length);\n\n const showcasePersistentIds = useMemo(\n () => new Set(showcasePersistent.map((o) => o.id)),\n [showcasePersistent]\n );\n const showcaseSpotlightIds = useMemo(() => {\n const beat = showcaseBeats[showcaseBeatIndex];\n return new Set((beat ?? []).map((o) => o.id));\n }, [showcaseBeats, showcaseBeatIndex]);\n\n const updateDims = useCallback(() => {\n const video = videoRef.current;\n if (video) {\n const computed = computeVideoDimensions(video);\n setDims((prev) => {\n if (\n !computed ||\n (prev &&\n prev.nativeWidth === computed.nativeWidth &&\n prev.nativeHeight === computed.nativeHeight &&\n prev.displayWidth === computed.displayWidth &&\n prev.displayHeight === computed.displayHeight &&\n prev.offsetX === computed.offsetX &&\n prev.offsetY === computed.offsetY)\n ) {\n return prev;\n }\n return computed;\n });\n }\n }, [videoRef]);\n\n useEffect(() => {\n updateDims();\n const interval = setInterval(updateDims, 500);\n\n const handleResize = () => {\n if (rafRef.current) cancelAnimationFrame(rafRef.current);\n rafRef.current = requestAnimationFrame(updateDims);\n };\n window.addEventListener(\"resize\", handleResize);\n\n return () => {\n clearInterval(interval);\n window.removeEventListener(\"resize\", handleResize);\n if (rafRef.current) cancelAnimationFrame(rafRef.current);\n };\n }, [updateDims]);\n\n const activeOverlays = useMemo(() => {\n if (showcaseMode) {\n const beat = showcaseBeats[showcaseBeatIndex] ?? [];\n return [...showcasePersistent, ...beat];\n }\n return overlays.filter((o) => isOverlayActive(o, currentTime));\n }, [\n overlays,\n currentTime,\n showcaseMode,\n showcasePersistent,\n showcaseBeats,\n showcaseBeatIndex,\n ]);\n\n useLayoutEffect(() => {\n const activeIds = new Set(activeOverlays.map((o) => o.id));\n\n setFadeMap((prev) => {\n const next = new Map(prev);\n\n for (const overlay of activeOverlays) {\n if (!next.has(overlay.id)) {\n next.set(overlay.id, { overlay, visible: false });\n } else {\n const existing = next.get(overlay.id)!;\n next.set(overlay.id, { ...existing, overlay });\n }\n }\n\n for (const [id, state] of next) {\n if (!activeIds.has(id) && state.visible) {\n next.set(id, { ...state, visible: false });\n if (!removeTimers.current.has(id)) {\n const timer = setTimeout(() => {\n setFadeMap((m) => {\n const updated = new Map(m);\n updated.delete(id);\n return updated;\n });\n removeTimers.current.delete(id);\n }, FADE_DURATION_MS);\n removeTimers.current.set(id, timer);\n }\n } else if (!activeIds.has(id) && !state.visible) {\n }\n }\n\n return next;\n });\n }, [activeOverlays]);\n\n useEffect(() => {\n const toFadeIn: number[] = [];\n for (const [id, state] of fadeMap) {\n if (!state.visible) {\n const isActive = activeOverlays.some((o) => o.id === id);\n if (isActive) toFadeIn.push(id);\n }\n }\n if (toFadeIn.length === 0) return;\n\n const raf = requestAnimationFrame(() => {\n setFadeMap((prev) => {\n const next = new Map(prev);\n for (const id of toFadeIn) {\n const state = next.get(id);\n if (state) next.set(id, { ...state, visible: true });\n }\n return next;\n });\n });\n return () => cancelAnimationFrame(raf);\n }, [fadeMap, activeOverlays]);\n\n useEffect(() => {\n return () => {\n for (const timer of removeTimers.current.values()) clearTimeout(timer);\n };\n }, []);\n\n if (!dims || fadeMap.size === 0) return null;\n\n return (\n <div\n aria-hidden=\"true\"\n style={{\n position: \"absolute\",\n left: `${dims.offsetX}px`,\n top: `${dims.offsetY}px`,\n width: `${dims.displayWidth}px`,\n height: `${dims.displayHeight}px`,\n pointerEvents: \"none\",\n overflow: \"hidden\",\n zIndex: 8,\n }}\n >\n {[...fadeMap.values()].map(({ overlay, visible }) => {\n const uniform =\n coordinateSpace &&\n coordinateSpace.width > 0 &&\n coordinateSpace.height > 0;\n const auth = uniform\n ? { width: coordinateSpace.width, height: coordinateSpace.height }\n : overlayAuthoringDimensions(\n overlay,\n dims.nativeWidth,\n dims.nativeHeight\n );\n const scaleX = dims.displayWidth / auth.width;\n const scaleY = dims.displayHeight / auth.height;\n const left = overlay.x * scaleX;\n const top = overlay.y * scaleY;\n const width = overlay.width * scaleX;\n const height = overlay.height * scaleY;\n const baseOpacity = Math.max(0, Math.min(100, overlay.opacity)) / 100;\n const isShowcasePersistent =\n showcaseMode && showcasePersistentIds.has(overlay.id);\n const isShowcaseSpotlight =\n showcaseMode && showcaseSpotlightIds.has(overlay.id);\n\n const isShowcaseDormant =\n showcaseMode && !isShowcasePersistent && !isShowcaseSpotlight;\n const useShowcasePop = isShowcaseSpotlight;\n const opacity = useShowcasePop\n ? baseOpacity *\n beatOpacity(showcaseBeatPhase, showcaseBeatMs)\n : isShowcaseDormant\n ? 0\n : visible\n ? baseOpacity\n : 0;\n const popScale = useShowcasePop\n ? beatScale(showcaseBeatPhase, showcaseBeatMs)\n : 1;\n const sz: OverlaySize = { w: width, h: height };\n\n return (\n <div\n key={overlay.id}\n style={{\n position: \"absolute\",\n left: `${left}px`,\n top: `${top}px`,\n width: `${width}px`,\n height: `${height}px`,\n opacity,\n transition: useShowcasePop\n ? \"none\"\n : `opacity ${FADE_DURATION_MS}ms ease`,\n transform: showcaseMode && useShowcasePop ? `scale(${popScale})` : undefined,\n transformOrigin: showcaseMode && useShowcasePop ? \"center center\" : undefined,\n zIndex: overlay.z_index,\n overflow: \"hidden\",\n }}\n >\n {overlay.type === \"image\" && <ImageOverlay overlay={overlay} />}\n {overlay.type === \"text\" && <TextOverlay overlay={overlay} />}\n {overlay.type === \"scroller\" && <ScrollerOverlay overlay={overlay} />}\n {overlay.type === \"shape\" && <ShapeOverlay overlay={overlay} size={sz} />}\n {overlay.type === \"score_bug\" && <ScoreBugOverlay overlay={overlay} size={sz} />}\n {overlay.type === \"lower_third\" && <LowerThirdOverlay overlay={overlay} size={sz} />}\n {overlay.type === \"qr_code\" && <QrCodeOverlay overlay={overlay} size={sz} />}\n {overlay.type === \"coming_up_next\" && <ComingUpNextOverlay overlay={overlay} size={sz} />}\n {overlay.type === \"contextual_trigger\" && <ContextualTriggerOverlay overlay={overlay} size={sz} />}\n {overlay.type === \"odds_betting\" && <OddsBettingOverlay overlay={overlay} size={sz} />}\n {overlay.type === \"breaking_news\" && <BreakingNewsOverlay overlay={overlay} size={sz} />}\n {overlay.type === \"countdown\" && <CountdownOverlay overlay={overlay} size={sz} />}\n </div>\n );\n })}\n </div>\n );\n};\n","export const OVERLAY_API_BASE = \"https://adstorm.co/api-adstorm-dev\";\n\nexport interface OverlayCoordinateSpace {\n width: number;\n height: number;\n}\n\nexport interface SwirlScrollerConfig {\n rss_url?: string;\n update_interval?: number;\n scroll_speed?: number;\n direction?: string;\n font_size?: number;\n font_family?: string;\n font_weight?: string;\n text_color?: string;\n background_color?: string;\n background_opacity?: number;\n border_color?: string;\n border_width?: number;\n border_radius?: number;\n padding?: number;\n margin?: number;\n show_title?: boolean;\n show_description?: boolean;\n show_timestamp?: boolean;\n show_author?: boolean;\n show_category?: boolean;\n max_items?: number;\n item_spacing?: number;\n fade_in_out?: boolean;\n fade_distance?: number;\n auto_refresh?: boolean;\n use_custom_text?: boolean;\n custom_text?: string;\n label?: string;\n label_line2?: string;\n label_color?: string;\n label_text_color?: string;\n accent_color?: string;\n show_accent_line?: boolean;\n separator_char?: string;\n preset?: string;\n}\n\nexport type SwirlOverlayType =\n | \"image\"\n | \"text\"\n | \"scroller\"\n | \"shape\"\n | \"score_bug\"\n | \"lower_third\"\n | \"qr_code\"\n | \"coming_up_next\"\n | \"contextual_trigger\"\n | \"odds_betting\"\n | \"breaking_news\"\n | \"countdown\";\n\nexport interface SwirlOverlay {\n id: number;\n project_id: number;\n name: string;\n type: SwirlOverlayType | string;\n visible: boolean;\n x: number;\n y: number;\n width: number;\n height: number;\n opacity: number;\n start_time: string;\n duration: string;\n content?: string;\n image_url?: string;\n scroller_config?: SwirlScrollerConfig;\n z_index: number;\n created_at?: string;\n updated_at?: string;\n}\n\nexport function timeStringToSeconds(timeStr: string): number {\n if (!timeStr) return 0;\n\n const parts = timeStr.split(\":\");\n\n if (parts.length >= 3) {\n const hours = parseInt(parts[0] ?? \"0\", 10) || 0;\n const minutes = parseInt(parts[1] ?? \"0\", 10) || 0;\n const secStr = parts[2] ?? \"0\";\n const dotIdx = secStr.indexOf(\".\");\n const seconds =\n parseInt(dotIdx >= 0 ? secStr.substring(0, dotIdx) : secStr, 10) || 0;\n const msFrag = dotIdx >= 0 ? secStr.substring(dotIdx + 1) : \"\";\n const ms = msFrag ? parseInt(msFrag.padEnd(3, \"0\").substring(0, 3), 10) || 0 : 0;\n return hours * 3600 + minutes * 60 + seconds + ms / 1000;\n }\n\n if (parts.length === 2) {\n const minutes = parseInt(parts[0] ?? \"0\", 10) || 0;\n const secStr = parts[1] ?? \"0\";\n const dotIdx = secStr.indexOf(\".\");\n const seconds =\n parseInt(dotIdx >= 0 ? secStr.substring(0, dotIdx) : secStr, 10) || 0;\n const msFrag = dotIdx >= 0 ? secStr.substring(dotIdx + 1) : \"\";\n const ms = msFrag ? parseInt(msFrag.padEnd(3, \"0\").substring(0, 3), 10) || 0 : 0;\n return minutes * 60 + seconds + ms / 1000;\n }\n\n const num = parseFloat(timeStr);\n return isFinite(num) ? Math.max(0, num) : 0;\n}\n\nexport function isOverlayActive(\n overlay: SwirlOverlay,\n currentTime: number\n): boolean {\n if (!overlay.visible) return false;\n const startSec = timeStringToSeconds(overlay.start_time);\n const durationSec = timeStringToSeconds(overlay.duration);\n if (durationSec <= 0) return false;\n return currentTime >= startSec && currentTime < startSec + durationSec;\n}\n\nexport const SWIRL_HD_AUTHORING_WIDTH = 1920;\nexport const SWIRL_HD_AUTHORING_HEIGHT = 1080;\n\nconst NAB_DEMO_NAME_PREFIX = \"NAB Demo — \";\n\nexport function overlayAuthoringDimensions(\n overlay: SwirlOverlay,\n decodeWidth: number,\n decodeHeight: number\n): { width: number; height: number } {\n if (overlay.name.startsWith(NAB_DEMO_NAME_PREFIX)) {\n return {\n width: SWIRL_HD_AUTHORING_WIDTH,\n height: SWIRL_HD_AUTHORING_HEIGHT,\n };\n }\n if (!decodeWidth || !decodeHeight) {\n return {\n width: decodeWidth || SWIRL_HD_AUTHORING_WIDTH,\n height: decodeHeight || SWIRL_HD_AUTHORING_HEIGHT,\n };\n }\n return { width: decodeWidth, height: decodeHeight };\n}\n\nfunction overlayExtents(overlays: SwirlOverlay[]): { maxR: number; maxB: number } {\n let maxR = 0;\n let maxB = 0;\n for (const o of overlays) {\n if (!o.visible) continue;\n maxR = Math.max(maxR, o.x + o.width);\n maxB = Math.max(maxB, o.y + o.height);\n }\n return { maxR, maxB };\n}\n\nexport function swirlProjectHasNabDemoMixedWithOther(overlays: SwirlOverlay[]): boolean {\n let hasNab = false;\n let hasOther = false;\n for (const o of overlays) {\n if (!o.visible) continue;\n if (o.name.startsWith(NAB_DEMO_NAME_PREFIX)) hasNab = true;\n else hasOther = true;\n }\n return hasNab && hasOther;\n}\n\nconst STANDARD_16_9_LADDER: ReadonlyArray<OverlayCoordinateSpace> = [\n { width: 384, height: 216 },\n { width: 640, height: 360 },\n { width: 854, height: 480 },\n { width: 1280, height: 720 },\n { width: 1920, height: 1080 },\n { width: 2560, height: 1440 },\n { width: 3840, height: 2160 },\n];\n\nfunction smallestLadderRungContaining(\n maxR: number,\n maxB: number,\n eps: number\n): OverlayCoordinateSpace | null {\n for (const rung of STANDARD_16_9_LADDER) {\n if (maxR <= rung.width + eps && maxB <= rung.height + eps) return rung;\n }\n return null;\n}\n\nexport function inferSwirlOverlayCoordinateSpace(\n overlays: SwirlOverlay[],\n videoWidth: number,\n videoHeight: number\n): OverlayCoordinateSpace {\n if (!videoWidth || !videoHeight) {\n return {\n width: videoWidth || SWIRL_HD_AUTHORING_WIDTH,\n height: videoHeight || SWIRL_HD_AUTHORING_HEIGHT,\n };\n }\n if (!overlays.length) {\n return { width: videoWidth, height: videoHeight };\n }\n\n const { maxR, maxB } = overlayExtents(overlays);\n const EPS = 1;\n\n if (maxR <= 0 || maxB <= 0) {\n return { width: videoWidth, height: videoHeight };\n }\n\n const mixed = swirlProjectHasNabDemoMixedWithOther(overlays);\n const fitsHdCanvas =\n maxR <= SWIRL_HD_AUTHORING_WIDTH + EPS &&\n maxB <= SWIRL_HD_AUTHORING_HEIGHT + EPS;\n const exceedsDecode = maxR > videoWidth + EPS || maxB > videoHeight + EPS;\n const decodeLargerThanHd =\n videoWidth > SWIRL_HD_AUTHORING_WIDTH + EPS ||\n videoHeight > SWIRL_HD_AUTHORING_HEIGHT + EPS;\n\n if (fitsHdCanvas && (decodeLargerThanHd || (exceedsDecode && !mixed))) {\n return {\n width: SWIRL_HD_AUTHORING_WIDTH,\n height: SWIRL_HD_AUTHORING_HEIGHT,\n };\n }\n\n if (mixed) {\n return { width: videoWidth, height: videoHeight };\n }\n\n const rung = smallestLadderRungContaining(maxR, maxB, EPS);\n if (rung && rung.width < videoWidth - EPS && rung.height < videoHeight - EPS) {\n return { width: rung.width, height: rung.height };\n }\n\n return { width: videoWidth, height: videoHeight };\n}\n\nfunction normalizeScrollerConfig(\n raw: SwirlScrollerConfig | Record<string, unknown> | undefined\n): SwirlScrollerConfig | undefined {\n if (!raw || typeof raw !== \"object\") return undefined;\n const r = raw as Record<string, unknown>;\n const merged: SwirlScrollerConfig = { ...(raw as SwirlScrollerConfig) };\n if (merged.use_custom_text === undefined && typeof r.useCustomText === \"boolean\") {\n merged.use_custom_text = r.useCustomText;\n }\n if ((merged.custom_text === undefined || merged.custom_text === \"\") && typeof r.customText === \"string\") {\n merged.custom_text = r.customText;\n }\n if (!merged.rss_url && typeof r.rssUrl === \"string\") {\n merged.rss_url = r.rssUrl;\n }\n return merged;\n}\n\nexport function normalizeSwirlOverlay(raw: SwirlOverlay & Record<string, unknown>): SwirlOverlay {\n const o = { ...raw };\n if (o.type === \"scroller\") {\n const sc = raw.scroller_config ?? (raw as Record<string, unknown>).scrollerConfig;\n const normalized = normalizeScrollerConfig(sc as SwirlScrollerConfig);\n if (normalized) o.scroller_config = normalized;\n }\n return o;\n}\n\nexport async function fetchProjectOverlays(\n projectId: number,\n apiBaseUrl: string = OVERLAY_API_BASE\n): Promise<SwirlOverlay[]> {\n const base = apiBaseUrl.replace(/\\/$/, \"\");\n const response = await fetch(\n `${base}/adstorm/swirl/projects/${projectId}/overlays`\n );\n if (!response.ok) {\n throw new Error(\n `Failed to fetch overlays: ${response.status} ${response.statusText}`\n );\n }\n const data = await response.json();\n if (!Array.isArray(data)) return [];\n return data.map((row: SwirlOverlay & Record<string, unknown>) =>\n normalizeSwirlOverlay(row)\n );\n}\n\nexport function resolveImageUrl(\n imageUrl: string,\n apiBaseUrl: string = OVERLAY_API_BASE\n): string {\n if (!imageUrl) return \"\";\n if (imageUrl.startsWith(\"http://\") || imageUrl.startsWith(\"https://\")) {\n return imageUrl;\n }\n if (imageUrl.startsWith(\"/\")) {\n try {\n const url = new URL(apiBaseUrl);\n return `${url.origin}${imageUrl}`;\n } catch {\n return imageUrl;\n }\n }\n return `${apiBaseUrl}/${imageUrl}`;\n}\n"]}
|