@xiboplayer/pwa 0.6.0 → 0.6.1
This diff represents the content of publicly available package versions that have been released to one of the supported registries. The information contained in this diff is provided for informational purposes only and reflects changes between package versions as they appear in their respective public registries.
- package/dist/assets/{chunk-config-yjpxlWy6.js → chunk-config-oTovBPVw.js} +2 -2
- package/dist/assets/{chunk-config-yjpxlWy6.js.map → chunk-config-oTovBPVw.js.map} +1 -1
- package/dist/assets/{index-BKKe321E.js → index-B6MdC-Qx.js} +2 -2
- package/dist/assets/{index-BKKe321E.js.map → index-B6MdC-Qx.js.map} +1 -1
- package/dist/assets/{index-BnKrYvmo.js → index-BVIXBw9z.js} +2 -2
- package/dist/assets/{index-BnKrYvmo.js.map → index-BVIXBw9z.js.map} +1 -1
- package/dist/assets/{index-DblGx8T7.js → index-BWvWWyDc.js} +2 -2
- package/dist/assets/{index-DblGx8T7.js.map → index-BWvWWyDc.js.map} +1 -1
- package/dist/assets/{index-B-_xo7Jd.js → index-C3Orblel.js} +2 -2
- package/dist/assets/{index-B-_xo7Jd.js.map → index-C3Orblel.js.map} +1 -1
- package/dist/assets/{index-D8pNyRvb.js → index-CleHw0Tc.js} +2 -2
- package/dist/assets/{index-D8pNyRvb.js.map → index-CleHw0Tc.js.map} +1 -1
- package/dist/assets/index-Cq9aOTTR.js +2 -0
- package/dist/assets/{index-DcGeUWV6.js.map → index-Cq9aOTTR.js.map} +1 -1
- package/dist/assets/index-D81Qhc3r.js +2 -0
- package/dist/assets/{index-C0V6Yuf7.js.map → index-D81Qhc3r.js.map} +1 -1
- package/dist/assets/{index-CjVBCq85.js → index-D_aTOqNE.js} +2 -2
- package/dist/assets/{index-CjVBCq85.js.map → index-D_aTOqNE.js.map} +1 -1
- package/dist/assets/index-Dj2ND9Mx.js +2 -0
- package/dist/assets/{index-ChxdOFuh.js.map → index-Dj2ND9Mx.js.map} +1 -1
- package/dist/assets/{index-BgzyAcAx.js → index-leM889oV.js} +2 -2
- package/dist/assets/{index-BgzyAcAx.js.map → index-leM889oV.js.map} +1 -1
- package/dist/assets/{main-BuxLonCL.js → main-DvteKhS8.js} +12 -17
- package/dist/assets/main-DvteKhS8.js.map +1 -0
- package/dist/assets/{setup-BKQoSJHa.js → setup-KpkKDvsj.js} +2 -2
- package/dist/assets/{setup-BKQoSJHa.js.map → setup-KpkKDvsj.js.map} +1 -1
- package/dist/assets/{sync-manager-Bxee4l4n.js → sync-manager-DEght5gK.js} +2 -2
- package/dist/assets/{sync-manager-Bxee4l4n.js.map → sync-manager-DEght5gK.js.map} +1 -1
- package/dist/assets/{widget-html-wxCcPbXd.js → widget-html-BAV4ZZd8.js} +2 -2
- package/dist/assets/{widget-html-wxCcPbXd.js.map → widget-html-BAV4ZZd8.js.map} +1 -1
- package/dist/assets/xmds-client-Dyy8ex-B.js +16 -0
- package/dist/assets/xmds-client-Dyy8ex-B.js.map +1 -0
- package/dist/index.html +3 -3
- package/dist/setup.html +3 -3
- package/dist/sw-pwa.js +1 -1
- package/package.json +13 -13
- package/dist/assets/index-C0V6Yuf7.js +0 -2
- package/dist/assets/index-ChxdOFuh.js +0 -2
- package/dist/assets/index-DcGeUWV6.js +0 -2
- package/dist/assets/main-BuxLonCL.js.map +0 -1
- package/dist/assets/xmds-client-xqEtcrNj.js +0 -16
- package/dist/assets/xmds-client-xqEtcrNj.js.map +0 -1
|
@@ -1,2 +1,2 @@
|
|
|
1
|
-
import{B as i,C as t,a as c,D as l,F as g,b as d,L as p,S as C,d as E,c as h,g as F,i as R}from"./widget-html-
|
|
2
|
-
//# sourceMappingURL=index-
|
|
1
|
+
import{B as i,C as t,a as c,D as l,F as g,b as d,L as p,S as C,d as E,c as h,g as F,i as R}from"./widget-html-BAV4ZZd8.js";import"./index-leM889oV.js";const a="0.6.1",e={version:a},o=e.version;export{i as BARRIER,t as CacheAnalyzer,c as CacheManager,l as DownloadManager,g as FILE_TYPES,d as FileDownload,p as LayoutTaskBuilder,C as StoreClient,o as VERSION,E as cacheManager,h as cacheWidgetHtml,F as getFileTypeConfig,R as isUrlExpired};
|
|
2
|
+
//# sourceMappingURL=index-C3Orblel.js.map
|
|
@@ -1 +1 @@
|
|
|
1
|
-
{"version":3,"file":"index-
|
|
1
|
+
{"version":3,"file":"index-C3Orblel.js","sources":["../../../cache/src/index.js"],"sourcesContent":["// @xiboplayer/cache - Offline caching and downloads\nimport pkg from '../package.json' with { type: 'json' };\nexport const VERSION = pkg.version;\nexport { CacheManager, cacheManager } from './cache.js';\nexport { StoreClient } from './store-client.js';\nexport { DownloadManager, FileDownload, LayoutTaskBuilder, BARRIER, isUrlExpired } from './download-manager.js';\nexport { CacheAnalyzer } from './cache-analyzer.js';\nexport { cacheWidgetHtml } from './widget-html.js';\nexport { FILE_TYPES, getFileTypeConfig } from './file-types.js';\n"],"names":["VERSION","pkg"],"mappings":"qLAEaA,EAAUC,EAAI"}
|
|
@@ -1,2 +1,2 @@
|
|
|
1
|
-
import{EventEmitter as r,createLogger as o}from"./index-
|
|
2
|
-
//# sourceMappingURL=index-
|
|
1
|
+
import{EventEmitter as r,createLogger as o}from"./index-leM889oV.js";const a="0.6.1",l={version:a},i=o("DisplaySettings");class c extends r{constructor(){super(),this.settings={collectInterval:300,displayName:"Unknown Display",sizeX:1920,sizeY:1080,statsEnabled:!1,aggregationLevel:"Individual",logLevel:"error",xmrNetworkAddress:null,xmrWebSocketAddress:null,xmrCmsKey:null,preventSleep:!0,embeddedServerPort:9696,screenshotInterval:120,downloadStartWindow:null,downloadEndWindow:null,licenceCode:null,isSspEnabled:!1}}applySettings(e){if(!e)return i.warn("No settings provided"),{changed:[],settings:this.settings};const t=[],s=this.settings.collectInterval;return this.settings.collectInterval=this.parseCollectInterval(e.collectInterval||e.CollectInterval),this.settings.displayName=e.displayName||e.DisplayName||this.settings.displayName,this.settings.sizeX=parseInt(e.sizeX||e.SizeX||this.settings.sizeX),this.settings.sizeY=parseInt(e.sizeY||e.SizeY||this.settings.sizeY),this.settings.statsEnabled=this.parseBoolean(e.statsEnabled||e.StatsEnabled),this.settings.aggregationLevel=e.aggregationLevel||e.AggregationLevel||this.settings.aggregationLevel,this.settings.logLevel=e.logLevel||e.LogLevel||this.settings.logLevel,this.settings.xmrNetworkAddress=e.xmrNetworkAddress||e.XmrNetworkAddress||this.settings.xmrNetworkAddress,this.settings.xmrWebSocketAddress=e.xmrWebSocketAddress||e.XmrWebSocketAddress||this.settings.xmrWebSocketAddress,this.settings.xmrCmsKey=e.xmrCmsKey||e.XmrCmsKey||this.settings.xmrCmsKey,this.settings.preventSleep=this.parseBoolean(e.preventSleep||e.PreventSleep,!0),this.settings.embeddedServerPort=parseInt(e.embeddedServerPort||e.EmbeddedServerPort||this.settings.embeddedServerPort),this.settings.screenshotInterval=parseInt(e.screenshotInterval||e.ScreenshotInterval||this.settings.screenshotInterval),this.settings.downloadStartWindow=e.downloadStartWindow||e.DownloadStartWindow||this.settings.downloadStartWindow,this.settings.downloadEndWindow=e.downloadEndWindow||e.DownloadEndWindow||this.settings.downloadEndWindow,this.settings.licenceCode=e.licenceCode||e.LicenceCode||this.settings.licenceCode,this.settings.isSspEnabled=this.parseBoolean(e.isAdspaceEnabled||e.IsAdspaceEnabled),s!==this.settings.collectInterval&&(t.push("collectInterval"),this.emit("interval-changed",this.settings.collectInterval)),this.emit("settings-applied",this.settings,t),i.info("Applied settings:",{collectInterval:this.settings.collectInterval,displayName:this.settings.displayName,statsEnabled:this.settings.statsEnabled,changes:t}),{changed:t,settings:this.settings}}parseCollectInterval(e){const t=parseInt(e,10);return isNaN(t)||t<60?300:t>86400?86400:t}parseBoolean(e,t=!1){return e===!0||e===!1?e:e==="1"||e===1?!0:e==="0"||e===0?!1:t}getCollectInterval(){return this.settings.collectInterval}getDisplayName(){return this.settings.displayName}getDisplaySize(){return{width:this.settings.sizeX,height:this.settings.sizeY}}isStatsEnabled(){return this.settings.statsEnabled}getAllSettings(){return{...this.settings}}getSetting(e,t=null){return this.settings[e]!==void 0?this.settings[e]:t}isInDownloadWindow(){if(!this.settings.downloadStartWindow||!this.settings.downloadEndWindow)return!0;try{const e=new Date,t=e.getHours()*60+e.getMinutes(),s=this.parseTimeWindow(this.settings.downloadStartWindow),n=this.parseTimeWindow(this.settings.downloadEndWindow);return s>n?t>=s||t<n:t>=s&&t<n}catch(e){return i.warn("Failed to parse download window:",e),!0}}parseTimeWindow(e){if(!e||typeof e!="string")throw new Error("Invalid time window format");const t=e.split(":");if(t.length!==2)throw new Error("Invalid time window format (expected HH:MM)");const s=parseInt(t[0],10),n=parseInt(t[1],10);if(isNaN(s)||isNaN(n)||s<0||s>23||n<0||n>59)throw new Error("Invalid time window values");return s*60+n}getNextDownloadWindow(){if(!this.settings.downloadStartWindow||!this.settings.downloadEndWindow)return null;try{const e=new Date,t=e.getHours()*60+e.getMinutes(),s=this.parseTimeWindow(this.settings.downloadStartWindow),n=new Date(e);return t<s||n.setDate(n.getDate()+1),n.setHours(Math.floor(s/60),s%60,0,0),n}catch(e){return i.warn("Failed to calculate next download window:",e),null}}shouldTakeScreenshot(e){return e?(Date.now()-e.getTime())/1e3>=this.settings.screenshotInterval:!0}}const w=l.version;export{c as DisplaySettings,w as VERSION};
|
|
2
|
+
//# sourceMappingURL=index-CleHw0Tc.js.map
|
|
@@ -1 +1 @@
|
|
|
1
|
-
{"version":3,"file":"index-D8pNyRvb.js","sources":["../../../settings/src/settings.js","../../../settings/src/index.js"],"sourcesContent":["/**\n * DisplaySettings - CMS display settings management\n *\n * Parses and applies configuration from RegisterDisplay response.\n * Based on upstream electron-player implementation.\n *\n * Architecture:\n * ┌─────────────────────────────────────────────────────┐\n * │ PlayerCore │\n * │ - Receives RegisterDisplay response │\n * │ - Passes to DisplaySettings.applySettings() │\n * └─────────────────────────────────────────────────────┘\n * ↓\n * ┌─────────────────────────────────────────────────────┐\n * │ DisplaySettings (this module) │\n * │ - Parse all CMS settings │\n * │ - Validate and normalize values │\n * │ - Apply collection interval │\n * │ - Check download windows │\n * │ - Handle screenshot requests │\n * │ - Emit events on changes │\n * └─────────────────────────────────────────────────────┘\n * ↓\n * ┌─────────────────────────────────────────────────────┐\n * │ Platform Layer (PWA/Electron/Mobile) │\n * │ - Listen for setting change events │\n * │ - Update UI with display name │\n * │ - Handle screenshot requests │\n * │ - Respect download windows │\n * └─────────────────────────────────────────────────────┘\n *\n * Usage:\n * const settings = new DisplaySettings();\n * settings.applySettings(regResult.settings);\n *\n * // Get settings\n * const collectInterval = settings.getCollectInterval();\n * const canDownload = settings.isInDownloadWindow();\n *\n * // Listen for changes\n * settings.on('interval-changed', (newInterval) => { ... });\n */\n\nimport { EventEmitter, createLogger } from '@xiboplayer/utils';\n\nconst log = createLogger('DisplaySettings');\n\nexport class DisplaySettings extends EventEmitter {\n constructor() {\n super();\n\n // Current settings (with defaults)\n this.settings = {\n // Collection\n collectInterval: 300, // seconds (5 minutes default)\n\n // Display info\n displayName: 'Unknown Display',\n sizeX: 1920,\n sizeY: 1080,\n\n // Stats\n statsEnabled: false,\n aggregationLevel: 'Individual', // or 'Aggregate'\n\n // Logging\n logLevel: 'error', // 'error', 'audit', 'info', 'debug'\n\n // XMR\n xmrNetworkAddress: null,\n xmrWebSocketAddress: null,\n xmrCmsKey: null,\n\n // Features\n preventSleep: true,\n embeddedServerPort: 9696,\n screenshotInterval: 120, // seconds\n\n // Download windows\n downloadStartWindow: null,\n downloadEndWindow: null,\n\n // License\n licenceCode: null,\n\n // SSP (ad space)\n isSspEnabled: false,\n };\n }\n\n /**\n * Apply settings from RegisterDisplay response\n * @param {Object} settings - Raw settings from CMS\n * @returns {Object} Applied settings with changes\n */\n applySettings(settings) {\n if (!settings) {\n log.warn('No settings provided');\n return { changed: [], settings: this.settings };\n }\n\n const changes = [];\n const oldInterval = this.settings.collectInterval;\n\n // Parse all settings with defaults\n // Handle both lowercase and CamelCase (uppercase first letter)\n this.settings.collectInterval = this.parseCollectInterval(settings.collectInterval || settings.CollectInterval);\n this.settings.displayName = settings.displayName || settings.DisplayName || this.settings.displayName;\n this.settings.sizeX = parseInt(settings.sizeX || settings.SizeX || this.settings.sizeX);\n this.settings.sizeY = parseInt(settings.sizeY || settings.SizeY || this.settings.sizeY);\n\n // Stats\n this.settings.statsEnabled = this.parseBoolean(settings.statsEnabled || settings.StatsEnabled);\n this.settings.aggregationLevel = settings.aggregationLevel || settings.AggregationLevel || this.settings.aggregationLevel;\n\n // Logging\n this.settings.logLevel = settings.logLevel || settings.LogLevel || this.settings.logLevel;\n\n // XMR\n this.settings.xmrNetworkAddress = settings.xmrNetworkAddress || settings.XmrNetworkAddress || this.settings.xmrNetworkAddress;\n this.settings.xmrWebSocketAddress = settings.xmrWebSocketAddress || settings.XmrWebSocketAddress || this.settings.xmrWebSocketAddress;\n this.settings.xmrCmsKey = settings.xmrCmsKey || settings.XmrCmsKey || this.settings.xmrCmsKey;\n\n // Features\n this.settings.preventSleep = this.parseBoolean(settings.preventSleep || settings.PreventSleep, true);\n this.settings.embeddedServerPort = parseInt(settings.embeddedServerPort || settings.EmbeddedServerPort || this.settings.embeddedServerPort);\n this.settings.screenshotInterval = parseInt(settings.screenshotInterval || settings.ScreenshotInterval || this.settings.screenshotInterval);\n\n // Download windows\n this.settings.downloadStartWindow = settings.downloadStartWindow || settings.DownloadStartWindow || this.settings.downloadStartWindow;\n this.settings.downloadEndWindow = settings.downloadEndWindow || settings.DownloadEndWindow || this.settings.downloadEndWindow;\n\n // License\n this.settings.licenceCode = settings.licenceCode || settings.LicenceCode || this.settings.licenceCode;\n\n // SSP\n this.settings.isSspEnabled = this.parseBoolean(settings.isAdspaceEnabled || settings.IsAdspaceEnabled);\n\n // Detect changes\n if (oldInterval !== this.settings.collectInterval) {\n changes.push('collectInterval');\n this.emit('interval-changed', this.settings.collectInterval);\n }\n\n // Emit generic settings-applied event\n this.emit('settings-applied', this.settings, changes);\n\n log.info('Applied settings:', {\n collectInterval: this.settings.collectInterval,\n displayName: this.settings.displayName,\n statsEnabled: this.settings.statsEnabled,\n changes\n });\n\n return { changed: changes, settings: this.settings };\n }\n\n /**\n * Parse collection interval (seconds)\n * @param {*} value - Raw value from CMS\n * @returns {number} Collection interval in seconds\n */\n parseCollectInterval(value) {\n const interval = parseInt(value, 10);\n\n // Validate range (minimum 60s, maximum 86400s = 24h)\n if (isNaN(interval) || interval < 60) {\n return 300; // 5 minutes default\n }\n\n if (interval > 86400) {\n return 86400; // 24 hours max\n }\n\n return interval;\n }\n\n /**\n * Parse boolean setting\n * @param {*} value - Raw value from CMS (string '1' or '0', or boolean)\n * @param {boolean} defaultValue - Default if not set\n * @returns {boolean}\n */\n parseBoolean(value, defaultValue = false) {\n if (value === true || value === false) {\n return value;\n }\n\n if (value === '1' || value === 1) {\n return true;\n }\n\n if (value === '0' || value === 0) {\n return false;\n }\n\n return defaultValue;\n }\n\n /**\n * Get collection interval in seconds\n * @returns {number}\n */\n getCollectInterval() {\n return this.settings.collectInterval;\n }\n\n /**\n * Get display name\n * @returns {string}\n */\n getDisplayName() {\n return this.settings.displayName;\n }\n\n /**\n * Get display size\n * @returns {{ width: number, height: number }}\n */\n getDisplaySize() {\n return {\n width: this.settings.sizeX,\n height: this.settings.sizeY\n };\n }\n\n /**\n * Check if stats are enabled\n * @returns {boolean}\n */\n isStatsEnabled() {\n return this.settings.statsEnabled;\n }\n\n /**\n * Get all settings\n * @returns {Object}\n */\n getAllSettings() {\n return { ...this.settings };\n }\n\n /**\n * Get a specific setting by key\n * @param {string} key - Setting key\n * @param {*} defaultValue - Default value if not set\n * @returns {*}\n */\n getSetting(key, defaultValue = null) {\n return this.settings[key] !== undefined ? this.settings[key] : defaultValue;\n }\n\n /**\n * Check if current time is within download window\n * @returns {boolean}\n */\n isInDownloadWindow() {\n // If no download window configured, always allow\n if (!this.settings.downloadStartWindow || !this.settings.downloadEndWindow) {\n return true;\n }\n\n try {\n const now = new Date();\n const currentTime = now.getHours() * 60 + now.getMinutes();\n\n const start = this.parseTimeWindow(this.settings.downloadStartWindow);\n const end = this.parseTimeWindow(this.settings.downloadEndWindow);\n\n // Handle overnight window (e.g., 22:00 - 06:00)\n if (start > end) {\n // Overnight: allow if AFTER start OR BEFORE end\n return currentTime >= start || currentTime < end;\n } else {\n // Same day: allow if AFTER start AND BEFORE end\n return currentTime >= start && currentTime < end;\n }\n } catch (error) {\n log.warn('Failed to parse download window:', error);\n return true; // Allow downloads if parsing fails\n }\n }\n\n /**\n * Parse time window string to minutes since midnight\n * @param {string} timeStr - Time string (e.g., \"14:30\", \"22:00\")\n * @returns {number} Minutes since midnight\n */\n parseTimeWindow(timeStr) {\n if (!timeStr || typeof timeStr !== 'string') {\n throw new Error('Invalid time window format');\n }\n\n const parts = timeStr.split(':');\n if (parts.length !== 2) {\n throw new Error('Invalid time window format (expected HH:MM)');\n }\n\n const hours = parseInt(parts[0], 10);\n const minutes = parseInt(parts[1], 10);\n\n if (isNaN(hours) || isNaN(minutes) || hours < 0 || hours > 23 || minutes < 0 || minutes > 59) {\n throw new Error('Invalid time window values');\n }\n\n return hours * 60 + minutes;\n }\n\n /**\n * Get next download window start time\n * @returns {Date|null} Next window start, or null if always allowed\n */\n getNextDownloadWindow() {\n if (!this.settings.downloadStartWindow || !this.settings.downloadEndWindow) {\n return null;\n }\n\n try {\n const now = new Date();\n const currentTime = now.getHours() * 60 + now.getMinutes();\n const start = this.parseTimeWindow(this.settings.downloadStartWindow);\n\n const nextWindow = new Date(now);\n\n if (currentTime < start) {\n // Window is later today\n nextWindow.setHours(Math.floor(start / 60), start % 60, 0, 0);\n } else {\n // Window is tomorrow\n nextWindow.setDate(nextWindow.getDate() + 1);\n nextWindow.setHours(Math.floor(start / 60), start % 60, 0, 0);\n }\n\n return nextWindow;\n } catch (error) {\n log.warn('Failed to calculate next download window:', error);\n return null;\n }\n }\n\n /**\n * Check if screenshot interval has elapsed\n * @param {Date} lastScreenshot - Last screenshot timestamp\n * @returns {boolean}\n */\n shouldTakeScreenshot(lastScreenshot) {\n if (!lastScreenshot) {\n return true;\n }\n\n const elapsed = (Date.now() - lastScreenshot.getTime()) / 1000;\n return elapsed >= this.settings.screenshotInterval;\n }\n}\n","// @xiboplayer/settings - CMS settings management\nimport pkg from '../package.json' with { type: 'json' };\nexport const VERSION = pkg.version;\n\n/**\n * Settings manager for Xibo Player\n * @module @xiboplayer/settings\n */\nexport { DisplaySettings } from './settings.js';\n"],"names":["log","createLogger","DisplaySettings","EventEmitter","settings","changes","oldInterval","value","interval","defaultValue","key","now","currentTime","start","end","error","timeStr","parts","hours","minutes","nextWindow","lastScreenshot","VERSION","pkg"],"mappings":"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"}
|
|
1
|
+
{"version":3,"file":"index-CleHw0Tc.js","sources":["../../../settings/src/settings.js","../../../settings/src/index.js"],"sourcesContent":["/**\n * DisplaySettings - CMS display settings management\n *\n * Parses and applies configuration from RegisterDisplay response.\n * Based on upstream electron-player implementation.\n *\n * Architecture:\n * ┌─────────────────────────────────────────────────────┐\n * │ PlayerCore │\n * │ - Receives RegisterDisplay response │\n * │ - Passes to DisplaySettings.applySettings() │\n * └─────────────────────────────────────────────────────┘\n * ↓\n * ┌─────────────────────────────────────────────────────┐\n * │ DisplaySettings (this module) │\n * │ - Parse all CMS settings │\n * │ - Validate and normalize values │\n * │ - Apply collection interval │\n * │ - Check download windows │\n * │ - Handle screenshot requests │\n * │ - Emit events on changes │\n * └─────────────────────────────────────────────────────┘\n * ↓\n * ┌─────────────────────────────────────────────────────┐\n * │ Platform Layer (PWA/Electron/Mobile) │\n * │ - Listen for setting change events │\n * │ - Update UI with display name │\n * │ - Handle screenshot requests │\n * │ - Respect download windows │\n * └─────────────────────────────────────────────────────┘\n *\n * Usage:\n * const settings = new DisplaySettings();\n * settings.applySettings(regResult.settings);\n *\n * // Get settings\n * const collectInterval = settings.getCollectInterval();\n * const canDownload = settings.isInDownloadWindow();\n *\n * // Listen for changes\n * settings.on('interval-changed', (newInterval) => { ... });\n */\n\nimport { EventEmitter, createLogger } from '@xiboplayer/utils';\n\nconst log = createLogger('DisplaySettings');\n\nexport class DisplaySettings extends EventEmitter {\n constructor() {\n super();\n\n // Current settings (with defaults)\n this.settings = {\n // Collection\n collectInterval: 300, // seconds (5 minutes default)\n\n // Display info\n displayName: 'Unknown Display',\n sizeX: 1920,\n sizeY: 1080,\n\n // Stats\n statsEnabled: false,\n aggregationLevel: 'Individual', // or 'Aggregate'\n\n // Logging\n logLevel: 'error', // 'error', 'audit', 'info', 'debug'\n\n // XMR\n xmrNetworkAddress: null,\n xmrWebSocketAddress: null,\n xmrCmsKey: null,\n\n // Features\n preventSleep: true,\n embeddedServerPort: 9696,\n screenshotInterval: 120, // seconds\n\n // Download windows\n downloadStartWindow: null,\n downloadEndWindow: null,\n\n // License\n licenceCode: null,\n\n // SSP (ad space)\n isSspEnabled: false,\n };\n }\n\n /**\n * Apply settings from RegisterDisplay response\n * @param {Object} settings - Raw settings from CMS\n * @returns {Object} Applied settings with changes\n */\n applySettings(settings) {\n if (!settings) {\n log.warn('No settings provided');\n return { changed: [], settings: this.settings };\n }\n\n const changes = [];\n const oldInterval = this.settings.collectInterval;\n\n // Parse all settings with defaults\n // Handle both lowercase and CamelCase (uppercase first letter)\n this.settings.collectInterval = this.parseCollectInterval(settings.collectInterval || settings.CollectInterval);\n this.settings.displayName = settings.displayName || settings.DisplayName || this.settings.displayName;\n this.settings.sizeX = parseInt(settings.sizeX || settings.SizeX || this.settings.sizeX);\n this.settings.sizeY = parseInt(settings.sizeY || settings.SizeY || this.settings.sizeY);\n\n // Stats\n this.settings.statsEnabled = this.parseBoolean(settings.statsEnabled || settings.StatsEnabled);\n this.settings.aggregationLevel = settings.aggregationLevel || settings.AggregationLevel || this.settings.aggregationLevel;\n\n // Logging\n this.settings.logLevel = settings.logLevel || settings.LogLevel || this.settings.logLevel;\n\n // XMR\n this.settings.xmrNetworkAddress = settings.xmrNetworkAddress || settings.XmrNetworkAddress || this.settings.xmrNetworkAddress;\n this.settings.xmrWebSocketAddress = settings.xmrWebSocketAddress || settings.XmrWebSocketAddress || this.settings.xmrWebSocketAddress;\n this.settings.xmrCmsKey = settings.xmrCmsKey || settings.XmrCmsKey || this.settings.xmrCmsKey;\n\n // Features\n this.settings.preventSleep = this.parseBoolean(settings.preventSleep || settings.PreventSleep, true);\n this.settings.embeddedServerPort = parseInt(settings.embeddedServerPort || settings.EmbeddedServerPort || this.settings.embeddedServerPort);\n this.settings.screenshotInterval = parseInt(settings.screenshotInterval || settings.ScreenshotInterval || this.settings.screenshotInterval);\n\n // Download windows\n this.settings.downloadStartWindow = settings.downloadStartWindow || settings.DownloadStartWindow || this.settings.downloadStartWindow;\n this.settings.downloadEndWindow = settings.downloadEndWindow || settings.DownloadEndWindow || this.settings.downloadEndWindow;\n\n // License\n this.settings.licenceCode = settings.licenceCode || settings.LicenceCode || this.settings.licenceCode;\n\n // SSP\n this.settings.isSspEnabled = this.parseBoolean(settings.isAdspaceEnabled || settings.IsAdspaceEnabled);\n\n // Detect changes\n if (oldInterval !== this.settings.collectInterval) {\n changes.push('collectInterval');\n this.emit('interval-changed', this.settings.collectInterval);\n }\n\n // Emit generic settings-applied event\n this.emit('settings-applied', this.settings, changes);\n\n log.info('Applied settings:', {\n collectInterval: this.settings.collectInterval,\n displayName: this.settings.displayName,\n statsEnabled: this.settings.statsEnabled,\n changes\n });\n\n return { changed: changes, settings: this.settings };\n }\n\n /**\n * Parse collection interval (seconds)\n * @param {*} value - Raw value from CMS\n * @returns {number} Collection interval in seconds\n */\n parseCollectInterval(value) {\n const interval = parseInt(value, 10);\n\n // Validate range (minimum 60s, maximum 86400s = 24h)\n if (isNaN(interval) || interval < 60) {\n return 300; // 5 minutes default\n }\n\n if (interval > 86400) {\n return 86400; // 24 hours max\n }\n\n return interval;\n }\n\n /**\n * Parse boolean setting\n * @param {*} value - Raw value from CMS (string '1' or '0', or boolean)\n * @param {boolean} defaultValue - Default if not set\n * @returns {boolean}\n */\n parseBoolean(value, defaultValue = false) {\n if (value === true || value === false) {\n return value;\n }\n\n if (value === '1' || value === 1) {\n return true;\n }\n\n if (value === '0' || value === 0) {\n return false;\n }\n\n return defaultValue;\n }\n\n /**\n * Get collection interval in seconds\n * @returns {number}\n */\n getCollectInterval() {\n return this.settings.collectInterval;\n }\n\n /**\n * Get display name\n * @returns {string}\n */\n getDisplayName() {\n return this.settings.displayName;\n }\n\n /**\n * Get display size\n * @returns {{ width: number, height: number }}\n */\n getDisplaySize() {\n return {\n width: this.settings.sizeX,\n height: this.settings.sizeY\n };\n }\n\n /**\n * Check if stats are enabled\n * @returns {boolean}\n */\n isStatsEnabled() {\n return this.settings.statsEnabled;\n }\n\n /**\n * Get all settings\n * @returns {Object}\n */\n getAllSettings() {\n return { ...this.settings };\n }\n\n /**\n * Get a specific setting by key\n * @param {string} key - Setting key\n * @param {*} defaultValue - Default value if not set\n * @returns {*}\n */\n getSetting(key, defaultValue = null) {\n return this.settings[key] !== undefined ? this.settings[key] : defaultValue;\n }\n\n /**\n * Check if current time is within download window\n * @returns {boolean}\n */\n isInDownloadWindow() {\n // If no download window configured, always allow\n if (!this.settings.downloadStartWindow || !this.settings.downloadEndWindow) {\n return true;\n }\n\n try {\n const now = new Date();\n const currentTime = now.getHours() * 60 + now.getMinutes();\n\n const start = this.parseTimeWindow(this.settings.downloadStartWindow);\n const end = this.parseTimeWindow(this.settings.downloadEndWindow);\n\n // Handle overnight window (e.g., 22:00 - 06:00)\n if (start > end) {\n // Overnight: allow if AFTER start OR BEFORE end\n return currentTime >= start || currentTime < end;\n } else {\n // Same day: allow if AFTER start AND BEFORE end\n return currentTime >= start && currentTime < end;\n }\n } catch (error) {\n log.warn('Failed to parse download window:', error);\n return true; // Allow downloads if parsing fails\n }\n }\n\n /**\n * Parse time window string to minutes since midnight\n * @param {string} timeStr - Time string (e.g., \"14:30\", \"22:00\")\n * @returns {number} Minutes since midnight\n */\n parseTimeWindow(timeStr) {\n if (!timeStr || typeof timeStr !== 'string') {\n throw new Error('Invalid time window format');\n }\n\n const parts = timeStr.split(':');\n if (parts.length !== 2) {\n throw new Error('Invalid time window format (expected HH:MM)');\n }\n\n const hours = parseInt(parts[0], 10);\n const minutes = parseInt(parts[1], 10);\n\n if (isNaN(hours) || isNaN(minutes) || hours < 0 || hours > 23 || minutes < 0 || minutes > 59) {\n throw new Error('Invalid time window values');\n }\n\n return hours * 60 + minutes;\n }\n\n /**\n * Get next download window start time\n * @returns {Date|null} Next window start, or null if always allowed\n */\n getNextDownloadWindow() {\n if (!this.settings.downloadStartWindow || !this.settings.downloadEndWindow) {\n return null;\n }\n\n try {\n const now = new Date();\n const currentTime = now.getHours() * 60 + now.getMinutes();\n const start = this.parseTimeWindow(this.settings.downloadStartWindow);\n\n const nextWindow = new Date(now);\n\n if (currentTime < start) {\n // Window is later today\n nextWindow.setHours(Math.floor(start / 60), start % 60, 0, 0);\n } else {\n // Window is tomorrow\n nextWindow.setDate(nextWindow.getDate() + 1);\n nextWindow.setHours(Math.floor(start / 60), start % 60, 0, 0);\n }\n\n return nextWindow;\n } catch (error) {\n log.warn('Failed to calculate next download window:', error);\n return null;\n }\n }\n\n /**\n * Check if screenshot interval has elapsed\n * @param {Date} lastScreenshot - Last screenshot timestamp\n * @returns {boolean}\n */\n shouldTakeScreenshot(lastScreenshot) {\n if (!lastScreenshot) {\n return true;\n }\n\n const elapsed = (Date.now() - lastScreenshot.getTime()) / 1000;\n return elapsed >= this.settings.screenshotInterval;\n }\n}\n","// @xiboplayer/settings - CMS settings management\nimport pkg from '../package.json' with { type: 'json' };\nexport const VERSION = pkg.version;\n\n/**\n * Settings manager for Xibo Player\n * @module @xiboplayer/settings\n */\nexport { DisplaySettings } from './settings.js';\n"],"names":["log","createLogger","DisplaySettings","EventEmitter","settings","changes","oldInterval","value","interval","defaultValue","key","now","currentTime","start","end","error","timeStr","parts","hours","minutes","nextWindow","lastScreenshot","VERSION","pkg"],"mappings":"mGA6CMA,EAAMC,EAAa,iBAAiB,EAEnC,MAAMC,UAAwBC,CAAa,CAChD,aAAc,CACZ,MAAK,EAGL,KAAK,SAAW,CAEd,gBAAiB,IAGjB,YAAa,kBACb,MAAO,KACP,MAAO,KAGP,aAAc,GACd,iBAAkB,aAGlB,SAAU,QAGV,kBAAmB,KACnB,oBAAqB,KACrB,UAAW,KAGX,aAAc,GACd,mBAAoB,KACpB,mBAAoB,IAGpB,oBAAqB,KACrB,kBAAmB,KAGnB,YAAa,KAGb,aAAc,EACpB,CACE,CAOA,cAAcC,EAAU,CACtB,GAAI,CAACA,EACH,OAAAJ,EAAI,KAAK,sBAAsB,EACxB,CAAE,QAAS,CAAA,EAAI,SAAU,KAAK,QAAQ,EAG/C,MAAMK,EAAU,CAAA,EACVC,EAAc,KAAK,SAAS,gBAIlC,YAAK,SAAS,gBAAkB,KAAK,qBAAqBF,EAAS,iBAAmBA,EAAS,eAAe,EAC9G,KAAK,SAAS,YAAcA,EAAS,aAAeA,EAAS,aAAe,KAAK,SAAS,YAC1F,KAAK,SAAS,MAAQ,SAASA,EAAS,OAASA,EAAS,OAAS,KAAK,SAAS,KAAK,EACtF,KAAK,SAAS,MAAQ,SAASA,EAAS,OAASA,EAAS,OAAS,KAAK,SAAS,KAAK,EAGtF,KAAK,SAAS,aAAe,KAAK,aAAaA,EAAS,cAAgBA,EAAS,YAAY,EAC7F,KAAK,SAAS,iBAAmBA,EAAS,kBAAoBA,EAAS,kBAAoB,KAAK,SAAS,iBAGzG,KAAK,SAAS,SAAWA,EAAS,UAAYA,EAAS,UAAY,KAAK,SAAS,SAGjF,KAAK,SAAS,kBAAoBA,EAAS,mBAAqBA,EAAS,mBAAqB,KAAK,SAAS,kBAC5G,KAAK,SAAS,oBAAsBA,EAAS,qBAAuBA,EAAS,qBAAuB,KAAK,SAAS,oBAClH,KAAK,SAAS,UAAYA,EAAS,WAAaA,EAAS,WAAa,KAAK,SAAS,UAGpF,KAAK,SAAS,aAAe,KAAK,aAAaA,EAAS,cAAgBA,EAAS,aAAc,EAAI,EACnG,KAAK,SAAS,mBAAqB,SAASA,EAAS,oBAAsBA,EAAS,oBAAsB,KAAK,SAAS,kBAAkB,EAC1I,KAAK,SAAS,mBAAqB,SAASA,EAAS,oBAAsBA,EAAS,oBAAsB,KAAK,SAAS,kBAAkB,EAG1I,KAAK,SAAS,oBAAsBA,EAAS,qBAAuBA,EAAS,qBAAuB,KAAK,SAAS,oBAClH,KAAK,SAAS,kBAAoBA,EAAS,mBAAqBA,EAAS,mBAAqB,KAAK,SAAS,kBAG5G,KAAK,SAAS,YAAcA,EAAS,aAAeA,EAAS,aAAe,KAAK,SAAS,YAG1F,KAAK,SAAS,aAAe,KAAK,aAAaA,EAAS,kBAAoBA,EAAS,gBAAgB,EAGjGE,IAAgB,KAAK,SAAS,kBAChCD,EAAQ,KAAK,iBAAiB,EAC9B,KAAK,KAAK,mBAAoB,KAAK,SAAS,eAAe,GAI7D,KAAK,KAAK,mBAAoB,KAAK,SAAUA,CAAO,EAEpDL,EAAI,KAAK,oBAAqB,CAC5B,gBAAiB,KAAK,SAAS,gBAC/B,YAAa,KAAK,SAAS,YAC3B,aAAc,KAAK,SAAS,aAC5B,QAAAK,CACN,CAAK,EAEM,CAAE,QAASA,EAAS,SAAU,KAAK,QAAQ,CACpD,CAOA,qBAAqBE,EAAO,CAC1B,MAAMC,EAAW,SAASD,EAAO,EAAE,EAGnC,OAAI,MAAMC,CAAQ,GAAKA,EAAW,GACzB,IAGLA,EAAW,MACN,MAGFA,CACT,CAQA,aAAaD,EAAOE,EAAe,GAAO,CACxC,OAAIF,IAAU,IAAQA,IAAU,GACvBA,EAGLA,IAAU,KAAOA,IAAU,EACtB,GAGLA,IAAU,KAAOA,IAAU,EACtB,GAGFE,CACT,CAMA,oBAAqB,CACnB,OAAO,KAAK,SAAS,eACvB,CAMA,gBAAiB,CACf,OAAO,KAAK,SAAS,WACvB,CAMA,gBAAiB,CACf,MAAO,CACL,MAAO,KAAK,SAAS,MACrB,OAAQ,KAAK,SAAS,KAC5B,CACE,CAMA,gBAAiB,CACf,OAAO,KAAK,SAAS,YACvB,CAMA,gBAAiB,CACf,MAAO,CAAE,GAAG,KAAK,QAAQ,CAC3B,CAQA,WAAWC,EAAKD,EAAe,KAAM,CACnC,OAAO,KAAK,SAASC,CAAG,IAAM,OAAY,KAAK,SAASA,CAAG,EAAID,CACjE,CAMA,oBAAqB,CAEnB,GAAI,CAAC,KAAK,SAAS,qBAAuB,CAAC,KAAK,SAAS,kBACvD,MAAO,GAGT,GAAI,CACF,MAAME,EAAM,IAAI,KACVC,EAAcD,EAAI,SAAQ,EAAK,GAAKA,EAAI,WAAU,EAElDE,EAAQ,KAAK,gBAAgB,KAAK,SAAS,mBAAmB,EAC9DC,EAAM,KAAK,gBAAgB,KAAK,SAAS,iBAAiB,EAGhE,OAAID,EAAQC,EAEHF,GAAeC,GAASD,EAAcE,EAGtCF,GAAeC,GAASD,EAAcE,CAEjD,OAASC,EAAO,CACd,OAAAf,EAAI,KAAK,mCAAoCe,CAAK,EAC3C,EACT,CACF,CAOA,gBAAgBC,EAAS,CACvB,GAAI,CAACA,GAAW,OAAOA,GAAY,SACjC,MAAM,IAAI,MAAM,4BAA4B,EAG9C,MAAMC,EAAQD,EAAQ,MAAM,GAAG,EAC/B,GAAIC,EAAM,SAAW,EACnB,MAAM,IAAI,MAAM,6CAA6C,EAG/D,MAAMC,EAAQ,SAASD,EAAM,CAAC,EAAG,EAAE,EAC7BE,EAAU,SAASF,EAAM,CAAC,EAAG,EAAE,EAErC,GAAI,MAAMC,CAAK,GAAK,MAAMC,CAAO,GAAKD,EAAQ,GAAKA,EAAQ,IAAMC,EAAU,GAAKA,EAAU,GACxF,MAAM,IAAI,MAAM,4BAA4B,EAG9C,OAAOD,EAAQ,GAAKC,CACtB,CAMA,uBAAwB,CACtB,GAAI,CAAC,KAAK,SAAS,qBAAuB,CAAC,KAAK,SAAS,kBACvD,OAAO,KAGT,GAAI,CACF,MAAMR,EAAM,IAAI,KACVC,EAAcD,EAAI,SAAQ,EAAK,GAAKA,EAAI,WAAU,EAClDE,EAAQ,KAAK,gBAAgB,KAAK,SAAS,mBAAmB,EAE9DO,EAAa,IAAI,KAAKT,CAAG,EAE/B,OAAIC,EAAcC,GAKhBO,EAAW,QAAQA,EAAW,QAAO,EAAK,CAAC,EAC3CA,EAAW,SAAS,KAAK,MAAMP,EAAQ,EAAE,EAAGA,EAAQ,GAAI,EAAG,CAAC,EAGvDO,CACT,OAASL,EAAO,CACd,OAAAf,EAAI,KAAK,4CAA6Ce,CAAK,EACpD,IACT,CACF,CAOA,qBAAqBM,EAAgB,CACnC,OAAKA,GAIY,KAAK,IAAG,EAAKA,EAAe,QAAO,GAAM,KACxC,KAAK,SAAS,mBAJvB,EAKX,CACF,CC/VY,MAACC,EAAUC,EAAI"}
|
|
@@ -0,0 +1,2 @@
|
|
|
1
|
+
import{L as p,a as m,R as L}from"./main-DvteKhS8.js";import"./modulepreload-polyfill-B5Qt9EMX.js";import"./index-leM889oV.js";import"./widget-html-BAV4ZZd8.js";const o="0.6.1",r={version:o},a=r.version;export{p as LayoutPool,m as LayoutTranslator,L as RendererLite,a as VERSION};
|
|
2
|
+
//# sourceMappingURL=index-Cq9aOTTR.js.map
|
|
@@ -1 +1 @@
|
|
|
1
|
-
{"version":3,"file":"index-
|
|
1
|
+
{"version":3,"file":"index-Cq9aOTTR.js","sources":["../../../renderer/src/index.js"],"sourcesContent":["// @xiboplayer/renderer - Layout rendering\nimport pkg from '../package.json' with { type: 'json' };\nexport const VERSION = pkg.version;\nexport { RendererLite } from './renderer-lite.js';\nexport { LayoutPool } from './layout-pool.js';\nexport { LayoutTranslator } from './layout.js';\n"],"names":["VERSION","pkg"],"mappings":"8LAEaA,EAAUC,EAAI"}
|
|
@@ -1 +1 @@
|
|
|
1
|
-
{"version":3,"file":"index-
|
|
1
|
+
{"version":3,"file":"index-D81Qhc3r.js","sources":["../../../xmds/src/index.js"],"sourcesContent":["// @xiboplayer/xmds - XMDS clients (REST and SOAP)\nimport pkg from '../package.json' with { type: 'json' };\nexport const VERSION = pkg.version;\nexport { RestClient } from './rest-client.js';\nexport { XmdsClient } from './xmds-client.js';\nexport { parseScheduleResponse } from './schedule-parser.js';\n"],"names":["VERSION","pkg"],"mappings":"sHAEaA,EAAUC,EAAI"}
|
|
@@ -1,2 +1,2 @@
|
|
|
1
|
-
import{D as y,P as m}from"./main-
|
|
2
|
-
//# sourceMappingURL=index-
|
|
1
|
+
import{D as y,P as m}from"./main-DvteKhS8.js";import{EventEmitter as i}from"./index-leM889oV.js";import"./modulepreload-polyfill-B5Qt9EMX.js";import"./widget-html-BAV4ZZd8.js";const r="0.6.1",a={version:r};class c extends i{constructor(){super(),this.currentLayoutId=null,this.currentScheduleId=null,this.displayName="",this.hardwareKey="",this.playerType="pwa",this.displayStatus="idle",this.screenWidth=0,this.screenHeight=0,this.lastCollectionTime=null,this.lastHeartbeat=null,this.isRegistered=!1}set(t,e){if(this[t]===e)return;const s=this[t];this[t]=e,this.emit("change",t,e,s)}toJSON(){return{currentLayoutId:this.currentLayoutId,currentScheduleId:this.currentScheduleId,displayName:this.displayName,hardwareKey:this.hardwareKey,playerType:this.playerType,displayStatus:this.displayStatus,screenWidth:this.screenWidth,screenHeight:this.screenHeight,lastCollectionTime:this.lastCollectionTime,lastHeartbeat:this.lastHeartbeat,isRegistered:this.isRegistered}}}const d=a.version;export{y as DataConnectorManager,m as PlayerCore,c as PlayerState,d as VERSION};
|
|
2
|
+
//# sourceMappingURL=index-D_aTOqNE.js.map
|
|
@@ -1 +1 @@
|
|
|
1
|
-
{"version":3,"file":"index-
|
|
1
|
+
{"version":3,"file":"index-D_aTOqNE.js","sources":["../../../core/src/state.js","../../../core/src/index.js"],"sourcesContent":["/**\n * Centralized player state\n *\n * Single source of truth for player status, dimensions, and display info.\n * Avoids scattered state across PlayerCore, renderer, and platform layer.\n */\n\nimport { EventEmitter } from '@xiboplayer/utils';\n\nexport class PlayerState extends EventEmitter {\n constructor() {\n super();\n this.currentLayoutId = null;\n this.currentScheduleId = null;\n this.displayName = '';\n this.hardwareKey = '';\n this.playerType = 'pwa';\n this.displayStatus = 'idle'; // idle | collecting | rendering | error\n this.screenWidth = 0;\n this.screenHeight = 0;\n this.lastCollectionTime = null;\n this.lastHeartbeat = null;\n this.isRegistered = false;\n }\n\n /**\n * Update a state property and emit change event\n */\n set(key, value) {\n if (this[key] === value) return;\n const old = this[key];\n this[key] = value;\n this.emit('change', key, value, old);\n }\n\n /**\n * Get snapshot of current state (for status reporting)\n */\n toJSON() {\n return {\n currentLayoutId: this.currentLayoutId,\n currentScheduleId: this.currentScheduleId,\n displayName: this.displayName,\n hardwareKey: this.hardwareKey,\n playerType: this.playerType,\n displayStatus: this.displayStatus,\n screenWidth: this.screenWidth,\n screenHeight: this.screenHeight,\n lastCollectionTime: this.lastCollectionTime,\n lastHeartbeat: this.lastHeartbeat,\n isRegistered: this.isRegistered\n };\n }\n}\n","// @xiboplayer/core - Player core orchestration\nimport pkg from '../package.json' with { type: 'json' };\nexport const VERSION = pkg.version;\nexport { PlayerCore } from './player-core.js';\nexport { PlayerState } from './state.js';\nexport { DataConnectorManager } from './data-connectors.js';\n"],"names":["PlayerState","EventEmitter","key","value","old","VERSION","pkg"],"mappings":"8MASO,MAAMA,UAAoBC,CAAa,CAC5C,aAAc,CACZ,MAAK,EACL,KAAK,gBAAkB,KACvB,KAAK,kBAAoB,KACzB,KAAK,YAAc,GACnB,KAAK,YAAc,GACnB,KAAK,WAAa,MAClB,KAAK,cAAgB,OACrB,KAAK,YAAc,EACnB,KAAK,aAAe,EACpB,KAAK,mBAAqB,KAC1B,KAAK,cAAgB,KACrB,KAAK,aAAe,EACtB,CAKA,IAAIC,EAAKC,EAAO,CACd,GAAI,KAAKD,CAAG,IAAMC,EAAO,OACzB,MAAMC,EAAM,KAAKF,CAAG,EACpB,KAAKA,CAAG,EAAIC,EACZ,KAAK,KAAK,SAAUD,EAAKC,EAAOC,CAAG,CACrC,CAKA,QAAS,CACP,MAAO,CACL,gBAAiB,KAAK,gBACtB,kBAAmB,KAAK,kBACxB,YAAa,KAAK,YAClB,YAAa,KAAK,YAClB,WAAY,KAAK,WACjB,cAAe,KAAK,cACpB,YAAa,KAAK,YAClB,aAAc,KAAK,aACnB,mBAAoB,KAAK,mBACzB,cAAe,KAAK,cACpB,aAAc,KAAK,YACzB,CACE,CACF,CCnDY,MAACC,EAAUC,EAAI"}
|
|
@@ -0,0 +1,2 @@
|
|
|
1
|
+
import{I as n,O as i,S as l,b as p,c as d,p as S,s as h}from"./main-DvteKhS8.js";import"./modulepreload-polyfill-B5Qt9EMX.js";import"./index-leM889oV.js";import"./widget-html-BAV4ZZd8.js";const e="0.6.1",r={version:e},t=r.version;export{n as InterruptScheduler,i as OverlayScheduler,l as ScheduleManager,t as VERSION,p as buildScheduleQueue,d as calculateTimeline,S as parseLayoutDuration,h as scheduleManager};
|
|
2
|
+
//# sourceMappingURL=index-Dj2ND9Mx.js.map
|
|
@@ -1 +1 @@
|
|
|
1
|
-
{"version":3,"file":"index-
|
|
1
|
+
{"version":3,"file":"index-Dj2ND9Mx.js","sources":["../../../schedule/src/index.js"],"sourcesContent":["// @xiboplayer/schedule - Campaign scheduling and advanced features\n// Basic scheduling, interrupts, overlays, and dayparting\nimport pkg from '../package.json' with { type: 'json' };\nexport const VERSION = pkg.version;\n\n/**\n * Core schedule manager for basic scheduling and dayparting\n * @module @xiboplayer/schedule\n */\nexport { ScheduleManager, scheduleManager } from './schedule.js';\n\n/**\n * Interrupt scheduler for shareOfVoice layouts\n * @module @xiboplayer/schedule/interrupts\n */\nexport { InterruptScheduler } from './interrupts.js';\n\n/**\n * Overlay layout scheduler\n * @module @xiboplayer/schedule/overlays\n */\nexport { OverlayScheduler } from './overlays.js';\n\n/**\n * Offline timeline calculator — duration parser + timeline simulator\n * @module @xiboplayer/schedule/timeline\n */\nexport { calculateTimeline, parseLayoutDuration, buildScheduleQueue } from './timeline.js';\n"],"names":["VERSION","pkg"],"mappings":"0NAGaA,EAAUC,EAAI"}
|
|
@@ -1,5 +1,5 @@
|
|
|
1
|
-
const D="0.6.
|
|
1
|
+
const D="0.6.1",R={version:D},y={DEBUG:0,INFO:1,WARNING:2,ERROR:3,NONE:4},f=[];class N{constructor(t,e=null){this.name=t,this.useGlobal=e===null,this.useGlobal||this.setLevel(e)}setLevel(t){this.useGlobal=!1,typeof t=="string"?this.level=y[t.toUpperCase()]??y.INFO:this.level=t}getEffectiveLevel(){return this.useGlobal?h.level:this.level}debug(...t){this.getEffectiveLevel()<=y.DEBUG&&console.log(`[${this.name}] DEBUG:`,...t),w("debug",this.name,t)}info(...t){this.getEffectiveLevel()<=y.INFO&&console.log(`[${this.name}]`,...t),w("info",this.name,t)}warn(...t){this.getEffectiveLevel()<=y.WARNING&&console.warn(`[${this.name}]`,...t),w("warning",this.name,t)}error(...t){this.getEffectiveLevel()<=y.ERROR&&console.error(`[${this.name}]`,...t),w("error",this.name,t)}log(t,...e){switch(t.toUpperCase()){case"DEBUG":return this.debug(...e);case"INFO":return this.info(...e);case"WARNING":case"WARN":return this.warn(...e);case"ERROR":return this.error(...e)}}}const h={level:y.WARNING,setGlobalLevel(i){typeof i=="string"?this.level=y[i.toUpperCase()]??y.INFO:this.level=i,console.log(`[Logger] Global log level set to: ${this.getLevelName(this.level)}`)},getLevelName(i){return Object.keys(y).find(t=>y[t]===i)||"UNKNOWN"}};let T=!1;if(typeof window<"u"){const t=new URLSearchParams(window.location.search).get("logLevel"),e=localStorage.getItem("xibo_log_level");t?(h.setGlobalLevel(t),T=!0):e?(h.setGlobalLevel(e),T=!0):h.setGlobalLevel("WARNING")}else typeof self<"u"&&self.swLogLevel&&h.setGlobalLevel(self.swLogLevel);function k(i,t=null){return new N(i,t)}function F(i){h.setGlobalLevel(i),typeof window<"u"&&localStorage.setItem("xibo_log_level",i.toUpperCase())}function H(){return h.getLevelName(h.level)}function W(){return h.level<=y.DEBUG}function Y(i){if(T||!i)return!1;const t=U(i);return h.setGlobalLevel(t),!0}function U(i){switch((i||"").toLowerCase()){case"debug":return"DEBUG";case"info":case"notice":case"audit":return"INFO";case"warning":return"WARNING";case"error":case"critical":case"alert":case"emergency":return"ERROR";default:return"INFO"}}function w(i,t,e){if(f.length!==0)for(const a of f)try{a({level:i,name:t,args:e})}catch{}}function z(i){f.push(i)}function X(i){const t=f.indexOf(i);t>=0&&f.splice(t,1)}class V{constructor(){this.events=new Map}on(t,e){this.events.has(t)||this.events.set(t,[]),this.events.get(t).push(e)}once(t,e){const a=(...s)=>{e(...s),this.off(t,a)};this.on(t,a)}off(t,e){if(!this.events.has(t))return;const a=this.events.get(t),s=a.indexOf(e);s!==-1&&a.splice(s,1)}emit(t,...e){if(!this.events.has(t))return;const a=this.events.get(t).slice();for(const s of a)s(...e)}removeAllListeners(t){t?this.events.delete(t):this.events.clear()}}function b(i,t){const e=new Uint8Array(i);let a="";for(let n=0;n<e.length;n++)a+=String.fromCharCode(e[n]);const s=btoa(a),r=[];for(let n=0;n<s.length;n+=64)r.push(s.substring(n,n+64));return`-----BEGIN ${t}-----
|
|
2
2
|
${r.join(`
|
|
3
3
|
`)}
|
|
4
4
|
-----END ${t}-----`}async function G(){const i=await crypto.subtle.generateKey({name:"RSA-OAEP",modulusLength:1024,publicExponent:new Uint8Array([1,0,1]),hash:"SHA-256"},!0,["encrypt","decrypt"]),t=await crypto.subtle.exportKey("spki",i.publicKey),e=await crypto.subtle.exportKey("pkcs8",i.privateKey);return{publicKeyPem:b(t,"PUBLIC KEY"),privateKeyPem:b(e,"PRIVATE KEY")}}function O(i){return!i||typeof i!="string"?!1:/^-----BEGIN (PUBLIC KEY|PRIVATE KEY)-----\n[A-Za-z0-9+/=\n]+\n-----END \1-----$/.test(i.trim())}var x={};const m="xibo_config",E="xibo-hw-backup",P=1;function _(){const i=typeof process<"u"&&x?x:{},t={cmsUrl:i.CMS_URL||"",cmsKey:i.CMS_KEY||"",displayName:i.DISPLAY_NAME||"",hardwareKey:i.HARDWARE_KEY||"",xmrChannel:i.XMR_CHANNEL||"",googleGeoApiKey:i.GOOGLE_GEO_API_KEY||""};return Object.values(t).some(a=>a!=="")?t:null}class I{constructor(){this.data=this.load(),this._fromEnv||this._restoreHardwareKeyFromBackup()}load(){const t=_();if(t)return this._fromEnv=!0,t;if(typeof localStorage>"u")return{cmsUrl:"",cmsKey:"",displayName:"",hardwareKey:"",xmrChannel:""};const e=localStorage.getItem(m);let a={};if(e)try{a=JSON.parse(e)}catch(r){console.error("[Config] Failed to parse localStorage config:",r)}let s=!1;return!a.hardwareKey||a.hardwareKey.length<10?(console.warn("[Config] Missing/invalid hardwareKey — generating"),a.hardwareKey=this.generateStableHardwareKey(),this._backupHardwareKey(a.hardwareKey),s=!0):console.log("[Config] ✓ Loaded existing hardwareKey:",a.hardwareKey),a.xmrChannel||(console.warn("[Config] Missing xmrChannel — generating"),a.xmrChannel=this.generateXmrChannel(),s=!0),a.cmsUrl=a.cmsUrl||"",a.cmsKey=a.cmsKey||"",a.displayName=a.displayName||"",s&&localStorage.setItem(m,JSON.stringify(a)),a}_backupKeys(t){try{const e=indexedDB.open(E,P);e.onupgradeneeded=()=>{const a=e.result;a.objectStoreNames.contains("keys")||a.createObjectStore("keys")},e.onsuccess=()=>{const a=e.result,s=a.transaction("keys","readwrite"),r=s.objectStore("keys");for(const[n,o]of Object.entries(t))r.put(o,n);s.oncomplete=()=>{console.log("[Config] Keys backed up to IndexedDB:",Object.keys(t).join(", ")),a.close()}}}catch{}}_backupHardwareKey(t){this._backupKeys({hardwareKey:t})}async _restoreHardwareKeyFromBackup(){if(!(typeof indexedDB>"u"))try{const t=await new Promise((r,n)=>{const o=indexedDB.open(E,P);o.onupgradeneeded=()=>{const c=o.result;c.objectStoreNames.contains("keys")||c.createObjectStore("keys")},o.onsuccess=()=>r(o.result),o.onerror=()=>n(o.error)}),a=t.transaction("keys","readonly").objectStore("keys"),s=await new Promise(r=>{const n=a.get("hardwareKey");n.onsuccess=()=>r(n.result),n.onerror=()=>r(null)});t.close(),s&&s!==this.data.hardwareKey?(console.log("[Config] Restoring hardware key from IndexedDB backup:",s),console.log("[Config] (was:",this.data.hardwareKey,")"),this.data.hardwareKey=s,this.save()):!s&&this.data.hardwareKey&&this._backupHardwareKey(this.data.hardwareKey)}catch{}}save(){typeof localStorage<"u"&&localStorage.setItem(m,JSON.stringify(this.data))}isConfigured(){return!!(this.data.cmsUrl&&this.data.cmsKey&&this.data.displayName)}generateStableHardwareKey(){if(typeof crypto<"u"&&crypto.randomUUID){const s="pwa-"+crypto.randomUUID().replace(/-/g,"").substring(0,28);return console.log("[Config] Generated new UUID-based hardware key:",s),s}const e="pwa-"+Array.from({length:28},()=>Math.floor(Math.random()*16).toString(16)).join("");return console.log("[Config] Generated new random hardware key:",e),e}getCanvasFingerprint(){try{const t=document.createElement("canvas"),e=t.getContext("2d");return e?(e.textBaseline="top",e.font="14px Arial",e.fillStyle="#f60",e.fillRect(125,1,62,20),e.fillStyle="#069",e.fillText("Xibo Player",2,15),t.toDataURL()):"no-canvas"}catch{return"canvas-error"}}generateHardwareKey(){return this.generateStableHardwareKey()}generateXmrChannel(){return"xxxxxxxx-xxxx-4xxx-yxxx-xxxxxxxxxxxx".replace(/[xy]/g,t=>{const e=Math.random()*16|0;return(t==="x"?e:e&3|8).toString(16)})}async ensureXmrKeyPair(){if(this.data.xmrPubKey&&O(this.data.xmrPubKey))return;console.log("[Config] Generating RSA key pair for XMR registration...");const{publicKeyPem:t,privateKeyPem:e}=await G();this.data.xmrPubKey=t,this.data.xmrPrivKey=e,this.save(),typeof indexedDB<"u"&&this._backupKeys({xmrPubKey:t,xmrPrivKey:e}),console.log("[Config] RSA key pair generated and saved")}hash(t){let e=2166136261;for(let s=0;s<t.length;s++)e^=t.charCodeAt(s),e+=(e<<1)+(e<<4)+(e<<7)+(e<<8)+(e<<24);e=e>>>0;let a="";for(let s=0;s<4;s++){let r=e+s*1234567;for(let n=0;n<t.length;n++)r^=t.charCodeAt(n)+s,r+=(r<<1)+(r<<4)+(r<<7)+(r<<8)+(r<<24);r=r>>>0,a+=r.toString(16).padStart(8,"0")}return a.substring(0,32)}get cmsUrl(){return this.data.cmsUrl}set cmsUrl(t){this.data.cmsUrl=t,this.save()}get cmsKey(){return this.data.cmsKey}set cmsKey(t){this.data.cmsKey=t,this.save()}get displayName(){return this.data.displayName}set displayName(t){this.data.displayName=t,this.save()}get hardwareKey(){return this.data.hardwareKey||(console.error("[Config] CRITICAL: hardwareKey missing! Generating emergency key."),this.data.hardwareKey=this.generateStableHardwareKey(),this.save()),this.data.hardwareKey}get xmrChannel(){return this.data.xmrChannel||(console.warn("[Config] xmrChannel missing at access time — generating"),this.data.xmrChannel=this.generateXmrChannel(),this.save()),this.data.xmrChannel}get xmrPubKey(){return this.data.xmrPubKey||""}get xmrPrivKey(){return this.data.xmrPrivKey||""}get googleGeoApiKey(){return this.data.googleGeoApiKey||""}set googleGeoApiKey(t){this.data.googleGeoApiKey=t,this.save()}}const M=new I,q=new Set(["serverPort","kioskMode","fullscreen","hideMouseCursor","preventSleep","width","height"]);function J(i,t){const e=new Set([...q,...t||[]]),a={};for(const[s,r]of Object.entries(i))e.has(s)||(a[s]=r);return Object.keys(a).length>0?a:void 0}const $=k("FetchRetry"),S=3e4,L=12e4;function v(i){if(!i)return S;const t=Number(i);if(!isNaN(t)&&t>=0)return Math.min(t*1e3,L);const e=new Date(i);if(!isNaN(e.getTime())){const a=e.getTime()-Date.now();return Math.min(Math.max(a,0),L)}return S}async function Z(i,t={},e={}){const{maxRetries:a=3,baseDelayMs:s=1e3,maxDelayMs:r=3e4}=e;let n,o;for(let c=0;c<=a;c++){try{const l=await fetch(i,t);if(l.status===429){const u=v(l.headers.get("Retry-After"));if($.debug(`429 Rate limited, waiting ${u}ms (Retry-After: ${l.headers.get("Retry-After")})`),o=l,n=new Error("HTTP 429: Too Many Requests"),n.status=429,c<a){await new Promise(d=>setTimeout(d,u));continue}break}if(l.ok||l.status>=400&&l.status<500)return l;if(o=l,n=new Error(`HTTP ${l.status}: ${l.statusText}`),n.status=l.status,l.status===503&&c<a){const u=l.headers.get("Retry-After");if(u){const d=v(u);$.debug(`503 Service Unavailable, Retry-After: ${u} (${d}ms)`),await new Promise(K=>setTimeout(K,d));continue}}}catch(l){n=l,o=null}if(c<a){const u=Math.min(s*Math.pow(2,c),r)*(.5+Math.random()*.5);$.debug(`Retry ${c+1}/${a} in ${Math.round(u)}ms:`,String(i).slice(0,80)),await new Promise(d=>setTimeout(d,u))}}if(o)return o;throw n}const p=k("CmsApi");class Q{constructor({baseUrl:t,clientId:e,clientSecret:a,apiToken:s}={}){this.baseUrl=(t||"").replace(/\/+$/,""),this.clientId=e||null,this.clientSecret=a||null,this.accessToken=s||null,this.tokenExpiry=s?1/0:0}async authenticate(){p.info("Authenticating with CMS API...");const t=await fetch(`${this.baseUrl}/api/authorize/access_token`,{method:"POST",headers:{"Content-Type":"application/x-www-form-urlencoded"},body:new URLSearchParams({grant_type:"client_credentials",client_id:this.clientId,client_secret:this.clientSecret})});if(!t.ok){const a=await t.text();throw new Error(`OAuth2 authentication failed (${t.status}): ${a}`)}const e=await t.json();return this.accessToken=e.access_token,this.tokenExpiry=Date.now()+(e.expires_in||3600)*1e3,p.info("Authenticated successfully, token expires in",e.expires_in,"s"),this.accessToken}async ensureToken(){if(!(this.accessToken&&Date.now()<this.tokenExpiry-6e4)){if(!this.clientId||!this.clientSecret){if(this.accessToken)return;throw new g("AUTH","/authorize",0,"No valid token and no OAuth2 credentials")}await this.authenticate()}}async request(t,e,a={}){var c;await this.ensureToken();const s=new URL(`${this.baseUrl}/api${e}`),r={method:t,headers:{Authorization:`Bearer ${this.accessToken}`}};if(t==="GET")for(const[l,u]of Object.entries(a))u!=null&&s.searchParams.set(l,String(u));else r.headers["Content-Type"]="application/x-www-form-urlencoded",r.body=new URLSearchParams(a);const n=await fetch(s,r);if(!n.ok){const l=await n.text();let u;try{const d=JSON.parse(l);u=((c=d.error)==null?void 0:c.message)||d.message||l}catch{u=l}throw new g(t,e,n.status,u)}return(n.headers.get("Content-Type")||"").includes("application/json")?n.json():null}get(t,e){return this.request("GET",t,e)}post(t,e){return this.request("POST",t,e)}put(t,e){return this.request("PUT",t,e)}del(t){return this.request("DELETE",t)}async findDisplay(t){p.info("Looking up display by hardwareKey:",t);const e=await this.request("GET","/display",{hardwareKey:t}),a=Array.isArray(e)?e:[];if(a.length===0)return p.info("No display found for hardwareKey:",t),null;const s=a[0];return p.info(`Found display: ${s.display} (ID: ${s.displayId}, licensed: ${s.licensed})`),s}async authorizeDisplay(t){p.info("Authorizing display:",t),await this.request("PUT",`/display/authorise/${t}`),p.info("Display authorized successfully")}async editDisplay(t,e){return p.info("Editing display:",t,e),this.request("PUT",`/display/${t}`,e)}async listDisplays(t={}){const e=await this.request("GET","/display",t);return Array.isArray(e)?e:[]}async requestScreenshot(t){await this.request("PUT",`/display/requestscreenshot/${t}`)}async getDisplayStatus(t){return this.request("GET",`/display/status/${t}`)}async requestMultipart(t,e,a){var o;await this.ensureToken();const s=`${this.baseUrl}/api${e}`,r=await fetch(s,{method:t,headers:{Authorization:`Bearer ${this.accessToken}`},body:a});if(!r.ok){const c=await r.text();let l;try{const u=JSON.parse(c);l=((o=u.error)==null?void 0:o.message)||u.message||c}catch{l=c}throw new Error(`CMS API ${t} ${e} failed (${r.status}): ${l}`)}return(r.headers.get("Content-Type")||"").includes("application/json")?r.json():null}async createLayout({name:t,resolutionId:e,description:a}){const s={name:t,resolutionId:e};return a&&(s.description=a),this.request("POST","/layout",s)}async listLayouts(t={}){const e=await this.request("GET","/layout",t);return Array.isArray(e)?e:[]}async getLayout(t){return this.request("GET",`/layout/${t}`)}async deleteLayout(t){await this.request("DELETE",`/layout/${t}`)}async publishLayout(t){await this.request("PUT",`/layout/publish/${t}`,{publishNow:1})}async checkoutLayout(t){return this.request("PUT",`/layout/checkout/${t}`)}async getDraftLayout(t){const e=await this.listLayouts({parentId:t});return e.length>0?e[0]:null}async editLayoutBackground(t,e){return this.request("PUT",`/layout/background/${t}`,e)}async addRegion(t,e){return this.request("POST",`/region/${t}`,e)}async editRegion(t,e){return this.request("PUT",`/region/${t}`,e)}async deleteRegion(t){await this.request("DELETE",`/region/${t}`)}async addWidget(t,e,a={}){const{templateId:s,displayOrder:r,...n}=a,o={};s!==void 0&&(o.templateId=s),r!==void 0&&(o.displayOrder=r);const c=await this.request("POST",`/playlist/widget/${t}/${e}`,o);return Object.keys(n).length>0?(n.duration!==void 0&&n.useDuration===void 0&&(n.useDuration=1),this.request("PUT",`/playlist/widget/${c.widgetId}`,n)):c}async editWidget(t,e){return this.request("PUT",`/playlist/widget/${t}`,e)}async deleteWidget(t){await this.request("DELETE",`/playlist/widget/${t}`)}async uploadMedia(t){return this.requestMultipart("POST","/library",t)}async listMedia(t={}){const e=await this.request("GET","/library",t);return Array.isArray(e)?e:[]}async getMedia(t){return this.request("GET",`/library/${t}`)}async deleteMedia(t){await this.request("DELETE",`/library/${t}`)}async createCampaign(t){return this.request("POST","/campaign",{name:t})}async listCampaigns(t={}){const e=await this.request("GET","/campaign",t);return Array.isArray(e)?e:[]}async deleteCampaign(t){await this.request("DELETE",`/campaign/${t}`)}async assignLayoutToCampaign(t,e,a){const s={layoutId:e};a!==void 0&&(s.displayOrder=a),await this.request("POST",`/campaign/layout/assign/${t}`,s)}async createSchedule(t){const e={...t};Array.isArray(e.displayGroupIds)&&delete e.displayGroupIds,await this.ensureToken();const a=`${this.baseUrl}/api/schedule`,s=new URLSearchParams;for(const[o,c]of Object.entries(e))c!=null&&s.set(o,String(c));if(Array.isArray(t.displayGroupIds))for(const o of t.displayGroupIds)s.append("displayGroupIds[]",String(o));const r=await fetch(a,{method:"POST",headers:{Authorization:`Bearer ${this.accessToken}`,"Content-Type":"application/x-www-form-urlencoded"},body:s});if(!r.ok){const o=await r.text();throw new Error(`CMS API POST /schedule failed (${r.status}): ${o}`)}return(r.headers.get("Content-Type")||"").includes("application/json")?r.json():null}async deleteSchedule(t){await this.request("DELETE",`/schedule/${t}`)}async listSchedules(t={}){const e=await this.request("GET","/schedule/data/events",t);return Array.isArray(e)?e:(e==null?void 0:e.events)||[]}async listDisplayGroups(t={}){const e=await this.request("GET","/displaygroup",t);return Array.isArray(e)?e:[]}async createDisplayGroup(t,e){const a={displayGroup:t};return e&&(a.description=e),this.request("POST","/displaygroup",a)}async deleteDisplayGroup(t){await this.request("DELETE",`/displaygroup/${t}`)}async assignDisplayToGroup(t,e){await this.ensureToken();const a=`${this.baseUrl}/api/displaygroup/${t}/display/assign`,s=new URLSearchParams;s.append("displayId[]",String(e));const r=await fetch(a,{method:"POST",headers:{Authorization:`Bearer ${this.accessToken}`,"Content-Type":"application/x-www-form-urlencoded"},body:s});if(!r.ok){const n=await r.text();throw new Error(`CMS API assign display to group failed (${r.status}): ${n}`)}}async unassignDisplayFromGroup(t,e){await this.ensureToken();const a=`${this.baseUrl}/api/displaygroup/${t}/display/unassign`,s=new URLSearchParams;s.append("displayId[]",String(e));const r=await fetch(a,{method:"POST",headers:{Authorization:`Bearer ${this.accessToken}`,"Content-Type":"application/x-www-form-urlencoded"},body:s});if(!r.ok){const n=await r.text();throw new Error(`CMS API unassign display from group failed (${r.status}): ${n}`)}}async listResolutions(){const t=await this.request("GET","/resolution");return Array.isArray(t)?t:[]}async listTemplates(t={}){const e=await this.request("GET","/template",t);return Array.isArray(e)?e:[]}async assignMediaToPlaylist(t,e){const a=Array.isArray(e)?e:[e];await this.ensureToken();const s=`${this.baseUrl}/api/playlist/library/assign/${t}`,r=new URLSearchParams;for(const c of a)r.append("media[]",String(c));const n=await fetch(s,{method:"POST",headers:{Authorization:`Bearer ${this.accessToken}`,"Content-Type":"application/x-www-form-urlencoded"},body:r});if(!n.ok){const c=await n.text();throw new g("POST",`/playlist/library/assign/${t}`,n.status,c)}return(n.headers.get("Content-Type")||"").includes("application/json")?n.json():null}async editLayout(t,e){return this.request("PUT",`/layout/${t}`,e)}async copyLayout(t,e={}){return this.post(`/layout/copy/${t}`,e)}async discardLayout(t){await this.put(`/layout/discard/${t}`)}async editCampaign(t,e){return this.put(`/campaign/${t}`,e)}async getCampaign(t){return this.get(`/campaign/${t}`)}async unassignLayoutFromCampaign(t,e){await this.post(`/campaign/layout/unassign/${t}`,{layoutId:e})}async editSchedule(t,e){return this.put(`/schedule/${t}`,e)}async retireLayout(t){await this.put(`/layout/retire/${t}`)}async unretireLayout(t){await this.put(`/layout/unretire/${t}`)}async getLayoutStatus(t){return this.get(`/layout/status/${t}`)}async tagLayout(t,e){await this.post(`/layout/${t}/tag`,{tag:e.join(",")})}async untagLayout(t,e){await this.post(`/layout/${t}/untag`,{tag:e.join(",")})}async listCommands(t={}){const e=await this.get("/command",t);return Array.isArray(e)?e:[]}async createCommand(t){return this.post("/command",t)}async editCommand(t,e){return this.put(`/command/${t}`,e)}async deleteCommand(t){await this.del(`/command/${t}`)}async deleteDisplay(t){await this.del(`/display/${t}`)}async wolDisplay(t){await this.post(`/display/wol/${t}`)}async setDefaultLayout(t,e){return this.put(`/display/${t}`,{defaultLayoutId:e})}async purgeDisplay(t){await this.post(`/display/purge/${t}`)}async listDayParts(t={}){const e=await this.get("/daypart",t);return Array.isArray(e)?e:[]}async createDayPart(t){return this.post("/daypart",t)}async editDayPart(t,e){return this.put(`/daypart/${t}`,e)}async deleteDayPart(t){await this.del(`/daypart/${t}`)}async uploadMediaUrl(t,e){return this.post("/library",{url:t,name:e,type:"uri"})}async copyMedia(t){return this.post(`/library/copy/${t}`)}async downloadMedia(t){await this.ensureToken();const e=`${this.baseUrl}/api/library/download/${t}`,a=await fetch(e,{headers:{Authorization:`Bearer ${this.accessToken}`}});if(!a.ok){const s=await a.text();throw new g("GET",`/library/download/${t}`,a.status,s)}return a}async editMedia(t,e){return this.put(`/library/${t}`,e)}async getMediaUsage(t){return this.get(`/library/usage/${t}`)}async tidyLibrary(){await this.post("/library/tidy")}async listPlaylists(t={}){const e=await this.get("/playlist",t);return Array.isArray(e)?e:[]}async createPlaylist(t){return this.post("/playlist",{name:t})}async getPlaylist(t){return this.get(`/playlist/${t}`)}async editPlaylist(t,e){return this.put(`/playlist/${t}`,e)}async deletePlaylist(t){await this.del(`/playlist/${t}`)}async reorderPlaylist(t,e){await this.ensureToken();const a=`${this.baseUrl}/api/playlist/order/${t}`,s=new URLSearchParams;for(const n of e)s.append("widgets[]",String(n));const r=await fetch(a,{method:"POST",headers:{Authorization:`Bearer ${this.accessToken}`,"Content-Type":"application/x-www-form-urlencoded"},body:s});if(!r.ok){const n=await r.text();throw new g("POST",`/playlist/order/${t}`,r.status,n)}}async copyPlaylist(t){return this.post(`/playlist/copy/${t}`)}async setWidgetTransition(t,e,a={}){return this.put(`/playlist/widget/transition/${t}`,{type:e,...a})}async setWidgetAudio(t,e){return this.put(`/playlist/widget/${t}/audio`,e)}async removeWidgetAudio(t){await this.del(`/playlist/widget/${t}/audio`)}async setWidgetExpiry(t,e){return this.put(`/playlist/widget/${t}/expiry`,e)}async saveAsTemplate(t,e){return this.post(`/template/${t}`,e)}async getTemplate(t){return this.get(`/template/${t}`)}async deleteTemplate(t){await this.del(`/template/${t}`)}async listDatasets(t={}){const e=await this.get("/dataset",t);return Array.isArray(e)?e:[]}async createDataset(t){return this.post("/dataset",t)}async editDataset(t,e){return this.put(`/dataset/${t}`,e)}async deleteDataset(t){await this.del(`/dataset/${t}`)}async listDatasetColumns(t){const e=await this.get(`/dataset/${t}/column`);return Array.isArray(e)?e:[]}async createDatasetColumn(t,e){return this.post(`/dataset/${t}/column`,e)}async editDatasetColumn(t,e,a){return this.put(`/dataset/${t}/column/${e}`,a)}async deleteDatasetColumn(t,e){await this.del(`/dataset/${t}/column/${e}`)}async listDatasetData(t,e={}){const a=await this.get(`/dataset/data/${t}`,e);return Array.isArray(a)?a:[]}async addDatasetRow(t,e){return this.post(`/dataset/data/${t}`,e)}async editDatasetRow(t,e,a){return this.put(`/dataset/data/${t}/${e}`,a)}async deleteDatasetRow(t,e){await this.del(`/dataset/data/${t}/${e}`)}async importDatasetCsv(t,e){return this.requestMultipart("POST",`/dataset/import/${t}`,e)}async clearDataset(t){await this.del(`/dataset/data/${t}`)}async listNotifications(t={}){const e=await this.get("/notification",t);return Array.isArray(e)?e:[]}async createNotification(t){return this.post("/notification",t)}async editNotification(t,e){return this.put(`/notification/${t}`,e)}async deleteNotification(t){await this.del(`/notification/${t}`)}async listFolders(t={}){const e=await this.get("/folder",t);return Array.isArray(e)?e:[]}async createFolder(t){return this.post("/folder",t)}async editFolder(t,e){return this.put(`/folder/${t}`,e)}async deleteFolder(t){await this.del(`/folder/${t}`)}async listTags(t={}){const e=await this.get("/tag",t);return Array.isArray(e)?e:[]}async createTag(t){return this.post("/tag",t)}async editTag(t,e){return this.put(`/tag/${t}`,e)}async deleteTag(t){await this.del(`/tag/${t}`)}async tagEntity(t,e,a){await this.post(`/${t}/${e}/tag`,{tag:a.join(",")})}async untagEntity(t,e,a){await this.post(`/${t}/${e}/untag`,{tag:a.join(",")})}async dgChangeLayout(t,e){await this.post(`/displaygroup/${t}/action/changeLayout`,{layoutId:e})}async dgOverlayLayout(t,e){await this.post(`/displaygroup/${t}/action/overlayLayout`,{layoutId:e})}async dgRevertToSchedule(t){await this.post(`/displaygroup/${t}/action/revertToSchedule`)}async dgCollectNow(t){await this.post(`/displaygroup/${t}/action/collectNow`)}async dgSendCommand(t,e){await this.post(`/displaygroup/${t}/action/command`,{commandId:e})}async editDisplayGroup(t,e){return this.put(`/displaygroup/${t}`,e)}}class g extends Error{constructor(t,e,a,s){super(`CMS API ${t} ${e} → ${a}: ${s}`),this.name="CmsApiError",this.method=t,this.path=e,this.status=a,this.detail=s}}const tt=R.version,j="/api/v2/player";var C;let A=((C=M.data)==null?void 0:C.playerApiBase)||j,B=A;function et(i){A=i.replace(/\/+$/,""),B=A}export{Q as CmsApiClient,g as CmsApiError,V as EventEmitter,y as LOG_LEVELS,B as PLAYER_API,q as SHELL_ONLY_KEYS,tt as VERSION,Y as applyCmsLogLevel,M as config,k as createLogger,J as extractPwaConfig,Z as fetchWithRetry,H as getLogLevel,W as isDebug,U as mapCmsLogLevel,z as registerLogSink,F as setLogLevel,et as setPlayerApi,X as unregisterLogSink};
|
|
5
|
-
//# sourceMappingURL=index-
|
|
5
|
+
//# sourceMappingURL=index-leM889oV.js.map
|