@reveldigital/player-client 2.2.0 → 2.3.0

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.
@@ -3,6 +3,7 @@ import { Subject } from 'rxjs';
3
3
  * Angular-friendly wrapper around the global `gadgets.reveldigital.datatable` library.
4
4
  *
5
5
  * Provides typed Promise-based methods and RxJS Observables for real-time events.
6
+ * All event emissions run inside Angular's zone to ensure change detection triggers.
6
7
  *
7
8
  * ```typescript
8
9
  * const dt = this.client.createDataTable('tbl_menu_items');
@@ -22,16 +23,18 @@ export class DataTableRef {
22
23
  * Creates a new DataTableRef.
23
24
  *
24
25
  * @param tableId - The data table ID (e.g. 'tbl_menu_items')
26
+ * @param zone - Angular NgZone for ensuring change detection on callbacks
25
27
  * @param options - Optional configuration overrides
26
28
  * @throws Error if the global datatable library is not loaded
27
29
  */
28
- constructor(tableId, options) {
30
+ constructor(tableId, zone, options) {
29
31
  /** Emits when an existing row is modified. */
30
32
  this.rowUpdated$ = new Subject();
31
33
  /** Emits when a new row is added. */
32
34
  this.rowCreated$ = new Subject();
33
35
  /** Emits when a row is removed. */
34
36
  this.rowDeleted$ = new Subject();
37
+ this._zone = zone;
35
38
  const lib = window.gadgets?.['reveldigital.datatable'];
36
39
  if (!lib || typeof lib.create !== 'function') {
37
40
  throw new Error('RevelDigital DataTable library is not available. ' +
@@ -112,20 +115,27 @@ export class DataTableRef {
112
115
  this.rowDeleted$.complete();
113
116
  }
114
117
  /** @ignore */
115
- static _fromInstance(instance) {
118
+ static _fromInstance(instance, zone) {
116
119
  const ref = Object.create(DataTableRef.prototype);
117
120
  ref.rowUpdated$ = new Subject();
118
121
  ref.rowCreated$ = new Subject();
119
122
  ref.rowDeleted$ = new Subject();
120
123
  ref._instance = instance;
124
+ ref._zone = zone;
121
125
  ref._wireEvents();
122
126
  return ref;
123
127
  }
124
128
  /** @ignore */
125
129
  _wireEvents() {
126
- this._onRowUpdated = (change) => this.rowUpdated$.next(change);
127
- this._onRowCreated = (change) => this.rowCreated$.next(change);
128
- this._onRowDeleted = (change) => this.rowDeleted$.next(change);
130
+ this._onRowUpdated = (change) => {
131
+ this._zone.run(() => this.rowUpdated$.next(change));
132
+ };
133
+ this._onRowCreated = (change) => {
134
+ this._zone.run(() => this.rowCreated$.next(change));
135
+ };
136
+ this._onRowDeleted = (change) => {
137
+ this._zone.run(() => this.rowDeleted$.next(change));
138
+ };
129
139
  this._instance.on('rowUpdated', this._onRowUpdated);
130
140
  this._instance.on('rowCreated', this._onRowCreated);
131
141
  this._instance.on('rowDeleted', this._onRowDeleted);
@@ -155,10 +165,11 @@ export class DataTablePrefRef {
155
165
  * Creates a new DataTablePrefRef from a gadget preference JSON string.
156
166
  *
157
167
  * @param prefValue - The raw gadget preference string (JSON)
168
+ * @param zone - Angular NgZone for ensuring change detection on callbacks
158
169
  * @param options - Optional configuration overrides
159
170
  * @throws Error if the global datatable library is not loaded
160
171
  */
161
- constructor(prefValue, options) {
172
+ constructor(prefValue, zone, options) {
162
173
  const lib = window.gadgets?.['reveldigital.datatable'];
163
174
  if (!lib || typeof lib.createFromPref !== 'function') {
164
175
  throw new Error('RevelDigital DataTable library is not available. ' +
@@ -166,7 +177,7 @@ export class DataTablePrefRef {
166
177
  }
167
178
  this._config = lib.createFromPref(prefValue, options);
168
179
  this.pref = this._config.pref;
169
- this.dataTable = DataTableRef._fromInstance(this._config.dt);
180
+ this.dataTable = DataTableRef._fromInstance(this._config.dt, zone);
170
181
  }
171
182
  /**
172
183
  * Fetches rows with the filter and sort settings from the preference automatically applied.
@@ -193,4 +204,4 @@ export class DataTablePrefRef {
193
204
  this.dataTable.dispose();
194
205
  }
195
206
  }
196
- //# sourceMappingURL=data:application/json;base64,{"version":3,"file":"datatable-ref.js","sourceRoot":"","sources":["../../../../../projects/reveldigital/player-client/src/lib/datatable-ref.ts"],"names":[],"mappings":"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","sourcesContent":["import { Subject } from 'rxjs';\nimport {\n  IDataTableChangeEvent,\n  IDataTableColumn,\n  IDataTableOptions,\n  IDataTablePref,\n  IDataTableQueryParams,\n  IDataTableResult,\n  IDataTableSchema\n} from './interfaces/datatable.interface';\n\n/**\n * Angular-friendly wrapper around the global `gadgets.reveldigital.datatable` library.\n *\n * Provides typed Promise-based methods and RxJS Observables for real-time events.\n *\n * ```typescript\n * const dt = this.client.createDataTable('tbl_menu_items');\n *\n * // Fetch rows\n * const result = await dt.getRows({ sort: 'price', sortDir: 'asc' });\n *\n * // Real-time updates\n * dt.rowUpdated$.subscribe(change => console.log('Updated:', change));\n *\n * // Cleanup\n * dt.dispose();\n * ```\n */\nexport class DataTableRef {\n\n  /** Emits when an existing row is modified. */\n  public rowUpdated$ = new Subject<IDataTableChangeEvent>();\n\n  /** Emits when a new row is added. */\n  public rowCreated$ = new Subject<IDataTableChangeEvent>();\n\n  /** Emits when a row is removed. */\n  public rowDeleted$ = new Subject<IDataTableChangeEvent>();\n\n  /** @ignore */\n  private _instance: any;\n\n  /** @ignore */\n  private _onRowUpdated!: (change: IDataTableChangeEvent) => void;\n  /** @ignore */\n  private _onRowCreated!: (change: IDataTableChangeEvent) => void;\n  /** @ignore */\n  private _onRowDeleted!: (change: IDataTableChangeEvent) => void;\n\n  /**\n   * Creates a new DataTableRef.\n   *\n   * @param tableId - The data table ID (e.g. 'tbl_menu_items')\n   * @param options - Optional configuration overrides\n   * @throws Error if the global datatable library is not loaded\n   */\n  constructor(tableId: string, options?: IDataTableOptions) {\n\n    const lib = (window as any).gadgets?.['reveldigital.datatable'];\n\n    if (!lib || typeof lib.create !== 'function') {\n      throw new Error(\n        'RevelDigital DataTable library is not available. ' +\n        'Ensure the datatable feature is enabled for this gadget.'\n      );\n    }\n\n    this._instance = lib.create(tableId, options);\n    this._wireEvents();\n  }\n\n  /**\n   * Fetches rows from the data table.\n   *\n   * @param params - Optional query parameters (filter, sort, pagination)\n   * @returns Promise resolving to the result set\n   *\n   * ```typescript\n   * const result = await dt.getRows({\n   *   filter: { category: 'Entree', price: { op: 'lte', value: 25 } },\n   *   sort: 'itemName',\n   *   sortDir: 'asc',\n   *   pageSize: 20\n   * });\n   * ```\n   */\n  public getRows(params?: IDataTableQueryParams): Promise<IDataTableResult> {\n    return this._instance.getRows(params);\n  }\n\n  /**\n   * Fetches the table schema (column definitions and metadata).\n   *\n   * @returns Promise resolving to the table schema\n   */\n  public getSchema(): Promise<IDataTableSchema> {\n    return this._instance.getSchema();\n  }\n\n  /**\n   * Gets visible (non-hidden) columns from the table schema.\n   *\n   * @returns Promise resolving to an array of visible column definitions\n   */\n  public getVisibleColumns(): Promise<IDataTableColumn[]> {\n    return this._instance.getVisibleColumns();\n  }\n\n  /**\n   * Fetches rows with hidden column data stripped.\n   *\n   * @param params - Optional query parameters (same as getRows)\n   * @returns Promise resolving to the result set with hidden fields removed\n   */\n  public getVisibleRows(params?: IDataTableQueryParams): Promise<IDataTableResult> {\n    return this._instance.getVisibleRows(params);\n  }\n\n  /**\n   * Starts polling for changes at the given interval.\n   * Emits on `rowUpdated$` when new data is detected.\n   *\n   * @param intervalMs - Polling interval in milliseconds (default 30000)\n   */\n  public startPolling(intervalMs?: number): void {\n    this._instance.startPolling(intervalMs);\n  }\n\n  /**\n   * Stops polling for changes.\n   */\n  public stopPolling(): void {\n    this._instance.stopPolling();\n  }\n\n  /**\n   * Releases all resources: stops polling, closes the real-time connection,\n   * removes event listeners, and completes all RxJS observables.\n   */\n  public dispose(): void {\n    this._instance.off('rowUpdated', this._onRowUpdated);\n    this._instance.off('rowCreated', this._onRowCreated);\n    this._instance.off('rowDeleted', this._onRowDeleted);\n\n    this._instance.dispose();\n\n    this.rowUpdated$.complete();\n    this.rowCreated$.complete();\n    this.rowDeleted$.complete();\n  }\n\n  /** @ignore */\n  static _fromInstance(instance: any): DataTableRef {\n    const ref = Object.create(DataTableRef.prototype) as DataTableRef;\n    ref.rowUpdated$ = new Subject<IDataTableChangeEvent>();\n    ref.rowCreated$ = new Subject<IDataTableChangeEvent>();\n    ref.rowDeleted$ = new Subject<IDataTableChangeEvent>();\n    ref._instance = instance;\n    ref._wireEvents();\n    return ref;\n  }\n\n  /** @ignore */\n  private _wireEvents(): void {\n    this._onRowUpdated = (change: IDataTableChangeEvent) => this.rowUpdated$.next(change);\n    this._onRowCreated = (change: IDataTableChangeEvent) => this.rowCreated$.next(change);\n    this._onRowDeleted = (change: IDataTableChangeEvent) => this.rowDeleted$.next(change);\n\n    this._instance.on('rowUpdated', this._onRowUpdated);\n    this._instance.on('rowCreated', this._onRowCreated);\n    this._instance.on('rowDeleted', this._onRowDeleted);\n  }\n}\n\n\n/**\n * Wrapper around a data table created from a gadget preference value.\n *\n * Automatically configures filter and sort settings from the preference,\n * and provides a `getFilteredRows()` convenience method that applies them.\n *\n * ```typescript\n * const cfg = this.client.createDataTableFromPref(prefs.getString('rdDataTable'));\n *\n * // Fetch rows with auto-wired filter + sort from the preference\n * const result = await cfg.getFilteredRows();\n *\n * // Access the underlying DataTableRef for schema, events, etc.\n * cfg.dataTable.rowUpdated$.subscribe(change => console.log(change));\n *\n * // Cleanup\n * cfg.dispose();\n * ```\n */\nexport class DataTablePrefRef {\n\n  /** The underlying DataTableRef with full access to schema, events, polling, etc. */\n  public readonly dataTable: DataTableRef;\n\n  /** The parsed preference object. */\n  public readonly pref: IDataTablePref;\n\n  /** @ignore */\n  private _config: any;\n\n  /**\n   * Creates a new DataTablePrefRef from a gadget preference JSON string.\n   *\n   * @param prefValue - The raw gadget preference string (JSON)\n   * @param options - Optional configuration overrides\n   * @throws Error if the global datatable library is not loaded\n   */\n  constructor(prefValue: string, options?: IDataTableOptions) {\n\n    const lib = (window as any).gadgets?.['reveldigital.datatable'];\n\n    if (!lib || typeof lib.createFromPref !== 'function') {\n      throw new Error(\n        'RevelDigital DataTable library is not available. ' +\n        'Ensure the datatable feature is enabled for this gadget.'\n      );\n    }\n\n    this._config = lib.createFromPref(prefValue, options);\n    this.pref = this._config.pref;\n    this.dataTable = DataTableRef._fromInstance(this._config.dt);\n  }\n\n  /**\n   * Fetches rows with the filter and sort settings from the preference automatically applied.\n   * Additional query parameters can override or supplement the preference settings.\n   *\n   * @param params - Optional additional query parameters\n   * @returns Promise resolving to the result set\n   *\n   * ```typescript\n   * // Use preference defaults\n   * const result = await cfg.getFilteredRows();\n   *\n   * // Override page size\n   * const page = await cfg.getFilteredRows({ pageSize: 10 });\n   * ```\n   */\n  public getFilteredRows(params?: IDataTableQueryParams): Promise<IDataTableResult> {\n    return this._config.getFilteredRows(params);\n  }\n\n  /**\n   * Releases all resources held by the underlying DataTableRef.\n   */\n  public dispose(): void {\n    this.dataTable.dispose();\n  }\n}\n"]}
207
+ //# sourceMappingURL=data:application/json;base64,{"version":3,"file":"datatable-ref.js","sourceRoot":"","sources":["../../../../../projects/reveldigital/player-client/src/lib/datatable-ref.ts"],"names":[],"mappings":"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","sourcesContent":["import { NgZone } from '@angular/core';\nimport { Subject } from 'rxjs';\nimport {\n  IDataTableChangeEvent,\n  IDataTableColumn,\n  IDataTableOptions,\n  IDataTablePref,\n  IDataTableQueryParams,\n  IDataTableResult,\n  IDataTableSchema\n} from './interfaces/datatable.interface';\n\n/**\n * Angular-friendly wrapper around the global `gadgets.reveldigital.datatable` library.\n *\n * Provides typed Promise-based methods and RxJS Observables for real-time events.\n * All event emissions run inside Angular's zone to ensure change detection triggers.\n *\n * ```typescript\n * const dt = this.client.createDataTable('tbl_menu_items');\n *\n * // Fetch rows\n * const result = await dt.getRows({ sort: 'price', sortDir: 'asc' });\n *\n * // Real-time updates\n * dt.rowUpdated$.subscribe(change => console.log('Updated:', change));\n *\n * // Cleanup\n * dt.dispose();\n * ```\n */\nexport class DataTableRef {\n\n  /** Emits when an existing row is modified. */\n  public rowUpdated$ = new Subject<IDataTableChangeEvent>();\n\n  /** Emits when a new row is added. */\n  public rowCreated$ = new Subject<IDataTableChangeEvent>();\n\n  /** Emits when a row is removed. */\n  public rowDeleted$ = new Subject<IDataTableChangeEvent>();\n\n  /** @ignore */\n  private _instance: any;\n\n  /** @ignore */\n  private _zone: NgZone;\n\n  /** @ignore */\n  private _onRowUpdated!: (change: IDataTableChangeEvent) => void;\n  /** @ignore */\n  private _onRowCreated!: (change: IDataTableChangeEvent) => void;\n  /** @ignore */\n  private _onRowDeleted!: (change: IDataTableChangeEvent) => void;\n\n  /**\n   * Creates a new DataTableRef.\n   *\n   * @param tableId - The data table ID (e.g. 'tbl_menu_items')\n   * @param zone - Angular NgZone for ensuring change detection on callbacks\n   * @param options - Optional configuration overrides\n   * @throws Error if the global datatable library is not loaded\n   */\n  constructor(tableId: string, zone: NgZone, options?: IDataTableOptions) {\n\n    this._zone = zone;\n\n    const lib = (window as any).gadgets?.['reveldigital.datatable'];\n\n    if (!lib || typeof lib.create !== 'function') {\n      throw new Error(\n        'RevelDigital DataTable library is not available. ' +\n        'Ensure the datatable feature is enabled for this gadget.'\n      );\n    }\n\n    this._instance = lib.create(tableId, options);\n    this._wireEvents();\n  }\n\n  /**\n   * Fetches rows from the data table.\n   *\n   * @param params - Optional query parameters (filter, sort, pagination)\n   * @returns Promise resolving to the result set\n   *\n   * ```typescript\n   * const result = await dt.getRows({\n   *   filter: { category: 'Entree', price: { op: 'lte', value: 25 } },\n   *   sort: 'itemName',\n   *   sortDir: 'asc',\n   *   pageSize: 20\n   * });\n   * ```\n   */\n  public getRows(params?: IDataTableQueryParams): Promise<IDataTableResult> {\n    return this._instance.getRows(params);\n  }\n\n  /**\n   * Fetches the table schema (column definitions and metadata).\n   *\n   * @returns Promise resolving to the table schema\n   */\n  public getSchema(): Promise<IDataTableSchema> {\n    return this._instance.getSchema();\n  }\n\n  /**\n   * Gets visible (non-hidden) columns from the table schema.\n   *\n   * @returns Promise resolving to an array of visible column definitions\n   */\n  public getVisibleColumns(): Promise<IDataTableColumn[]> {\n    return this._instance.getVisibleColumns();\n  }\n\n  /**\n   * Fetches rows with hidden column data stripped.\n   *\n   * @param params - Optional query parameters (same as getRows)\n   * @returns Promise resolving to the result set with hidden fields removed\n   */\n  public getVisibleRows(params?: IDataTableQueryParams): Promise<IDataTableResult> {\n    return this._instance.getVisibleRows(params);\n  }\n\n  /**\n   * Starts polling for changes at the given interval.\n   * Emits on `rowUpdated$` when new data is detected.\n   *\n   * @param intervalMs - Polling interval in milliseconds (default 30000)\n   */\n  public startPolling(intervalMs?: number): void {\n    this._instance.startPolling(intervalMs);\n  }\n\n  /**\n   * Stops polling for changes.\n   */\n  public stopPolling(): void {\n    this._instance.stopPolling();\n  }\n\n  /**\n   * Releases all resources: stops polling, closes the real-time connection,\n   * removes event listeners, and completes all RxJS observables.\n   */\n  public dispose(): void {\n    this._instance.off('rowUpdated', this._onRowUpdated);\n    this._instance.off('rowCreated', this._onRowCreated);\n    this._instance.off('rowDeleted', this._onRowDeleted);\n\n    this._instance.dispose();\n\n    this.rowUpdated$.complete();\n    this.rowCreated$.complete();\n    this.rowDeleted$.complete();\n  }\n\n  /** @ignore */\n  static _fromInstance(instance: any, zone: NgZone): DataTableRef {\n    const ref = Object.create(DataTableRef.prototype) as DataTableRef;\n    ref.rowUpdated$ = new Subject<IDataTableChangeEvent>();\n    ref.rowCreated$ = new Subject<IDataTableChangeEvent>();\n    ref.rowDeleted$ = new Subject<IDataTableChangeEvent>();\n    ref._instance = instance;\n    ref._zone = zone;\n    ref._wireEvents();\n    return ref;\n  }\n\n  /** @ignore */\n  private _wireEvents(): void {\n    this._onRowUpdated = (change: IDataTableChangeEvent) => {\n      this._zone.run(() => this.rowUpdated$.next(change));\n    };\n    this._onRowCreated = (change: IDataTableChangeEvent) => {\n      this._zone.run(() => this.rowCreated$.next(change));\n    };\n    this._onRowDeleted = (change: IDataTableChangeEvent) => {\n      this._zone.run(() => this.rowDeleted$.next(change));\n    };\n\n    this._instance.on('rowUpdated', this._onRowUpdated);\n    this._instance.on('rowCreated', this._onRowCreated);\n    this._instance.on('rowDeleted', this._onRowDeleted);\n  }\n}\n\n\n/**\n * Wrapper around a data table created from a gadget preference value.\n *\n * Automatically configures filter and sort settings from the preference,\n * and provides a `getFilteredRows()` convenience method that applies them.\n *\n * ```typescript\n * const cfg = this.client.createDataTableFromPref(prefs.getString('rdDataTable'));\n *\n * // Fetch rows with auto-wired filter + sort from the preference\n * const result = await cfg.getFilteredRows();\n *\n * // Access the underlying DataTableRef for schema, events, etc.\n * cfg.dataTable.rowUpdated$.subscribe(change => console.log(change));\n *\n * // Cleanup\n * cfg.dispose();\n * ```\n */\nexport class DataTablePrefRef {\n\n  /** The underlying DataTableRef with full access to schema, events, polling, etc. */\n  public readonly dataTable: DataTableRef;\n\n  /** The parsed preference object. */\n  public readonly pref: IDataTablePref;\n\n  /** @ignore */\n  private _config: any;\n\n  /**\n   * Creates a new DataTablePrefRef from a gadget preference JSON string.\n   *\n   * @param prefValue - The raw gadget preference string (JSON)\n   * @param zone - Angular NgZone for ensuring change detection on callbacks\n   * @param options - Optional configuration overrides\n   * @throws Error if the global datatable library is not loaded\n   */\n  constructor(prefValue: string, zone: NgZone, options?: IDataTableOptions) {\n\n    const lib = (window as any).gadgets?.['reveldigital.datatable'];\n\n    if (!lib || typeof lib.createFromPref !== 'function') {\n      throw new Error(\n        'RevelDigital DataTable library is not available. ' +\n        'Ensure the datatable feature is enabled for this gadget.'\n      );\n    }\n\n    this._config = lib.createFromPref(prefValue, options);\n    this.pref = this._config.pref;\n    this.dataTable = DataTableRef._fromInstance(this._config.dt, zone);\n  }\n\n  /**\n   * Fetches rows with the filter and sort settings from the preference automatically applied.\n   * Additional query parameters can override or supplement the preference settings.\n   *\n   * @param params - Optional additional query parameters\n   * @returns Promise resolving to the result set\n   *\n   * ```typescript\n   * // Use preference defaults\n   * const result = await cfg.getFilteredRows();\n   *\n   * // Override page size\n   * const page = await cfg.getFilteredRows({ pageSize: 10 });\n   * ```\n   */\n  public getFilteredRows(params?: IDataTableQueryParams): Promise<IDataTableResult> {\n    return this._config.getFilteredRows(params);\n  }\n\n  /**\n   * Releases all resources held by the underlying DataTableRef.\n   */\n  public dispose(): void {\n    this.dataTable.dispose();\n  }\n}\n"]}
@@ -135,6 +135,7 @@ export class PlayerClientService {
135
135
  this.onConfig$.next(null);
136
136
  }
137
137
  }), tap(e => this.onPostMessage$.next(e)));
138
+ this.zone = zone;
138
139
  window.RevelDigital = {
139
140
  Controller: {
140
141
  onCommand: (name, arg) => {
@@ -822,7 +823,7 @@ export class PlayerClientService {
822
823
  * ```
823
824
  */
824
825
  createDataTable(tableId, options) {
825
- return new DataTableRef(tableId, options);
826
+ return new DataTableRef(tableId, this.zone, options);
826
827
  }
827
828
  /**
828
829
  * Creates a typed data table wrapper from a gadget preference value.
@@ -851,7 +852,7 @@ export class PlayerClientService {
851
852
  * ```
852
853
  */
853
854
  createDataTableFromPref(prefValue, options) {
854
- return new DataTablePrefRef(prefValue, options);
855
+ return new DataTablePrefRef(prefValue, this.zone, options);
855
856
  }
856
857
  // ---
857
858
  // PRIVATE METHODS.
@@ -988,4 +989,4 @@ class NoopClient {
988
989
  }
989
990
  }
990
991
  }
991
- //# sourceMappingURL=data:application/json;base64,{"version":3,"file":"player-client.service.js","sourceRoot":"","sources":["../../../../../projects/reveldigital/player-client/src/lib/player-client.service.ts"],"names":[],"mappings":"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","sourcesContent":["import { Injectable, NgZone, OnDestroy } from '@angular/core';\nimport { gadgets } from '@reveldigital/gadget-types';\nimport { BehaviorSubject, fromEvent, Subject, Subscription } from 'rxjs';\nimport { filter, map, share, tap } from 'rxjs/operators';\nimport { IClient } from './interfaces/client.interface';\nimport { ICommand } from './interfaces/command.interface';\nimport { IDictionary } from './interfaces/config.interface';\nimport { IDataTableOptions } from './interfaces/datatable.interface';\nimport { IDevice } from './interfaces/device.interface';\nimport { IEventProperties } from './interfaces/event-properties.interface';\nimport { DataTableRef, DataTablePrefRef } from './datatable-ref';\nimport { version } from './version';\n\n//import { version } from './version.js';\n\n// So that TypeScript doesn't complain, we're going to augment the GLOBAL / WINDOW \n// name-space definition to include the Tracker API. This also provides us with a place\n// to actually DOCUMENT the API so that our developers aren't guessing about what's\n// available on the library.\n\n/** @ignore */\ndeclare global {\n  let Client: IClient;\n}\n\n/**\n * Service for interacting with the Revel Digital player client.\n * \n * This service provides a comprehensive interface for gadgets and web applications\n * to communicate with the Revel Digital player environment. It handles device\n * information, commands, events, analytics tracking, and configuration management.\n * \n * The service supports both direct API calls and event-based communication patterns,\n * allowing gadgets to respond to player lifecycle events (start, stop, commands) and\n * send data back to the player or remote devices.\n * \n * ```typescript\n * constructor(private client: PlayerClientService) {\n *   // Subscribe to player events\n *   this.client.onStart$.subscribe(() => {\n *     console.log('Gadget started');\n *   });\n *   \n *   this.client.onCommand$.subscribe(command => {\n *     console.log('Received command:', command);\n *   });\n * }\n * \n * async ngOnInit() {\n *   // Get device information\n *   const device = await this.client.getDevice();\n *   const deviceTime = await this.client.getDeviceTime();\n *   \n *   // Track analytics\n *   this.client.track('gadget_loaded', { version: '1.0' });\n * }\n * ```\n * \n * @since 1.0.0\n */\n@Injectable({\n  providedIn: 'root'\n})\nexport class PlayerClientService implements OnDestroy {\n\n  /** @ignore */\n  private clientPromise: Promise<IClient> | null;\n\n  /**\n   * Observable stream of commands sent to this player from the Revel Digital platform.\n   * Subscribe to this to handle custom commands sent from templates, playlists, or remote devices.\n   * \n   * ```typescript\n   * this.client.onCommand$.subscribe(command => {\n   *   if (command.name === 'customAction') {\n   *     this.handleCustomAction(command.arg);\n   *   }\n   * });\n   * ```\n   */\n  public onCommand$ = new Subject<ICommand>();\n\n  /**\n   * Observable that signals when the gadget has been loaded and is ready to start.\n   * Emits `true` when ready, `false` when destroyed.\n   * \n   * ```typescript\n   * this.client.onReady$.subscribe(isReady => {\n   *   if (isReady) {\n   *     console.log('Client is ready');\n   *     this.initializeGadget();\n   *   }\n   * });\n   * ```\n   */\n  public onReady$ = new BehaviorSubject(false);\n\n  /**\n   * Observable that signals when the gadget has been started by the player.\n   * This event occurs when the player begins execution of the gadget content.\n   * \n   * ```typescript\n   * this.client.onStart$.subscribe(() => {\n   *   console.log('Gadget started');\n   *   this.startAnimation();\n   * });\n   * ```\n   */\n  public onStart$ = new Subject();\n\n  /**\n   * Observable that signals when the gadget has been stopped by the player.\n   * This event occurs when the player stops execution, typically when moving\n   * to the next item in a playlist or when the content duration expires.\n   * \n   * ```typescript\n   * this.client.onStop$.subscribe(() => {\n   *   console.log('Gadget stopped');\n   *   this.cleanup();\n   * });\n   * ```\n   */\n  public onStop$ = new Subject();\n\n  /**\n   * Observable that signals when the gadget should open the configuration window.\n   * This allows gadgets to respond to configuration requests from the player.\n   * \n   * ```typescript\n   * this.client.onConfig$.subscribe(() => {\n   *   this.openConfigurationDialog();\n   * });\n   * ```\n   */\n  public onConfig$ = new Subject();\n\n  /**\n   * Observable that signals when the gadget has received a postMessage event from the player.\n   * This handles communication between the gadget and player via the postMessage API.\n   * \n   * ```typescript\n   * this.client.onPostMessage$.subscribe(message => {\n   *   console.log('Received message:', message);\n   *   this.handlePlayerMessage(message);\n   * });\n   * ```\n   */\n  public onPostMessage$ = new Subject();\n\n  //\n  // Two methods available for calling into the library:\n  //\n  // 1) Using dispatchEvent() with the following custom events\n  // 2) Using the window scoped RevelDigital object as defined in the constructor\n  //\n  /** @ignore */\n  private onStartSub: Subscription;\n  /** @ignore */\n  private onStartEvt$ = fromEvent(window, 'RevelDigital.Start').pipe(\n    share(),\n    tap(this.onStart$)\n  );\n  /** @ignore */\n  private onStopSub: Subscription;\n  /** @ignore */\n  private onStopEvt$ = fromEvent(window, 'RevelDigital.Stop').pipe(\n    share(),\n    tap(this.onStop$)\n  );\n  /** @ignore */\n  private onCommandSub: Subscription;\n  /** @ignore */\n  private onCommandEvt$ = fromEvent<ICommand>(window, 'RevelDigital.Command').pipe(\n    map((e: any) => { return { name: e.detail.name, arg: e.detail.arg } as ICommand }),\n    share(),\n    tap(this.onCommand$)\n  );\n  /** @ignore */\n  // private onConfigSub: Subscription;\n  // /** @ignore */\n  // private onConfigEvt$ = fromEvent(window, 'RevelDigital.Config').pipe(\n  //   share(),\n  //   tap((e: CustomEvent) => {\n  //     console.log(e);\n\n  //     if (e.detail.type === 'applyConfig' && e.detail.isOpener) {\n  //       this.applyConfig(e.detail.config); // propagate config to iframe parent from the popup window\n  //     } else {\n  //       this.onConfig$.next(e.detail);\n  //     }\n  //   })\n  // );\n  // private onPostMessageSub: Subscription;\n  // private onPostMessageEvt$ = fromEvent(window, 'message').pipe(\n  //   filter((messageEvent: MessageEvent) =>\n  //     messageEvent.source !== window.parent &&\n  //     typeof messageEvent.data === 'string' &&\n  //     messageEvent.data.startsWith('reveldigital:')),\n  //   map((e: any) => { return JSON.parse(e.substring(13)) as Command }),\n  //   share(),\n  //   tap(this.onCommand$)\n  // );\n\n  private onPostMessageSub: Subscription;\n  private onPostMessageEvt$ = fromEvent(window, 'message').pipe(\n    filter((messageEvent: MessageEvent) =>\n      //messageEvent.source !== window.parent &&\n      typeof messageEvent.data === 'string'),\n    map((e: any) => JSON.parse(e.data)),\n    share(),\n    tap((e: any) => {\n      if (e.type === 'applyConfig' && e.isOpener) {\n        e.isOpener = false;\n        // propagate config to iframe parent from the popup window\n        window.parent.postMessage(\n          JSON.stringify(e),\n          '*'\n        );\n      } else if (e.type === 'openConfig') {\n        this.onConfig$.next(null);\n      }\n    }),\n    tap(e => this.onPostMessage$.next(e))\n  );\n\n\n  /** @ignore */\n  constructor(zone: NgZone) {\n\n    (window as any).RevelDigital = {\n      Controller: {\n        onCommand: (name: string, arg: string) => {\n          zone.run(() => {\n            this.onCommand$.next({ name: name, arg: arg });\n          });\n        },\n        onStart: () => {\n          zone.run(() => {\n            this.onStart$.next(null);\n          });\n        },\n        onStop: () => {\n          zone.run(() => {\n            this.onStop$.next(null);\n          });\n        }\n      }\n    }\n\n    this.onStartSub = this.onStartEvt$.subscribe(() => { });\n    this.onStopSub = this.onStopEvt$.subscribe(() => { });\n    this.onCommandSub = this.onCommandEvt$.subscribe(() => { });\n    //this.onConfigSub = this.onConfigEvt$.subscribe(() => { });\n    this.onPostMessageSub = this.onPostMessageEvt$.subscribe(() => { });\n\n    this.clientPromise = null;\n\n    this.onReady$.next(true);\n  }\n\n  /** @ignore */\n  ngOnDestroy(): void {\n\n    this.onStartSub?.unsubscribe();\n    this.onStopSub?.unsubscribe();\n    this.onCommandSub?.unsubscribe();\n    //this.onConfigSub?.unsubscribe();\n    this.onPostMessageSub?.unsubscribe();\n\n    this.onReady$.next(false);\n  }\n\n  /** @ignore */\n  public static init(data: any) {\n\n    console.log(\n      '%c⚙️ Initializing Revel Digital client library',\n      'background-color:blue; color:yellow;'\n    );\n  }\n\n  /**\n   * Sends a callback to player scripting with variable arguments.\n   * \n   * This method allows the gadget to communicate with player scripting.\n   * If the appropriate scripting is in place in the currently running template, \n   * calling this method will initiate a callback which can be acted upon in player script.\n   * \n   * @param args - Variable number of arguments to pass to the callback (max 5 arguments)\n   * \n   * ```typescript\n   * // Send a simple callback\n   * this.client.callback('test', 'data');\n   * \n   * // Send multiple parameters\n   * this.client.callback('action', 'value1', 'value2', { data: 'object' });\n   * ```\n   */\n  public callback(...args: any[]): void {\n\n    this.getClient().then((client) => {\n\n      switch (args.length) {\n        case 0:\n          client.callback();\n          break;\n        case 1:\n          client.callback(args[0]);\n          break;\n        case 2:\n          client.callback(args[1]);\n          break;\n        case 3:\n          client.callback(args[2]);\n          break;\n        case 4:\n          client.callback(args[3]);\n          break;\n        case 5:\n          client.callback(args[4]);\n          break;\n      }\n    })\n  }\n\n  /**\n   * Gets the user preferences interface exposed by the Google Gadgets API.\n   * \n   * This method provides access to gadget preferences which can be configured\n   * in the Revel Digital CMS. Use this to retrieve user-configurable settings\n   * for your gadget.\n   * \n   * @returns The Gadgets API Prefs object for accessing preference values\n   * \n   * ```typescript\n   * constructor(public client: PlayerClientService) {\n   *   const prefs = client.getPrefs();\n   *   const myString = prefs.getString('myStringPref');\n   *   const myNumber = prefs.getInt('myNumberPref');\n   *   const myBool = prefs.getBool('myBoolPref');\n   * }\n   * ```\n   * \n   * @see {@link https://developers.google.com/gadgets/docs/basic} Google Gadgets API documentation\n   */\n  public getPrefs(): gadgets.Prefs {\n\n    return new window['gadgets']['Prefs']();\n  }\n\n  /**\n   * Gets the current device time in ISO8601 format.\n   * \n   * Current device time is determined by the device timezone assigned to the device in the CMS.\n   * This is useful for displaying time-sensitive content or scheduling operations based on\n   * the device's local time rather than browser time.\n   * \n   * @param date - Optional. If supplied, will translate the supplied date/time to device time based on respective timezones\n   * @returns Promise resolving to date/time in ISO8601 format\n   * \n   * ```typescript\n   * // Get current device time\n   * const currentTime = await this.client.getDeviceTime();\n   * console.log('Device time:', currentTime);\n   * \n   * // Convert a specific date to device time\n   * const specificDate = new Date('2023-01-01T12:00:00Z');\n   * const deviceTime = await this.client.getDeviceTime(specificDate);\n   * ```\n   */\n  public async getDeviceTime(date?: Date): Promise<string> {\n\n    const client = await this.getClient();\n\n    if (date !== undefined) {\n      return client.getDeviceTime(date);\n    }\n    return client.getDeviceTime();\n  }\n\n  /**\n   * Gets the timezone name currently assigned to the device.\n   * \n   * @returns Promise resolving to the timezone name (e.g., \"America/New_York\")\n   * \n   * ```typescript\n   * const timezoneName = await this.client.getDeviceTimeZoneName();\n   * console.log('Device timezone:', timezoneName);\n   * ```\n   */\n  public async getDeviceTimeZoneName(): Promise<string> {\n\n    const client = await this.getClient();\n\n    return client.getDeviceTimeZoneName();\n  }\n\n  /**\n   * Gets the timezone ID currently assigned to the device.\n   * \n   * @returns Promise resolving to the timezone ID\n   * \n   * ```typescript\n   * const timezoneId = await this.client.getDeviceTimeZoneID();\n   * console.log('Device timezone ID:', timezoneId);\n   * ```\n   */\n  public async getDeviceTimeZoneID(): Promise<string> {\n\n    const client = await this.getClient();\n\n    return client.getDeviceTimeZoneID();\n  }\n\n  /**\n   * Gets the numerical offset from GMT of the timezone currently assigned to the device.\n   * \n   * @returns Promise resolving to the timezone offset in hours (e.g., -5 for EST)\n   * \n   * ```typescript\n   * const offset = await this.client.getDeviceTimeZoneOffset();\n   * console.log('Device timezone offset:', offset, 'hours from GMT');\n   * ```\n   */\n  public async getDeviceTimeZoneOffset(): Promise<number> {\n\n    const client = await this.getClient();\n\n    return client.getDeviceTimeZoneOffset();\n  }\n\n  /**\n   * Gets the language code of the language currently assigned to the device.\n   * \n   * @returns Promise resolving to the language code (e.g., \"en-US\", \"fr-FR\")\n   * \n   * ```typescript\n   * const languageCode = await this.client.getLanguageCode();\n   * console.log('Device language:', languageCode);\n   * // Use for localization\n   * this.loadLanguageResources(languageCode);\n   * ```\n   */\n  public async getLanguageCode(): Promise<string> {\n\n    const client = await this.getClient();\n\n    return client.getLanguageCode();\n  }\n\n  /**\n   * Gets the unique Revel Digital device key associated with the device.\n   * \n   * The device key is a unique identifier for this specific player device\n   * and can be used for device-specific operations or remote commands.\n   * \n   * @returns Promise resolving to the device key string\n   * \n   * ```typescript\n   * const deviceKey = await this.client.getDeviceKey();\n   * console.log('Device key:', deviceKey);\n   * ```\n   */\n  public async getDeviceKey(): Promise<string> {\n\n    const client = await this.getClient();\n\n    return client.getDeviceKey();\n  }\n\n  /**\n   * Sends a command to the local player device.\n   * \n   * Commands can be used to control various aspects of the player or trigger\n   * specific behaviors. The command is processed by the local player only.\n   * \n   * @param name - The command name/identifier\n   * @param arg - The command argument/payload\n   * \n   * ```typescript\n   * // Send a simple command\n   * this.client.sendCommand('refresh', '');\n   * \n   * // Send a command with data\n   * this.client.sendCommand('setVolume', '50');\n   * \n   * // Send a command with JSON data\n   * this.client.sendCommand('configure', JSON.stringify({ setting: 'value' }));\n   * ```\n   */\n  public sendCommand(name: string, arg: string): void {\n\n    this.getClient().then((client) => {\n      client.sendCommand(name, arg);\n    })\n  }\n\n  /**\n   * Sends a command to remote player devices with the specified device keys.\n   * \n   * Remote commands allow cross-device communication within the same Revel Digital account.\n   * This is useful for synchronized displays, device coordination, or remote control scenarios.\n   * \n   * Note: Remote commands can only be delivered to devices within the same account as the sender device.\n   * \n   * @param deviceKeys - Array of target device keys to send the command to\n   * @param name - The command name/identifier\n   * @param arg - The command argument/payload\n   * \n   * ```typescript\n   * // Send command to specific devices\n   * const targetDevices = ['device-key-1', 'device-key-2'];\n   * this.client.sendRemoteCommand(targetDevices, 'syncAction', 'start');\n   * \n   * // Broadcast to multiple devices\n   * this.client.sendRemoteCommand(\n   *   ['lobby-display', 'kiosk-1', 'kiosk-2'], \n   *   'updateContent', \n   *   JSON.stringify({ contentId: '12345' })\n   * );\n   * ```\n   */\n  public sendRemoteCommand(deviceKeys: string[], name: string, arg: string): void {\n\n    this.getClient().then((client) => {\n      client.sendRemoteCommand(deviceKeys, name, arg);\n    });\n  }\n\n  /**\n   * Logs an analytics event for use with AdHawk analytics and reporting.\n   * \n   * Events are used for tracking various metrics including usage statistics, \n   * player condition, state changes, user interactions, and custom business metrics.\n   * These events can be viewed in the Revel Digital analytics dashboard.\n   * \n   * @param eventName - Unique name for this event (should be descriptive and consistent)\n   * @param properties - Optional map of user-defined properties to associate with this event\n   * \n   * ```typescript\n   * // Simple event tracking\n   * this.client.track('gadget_loaded');\n   * \n   * // Event with properties\n   * this.client.track('user_interaction', {\n   *   action: 'button_click',\n   *   button_id: 'start_button',\n   *   timestamp: new Date().toISOString()\n   * });\n   * \n   * // Performance tracking\n   * this.client.track('content_displayed', {\n   *   content_type: 'video',\n   *   duration: 30,\n   *   quality: 'HD'\n   * });\n   * ```\n   */\n  public track(eventName: string, properties?: IEventProperties): void {\n\n    this.getClient().then((client) => {\n      client.track(eventName, JSON.stringify(properties));\n    })\n  }\n\n  /**\n   * Initiates a timed event for duration tracking.\n   * \n   * Timed events are useful for tracking the duration of operations or user interactions.\n   * This method must be followed by a call to track() with the same event name to complete\n   * the timing measurement. The duration will be automatically calculated and included\n   * in the event properties.\n   * \n   * @param eventName - Unique name for this timed event (must match the subsequent track() call)\n   * \n   * ```typescript\n   * // Start timing an event\n   * this.client.timeEvent('video_playback');\n   * \n   * // ... video plays for some duration ...\n   * \n   * // End timing and log the event with duration\n   * this.client.track('video_playback', {\n   *   video_id: 'abc123',\n   *   quality: 'HD'\n   * }); // Duration will be automatically added\n   * \n   * // Example for user interaction timing\n   * this.client.timeEvent('form_completion');\n   * // ... user fills out form ...\n   * this.client.track('form_completion', { form_type: 'contact' });\n   * ```\n   */\n  public timeEvent(eventName: string): void {\n\n    this.getClient().then((client) => {\n      client.timeEvent(eventName);\n    })\n  }\n\n  /**\n   * Creates a new analytics event session for grouping related events.\n   * \n   * A session is a way of grouping events together for analysis. Each event tracked\n   * after calling this method will have the same session ID until a new session is created.\n   * Session IDs are randomly generated unless explicitly provided.\n   * \n   * This is useful for tracking user journeys, workflow completion, or grouping\n   * related interactions within a specific time period.\n   * \n   * @param id - Optional user-supplied session ID. If not provided, a random session ID will be generated\n   * \n   * ```typescript\n   * // Start a new session with auto-generated ID\n   * this.client.newEventSession();\n   * this.client.track('session_start');\n   * this.client.track('user_action_1');\n   * this.client.track('user_action_2');\n   * \n   * // Start a session with custom ID\n   * this.client.newEventSession('user-workflow-12345');\n   * this.client.track('workflow_start');\n   * this.client.track('step_completed', { step: 1 });\n   * \n   * // Start a new session for different workflow\n   * this.client.newEventSession();\n   * this.client.track('different_workflow_start');\n   * ```\n   */\n  public newEventSession(id?: string): void {\n\n    this.getClient().then((client) => {\n      if (id !== undefined) {\n        client.newEventSession();\n      } else {\n        client.newEventSession(id);\n      }\n    })\n  }\n\n  /**\n   * Gets the root folder path utilized by this player device.\n   * \n   * This returns the base directory path where the Revel Digital player\n   * stores its files and resources on the local device.\n   * \n   * @returns Promise resolving to the path of the root folder\n   * \n   * ```typescript\n   * const rootPath = await this.client.getRevelRoot();\n   * console.log('Player root directory:', rootPath);\n   * // Use for constructing file paths or understanding storage structure\n   * ```\n   */\n  public async getRevelRoot(): Promise<string> {\n\n    const client = await this.getClient();\n\n    return client.getRevelRoot();\n  }\n\n  /**\n   * Gets a map of commands currently active for this device.\n   * \n   * This returns the current command configuration that defines how the device\n   * responds to various command triggers and remote commands.\n   * \n   * @returns Promise resolving to a map of currently active commands\n   * \n   * ```typescript\n   * const commandMap = await this.client.getCommandMap();\n   * console.log('Active commands:', commandMap);\n   * \n   * // Check if specific command is available\n   * if (commandMap['customCommand']) {\n   *   console.log('Custom command is available');\n   * }\n   * ```\n   */\n  public async getCommandMap(): Promise<any> {\n\n    const client = await this.getClient();\n\n    return JSON.parse(await client.getCommandMap());\n  }\n\n  /**\n   * Signals to the player that this gadget has completed its visualization.\n   * \n   * This method notifies the player that the current gadget has finished its\n   * content display or interaction cycle. The player can then proceed with\n   * the next item in a playlist if applicable, or handle the completion\n   * according to its configuration.\n   * \n   * Call this method when your gadget has completed its intended function,\n   * such as finishing an animation, completing a form, or reaching a natural\n   * stopping point.\n   * \n   * ```typescript\n   * // After completing an animation\n   * private onAnimationComplete(): void {\n   *   this.client.finish();\n   * }\n   * \n   * // After user interaction is complete\n   * private onFormSubmitted(): void {\n   *   this.saveFormData();\n   *   this.client.finish();\n   * }\n   * \n   * // After a timed display period\n   * setTimeout(() => {\n   *   this.client.finish();\n   * }, 30000); // 30 seconds\n   * ```\n   */\n  public finish(): void {\n\n    this.getClient().then((client) => {\n\n      client.finish();\n    })\n  }\n\n  /**\n   * Checks if the gadget is running in preview mode.\n   * \n   * Preview mode is enabled when the gadget is being edited in the Revel Digital CMS,\n   * tested in the gadget editor, or otherwise not running in a normal player environment.\n   * This is useful for providing different behavior during development/testing versus\n   * production deployment.\n   * \n   * @returns Promise resolving to true if running in preview mode, false if running on actual player\n   * \n   * ```typescript\n   * const isPreview = await this.client.isPreviewMode();\n   * \n   * if (isPreview) {\n   *   console.log('Running in preview mode - using mock data');\n   *   this.loadMockData();\n   * } else {\n   *   console.log('Running on player device - using live data');\n   *   this.loadLiveData();\n   * }\n   * \n   * // Show different UI in preview\n   * this.showPreviewIndicator = isPreview;\n   * ```\n   */\n  public async isPreviewMode(): Promise<boolean> {\n\n    const client = await this.getClient();\n\n    return client instanceof NoopClient;\n  }\n\n  /**\n   * Gets detailed information about the device running the player.\n   * \n   * Returns comprehensive device details including name, registration key, type,\n   * service date, language, timezone, tags, and location information. This data\n   * is configured in the Revel Digital CMS for each device.\n   * \n   * @returns Promise resolving to device details object, or null if not available\n   * \n   * ```typescript\n   * const device = await this.client.getDevice();\n   * \n   * if (device) {\n   *   console.log('Device name:', device.name);\n   *   console.log('Device type:', device.deviceType);\n   *   console.log('Location:', device.location.city, device.location.state);\n   *   console.log('Tags:', device.tags);\n   *   \n   *   // Use device info for customization\n   *   if (device.location.country === 'US') {\n   *     this.loadUSContent();\n   *   }\n   *   \n   *   // Check device capabilities based on type\n   *   if (device.deviceType === 'android') {\n   *     this.enableTouchFeatures();\n   *   }\n   * }\n   * ```\n   */\n  public async getDevice(): Promise<IDevice | null> {\n\n    const client = await this.getClient();\n\n    const obj: any = JSON.parse(<string>await client.getDevice());\n\n    const device: IDevice[] = [obj].map((device: any) => {\n\n      return {\n        name: device.name,\n        registrationKey: device.key,\n        deviceType: device.devicetype,\n        enteredService: new Date(device.enteredservice),\n        langCode: device.langcode,\n        timeZone: device.timezone,\n        tags: device.description?.split('\\n'),\n        location: {\n          city: device.location?.city,\n          state: device.location?.state,\n          country: device.location?.country,\n          postalCode: device.location?.postalcode,\n          address: device.location?.address,\n          latitude: device.location?.latitude,\n          longitude: device.location?.longitude\n        }\n      }\n    });\n    return device[0];\n  }\n\n  /**\n   * Gets the width of the visualization area in pixels.\n   * \n   * This returns the available width for content display, which may be\n   * different from the full screen width depending on player configuration\n   * and template layout.\n   * \n   * @returns Promise resolving to width in pixels, or null if not available\n   * \n   * ```typescript\n   * const width = await this.client.getWidth();\n   * \n   * if (width) {\n   *   console.log('Available width:', width, 'pixels');\n   *   \n   *   // Adapt content layout based on width\n   *   if (width < 800) {\n   *     this.enableMobileLayout();\n   *   } else {\n   *     this.enableDesktopLayout();\n   *   }\n   * }\n   * ```\n   */\n  public async getWidth(): Promise<number | null> {\n\n    const client = await this.getClient();\n\n    return client.getWidth();\n  }\n\n  /**\n   * Gets the height of the visualization area in pixels.\n   * \n   * This returns the available height for content display, which may be\n   * different from the full screen height depending on player configuration\n   * and template layout.\n   * \n   * @returns Promise resolving to height in pixels, or null if not available\n   * \n   * ```typescript\n   * const height = await this.client.getHeight();\n   * \n   * if (height) {\n   *   console.log('Available height:', height, 'pixels');\n   *   \n   *   // Calculate aspect ratio for responsive design\n   *   const width = await this.client.getWidth();\n   *   const aspectRatio = width / height;\n   *   this.adjustContentForAspectRatio(aspectRatio);\n   * }\n   * ```\n   */\n  public async getHeight(): Promise<number | null> {\n\n    const client = await this.getClient();\n\n    return client.getHeight();\n  }\n\n  /**\n   * Gets the duration of the currently playing content item.\n   * \n   * This method is only applicable when the gadget is associated with a playlist\n   * and returns the duration assigned to the current playlist item. The duration\n   * determines how long the content should be displayed before moving to the next item.\n   * \n   * @returns Promise resolving to duration in milliseconds, or null if not applicable/available\n   * \n   * ```typescript\n   * const duration = await this.client.getDuration();\n   * \n   * if (duration) {\n   *   console.log('Content duration:', duration, 'milliseconds');\n   *   \n   *   // Set up auto-finish timer\n   *   setTimeout(() => {\n   *     this.client.finish();\n   *   }, duration);\n   *   \n   *   // Show progress indicator\n   *   this.startProgressIndicator(duration);\n   * }\n   * ```\n   */\n  public async getDuration(): Promise<number | null> {\n\n    const client = await this.getClient();\n\n    return client.getDuration();\n  }\n\n  /**\n   * Gets the current version of the Revel Digital SDK.\n   * \n   * @returns Promise resolving to the SDK version string\n   * \n   * ```typescript\n   * const version = await this.client.getSdkVersion();\n   * console.log('SDK Version:', version);\n   * \n   * // Use for compatibility checks or logging\n   * this.client.track('gadget_loaded', { sdkVersion: version });\n   * ```\n   */\n  public async getSdkVersion(): Promise<string> {\n\n    return Promise.resolve(version);\n  }\n\n  /**\n   * Applies configuration preferences to the gadget (preview mode only).\n   * \n   * This method is only available when running in preview mode (typically during\n   * gadget development or testing in the CMS). It allows applying configuration\n   * changes that would normally come from the gadget's preference settings.\n   * \n   * @param prefs - Dictionary of preference key-value pairs to apply\n   * \n   * ```typescript\n   * if (await this.client.isPreviewMode()) {\n   *   // Apply test configuration in preview\n   *   await this.client.applyConfig({\n   *     'title': 'Test Title',\n   *     'backgroundColor': '#ff0000',\n   *     'showBorder': true,\n   *     'refreshInterval': 30\n   *   });\n   * }\n   * ```\n   */\n  public async applyConfig(prefs: IDictionary<any>) {\n\n    if (await this.isPreviewMode()) {\n      const client = await this.getClient();\n      client.applyConfig(prefs);\n    } else {\n      console.log(\n        '%capplyConfig() is only available in preview mode.',\n        'background-color:blue; color:yellow;'\n      );\n    }\n  }\n\n  /**\n   * Creates a typed wrapper for a Revel Digital data table.\n   *\n   * The data table feature must be enabled for the gadget. The returned\n   * {@link DataTableRef} provides typed Promise-based methods and RxJS\n   * Observables for real-time row change events.\n   *\n   * @param tableId - The data table ID (e.g. 'tbl_menu_items')\n   * @param options - Optional configuration overrides\n   * @returns A {@link DataTableRef} instance\n   * @throws Error if the global datatable library is not loaded\n   *\n   * ```typescript\n   * const dt = this.client.createDataTable('tbl_menu_items');\n   *\n   * // Fetch rows with filtering and sorting\n   * const result = await dt.getRows({\n   *   filter: { category: 'Entree', price: { op: 'lte', value: 25 } },\n   *   sort: 'price',\n   *   sortDir: 'asc'\n   * });\n   *\n   * // Subscribe to real-time updates\n   * dt.rowUpdated$.subscribe(change => console.log('Row updated:', change));\n   * dt.rowCreated$.subscribe(change => console.log('Row created:', change));\n   * dt.rowDeleted$.subscribe(change => console.log('Row deleted:', change));\n   *\n   * // Cleanup when done\n   * dt.dispose();\n   * ```\n   */\n  public createDataTable(tableId: string, options?: IDataTableOptions): DataTableRef {\n    return new DataTableRef(tableId, options);\n  }\n\n  /**\n   * Creates a typed data table wrapper from a gadget preference value.\n   *\n   * The preference JSON string (as serialized by the template editor's datatable\n   * option) is parsed and used to auto-configure filter, sort, and logic settings.\n   * The returned {@link DataTablePrefRef} provides a `getFilteredRows()` convenience\n   * method that applies these settings automatically.\n   *\n   * @param prefValue - The raw gadget preference string (JSON)\n   * @param options - Optional configuration overrides\n   * @returns A {@link DataTablePrefRef} instance\n   * @throws Error if the global datatable library is not loaded\n   *\n   * ```typescript\n   * const cfg = this.client.createDataTableFromPref(prefs.getString('rdDataTable'));\n   *\n   * // Fetch rows with auto-wired filter + sort\n   * const result = await cfg.getFilteredRows();\n   *\n   * // Access the underlying DataTableRef for events, schema, etc.\n   * cfg.dataTable.rowUpdated$.subscribe(change => console.log(change));\n   *\n   * // Cleanup when done\n   * cfg.dispose();\n   * ```\n   */\n  public createDataTableFromPref(prefValue: string, options?: IDataTableOptions): DataTablePrefRef {\n    return new DataTablePrefRef(prefValue, options);\n  }\n\n  // ---\n  // PRIVATE METHODS.\n  // ---\n  /** @ignore */\n  private getClient(): Promise<IClient> {\n\n    if (this.clientPromise) {\n\n      return (this.clientPromise);\n    }\n\n    if ((window as any).Client) {\n\n      return (this.clientPromise = Promise.resolve((window as any).Client));\n    }\n\n    // A \"complete\" status indicates that the \"load\" event has been fired on the\n    // window; and, that all sub-resources such as Scripts, Images, and Frames have\n    // been loaded.\n    if (window.document.readyState === \"complete\") {\n\n      // If this event has fired AND the 3rd-party script isn't available (see IF-\n      // condition BEFORE this one), it means that the 3rd-party script either\n      // failed on the network or was BLOCKED by an ad-blocker. As such, we have to\n      // fall-back to using a mock API.\n      return (this.clientPromise = Promise.resolve(new NoopClient()));\n    }\n\n    // ASSERT: If we made it this far, the document has not completed loading (but it\n    // may be in an \"interactive\" state which is when I believe that the Angular app\n    // gets bootstrapped). As such, we need bind to the LOAD event to wait for our\n    // third-party scripts to load (or fail to load, or be blocked).\n    this.clientPromise = new Promise<IClient>(\n      (resolve) => {\n\n        window.addEventListener(\n          \"load\",\n          function handleWindowLoad() {\n\n            // At this point, the 3rd-party library is either available or\n            // it's not - there's no further loading to do. If it's not\n            // present on the global scope, we're going to fall-back to using\n            // a mock API.\n            resolve((window as any).Client || new NoopClient());\n          }\n        );\n\n      }\n    );\n\n    return (this.clientPromise);\n  }\n}\n\n\n\n// ----------------------------------------------------------------------------------- //\n// ----------------------------------------------------------------------------------- //\n\n/**\n * Mock implementation of the IClient interface.\n * \n * This class provides a no-operation (NOOP) implementation of the client API\n * that allows consuming code to function normally even when the actual player\n * API is not available. This typically occurs during development, testing,\n * or when the player script is blocked by ad-blockers.\n * \n * All methods in this class either return null/empty values or perform no\n * operations, allowing gadgets to function without errors while providing\n * appropriate fallback behavior.\n * \n * @private\n */\nclass NoopClient implements IClient {\n\n  constructor() {\n\n    console.log(\n      '%cClient API not available, falling back to mock API',\n      'background-color:blue; color:yellow;'\n    );\n  }\n\n  public callback(...args: any[]): void {\n\n    // NOOP implement, nothing to do....\n  }\n\n  public getDeviceTime(date?: Date): Promise<string> {\n\n    return Promise.resolve(new Date().toISOString());\n  }\n\n  public async getDeviceTimeZoneName(): Promise<string> {\n\n    return Promise.resolve(null);\n  }\n\n  public async getDeviceTimeZoneID(): Promise<string> {\n\n    return Promise.resolve(null);\n  }\n\n  public async getDeviceTimeZoneOffset(): Promise<number> {\n\n    return Promise.resolve(null);\n  }\n\n  public async getLanguageCode(): Promise<string> {\n\n    return Promise.resolve(null);\n  }\n\n  public async getDeviceKey(): Promise<string> {\n\n    return Promise.resolve(null);\n  }\n\n  public sendCommand(name: string, arg: string): void {\n\n    // NOOP implement, nothing to do....\n  }\n\n  public sendRemoteCommand(deviceKeys: string[], name: string, arg: string) {\n\n    // NOOP implement, nothing to do....\n  }\n\n  public track(eventName: string, properties?: string): void {\n\n    // NOOP implement, nothing to do....\n  }\n\n  public timeEvent(eventName: string): void {\n\n    // NOOP implement, nothing to do....\n  }\n\n  public newEventSession(id?: string): void {\n\n    // NOOP implement, nothing to do....\n  }\n\n  public async getRevelRoot(): Promise<string> {\n\n    return Promise.resolve(null);\n  }\n\n  public async getCommandMap(): Promise<string> {\n\n    return Promise.resolve('{}');\n  }\n\n  public finish(): void {\n\n    // NOOP implement, nothing to do....\n  }\n\n  public async getDevice(): Promise<string | null> {\n\n    return Promise.resolve(null);\n  }\n\n  public async getWidth(): Promise<number | null> {\n\n    return Promise.resolve(null);\n  }\n\n  public async getHeight(): Promise<number | null> {\n\n    return Promise.resolve(null);\n  }\n\n  public async getDuration(): Promise<number | null> {\n\n    return Promise.resolve(null);\n  }\n\n  public async getSdkVersion(): Promise<string> {\n\n    return Promise.resolve(version);\n  }\n\n  public applyConfig(prefs: IDictionary<any>): void {\n\n    const evt = { type: 'applyConfig', prefs: prefs, isOpener: window.opener !== null };\n\n    if (window.opener) {\n      window.opener.postMessage(\n        JSON.stringify(evt),\n        '*'\n      );\n    } else {\n      window.parent.postMessage(\n        JSON.stringify(evt),\n        '*'\n      );\n    }\n  }\n}\n"]}
992
+ //# sourceMappingURL=data:application/json;base64,{"version":3,"file":"player-client.service.js","sourceRoot":"","sources":["../../../../../projects/reveldigital/player-client/src/lib/player-client.service.ts"],"names":[],"mappings":"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","sourcesContent":["import { Injectable, NgZone, OnDestroy } from '@angular/core';\nimport { gadgets } from '@reveldigital/gadget-types';\nimport { BehaviorSubject, fromEvent, Subject, Subscription } from 'rxjs';\nimport { filter, map, share, tap } from 'rxjs/operators';\nimport { IClient } from './interfaces/client.interface';\nimport { ICommand } from './interfaces/command.interface';\nimport { IDictionary } from './interfaces/config.interface';\nimport { IDataTableOptions } from './interfaces/datatable.interface';\nimport { IDevice } from './interfaces/device.interface';\nimport { IEventProperties } from './interfaces/event-properties.interface';\nimport { DataTableRef, DataTablePrefRef } from './datatable-ref';\nimport { version } from './version';\n\n//import { version } from './version.js';\n\n// So that TypeScript doesn't complain, we're going to augment the GLOBAL / WINDOW \n// name-space definition to include the Tracker API. This also provides us with a place\n// to actually DOCUMENT the API so that our developers aren't guessing about what's\n// available on the library.\n\n/** @ignore */\ndeclare global {\n  let Client: IClient;\n}\n\n/**\n * Service for interacting with the Revel Digital player client.\n * \n * This service provides a comprehensive interface for gadgets and web applications\n * to communicate with the Revel Digital player environment. It handles device\n * information, commands, events, analytics tracking, and configuration management.\n * \n * The service supports both direct API calls and event-based communication patterns,\n * allowing gadgets to respond to player lifecycle events (start, stop, commands) and\n * send data back to the player or remote devices.\n * \n * ```typescript\n * constructor(private client: PlayerClientService) {\n *   // Subscribe to player events\n *   this.client.onStart$.subscribe(() => {\n *     console.log('Gadget started');\n *   });\n *   \n *   this.client.onCommand$.subscribe(command => {\n *     console.log('Received command:', command);\n *   });\n * }\n * \n * async ngOnInit() {\n *   // Get device information\n *   const device = await this.client.getDevice();\n *   const deviceTime = await this.client.getDeviceTime();\n *   \n *   // Track analytics\n *   this.client.track('gadget_loaded', { version: '1.0' });\n * }\n * ```\n * \n * @since 1.0.0\n */\n@Injectable({\n  providedIn: 'root'\n})\nexport class PlayerClientService implements OnDestroy {\n\n  /** @ignore */\n  private zone: NgZone;\n  /** @ignore */\n  private clientPromise: Promise<IClient> | null;\n\n  /**\n   * Observable stream of commands sent to this player from the Revel Digital platform.\n   * Subscribe to this to handle custom commands sent from templates, playlists, or remote devices.\n   * \n   * ```typescript\n   * this.client.onCommand$.subscribe(command => {\n   *   if (command.name === 'customAction') {\n   *     this.handleCustomAction(command.arg);\n   *   }\n   * });\n   * ```\n   */\n  public onCommand$ = new Subject<ICommand>();\n\n  /**\n   * Observable that signals when the gadget has been loaded and is ready to start.\n   * Emits `true` when ready, `false` when destroyed.\n   * \n   * ```typescript\n   * this.client.onReady$.subscribe(isReady => {\n   *   if (isReady) {\n   *     console.log('Client is ready');\n   *     this.initializeGadget();\n   *   }\n   * });\n   * ```\n   */\n  public onReady$ = new BehaviorSubject(false);\n\n  /**\n   * Observable that signals when the gadget has been started by the player.\n   * This event occurs when the player begins execution of the gadget content.\n   * \n   * ```typescript\n   * this.client.onStart$.subscribe(() => {\n   *   console.log('Gadget started');\n   *   this.startAnimation();\n   * });\n   * ```\n   */\n  public onStart$ = new Subject();\n\n  /**\n   * Observable that signals when the gadget has been stopped by the player.\n   * This event occurs when the player stops execution, typically when moving\n   * to the next item in a playlist or when the content duration expires.\n   * \n   * ```typescript\n   * this.client.onStop$.subscribe(() => {\n   *   console.log('Gadget stopped');\n   *   this.cleanup();\n   * });\n   * ```\n   */\n  public onStop$ = new Subject();\n\n  /**\n   * Observable that signals when the gadget should open the configuration window.\n   * This allows gadgets to respond to configuration requests from the player.\n   * \n   * ```typescript\n   * this.client.onConfig$.subscribe(() => {\n   *   this.openConfigurationDialog();\n   * });\n   * ```\n   */\n  public onConfig$ = new Subject();\n\n  /**\n   * Observable that signals when the gadget has received a postMessage event from the player.\n   * This handles communication between the gadget and player via the postMessage API.\n   * \n   * ```typescript\n   * this.client.onPostMessage$.subscribe(message => {\n   *   console.log('Received message:', message);\n   *   this.handlePlayerMessage(message);\n   * });\n   * ```\n   */\n  public onPostMessage$ = new Subject();\n\n  //\n  // Two methods available for calling into the library:\n  //\n  // 1) Using dispatchEvent() with the following custom events\n  // 2) Using the window scoped RevelDigital object as defined in the constructor\n  //\n  /** @ignore */\n  private onStartSub: Subscription;\n  /** @ignore */\n  private onStartEvt$ = fromEvent(window, 'RevelDigital.Start').pipe(\n    share(),\n    tap(this.onStart$)\n  );\n  /** @ignore */\n  private onStopSub: Subscription;\n  /** @ignore */\n  private onStopEvt$ = fromEvent(window, 'RevelDigital.Stop').pipe(\n    share(),\n    tap(this.onStop$)\n  );\n  /** @ignore */\n  private onCommandSub: Subscription;\n  /** @ignore */\n  private onCommandEvt$ = fromEvent<ICommand>(window, 'RevelDigital.Command').pipe(\n    map((e: any) => { return { name: e.detail.name, arg: e.detail.arg } as ICommand }),\n    share(),\n    tap(this.onCommand$)\n  );\n  /** @ignore */\n  // private onConfigSub: Subscription;\n  // /** @ignore */\n  // private onConfigEvt$ = fromEvent(window, 'RevelDigital.Config').pipe(\n  //   share(),\n  //   tap((e: CustomEvent) => {\n  //     console.log(e);\n\n  //     if (e.detail.type === 'applyConfig' && e.detail.isOpener) {\n  //       this.applyConfig(e.detail.config); // propagate config to iframe parent from the popup window\n  //     } else {\n  //       this.onConfig$.next(e.detail);\n  //     }\n  //   })\n  // );\n  // private onPostMessageSub: Subscription;\n  // private onPostMessageEvt$ = fromEvent(window, 'message').pipe(\n  //   filter((messageEvent: MessageEvent) =>\n  //     messageEvent.source !== window.parent &&\n  //     typeof messageEvent.data === 'string' &&\n  //     messageEvent.data.startsWith('reveldigital:')),\n  //   map((e: any) => { return JSON.parse(e.substring(13)) as Command }),\n  //   share(),\n  //   tap(this.onCommand$)\n  // );\n\n  private onPostMessageSub: Subscription;\n  private onPostMessageEvt$ = fromEvent(window, 'message').pipe(\n    filter((messageEvent: MessageEvent) =>\n      //messageEvent.source !== window.parent &&\n      typeof messageEvent.data === 'string'),\n    map((e: any) => JSON.parse(e.data)),\n    share(),\n    tap((e: any) => {\n      if (e.type === 'applyConfig' && e.isOpener) {\n        e.isOpener = false;\n        // propagate config to iframe parent from the popup window\n        window.parent.postMessage(\n          JSON.stringify(e),\n          '*'\n        );\n      } else if (e.type === 'openConfig') {\n        this.onConfig$.next(null);\n      }\n    }),\n    tap(e => this.onPostMessage$.next(e))\n  );\n\n\n  /** @ignore */\n  constructor(zone: NgZone) {\n\n    this.zone = zone;\n\n    (window as any).RevelDigital = {\n      Controller: {\n        onCommand: (name: string, arg: string) => {\n          zone.run(() => {\n            this.onCommand$.next({ name: name, arg: arg });\n          });\n        },\n        onStart: () => {\n          zone.run(() => {\n            this.onStart$.next(null);\n          });\n        },\n        onStop: () => {\n          zone.run(() => {\n            this.onStop$.next(null);\n          });\n        }\n      }\n    }\n\n    this.onStartSub = this.onStartEvt$.subscribe(() => { });\n    this.onStopSub = this.onStopEvt$.subscribe(() => { });\n    this.onCommandSub = this.onCommandEvt$.subscribe(() => { });\n    //this.onConfigSub = this.onConfigEvt$.subscribe(() => { });\n    this.onPostMessageSub = this.onPostMessageEvt$.subscribe(() => { });\n\n    this.clientPromise = null;\n\n    this.onReady$.next(true);\n  }\n\n  /** @ignore */\n  ngOnDestroy(): void {\n\n    this.onStartSub?.unsubscribe();\n    this.onStopSub?.unsubscribe();\n    this.onCommandSub?.unsubscribe();\n    //this.onConfigSub?.unsubscribe();\n    this.onPostMessageSub?.unsubscribe();\n\n    this.onReady$.next(false);\n  }\n\n  /** @ignore */\n  public static init(data: any) {\n\n    console.log(\n      '%c⚙️ Initializing Revel Digital client library',\n      'background-color:blue; color:yellow;'\n    );\n  }\n\n  /**\n   * Sends a callback to player scripting with variable arguments.\n   * \n   * This method allows the gadget to communicate with player scripting.\n   * If the appropriate scripting is in place in the currently running template, \n   * calling this method will initiate a callback which can be acted upon in player script.\n   * \n   * @param args - Variable number of arguments to pass to the callback (max 5 arguments)\n   * \n   * ```typescript\n   * // Send a simple callback\n   * this.client.callback('test', 'data');\n   * \n   * // Send multiple parameters\n   * this.client.callback('action', 'value1', 'value2', { data: 'object' });\n   * ```\n   */\n  public callback(...args: any[]): void {\n\n    this.getClient().then((client) => {\n\n      switch (args.length) {\n        case 0:\n          client.callback();\n          break;\n        case 1:\n          client.callback(args[0]);\n          break;\n        case 2:\n          client.callback(args[1]);\n          break;\n        case 3:\n          client.callback(args[2]);\n          break;\n        case 4:\n          client.callback(args[3]);\n          break;\n        case 5:\n          client.callback(args[4]);\n          break;\n      }\n    })\n  }\n\n  /**\n   * Gets the user preferences interface exposed by the Google Gadgets API.\n   * \n   * This method provides access to gadget preferences which can be configured\n   * in the Revel Digital CMS. Use this to retrieve user-configurable settings\n   * for your gadget.\n   * \n   * @returns The Gadgets API Prefs object for accessing preference values\n   * \n   * ```typescript\n   * constructor(public client: PlayerClientService) {\n   *   const prefs = client.getPrefs();\n   *   const myString = prefs.getString('myStringPref');\n   *   const myNumber = prefs.getInt('myNumberPref');\n   *   const myBool = prefs.getBool('myBoolPref');\n   * }\n   * ```\n   * \n   * @see {@link https://developers.google.com/gadgets/docs/basic} Google Gadgets API documentation\n   */\n  public getPrefs(): gadgets.Prefs {\n\n    return new window['gadgets']['Prefs']();\n  }\n\n  /**\n   * Gets the current device time in ISO8601 format.\n   * \n   * Current device time is determined by the device timezone assigned to the device in the CMS.\n   * This is useful for displaying time-sensitive content or scheduling operations based on\n   * the device's local time rather than browser time.\n   * \n   * @param date - Optional. If supplied, will translate the supplied date/time to device time based on respective timezones\n   * @returns Promise resolving to date/time in ISO8601 format\n   * \n   * ```typescript\n   * // Get current device time\n   * const currentTime = await this.client.getDeviceTime();\n   * console.log('Device time:', currentTime);\n   * \n   * // Convert a specific date to device time\n   * const specificDate = new Date('2023-01-01T12:00:00Z');\n   * const deviceTime = await this.client.getDeviceTime(specificDate);\n   * ```\n   */\n  public async getDeviceTime(date?: Date): Promise<string> {\n\n    const client = await this.getClient();\n\n    if (date !== undefined) {\n      return client.getDeviceTime(date);\n    }\n    return client.getDeviceTime();\n  }\n\n  /**\n   * Gets the timezone name currently assigned to the device.\n   * \n   * @returns Promise resolving to the timezone name (e.g., \"America/New_York\")\n   * \n   * ```typescript\n   * const timezoneName = await this.client.getDeviceTimeZoneName();\n   * console.log('Device timezone:', timezoneName);\n   * ```\n   */\n  public async getDeviceTimeZoneName(): Promise<string> {\n\n    const client = await this.getClient();\n\n    return client.getDeviceTimeZoneName();\n  }\n\n  /**\n   * Gets the timezone ID currently assigned to the device.\n   * \n   * @returns Promise resolving to the timezone ID\n   * \n   * ```typescript\n   * const timezoneId = await this.client.getDeviceTimeZoneID();\n   * console.log('Device timezone ID:', timezoneId);\n   * ```\n   */\n  public async getDeviceTimeZoneID(): Promise<string> {\n\n    const client = await this.getClient();\n\n    return client.getDeviceTimeZoneID();\n  }\n\n  /**\n   * Gets the numerical offset from GMT of the timezone currently assigned to the device.\n   * \n   * @returns Promise resolving to the timezone offset in hours (e.g., -5 for EST)\n   * \n   * ```typescript\n   * const offset = await this.client.getDeviceTimeZoneOffset();\n   * console.log('Device timezone offset:', offset, 'hours from GMT');\n   * ```\n   */\n  public async getDeviceTimeZoneOffset(): Promise<number> {\n\n    const client = await this.getClient();\n\n    return client.getDeviceTimeZoneOffset();\n  }\n\n  /**\n   * Gets the language code of the language currently assigned to the device.\n   * \n   * @returns Promise resolving to the language code (e.g., \"en-US\", \"fr-FR\")\n   * \n   * ```typescript\n   * const languageCode = await this.client.getLanguageCode();\n   * console.log('Device language:', languageCode);\n   * // Use for localization\n   * this.loadLanguageResources(languageCode);\n   * ```\n   */\n  public async getLanguageCode(): Promise<string> {\n\n    const client = await this.getClient();\n\n    return client.getLanguageCode();\n  }\n\n  /**\n   * Gets the unique Revel Digital device key associated with the device.\n   * \n   * The device key is a unique identifier for this specific player device\n   * and can be used for device-specific operations or remote commands.\n   * \n   * @returns Promise resolving to the device key string\n   * \n   * ```typescript\n   * const deviceKey = await this.client.getDeviceKey();\n   * console.log('Device key:', deviceKey);\n   * ```\n   */\n  public async getDeviceKey(): Promise<string> {\n\n    const client = await this.getClient();\n\n    return client.getDeviceKey();\n  }\n\n  /**\n   * Sends a command to the local player device.\n   * \n   * Commands can be used to control various aspects of the player or trigger\n   * specific behaviors. The command is processed by the local player only.\n   * \n   * @param name - The command name/identifier\n   * @param arg - The command argument/payload\n   * \n   * ```typescript\n   * // Send a simple command\n   * this.client.sendCommand('refresh', '');\n   * \n   * // Send a command with data\n   * this.client.sendCommand('setVolume', '50');\n   * \n   * // Send a command with JSON data\n   * this.client.sendCommand('configure', JSON.stringify({ setting: 'value' }));\n   * ```\n   */\n  public sendCommand(name: string, arg: string): void {\n\n    this.getClient().then((client) => {\n      client.sendCommand(name, arg);\n    })\n  }\n\n  /**\n   * Sends a command to remote player devices with the specified device keys.\n   * \n   * Remote commands allow cross-device communication within the same Revel Digital account.\n   * This is useful for synchronized displays, device coordination, or remote control scenarios.\n   * \n   * Note: Remote commands can only be delivered to devices within the same account as the sender device.\n   * \n   * @param deviceKeys - Array of target device keys to send the command to\n   * @param name - The command name/identifier\n   * @param arg - The command argument/payload\n   * \n   * ```typescript\n   * // Send command to specific devices\n   * const targetDevices = ['device-key-1', 'device-key-2'];\n   * this.client.sendRemoteCommand(targetDevices, 'syncAction', 'start');\n   * \n   * // Broadcast to multiple devices\n   * this.client.sendRemoteCommand(\n   *   ['lobby-display', 'kiosk-1', 'kiosk-2'], \n   *   'updateContent', \n   *   JSON.stringify({ contentId: '12345' })\n   * );\n   * ```\n   */\n  public sendRemoteCommand(deviceKeys: string[], name: string, arg: string): void {\n\n    this.getClient().then((client) => {\n      client.sendRemoteCommand(deviceKeys, name, arg);\n    });\n  }\n\n  /**\n   * Logs an analytics event for use with AdHawk analytics and reporting.\n   * \n   * Events are used for tracking various metrics including usage statistics, \n   * player condition, state changes, user interactions, and custom business metrics.\n   * These events can be viewed in the Revel Digital analytics dashboard.\n   * \n   * @param eventName - Unique name for this event (should be descriptive and consistent)\n   * @param properties - Optional map of user-defined properties to associate with this event\n   * \n   * ```typescript\n   * // Simple event tracking\n   * this.client.track('gadget_loaded');\n   * \n   * // Event with properties\n   * this.client.track('user_interaction', {\n   *   action: 'button_click',\n   *   button_id: 'start_button',\n   *   timestamp: new Date().toISOString()\n   * });\n   * \n   * // Performance tracking\n   * this.client.track('content_displayed', {\n   *   content_type: 'video',\n   *   duration: 30,\n   *   quality: 'HD'\n   * });\n   * ```\n   */\n  public track(eventName: string, properties?: IEventProperties): void {\n\n    this.getClient().then((client) => {\n      client.track(eventName, JSON.stringify(properties));\n    })\n  }\n\n  /**\n   * Initiates a timed event for duration tracking.\n   * \n   * Timed events are useful for tracking the duration of operations or user interactions.\n   * This method must be followed by a call to track() with the same event name to complete\n   * the timing measurement. The duration will be automatically calculated and included\n   * in the event properties.\n   * \n   * @param eventName - Unique name for this timed event (must match the subsequent track() call)\n   * \n   * ```typescript\n   * // Start timing an event\n   * this.client.timeEvent('video_playback');\n   * \n   * // ... video plays for some duration ...\n   * \n   * // End timing and log the event with duration\n   * this.client.track('video_playback', {\n   *   video_id: 'abc123',\n   *   quality: 'HD'\n   * }); // Duration will be automatically added\n   * \n   * // Example for user interaction timing\n   * this.client.timeEvent('form_completion');\n   * // ... user fills out form ...\n   * this.client.track('form_completion', { form_type: 'contact' });\n   * ```\n   */\n  public timeEvent(eventName: string): void {\n\n    this.getClient().then((client) => {\n      client.timeEvent(eventName);\n    })\n  }\n\n  /**\n   * Creates a new analytics event session for grouping related events.\n   * \n   * A session is a way of grouping events together for analysis. Each event tracked\n   * after calling this method will have the same session ID until a new session is created.\n   * Session IDs are randomly generated unless explicitly provided.\n   * \n   * This is useful for tracking user journeys, workflow completion, or grouping\n   * related interactions within a specific time period.\n   * \n   * @param id - Optional user-supplied session ID. If not provided, a random session ID will be generated\n   * \n   * ```typescript\n   * // Start a new session with auto-generated ID\n   * this.client.newEventSession();\n   * this.client.track('session_start');\n   * this.client.track('user_action_1');\n   * this.client.track('user_action_2');\n   * \n   * // Start a session with custom ID\n   * this.client.newEventSession('user-workflow-12345');\n   * this.client.track('workflow_start');\n   * this.client.track('step_completed', { step: 1 });\n   * \n   * // Start a new session for different workflow\n   * this.client.newEventSession();\n   * this.client.track('different_workflow_start');\n   * ```\n   */\n  public newEventSession(id?: string): void {\n\n    this.getClient().then((client) => {\n      if (id !== undefined) {\n        client.newEventSession();\n      } else {\n        client.newEventSession(id);\n      }\n    })\n  }\n\n  /**\n   * Gets the root folder path utilized by this player device.\n   * \n   * This returns the base directory path where the Revel Digital player\n   * stores its files and resources on the local device.\n   * \n   * @returns Promise resolving to the path of the root folder\n   * \n   * ```typescript\n   * const rootPath = await this.client.getRevelRoot();\n   * console.log('Player root directory:', rootPath);\n   * // Use for constructing file paths or understanding storage structure\n   * ```\n   */\n  public async getRevelRoot(): Promise<string> {\n\n    const client = await this.getClient();\n\n    return client.getRevelRoot();\n  }\n\n  /**\n   * Gets a map of commands currently active for this device.\n   * \n   * This returns the current command configuration that defines how the device\n   * responds to various command triggers and remote commands.\n   * \n   * @returns Promise resolving to a map of currently active commands\n   * \n   * ```typescript\n   * const commandMap = await this.client.getCommandMap();\n   * console.log('Active commands:', commandMap);\n   * \n   * // Check if specific command is available\n   * if (commandMap['customCommand']) {\n   *   console.log('Custom command is available');\n   * }\n   * ```\n   */\n  public async getCommandMap(): Promise<any> {\n\n    const client = await this.getClient();\n\n    return JSON.parse(await client.getCommandMap());\n  }\n\n  /**\n   * Signals to the player that this gadget has completed its visualization.\n   * \n   * This method notifies the player that the current gadget has finished its\n   * content display or interaction cycle. The player can then proceed with\n   * the next item in a playlist if applicable, or handle the completion\n   * according to its configuration.\n   * \n   * Call this method when your gadget has completed its intended function,\n   * such as finishing an animation, completing a form, or reaching a natural\n   * stopping point.\n   * \n   * ```typescript\n   * // After completing an animation\n   * private onAnimationComplete(): void {\n   *   this.client.finish();\n   * }\n   * \n   * // After user interaction is complete\n   * private onFormSubmitted(): void {\n   *   this.saveFormData();\n   *   this.client.finish();\n   * }\n   * \n   * // After a timed display period\n   * setTimeout(() => {\n   *   this.client.finish();\n   * }, 30000); // 30 seconds\n   * ```\n   */\n  public finish(): void {\n\n    this.getClient().then((client) => {\n\n      client.finish();\n    })\n  }\n\n  /**\n   * Checks if the gadget is running in preview mode.\n   * \n   * Preview mode is enabled when the gadget is being edited in the Revel Digital CMS,\n   * tested in the gadget editor, or otherwise not running in a normal player environment.\n   * This is useful for providing different behavior during development/testing versus\n   * production deployment.\n   * \n   * @returns Promise resolving to true if running in preview mode, false if running on actual player\n   * \n   * ```typescript\n   * const isPreview = await this.client.isPreviewMode();\n   * \n   * if (isPreview) {\n   *   console.log('Running in preview mode - using mock data');\n   *   this.loadMockData();\n   * } else {\n   *   console.log('Running on player device - using live data');\n   *   this.loadLiveData();\n   * }\n   * \n   * // Show different UI in preview\n   * this.showPreviewIndicator = isPreview;\n   * ```\n   */\n  public async isPreviewMode(): Promise<boolean> {\n\n    const client = await this.getClient();\n\n    return client instanceof NoopClient;\n  }\n\n  /**\n   * Gets detailed information about the device running the player.\n   * \n   * Returns comprehensive device details including name, registration key, type,\n   * service date, language, timezone, tags, and location information. This data\n   * is configured in the Revel Digital CMS for each device.\n   * \n   * @returns Promise resolving to device details object, or null if not available\n   * \n   * ```typescript\n   * const device = await this.client.getDevice();\n   * \n   * if (device) {\n   *   console.log('Device name:', device.name);\n   *   console.log('Device type:', device.deviceType);\n   *   console.log('Location:', device.location.city, device.location.state);\n   *   console.log('Tags:', device.tags);\n   *   \n   *   // Use device info for customization\n   *   if (device.location.country === 'US') {\n   *     this.loadUSContent();\n   *   }\n   *   \n   *   // Check device capabilities based on type\n   *   if (device.deviceType === 'android') {\n   *     this.enableTouchFeatures();\n   *   }\n   * }\n   * ```\n   */\n  public async getDevice(): Promise<IDevice | null> {\n\n    const client = await this.getClient();\n\n    const obj: any = JSON.parse(<string>await client.getDevice());\n\n    const device: IDevice[] = [obj].map((device: any) => {\n\n      return {\n        name: device.name,\n        registrationKey: device.key,\n        deviceType: device.devicetype,\n        enteredService: new Date(device.enteredservice),\n        langCode: device.langcode,\n        timeZone: device.timezone,\n        tags: device.description?.split('\\n'),\n        location: {\n          city: device.location?.city,\n          state: device.location?.state,\n          country: device.location?.country,\n          postalCode: device.location?.postalcode,\n          address: device.location?.address,\n          latitude: device.location?.latitude,\n          longitude: device.location?.longitude\n        }\n      }\n    });\n    return device[0];\n  }\n\n  /**\n   * Gets the width of the visualization area in pixels.\n   * \n   * This returns the available width for content display, which may be\n   * different from the full screen width depending on player configuration\n   * and template layout.\n   * \n   * @returns Promise resolving to width in pixels, or null if not available\n   * \n   * ```typescript\n   * const width = await this.client.getWidth();\n   * \n   * if (width) {\n   *   console.log('Available width:', width, 'pixels');\n   *   \n   *   // Adapt content layout based on width\n   *   if (width < 800) {\n   *     this.enableMobileLayout();\n   *   } else {\n   *     this.enableDesktopLayout();\n   *   }\n   * }\n   * ```\n   */\n  public async getWidth(): Promise<number | null> {\n\n    const client = await this.getClient();\n\n    return client.getWidth();\n  }\n\n  /**\n   * Gets the height of the visualization area in pixels.\n   * \n   * This returns the available height for content display, which may be\n   * different from the full screen height depending on player configuration\n   * and template layout.\n   * \n   * @returns Promise resolving to height in pixels, or null if not available\n   * \n   * ```typescript\n   * const height = await this.client.getHeight();\n   * \n   * if (height) {\n   *   console.log('Available height:', height, 'pixels');\n   *   \n   *   // Calculate aspect ratio for responsive design\n   *   const width = await this.client.getWidth();\n   *   const aspectRatio = width / height;\n   *   this.adjustContentForAspectRatio(aspectRatio);\n   * }\n   * ```\n   */\n  public async getHeight(): Promise<number | null> {\n\n    const client = await this.getClient();\n\n    return client.getHeight();\n  }\n\n  /**\n   * Gets the duration of the currently playing content item.\n   * \n   * This method is only applicable when the gadget is associated with a playlist\n   * and returns the duration assigned to the current playlist item. The duration\n   * determines how long the content should be displayed before moving to the next item.\n   * \n   * @returns Promise resolving to duration in milliseconds, or null if not applicable/available\n   * \n   * ```typescript\n   * const duration = await this.client.getDuration();\n   * \n   * if (duration) {\n   *   console.log('Content duration:', duration, 'milliseconds');\n   *   \n   *   // Set up auto-finish timer\n   *   setTimeout(() => {\n   *     this.client.finish();\n   *   }, duration);\n   *   \n   *   // Show progress indicator\n   *   this.startProgressIndicator(duration);\n   * }\n   * ```\n   */\n  public async getDuration(): Promise<number | null> {\n\n    const client = await this.getClient();\n\n    return client.getDuration();\n  }\n\n  /**\n   * Gets the current version of the Revel Digital SDK.\n   * \n   * @returns Promise resolving to the SDK version string\n   * \n   * ```typescript\n   * const version = await this.client.getSdkVersion();\n   * console.log('SDK Version:', version);\n   * \n   * // Use for compatibility checks or logging\n   * this.client.track('gadget_loaded', { sdkVersion: version });\n   * ```\n   */\n  public async getSdkVersion(): Promise<string> {\n\n    return Promise.resolve(version);\n  }\n\n  /**\n   * Applies configuration preferences to the gadget (preview mode only).\n   * \n   * This method is only available when running in preview mode (typically during\n   * gadget development or testing in the CMS). It allows applying configuration\n   * changes that would normally come from the gadget's preference settings.\n   * \n   * @param prefs - Dictionary of preference key-value pairs to apply\n   * \n   * ```typescript\n   * if (await this.client.isPreviewMode()) {\n   *   // Apply test configuration in preview\n   *   await this.client.applyConfig({\n   *     'title': 'Test Title',\n   *     'backgroundColor': '#ff0000',\n   *     'showBorder': true,\n   *     'refreshInterval': 30\n   *   });\n   * }\n   * ```\n   */\n  public async applyConfig(prefs: IDictionary<any>) {\n\n    if (await this.isPreviewMode()) {\n      const client = await this.getClient();\n      client.applyConfig(prefs);\n    } else {\n      console.log(\n        '%capplyConfig() is only available in preview mode.',\n        'background-color:blue; color:yellow;'\n      );\n    }\n  }\n\n  /**\n   * Creates a typed wrapper for a Revel Digital data table.\n   *\n   * The data table feature must be enabled for the gadget. The returned\n   * {@link DataTableRef} provides typed Promise-based methods and RxJS\n   * Observables for real-time row change events.\n   *\n   * @param tableId - The data table ID (e.g. 'tbl_menu_items')\n   * @param options - Optional configuration overrides\n   * @returns A {@link DataTableRef} instance\n   * @throws Error if the global datatable library is not loaded\n   *\n   * ```typescript\n   * const dt = this.client.createDataTable('tbl_menu_items');\n   *\n   * // Fetch rows with filtering and sorting\n   * const result = await dt.getRows({\n   *   filter: { category: 'Entree', price: { op: 'lte', value: 25 } },\n   *   sort: 'price',\n   *   sortDir: 'asc'\n   * });\n   *\n   * // Subscribe to real-time updates\n   * dt.rowUpdated$.subscribe(change => console.log('Row updated:', change));\n   * dt.rowCreated$.subscribe(change => console.log('Row created:', change));\n   * dt.rowDeleted$.subscribe(change => console.log('Row deleted:', change));\n   *\n   * // Cleanup when done\n   * dt.dispose();\n   * ```\n   */\n  public createDataTable(tableId: string, options?: IDataTableOptions): DataTableRef {\n    return new DataTableRef(tableId, this.zone, options);\n  }\n\n  /**\n   * Creates a typed data table wrapper from a gadget preference value.\n   *\n   * The preference JSON string (as serialized by the template editor's datatable\n   * option) is parsed and used to auto-configure filter, sort, and logic settings.\n   * The returned {@link DataTablePrefRef} provides a `getFilteredRows()` convenience\n   * method that applies these settings automatically.\n   *\n   * @param prefValue - The raw gadget preference string (JSON)\n   * @param options - Optional configuration overrides\n   * @returns A {@link DataTablePrefRef} instance\n   * @throws Error if the global datatable library is not loaded\n   *\n   * ```typescript\n   * const cfg = this.client.createDataTableFromPref(prefs.getString('rdDataTable'));\n   *\n   * // Fetch rows with auto-wired filter + sort\n   * const result = await cfg.getFilteredRows();\n   *\n   * // Access the underlying DataTableRef for events, schema, etc.\n   * cfg.dataTable.rowUpdated$.subscribe(change => console.log(change));\n   *\n   * // Cleanup when done\n   * cfg.dispose();\n   * ```\n   */\n  public createDataTableFromPref(prefValue: string, options?: IDataTableOptions): DataTablePrefRef {\n    return new DataTablePrefRef(prefValue, this.zone, options);\n  }\n\n  // ---\n  // PRIVATE METHODS.\n  // ---\n  /** @ignore */\n  private getClient(): Promise<IClient> {\n\n    if (this.clientPromise) {\n\n      return (this.clientPromise);\n    }\n\n    if ((window as any).Client) {\n\n      return (this.clientPromise = Promise.resolve((window as any).Client));\n    }\n\n    // A \"complete\" status indicates that the \"load\" event has been fired on the\n    // window; and, that all sub-resources such as Scripts, Images, and Frames have\n    // been loaded.\n    if (window.document.readyState === \"complete\") {\n\n      // If this event has fired AND the 3rd-party script isn't available (see IF-\n      // condition BEFORE this one), it means that the 3rd-party script either\n      // failed on the network or was BLOCKED by an ad-blocker. As such, we have to\n      // fall-back to using a mock API.\n      return (this.clientPromise = Promise.resolve(new NoopClient()));\n    }\n\n    // ASSERT: If we made it this far, the document has not completed loading (but it\n    // may be in an \"interactive\" state which is when I believe that the Angular app\n    // gets bootstrapped). As such, we need bind to the LOAD event to wait for our\n    // third-party scripts to load (or fail to load, or be blocked).\n    this.clientPromise = new Promise<IClient>(\n      (resolve) => {\n\n        window.addEventListener(\n          \"load\",\n          function handleWindowLoad() {\n\n            // At this point, the 3rd-party library is either available or\n            // it's not - there's no further loading to do. If it's not\n            // present on the global scope, we're going to fall-back to using\n            // a mock API.\n            resolve((window as any).Client || new NoopClient());\n          }\n        );\n\n      }\n    );\n\n    return (this.clientPromise);\n  }\n}\n\n\n\n// ----------------------------------------------------------------------------------- //\n// ----------------------------------------------------------------------------------- //\n\n/**\n * Mock implementation of the IClient interface.\n * \n * This class provides a no-operation (NOOP) implementation of the client API\n * that allows consuming code to function normally even when the actual player\n * API is not available. This typically occurs during development, testing,\n * or when the player script is blocked by ad-blockers.\n * \n * All methods in this class either return null/empty values or perform no\n * operations, allowing gadgets to function without errors while providing\n * appropriate fallback behavior.\n * \n * @private\n */\nclass NoopClient implements IClient {\n\n  constructor() {\n\n    console.log(\n      '%cClient API not available, falling back to mock API',\n      'background-color:blue; color:yellow;'\n    );\n  }\n\n  public callback(...args: any[]): void {\n\n    // NOOP implement, nothing to do....\n  }\n\n  public getDeviceTime(date?: Date): Promise<string> {\n\n    return Promise.resolve(new Date().toISOString());\n  }\n\n  public async getDeviceTimeZoneName(): Promise<string> {\n\n    return Promise.resolve(null);\n  }\n\n  public async getDeviceTimeZoneID(): Promise<string> {\n\n    return Promise.resolve(null);\n  }\n\n  public async getDeviceTimeZoneOffset(): Promise<number> {\n\n    return Promise.resolve(null);\n  }\n\n  public async getLanguageCode(): Promise<string> {\n\n    return Promise.resolve(null);\n  }\n\n  public async getDeviceKey(): Promise<string> {\n\n    return Promise.resolve(null);\n  }\n\n  public sendCommand(name: string, arg: string): void {\n\n    // NOOP implement, nothing to do....\n  }\n\n  public sendRemoteCommand(deviceKeys: string[], name: string, arg: string) {\n\n    // NOOP implement, nothing to do....\n  }\n\n  public track(eventName: string, properties?: string): void {\n\n    // NOOP implement, nothing to do....\n  }\n\n  public timeEvent(eventName: string): void {\n\n    // NOOP implement, nothing to do....\n  }\n\n  public newEventSession(id?: string): void {\n\n    // NOOP implement, nothing to do....\n  }\n\n  public async getRevelRoot(): Promise<string> {\n\n    return Promise.resolve(null);\n  }\n\n  public async getCommandMap(): Promise<string> {\n\n    return Promise.resolve('{}');\n  }\n\n  public finish(): void {\n\n    // NOOP implement, nothing to do....\n  }\n\n  public async getDevice(): Promise<string | null> {\n\n    return Promise.resolve(null);\n  }\n\n  public async getWidth(): Promise<number | null> {\n\n    return Promise.resolve(null);\n  }\n\n  public async getHeight(): Promise<number | null> {\n\n    return Promise.resolve(null);\n  }\n\n  public async getDuration(): Promise<number | null> {\n\n    return Promise.resolve(null);\n  }\n\n  public async getSdkVersion(): Promise<string> {\n\n    return Promise.resolve(version);\n  }\n\n  public applyConfig(prefs: IDictionary<any>): void {\n\n    const evt = { type: 'applyConfig', prefs: prefs, isOpener: window.opener !== null };\n\n    if (window.opener) {\n      window.opener.postMessage(\n        JSON.stringify(evt),\n        '*'\n      );\n    } else {\n      window.parent.postMessage(\n        JSON.stringify(evt),\n        '*'\n      );\n    }\n  }\n}\n"]}
@@ -1,3 +1,3 @@
1
1
  // Generated by genversion.
2
- export const version = '2.2.0';
3
- //# sourceMappingURL=data:application/json;base64,eyJ2ZXJzaW9uIjozLCJmaWxlIjoidmVyc2lvbi5qcyIsInNvdXJjZVJvb3QiOiIiLCJzb3VyY2VzIjpbIi4uLy4uLy4uLy4uLy4uL3Byb2plY3RzL3JldmVsZGlnaXRhbC9wbGF5ZXItY2xpZW50L3NyYy9saWIvdmVyc2lvbi50cyJdLCJuYW1lcyI6W10sIm1hcHBpbmdzIjoiQUFBQSwyQkFBMkI7QUFDM0IsTUFBTSxDQUFDLE1BQU0sT0FBTyxHQUFHLE9BQU8sQ0FBQSIsInNvdXJjZXNDb250ZW50IjpbIi8vIEdlbmVyYXRlZCBieSBnZW52ZXJzaW9uLlxuZXhwb3J0IGNvbnN0IHZlcnNpb24gPSAnMi4yLjAnXG4iXX0=
2
+ export const version = '2.3.0';
3
+ //# sourceMappingURL=data:application/json;base64,eyJ2ZXJzaW9uIjozLCJmaWxlIjoidmVyc2lvbi5qcyIsInNvdXJjZVJvb3QiOiIiLCJzb3VyY2VzIjpbIi4uLy4uLy4uLy4uLy4uL3Byb2plY3RzL3JldmVsZGlnaXRhbC9wbGF5ZXItY2xpZW50L3NyYy9saWIvdmVyc2lvbi50cyJdLCJuYW1lcyI6W10sIm1hcHBpbmdzIjoiQUFBQSwyQkFBMkI7QUFDM0IsTUFBTSxDQUFDLE1BQU0sT0FBTyxHQUFHLE9BQU8sQ0FBQSIsInNvdXJjZXNDb250ZW50IjpbIi8vIEdlbmVyYXRlZCBieSBnZW52ZXJzaW9uLlxuZXhwb3J0IGNvbnN0IHZlcnNpb24gPSAnMi4zLjAnXG4iXX0=
@@ -15,6 +15,7 @@ import * as i1$1 from '@angular/platform-browser';
15
15
  * Angular-friendly wrapper around the global `gadgets.reveldigital.datatable` library.
16
16
  *
17
17
  * Provides typed Promise-based methods and RxJS Observables for real-time events.
18
+ * All event emissions run inside Angular's zone to ensure change detection triggers.
18
19
  *
19
20
  * ```typescript
20
21
  * const dt = this.client.createDataTable('tbl_menu_items');
@@ -34,16 +35,18 @@ class DataTableRef {
34
35
  * Creates a new DataTableRef.
35
36
  *
36
37
  * @param tableId - The data table ID (e.g. 'tbl_menu_items')
38
+ * @param zone - Angular NgZone for ensuring change detection on callbacks
37
39
  * @param options - Optional configuration overrides
38
40
  * @throws Error if the global datatable library is not loaded
39
41
  */
40
- constructor(tableId, options) {
42
+ constructor(tableId, zone, options) {
41
43
  /** Emits when an existing row is modified. */
42
44
  this.rowUpdated$ = new Subject();
43
45
  /** Emits when a new row is added. */
44
46
  this.rowCreated$ = new Subject();
45
47
  /** Emits when a row is removed. */
46
48
  this.rowDeleted$ = new Subject();
49
+ this._zone = zone;
47
50
  const lib = window.gadgets?.['reveldigital.datatable'];
48
51
  if (!lib || typeof lib.create !== 'function') {
49
52
  throw new Error('RevelDigital DataTable library is not available. ' +
@@ -124,20 +127,27 @@ class DataTableRef {
124
127
  this.rowDeleted$.complete();
125
128
  }
126
129
  /** @ignore */
127
- static _fromInstance(instance) {
130
+ static _fromInstance(instance, zone) {
128
131
  const ref = Object.create(DataTableRef.prototype);
129
132
  ref.rowUpdated$ = new Subject();
130
133
  ref.rowCreated$ = new Subject();
131
134
  ref.rowDeleted$ = new Subject();
132
135
  ref._instance = instance;
136
+ ref._zone = zone;
133
137
  ref._wireEvents();
134
138
  return ref;
135
139
  }
136
140
  /** @ignore */
137
141
  _wireEvents() {
138
- this._onRowUpdated = (change) => this.rowUpdated$.next(change);
139
- this._onRowCreated = (change) => this.rowCreated$.next(change);
140
- this._onRowDeleted = (change) => this.rowDeleted$.next(change);
142
+ this._onRowUpdated = (change) => {
143
+ this._zone.run(() => this.rowUpdated$.next(change));
144
+ };
145
+ this._onRowCreated = (change) => {
146
+ this._zone.run(() => this.rowCreated$.next(change));
147
+ };
148
+ this._onRowDeleted = (change) => {
149
+ this._zone.run(() => this.rowDeleted$.next(change));
150
+ };
141
151
  this._instance.on('rowUpdated', this._onRowUpdated);
142
152
  this._instance.on('rowCreated', this._onRowCreated);
143
153
  this._instance.on('rowDeleted', this._onRowDeleted);
@@ -167,10 +177,11 @@ class DataTablePrefRef {
167
177
  * Creates a new DataTablePrefRef from a gadget preference JSON string.
168
178
  *
169
179
  * @param prefValue - The raw gadget preference string (JSON)
180
+ * @param zone - Angular NgZone for ensuring change detection on callbacks
170
181
  * @param options - Optional configuration overrides
171
182
  * @throws Error if the global datatable library is not loaded
172
183
  */
173
- constructor(prefValue, options) {
184
+ constructor(prefValue, zone, options) {
174
185
  const lib = window.gadgets?.['reveldigital.datatable'];
175
186
  if (!lib || typeof lib.createFromPref !== 'function') {
176
187
  throw new Error('RevelDigital DataTable library is not available. ' +
@@ -178,7 +189,7 @@ class DataTablePrefRef {
178
189
  }
179
190
  this._config = lib.createFromPref(prefValue, options);
180
191
  this.pref = this._config.pref;
181
- this.dataTable = DataTableRef._fromInstance(this._config.dt);
192
+ this.dataTable = DataTableRef._fromInstance(this._config.dt, zone);
182
193
  }
183
194
  /**
184
195
  * Fetches rows with the filter and sort settings from the preference automatically applied.
@@ -207,7 +218,7 @@ class DataTablePrefRef {
207
218
  }
208
219
 
209
220
  // Generated by genversion.
210
- const version = '2.2.0';
221
+ const version = '2.3.0';
211
222
 
212
223
  /**
213
224
  * Service for interacting with the Revel Digital player client.
@@ -340,6 +351,7 @@ class PlayerClientService {
340
351
  this.onConfig$.next(null);
341
352
  }
342
353
  }), tap(e => this.onPostMessage$.next(e)));
354
+ this.zone = zone;
343
355
  window.RevelDigital = {
344
356
  Controller: {
345
357
  onCommand: (name, arg) => {
@@ -1027,7 +1039,7 @@ class PlayerClientService {
1027
1039
  * ```
1028
1040
  */
1029
1041
  createDataTable(tableId, options) {
1030
- return new DataTableRef(tableId, options);
1042
+ return new DataTableRef(tableId, this.zone, options);
1031
1043
  }
1032
1044
  /**
1033
1045
  * Creates a typed data table wrapper from a gadget preference value.
@@ -1056,7 +1068,7 @@ class PlayerClientService {
1056
1068
  * ```
1057
1069
  */
1058
1070
  createDataTableFromPref(prefValue, options) {
1059
- return new DataTablePrefRef(prefValue, options);
1071
+ return new DataTablePrefRef(prefValue, this.zone, options);
1060
1072
  }
1061
1073
  // ---
1062
1074
  // PRIVATE METHODS.
@@ -1 +1 @@
1
- {"version":3,"file":"reveldigital-player-client.mjs","sources":["../../../../projects/reveldigital/player-client/src/lib/datatable-ref.ts","../../../../projects/reveldigital/player-client/src/lib/version.ts","../../../../projects/reveldigital/player-client/src/lib/player-client.service.ts","../../../../projects/reveldigital/player-client/src/lib/app-init.service.ts","../../../../projects/reveldigital/player-client/src/lib/safe-style.pipe.ts","../../../../projects/reveldigital/player-client/src/lib/player-client.module.ts","../../../../projects/reveldigital/player-client/src/public-api.ts","../../../../projects/reveldigital/player-client/src/reveldigital-player-client.ts"],"sourcesContent":["import { Subject } from 'rxjs';\nimport {\n IDataTableChangeEvent,\n IDataTableColumn,\n IDataTableOptions,\n IDataTablePref,\n IDataTableQueryParams,\n IDataTableResult,\n IDataTableSchema\n} from './interfaces/datatable.interface';\n\n/**\n * Angular-friendly wrapper around the global `gadgets.reveldigital.datatable` library.\n *\n * Provides typed Promise-based methods and RxJS Observables for real-time events.\n *\n * ```typescript\n * const dt = this.client.createDataTable('tbl_menu_items');\n *\n * // Fetch rows\n * const result = await dt.getRows({ sort: 'price', sortDir: 'asc' });\n *\n * // Real-time updates\n * dt.rowUpdated$.subscribe(change => console.log('Updated:', change));\n *\n * // Cleanup\n * dt.dispose();\n * ```\n */\nexport class DataTableRef {\n\n /** Emits when an existing row is modified. */\n public rowUpdated$ = new Subject<IDataTableChangeEvent>();\n\n /** Emits when a new row is added. */\n public rowCreated$ = new Subject<IDataTableChangeEvent>();\n\n /** Emits when a row is removed. */\n public rowDeleted$ = new Subject<IDataTableChangeEvent>();\n\n /** @ignore */\n private _instance: any;\n\n /** @ignore */\n private _onRowUpdated!: (change: IDataTableChangeEvent) => void;\n /** @ignore */\n private _onRowCreated!: (change: IDataTableChangeEvent) => void;\n /** @ignore */\n private _onRowDeleted!: (change: IDataTableChangeEvent) => void;\n\n /**\n * Creates a new DataTableRef.\n *\n * @param tableId - The data table ID (e.g. 'tbl_menu_items')\n * @param options - Optional configuration overrides\n * @throws Error if the global datatable library is not loaded\n */\n constructor(tableId: string, options?: IDataTableOptions) {\n\n const lib = (window as any).gadgets?.['reveldigital.datatable'];\n\n if (!lib || typeof lib.create !== 'function') {\n throw new Error(\n 'RevelDigital DataTable library is not available. ' +\n 'Ensure the datatable feature is enabled for this gadget.'\n );\n }\n\n this._instance = lib.create(tableId, options);\n this._wireEvents();\n }\n\n /**\n * Fetches rows from the data table.\n *\n * @param params - Optional query parameters (filter, sort, pagination)\n * @returns Promise resolving to the result set\n *\n * ```typescript\n * const result = await dt.getRows({\n * filter: { category: 'Entree', price: { op: 'lte', value: 25 } },\n * sort: 'itemName',\n * sortDir: 'asc',\n * pageSize: 20\n * });\n * ```\n */\n public getRows(params?: IDataTableQueryParams): Promise<IDataTableResult> {\n return this._instance.getRows(params);\n }\n\n /**\n * Fetches the table schema (column definitions and metadata).\n *\n * @returns Promise resolving to the table schema\n */\n public getSchema(): Promise<IDataTableSchema> {\n return this._instance.getSchema();\n }\n\n /**\n * Gets visible (non-hidden) columns from the table schema.\n *\n * @returns Promise resolving to an array of visible column definitions\n */\n public getVisibleColumns(): Promise<IDataTableColumn[]> {\n return this._instance.getVisibleColumns();\n }\n\n /**\n * Fetches rows with hidden column data stripped.\n *\n * @param params - Optional query parameters (same as getRows)\n * @returns Promise resolving to the result set with hidden fields removed\n */\n public getVisibleRows(params?: IDataTableQueryParams): Promise<IDataTableResult> {\n return this._instance.getVisibleRows(params);\n }\n\n /**\n * Starts polling for changes at the given interval.\n * Emits on `rowUpdated$` when new data is detected.\n *\n * @param intervalMs - Polling interval in milliseconds (default 30000)\n */\n public startPolling(intervalMs?: number): void {\n this._instance.startPolling(intervalMs);\n }\n\n /**\n * Stops polling for changes.\n */\n public stopPolling(): void {\n this._instance.stopPolling();\n }\n\n /**\n * Releases all resources: stops polling, closes the real-time connection,\n * removes event listeners, and completes all RxJS observables.\n */\n public dispose(): void {\n this._instance.off('rowUpdated', this._onRowUpdated);\n this._instance.off('rowCreated', this._onRowCreated);\n this._instance.off('rowDeleted', this._onRowDeleted);\n\n this._instance.dispose();\n\n this.rowUpdated$.complete();\n this.rowCreated$.complete();\n this.rowDeleted$.complete();\n }\n\n /** @ignore */\n static _fromInstance(instance: any): DataTableRef {\n const ref = Object.create(DataTableRef.prototype) as DataTableRef;\n ref.rowUpdated$ = new Subject<IDataTableChangeEvent>();\n ref.rowCreated$ = new Subject<IDataTableChangeEvent>();\n ref.rowDeleted$ = new Subject<IDataTableChangeEvent>();\n ref._instance = instance;\n ref._wireEvents();\n return ref;\n }\n\n /** @ignore */\n private _wireEvents(): void {\n this._onRowUpdated = (change: IDataTableChangeEvent) => this.rowUpdated$.next(change);\n this._onRowCreated = (change: IDataTableChangeEvent) => this.rowCreated$.next(change);\n this._onRowDeleted = (change: IDataTableChangeEvent) => this.rowDeleted$.next(change);\n\n this._instance.on('rowUpdated', this._onRowUpdated);\n this._instance.on('rowCreated', this._onRowCreated);\n this._instance.on('rowDeleted', this._onRowDeleted);\n }\n}\n\n\n/**\n * Wrapper around a data table created from a gadget preference value.\n *\n * Automatically configures filter and sort settings from the preference,\n * and provides a `getFilteredRows()` convenience method that applies them.\n *\n * ```typescript\n * const cfg = this.client.createDataTableFromPref(prefs.getString('rdDataTable'));\n *\n * // Fetch rows with auto-wired filter + sort from the preference\n * const result = await cfg.getFilteredRows();\n *\n * // Access the underlying DataTableRef for schema, events, etc.\n * cfg.dataTable.rowUpdated$.subscribe(change => console.log(change));\n *\n * // Cleanup\n * cfg.dispose();\n * ```\n */\nexport class DataTablePrefRef {\n\n /** The underlying DataTableRef with full access to schema, events, polling, etc. */\n public readonly dataTable: DataTableRef;\n\n /** The parsed preference object. */\n public readonly pref: IDataTablePref;\n\n /** @ignore */\n private _config: any;\n\n /**\n * Creates a new DataTablePrefRef from a gadget preference JSON string.\n *\n * @param prefValue - The raw gadget preference string (JSON)\n * @param options - Optional configuration overrides\n * @throws Error if the global datatable library is not loaded\n */\n constructor(prefValue: string, options?: IDataTableOptions) {\n\n const lib = (window as any).gadgets?.['reveldigital.datatable'];\n\n if (!lib || typeof lib.createFromPref !== 'function') {\n throw new Error(\n 'RevelDigital DataTable library is not available. ' +\n 'Ensure the datatable feature is enabled for this gadget.'\n );\n }\n\n this._config = lib.createFromPref(prefValue, options);\n this.pref = this._config.pref;\n this.dataTable = DataTableRef._fromInstance(this._config.dt);\n }\n\n /**\n * Fetches rows with the filter and sort settings from the preference automatically applied.\n * Additional query parameters can override or supplement the preference settings.\n *\n * @param params - Optional additional query parameters\n * @returns Promise resolving to the result set\n *\n * ```typescript\n * // Use preference defaults\n * const result = await cfg.getFilteredRows();\n *\n * // Override page size\n * const page = await cfg.getFilteredRows({ pageSize: 10 });\n * ```\n */\n public getFilteredRows(params?: IDataTableQueryParams): Promise<IDataTableResult> {\n return this._config.getFilteredRows(params);\n }\n\n /**\n * Releases all resources held by the underlying DataTableRef.\n */\n public dispose(): void {\n this.dataTable.dispose();\n }\n}\n","// Generated by genversion.\nexport const version = '2.2.0'\n","import { Injectable, NgZone, OnDestroy } from '@angular/core';\nimport { gadgets } from '@reveldigital/gadget-types';\nimport { BehaviorSubject, fromEvent, Subject, Subscription } from 'rxjs';\nimport { filter, map, share, tap } from 'rxjs/operators';\nimport { IClient } from './interfaces/client.interface';\nimport { ICommand } from './interfaces/command.interface';\nimport { IDictionary } from './interfaces/config.interface';\nimport { IDataTableOptions } from './interfaces/datatable.interface';\nimport { IDevice } from './interfaces/device.interface';\nimport { IEventProperties } from './interfaces/event-properties.interface';\nimport { DataTableRef, DataTablePrefRef } from './datatable-ref';\nimport { version } from './version';\n\n//import { version } from './version.js';\n\n// So that TypeScript doesn't complain, we're going to augment the GLOBAL / WINDOW \n// name-space definition to include the Tracker API. This also provides us with a place\n// to actually DOCUMENT the API so that our developers aren't guessing about what's\n// available on the library.\n\n/** @ignore */\ndeclare global {\n let Client: IClient;\n}\n\n/**\n * Service for interacting with the Revel Digital player client.\n * \n * This service provides a comprehensive interface for gadgets and web applications\n * to communicate with the Revel Digital player environment. It handles device\n * information, commands, events, analytics tracking, and configuration management.\n * \n * The service supports both direct API calls and event-based communication patterns,\n * allowing gadgets to respond to player lifecycle events (start, stop, commands) and\n * send data back to the player or remote devices.\n * \n * ```typescript\n * constructor(private client: PlayerClientService) {\n * // Subscribe to player events\n * this.client.onStart$.subscribe(() => {\n * console.log('Gadget started');\n * });\n * \n * this.client.onCommand$.subscribe(command => {\n * console.log('Received command:', command);\n * });\n * }\n * \n * async ngOnInit() {\n * // Get device information\n * const device = await this.client.getDevice();\n * const deviceTime = await this.client.getDeviceTime();\n * \n * // Track analytics\n * this.client.track('gadget_loaded', { version: '1.0' });\n * }\n * ```\n * \n * @since 1.0.0\n */\n@Injectable({\n providedIn: 'root'\n})\nexport class PlayerClientService implements OnDestroy {\n\n /** @ignore */\n private clientPromise: Promise<IClient> | null;\n\n /**\n * Observable stream of commands sent to this player from the Revel Digital platform.\n * Subscribe to this to handle custom commands sent from templates, playlists, or remote devices.\n * \n * ```typescript\n * this.client.onCommand$.subscribe(command => {\n * if (command.name === 'customAction') {\n * this.handleCustomAction(command.arg);\n * }\n * });\n * ```\n */\n public onCommand$ = new Subject<ICommand>();\n\n /**\n * Observable that signals when the gadget has been loaded and is ready to start.\n * Emits `true` when ready, `false` when destroyed.\n * \n * ```typescript\n * this.client.onReady$.subscribe(isReady => {\n * if (isReady) {\n * console.log('Client is ready');\n * this.initializeGadget();\n * }\n * });\n * ```\n */\n public onReady$ = new BehaviorSubject(false);\n\n /**\n * Observable that signals when the gadget has been started by the player.\n * This event occurs when the player begins execution of the gadget content.\n * \n * ```typescript\n * this.client.onStart$.subscribe(() => {\n * console.log('Gadget started');\n * this.startAnimation();\n * });\n * ```\n */\n public onStart$ = new Subject();\n\n /**\n * Observable that signals when the gadget has been stopped by the player.\n * This event occurs when the player stops execution, typically when moving\n * to the next item in a playlist or when the content duration expires.\n * \n * ```typescript\n * this.client.onStop$.subscribe(() => {\n * console.log('Gadget stopped');\n * this.cleanup();\n * });\n * ```\n */\n public onStop$ = new Subject();\n\n /**\n * Observable that signals when the gadget should open the configuration window.\n * This allows gadgets to respond to configuration requests from the player.\n * \n * ```typescript\n * this.client.onConfig$.subscribe(() => {\n * this.openConfigurationDialog();\n * });\n * ```\n */\n public onConfig$ = new Subject();\n\n /**\n * Observable that signals when the gadget has received a postMessage event from the player.\n * This handles communication between the gadget and player via the postMessage API.\n * \n * ```typescript\n * this.client.onPostMessage$.subscribe(message => {\n * console.log('Received message:', message);\n * this.handlePlayerMessage(message);\n * });\n * ```\n */\n public onPostMessage$ = new Subject();\n\n //\n // Two methods available for calling into the library:\n //\n // 1) Using dispatchEvent() with the following custom events\n // 2) Using the window scoped RevelDigital object as defined in the constructor\n //\n /** @ignore */\n private onStartSub: Subscription;\n /** @ignore */\n private onStartEvt$ = fromEvent(window, 'RevelDigital.Start').pipe(\n share(),\n tap(this.onStart$)\n );\n /** @ignore */\n private onStopSub: Subscription;\n /** @ignore */\n private onStopEvt$ = fromEvent(window, 'RevelDigital.Stop').pipe(\n share(),\n tap(this.onStop$)\n );\n /** @ignore */\n private onCommandSub: Subscription;\n /** @ignore */\n private onCommandEvt$ = fromEvent<ICommand>(window, 'RevelDigital.Command').pipe(\n map((e: any) => { return { name: e.detail.name, arg: e.detail.arg } as ICommand }),\n share(),\n tap(this.onCommand$)\n );\n /** @ignore */\n // private onConfigSub: Subscription;\n // /** @ignore */\n // private onConfigEvt$ = fromEvent(window, 'RevelDigital.Config').pipe(\n // share(),\n // tap((e: CustomEvent) => {\n // console.log(e);\n\n // if (e.detail.type === 'applyConfig' && e.detail.isOpener) {\n // this.applyConfig(e.detail.config); // propagate config to iframe parent from the popup window\n // } else {\n // this.onConfig$.next(e.detail);\n // }\n // })\n // );\n // private onPostMessageSub: Subscription;\n // private onPostMessageEvt$ = fromEvent(window, 'message').pipe(\n // filter((messageEvent: MessageEvent) =>\n // messageEvent.source !== window.parent &&\n // typeof messageEvent.data === 'string' &&\n // messageEvent.data.startsWith('reveldigital:')),\n // map((e: any) => { return JSON.parse(e.substring(13)) as Command }),\n // share(),\n // tap(this.onCommand$)\n // );\n\n private onPostMessageSub: Subscription;\n private onPostMessageEvt$ = fromEvent(window, 'message').pipe(\n filter((messageEvent: MessageEvent) =>\n //messageEvent.source !== window.parent &&\n typeof messageEvent.data === 'string'),\n map((e: any) => JSON.parse(e.data)),\n share(),\n tap((e: any) => {\n if (e.type === 'applyConfig' && e.isOpener) {\n e.isOpener = false;\n // propagate config to iframe parent from the popup window\n window.parent.postMessage(\n JSON.stringify(e),\n '*'\n );\n } else if (e.type === 'openConfig') {\n this.onConfig$.next(null);\n }\n }),\n tap(e => this.onPostMessage$.next(e))\n );\n\n\n /** @ignore */\n constructor(zone: NgZone) {\n\n (window as any).RevelDigital = {\n Controller: {\n onCommand: (name: string, arg: string) => {\n zone.run(() => {\n this.onCommand$.next({ name: name, arg: arg });\n });\n },\n onStart: () => {\n zone.run(() => {\n this.onStart$.next(null);\n });\n },\n onStop: () => {\n zone.run(() => {\n this.onStop$.next(null);\n });\n }\n }\n }\n\n this.onStartSub = this.onStartEvt$.subscribe(() => { });\n this.onStopSub = this.onStopEvt$.subscribe(() => { });\n this.onCommandSub = this.onCommandEvt$.subscribe(() => { });\n //this.onConfigSub = this.onConfigEvt$.subscribe(() => { });\n this.onPostMessageSub = this.onPostMessageEvt$.subscribe(() => { });\n\n this.clientPromise = null;\n\n this.onReady$.next(true);\n }\n\n /** @ignore */\n ngOnDestroy(): void {\n\n this.onStartSub?.unsubscribe();\n this.onStopSub?.unsubscribe();\n this.onCommandSub?.unsubscribe();\n //this.onConfigSub?.unsubscribe();\n this.onPostMessageSub?.unsubscribe();\n\n this.onReady$.next(false);\n }\n\n /** @ignore */\n public static init(data: any) {\n\n console.log(\n '%c⚙️ Initializing Revel Digital client library',\n 'background-color:blue; color:yellow;'\n );\n }\n\n /**\n * Sends a callback to player scripting with variable arguments.\n * \n * This method allows the gadget to communicate with player scripting.\n * If the appropriate scripting is in place in the currently running template, \n * calling this method will initiate a callback which can be acted upon in player script.\n * \n * @param args - Variable number of arguments to pass to the callback (max 5 arguments)\n * \n * ```typescript\n * // Send a simple callback\n * this.client.callback('test', 'data');\n * \n * // Send multiple parameters\n * this.client.callback('action', 'value1', 'value2', { data: 'object' });\n * ```\n */\n public callback(...args: any[]): void {\n\n this.getClient().then((client) => {\n\n switch (args.length) {\n case 0:\n client.callback();\n break;\n case 1:\n client.callback(args[0]);\n break;\n case 2:\n client.callback(args[1]);\n break;\n case 3:\n client.callback(args[2]);\n break;\n case 4:\n client.callback(args[3]);\n break;\n case 5:\n client.callback(args[4]);\n break;\n }\n })\n }\n\n /**\n * Gets the user preferences interface exposed by the Google Gadgets API.\n * \n * This method provides access to gadget preferences which can be configured\n * in the Revel Digital CMS. Use this to retrieve user-configurable settings\n * for your gadget.\n * \n * @returns The Gadgets API Prefs object for accessing preference values\n * \n * ```typescript\n * constructor(public client: PlayerClientService) {\n * const prefs = client.getPrefs();\n * const myString = prefs.getString('myStringPref');\n * const myNumber = prefs.getInt('myNumberPref');\n * const myBool = prefs.getBool('myBoolPref');\n * }\n * ```\n * \n * @see {@link https://developers.google.com/gadgets/docs/basic} Google Gadgets API documentation\n */\n public getPrefs(): gadgets.Prefs {\n\n return new window['gadgets']['Prefs']();\n }\n\n /**\n * Gets the current device time in ISO8601 format.\n * \n * Current device time is determined by the device timezone assigned to the device in the CMS.\n * This is useful for displaying time-sensitive content or scheduling operations based on\n * the device's local time rather than browser time.\n * \n * @param date - Optional. If supplied, will translate the supplied date/time to device time based on respective timezones\n * @returns Promise resolving to date/time in ISO8601 format\n * \n * ```typescript\n * // Get current device time\n * const currentTime = await this.client.getDeviceTime();\n * console.log('Device time:', currentTime);\n * \n * // Convert a specific date to device time\n * const specificDate = new Date('2023-01-01T12:00:00Z');\n * const deviceTime = await this.client.getDeviceTime(specificDate);\n * ```\n */\n public async getDeviceTime(date?: Date): Promise<string> {\n\n const client = await this.getClient();\n\n if (date !== undefined) {\n return client.getDeviceTime(date);\n }\n return client.getDeviceTime();\n }\n\n /**\n * Gets the timezone name currently assigned to the device.\n * \n * @returns Promise resolving to the timezone name (e.g., \"America/New_York\")\n * \n * ```typescript\n * const timezoneName = await this.client.getDeviceTimeZoneName();\n * console.log('Device timezone:', timezoneName);\n * ```\n */\n public async getDeviceTimeZoneName(): Promise<string> {\n\n const client = await this.getClient();\n\n return client.getDeviceTimeZoneName();\n }\n\n /**\n * Gets the timezone ID currently assigned to the device.\n * \n * @returns Promise resolving to the timezone ID\n * \n * ```typescript\n * const timezoneId = await this.client.getDeviceTimeZoneID();\n * console.log('Device timezone ID:', timezoneId);\n * ```\n */\n public async getDeviceTimeZoneID(): Promise<string> {\n\n const client = await this.getClient();\n\n return client.getDeviceTimeZoneID();\n }\n\n /**\n * Gets the numerical offset from GMT of the timezone currently assigned to the device.\n * \n * @returns Promise resolving to the timezone offset in hours (e.g., -5 for EST)\n * \n * ```typescript\n * const offset = await this.client.getDeviceTimeZoneOffset();\n * console.log('Device timezone offset:', offset, 'hours from GMT');\n * ```\n */\n public async getDeviceTimeZoneOffset(): Promise<number> {\n\n const client = await this.getClient();\n\n return client.getDeviceTimeZoneOffset();\n }\n\n /**\n * Gets the language code of the language currently assigned to the device.\n * \n * @returns Promise resolving to the language code (e.g., \"en-US\", \"fr-FR\")\n * \n * ```typescript\n * const languageCode = await this.client.getLanguageCode();\n * console.log('Device language:', languageCode);\n * // Use for localization\n * this.loadLanguageResources(languageCode);\n * ```\n */\n public async getLanguageCode(): Promise<string> {\n\n const client = await this.getClient();\n\n return client.getLanguageCode();\n }\n\n /**\n * Gets the unique Revel Digital device key associated with the device.\n * \n * The device key is a unique identifier for this specific player device\n * and can be used for device-specific operations or remote commands.\n * \n * @returns Promise resolving to the device key string\n * \n * ```typescript\n * const deviceKey = await this.client.getDeviceKey();\n * console.log('Device key:', deviceKey);\n * ```\n */\n public async getDeviceKey(): Promise<string> {\n\n const client = await this.getClient();\n\n return client.getDeviceKey();\n }\n\n /**\n * Sends a command to the local player device.\n * \n * Commands can be used to control various aspects of the player or trigger\n * specific behaviors. The command is processed by the local player only.\n * \n * @param name - The command name/identifier\n * @param arg - The command argument/payload\n * \n * ```typescript\n * // Send a simple command\n * this.client.sendCommand('refresh', '');\n * \n * // Send a command with data\n * this.client.sendCommand('setVolume', '50');\n * \n * // Send a command with JSON data\n * this.client.sendCommand('configure', JSON.stringify({ setting: 'value' }));\n * ```\n */\n public sendCommand(name: string, arg: string): void {\n\n this.getClient().then((client) => {\n client.sendCommand(name, arg);\n })\n }\n\n /**\n * Sends a command to remote player devices with the specified device keys.\n * \n * Remote commands allow cross-device communication within the same Revel Digital account.\n * This is useful for synchronized displays, device coordination, or remote control scenarios.\n * \n * Note: Remote commands can only be delivered to devices within the same account as the sender device.\n * \n * @param deviceKeys - Array of target device keys to send the command to\n * @param name - The command name/identifier\n * @param arg - The command argument/payload\n * \n * ```typescript\n * // Send command to specific devices\n * const targetDevices = ['device-key-1', 'device-key-2'];\n * this.client.sendRemoteCommand(targetDevices, 'syncAction', 'start');\n * \n * // Broadcast to multiple devices\n * this.client.sendRemoteCommand(\n * ['lobby-display', 'kiosk-1', 'kiosk-2'], \n * 'updateContent', \n * JSON.stringify({ contentId: '12345' })\n * );\n * ```\n */\n public sendRemoteCommand(deviceKeys: string[], name: string, arg: string): void {\n\n this.getClient().then((client) => {\n client.sendRemoteCommand(deviceKeys, name, arg);\n });\n }\n\n /**\n * Logs an analytics event for use with AdHawk analytics and reporting.\n * \n * Events are used for tracking various metrics including usage statistics, \n * player condition, state changes, user interactions, and custom business metrics.\n * These events can be viewed in the Revel Digital analytics dashboard.\n * \n * @param eventName - Unique name for this event (should be descriptive and consistent)\n * @param properties - Optional map of user-defined properties to associate with this event\n * \n * ```typescript\n * // Simple event tracking\n * this.client.track('gadget_loaded');\n * \n * // Event with properties\n * this.client.track('user_interaction', {\n * action: 'button_click',\n * button_id: 'start_button',\n * timestamp: new Date().toISOString()\n * });\n * \n * // Performance tracking\n * this.client.track('content_displayed', {\n * content_type: 'video',\n * duration: 30,\n * quality: 'HD'\n * });\n * ```\n */\n public track(eventName: string, properties?: IEventProperties): void {\n\n this.getClient().then((client) => {\n client.track(eventName, JSON.stringify(properties));\n })\n }\n\n /**\n * Initiates a timed event for duration tracking.\n * \n * Timed events are useful for tracking the duration of operations or user interactions.\n * This method must be followed by a call to track() with the same event name to complete\n * the timing measurement. The duration will be automatically calculated and included\n * in the event properties.\n * \n * @param eventName - Unique name for this timed event (must match the subsequent track() call)\n * \n * ```typescript\n * // Start timing an event\n * this.client.timeEvent('video_playback');\n * \n * // ... video plays for some duration ...\n * \n * // End timing and log the event with duration\n * this.client.track('video_playback', {\n * video_id: 'abc123',\n * quality: 'HD'\n * }); // Duration will be automatically added\n * \n * // Example for user interaction timing\n * this.client.timeEvent('form_completion');\n * // ... user fills out form ...\n * this.client.track('form_completion', { form_type: 'contact' });\n * ```\n */\n public timeEvent(eventName: string): void {\n\n this.getClient().then((client) => {\n client.timeEvent(eventName);\n })\n }\n\n /**\n * Creates a new analytics event session for grouping related events.\n * \n * A session is a way of grouping events together for analysis. Each event tracked\n * after calling this method will have the same session ID until a new session is created.\n * Session IDs are randomly generated unless explicitly provided.\n * \n * This is useful for tracking user journeys, workflow completion, or grouping\n * related interactions within a specific time period.\n * \n * @param id - Optional user-supplied session ID. If not provided, a random session ID will be generated\n * \n * ```typescript\n * // Start a new session with auto-generated ID\n * this.client.newEventSession();\n * this.client.track('session_start');\n * this.client.track('user_action_1');\n * this.client.track('user_action_2');\n * \n * // Start a session with custom ID\n * this.client.newEventSession('user-workflow-12345');\n * this.client.track('workflow_start');\n * this.client.track('step_completed', { step: 1 });\n * \n * // Start a new session for different workflow\n * this.client.newEventSession();\n * this.client.track('different_workflow_start');\n * ```\n */\n public newEventSession(id?: string): void {\n\n this.getClient().then((client) => {\n if (id !== undefined) {\n client.newEventSession();\n } else {\n client.newEventSession(id);\n }\n })\n }\n\n /**\n * Gets the root folder path utilized by this player device.\n * \n * This returns the base directory path where the Revel Digital player\n * stores its files and resources on the local device.\n * \n * @returns Promise resolving to the path of the root folder\n * \n * ```typescript\n * const rootPath = await this.client.getRevelRoot();\n * console.log('Player root directory:', rootPath);\n * // Use for constructing file paths or understanding storage structure\n * ```\n */\n public async getRevelRoot(): Promise<string> {\n\n const client = await this.getClient();\n\n return client.getRevelRoot();\n }\n\n /**\n * Gets a map of commands currently active for this device.\n * \n * This returns the current command configuration that defines how the device\n * responds to various command triggers and remote commands.\n * \n * @returns Promise resolving to a map of currently active commands\n * \n * ```typescript\n * const commandMap = await this.client.getCommandMap();\n * console.log('Active commands:', commandMap);\n * \n * // Check if specific command is available\n * if (commandMap['customCommand']) {\n * console.log('Custom command is available');\n * }\n * ```\n */\n public async getCommandMap(): Promise<any> {\n\n const client = await this.getClient();\n\n return JSON.parse(await client.getCommandMap());\n }\n\n /**\n * Signals to the player that this gadget has completed its visualization.\n * \n * This method notifies the player that the current gadget has finished its\n * content display or interaction cycle. The player can then proceed with\n * the next item in a playlist if applicable, or handle the completion\n * according to its configuration.\n * \n * Call this method when your gadget has completed its intended function,\n * such as finishing an animation, completing a form, or reaching a natural\n * stopping point.\n * \n * ```typescript\n * // After completing an animation\n * private onAnimationComplete(): void {\n * this.client.finish();\n * }\n * \n * // After user interaction is complete\n * private onFormSubmitted(): void {\n * this.saveFormData();\n * this.client.finish();\n * }\n * \n * // After a timed display period\n * setTimeout(() => {\n * this.client.finish();\n * }, 30000); // 30 seconds\n * ```\n */\n public finish(): void {\n\n this.getClient().then((client) => {\n\n client.finish();\n })\n }\n\n /**\n * Checks if the gadget is running in preview mode.\n * \n * Preview mode is enabled when the gadget is being edited in the Revel Digital CMS,\n * tested in the gadget editor, or otherwise not running in a normal player environment.\n * This is useful for providing different behavior during development/testing versus\n * production deployment.\n * \n * @returns Promise resolving to true if running in preview mode, false if running on actual player\n * \n * ```typescript\n * const isPreview = await this.client.isPreviewMode();\n * \n * if (isPreview) {\n * console.log('Running in preview mode - using mock data');\n * this.loadMockData();\n * } else {\n * console.log('Running on player device - using live data');\n * this.loadLiveData();\n * }\n * \n * // Show different UI in preview\n * this.showPreviewIndicator = isPreview;\n * ```\n */\n public async isPreviewMode(): Promise<boolean> {\n\n const client = await this.getClient();\n\n return client instanceof NoopClient;\n }\n\n /**\n * Gets detailed information about the device running the player.\n * \n * Returns comprehensive device details including name, registration key, type,\n * service date, language, timezone, tags, and location information. This data\n * is configured in the Revel Digital CMS for each device.\n * \n * @returns Promise resolving to device details object, or null if not available\n * \n * ```typescript\n * const device = await this.client.getDevice();\n * \n * if (device) {\n * console.log('Device name:', device.name);\n * console.log('Device type:', device.deviceType);\n * console.log('Location:', device.location.city, device.location.state);\n * console.log('Tags:', device.tags);\n * \n * // Use device info for customization\n * if (device.location.country === 'US') {\n * this.loadUSContent();\n * }\n * \n * // Check device capabilities based on type\n * if (device.deviceType === 'android') {\n * this.enableTouchFeatures();\n * }\n * }\n * ```\n */\n public async getDevice(): Promise<IDevice | null> {\n\n const client = await this.getClient();\n\n const obj: any = JSON.parse(<string>await client.getDevice());\n\n const device: IDevice[] = [obj].map((device: any) => {\n\n return {\n name: device.name,\n registrationKey: device.key,\n deviceType: device.devicetype,\n enteredService: new Date(device.enteredservice),\n langCode: device.langcode,\n timeZone: device.timezone,\n tags: device.description?.split('\\n'),\n location: {\n city: device.location?.city,\n state: device.location?.state,\n country: device.location?.country,\n postalCode: device.location?.postalcode,\n address: device.location?.address,\n latitude: device.location?.latitude,\n longitude: device.location?.longitude\n }\n }\n });\n return device[0];\n }\n\n /**\n * Gets the width of the visualization area in pixels.\n * \n * This returns the available width for content display, which may be\n * different from the full screen width depending on player configuration\n * and template layout.\n * \n * @returns Promise resolving to width in pixels, or null if not available\n * \n * ```typescript\n * const width = await this.client.getWidth();\n * \n * if (width) {\n * console.log('Available width:', width, 'pixels');\n * \n * // Adapt content layout based on width\n * if (width < 800) {\n * this.enableMobileLayout();\n * } else {\n * this.enableDesktopLayout();\n * }\n * }\n * ```\n */\n public async getWidth(): Promise<number | null> {\n\n const client = await this.getClient();\n\n return client.getWidth();\n }\n\n /**\n * Gets the height of the visualization area in pixels.\n * \n * This returns the available height for content display, which may be\n * different from the full screen height depending on player configuration\n * and template layout.\n * \n * @returns Promise resolving to height in pixels, or null if not available\n * \n * ```typescript\n * const height = await this.client.getHeight();\n * \n * if (height) {\n * console.log('Available height:', height, 'pixels');\n * \n * // Calculate aspect ratio for responsive design\n * const width = await this.client.getWidth();\n * const aspectRatio = width / height;\n * this.adjustContentForAspectRatio(aspectRatio);\n * }\n * ```\n */\n public async getHeight(): Promise<number | null> {\n\n const client = await this.getClient();\n\n return client.getHeight();\n }\n\n /**\n * Gets the duration of the currently playing content item.\n * \n * This method is only applicable when the gadget is associated with a playlist\n * and returns the duration assigned to the current playlist item. The duration\n * determines how long the content should be displayed before moving to the next item.\n * \n * @returns Promise resolving to duration in milliseconds, or null if not applicable/available\n * \n * ```typescript\n * const duration = await this.client.getDuration();\n * \n * if (duration) {\n * console.log('Content duration:', duration, 'milliseconds');\n * \n * // Set up auto-finish timer\n * setTimeout(() => {\n * this.client.finish();\n * }, duration);\n * \n * // Show progress indicator\n * this.startProgressIndicator(duration);\n * }\n * ```\n */\n public async getDuration(): Promise<number | null> {\n\n const client = await this.getClient();\n\n return client.getDuration();\n }\n\n /**\n * Gets the current version of the Revel Digital SDK.\n * \n * @returns Promise resolving to the SDK version string\n * \n * ```typescript\n * const version = await this.client.getSdkVersion();\n * console.log('SDK Version:', version);\n * \n * // Use for compatibility checks or logging\n * this.client.track('gadget_loaded', { sdkVersion: version });\n * ```\n */\n public async getSdkVersion(): Promise<string> {\n\n return Promise.resolve(version);\n }\n\n /**\n * Applies configuration preferences to the gadget (preview mode only).\n * \n * This method is only available when running in preview mode (typically during\n * gadget development or testing in the CMS). It allows applying configuration\n * changes that would normally come from the gadget's preference settings.\n * \n * @param prefs - Dictionary of preference key-value pairs to apply\n * \n * ```typescript\n * if (await this.client.isPreviewMode()) {\n * // Apply test configuration in preview\n * await this.client.applyConfig({\n * 'title': 'Test Title',\n * 'backgroundColor': '#ff0000',\n * 'showBorder': true,\n * 'refreshInterval': 30\n * });\n * }\n * ```\n */\n public async applyConfig(prefs: IDictionary<any>) {\n\n if (await this.isPreviewMode()) {\n const client = await this.getClient();\n client.applyConfig(prefs);\n } else {\n console.log(\n '%capplyConfig() is only available in preview mode.',\n 'background-color:blue; color:yellow;'\n );\n }\n }\n\n /**\n * Creates a typed wrapper for a Revel Digital data table.\n *\n * The data table feature must be enabled for the gadget. The returned\n * {@link DataTableRef} provides typed Promise-based methods and RxJS\n * Observables for real-time row change events.\n *\n * @param tableId - The data table ID (e.g. 'tbl_menu_items')\n * @param options - Optional configuration overrides\n * @returns A {@link DataTableRef} instance\n * @throws Error if the global datatable library is not loaded\n *\n * ```typescript\n * const dt = this.client.createDataTable('tbl_menu_items');\n *\n * // Fetch rows with filtering and sorting\n * const result = await dt.getRows({\n * filter: { category: 'Entree', price: { op: 'lte', value: 25 } },\n * sort: 'price',\n * sortDir: 'asc'\n * });\n *\n * // Subscribe to real-time updates\n * dt.rowUpdated$.subscribe(change => console.log('Row updated:', change));\n * dt.rowCreated$.subscribe(change => console.log('Row created:', change));\n * dt.rowDeleted$.subscribe(change => console.log('Row deleted:', change));\n *\n * // Cleanup when done\n * dt.dispose();\n * ```\n */\n public createDataTable(tableId: string, options?: IDataTableOptions): DataTableRef {\n return new DataTableRef(tableId, options);\n }\n\n /**\n * Creates a typed data table wrapper from a gadget preference value.\n *\n * The preference JSON string (as serialized by the template editor's datatable\n * option) is parsed and used to auto-configure filter, sort, and logic settings.\n * The returned {@link DataTablePrefRef} provides a `getFilteredRows()` convenience\n * method that applies these settings automatically.\n *\n * @param prefValue - The raw gadget preference string (JSON)\n * @param options - Optional configuration overrides\n * @returns A {@link DataTablePrefRef} instance\n * @throws Error if the global datatable library is not loaded\n *\n * ```typescript\n * const cfg = this.client.createDataTableFromPref(prefs.getString('rdDataTable'));\n *\n * // Fetch rows with auto-wired filter + sort\n * const result = await cfg.getFilteredRows();\n *\n * // Access the underlying DataTableRef for events, schema, etc.\n * cfg.dataTable.rowUpdated$.subscribe(change => console.log(change));\n *\n * // Cleanup when done\n * cfg.dispose();\n * ```\n */\n public createDataTableFromPref(prefValue: string, options?: IDataTableOptions): DataTablePrefRef {\n return new DataTablePrefRef(prefValue, options);\n }\n\n // ---\n // PRIVATE METHODS.\n // ---\n /** @ignore */\n private getClient(): Promise<IClient> {\n\n if (this.clientPromise) {\n\n return (this.clientPromise);\n }\n\n if ((window as any).Client) {\n\n return (this.clientPromise = Promise.resolve((window as any).Client));\n }\n\n // A \"complete\" status indicates that the \"load\" event has been fired on the\n // window; and, that all sub-resources such as Scripts, Images, and Frames have\n // been loaded.\n if (window.document.readyState === \"complete\") {\n\n // If this event has fired AND the 3rd-party script isn't available (see IF-\n // condition BEFORE this one), it means that the 3rd-party script either\n // failed on the network or was BLOCKED by an ad-blocker. As such, we have to\n // fall-back to using a mock API.\n return (this.clientPromise = Promise.resolve(new NoopClient()));\n }\n\n // ASSERT: If we made it this far, the document has not completed loading (but it\n // may be in an \"interactive\" state which is when I believe that the Angular app\n // gets bootstrapped). As such, we need bind to the LOAD event to wait for our\n // third-party scripts to load (or fail to load, or be blocked).\n this.clientPromise = new Promise<IClient>(\n (resolve) => {\n\n window.addEventListener(\n \"load\",\n function handleWindowLoad() {\n\n // At this point, the 3rd-party library is either available or\n // it's not - there's no further loading to do. If it's not\n // present on the global scope, we're going to fall-back to using\n // a mock API.\n resolve((window as any).Client || new NoopClient());\n }\n );\n\n }\n );\n\n return (this.clientPromise);\n }\n}\n\n\n\n// ----------------------------------------------------------------------------------- //\n// ----------------------------------------------------------------------------------- //\n\n/**\n * Mock implementation of the IClient interface.\n * \n * This class provides a no-operation (NOOP) implementation of the client API\n * that allows consuming code to function normally even when the actual player\n * API is not available. This typically occurs during development, testing,\n * or when the player script is blocked by ad-blockers.\n * \n * All methods in this class either return null/empty values or perform no\n * operations, allowing gadgets to function without errors while providing\n * appropriate fallback behavior.\n * \n * @private\n */\nclass NoopClient implements IClient {\n\n constructor() {\n\n console.log(\n '%cClient API not available, falling back to mock API',\n 'background-color:blue; color:yellow;'\n );\n }\n\n public callback(...args: any[]): void {\n\n // NOOP implement, nothing to do....\n }\n\n public getDeviceTime(date?: Date): Promise<string> {\n\n return Promise.resolve(new Date().toISOString());\n }\n\n public async getDeviceTimeZoneName(): Promise<string> {\n\n return Promise.resolve(null);\n }\n\n public async getDeviceTimeZoneID(): Promise<string> {\n\n return Promise.resolve(null);\n }\n\n public async getDeviceTimeZoneOffset(): Promise<number> {\n\n return Promise.resolve(null);\n }\n\n public async getLanguageCode(): Promise<string> {\n\n return Promise.resolve(null);\n }\n\n public async getDeviceKey(): Promise<string> {\n\n return Promise.resolve(null);\n }\n\n public sendCommand(name: string, arg: string): void {\n\n // NOOP implement, nothing to do....\n }\n\n public sendRemoteCommand(deviceKeys: string[], name: string, arg: string) {\n\n // NOOP implement, nothing to do....\n }\n\n public track(eventName: string, properties?: string): void {\n\n // NOOP implement, nothing to do....\n }\n\n public timeEvent(eventName: string): void {\n\n // NOOP implement, nothing to do....\n }\n\n public newEventSession(id?: string): void {\n\n // NOOP implement, nothing to do....\n }\n\n public async getRevelRoot(): Promise<string> {\n\n return Promise.resolve(null);\n }\n\n public async getCommandMap(): Promise<string> {\n\n return Promise.resolve('{}');\n }\n\n public finish(): void {\n\n // NOOP implement, nothing to do....\n }\n\n public async getDevice(): Promise<string | null> {\n\n return Promise.resolve(null);\n }\n\n public async getWidth(): Promise<number | null> {\n\n return Promise.resolve(null);\n }\n\n public async getHeight(): Promise<number | null> {\n\n return Promise.resolve(null);\n }\n\n public async getDuration(): Promise<number | null> {\n\n return Promise.resolve(null);\n }\n\n public async getSdkVersion(): Promise<string> {\n\n return Promise.resolve(version);\n }\n\n public applyConfig(prefs: IDictionary<any>): void {\n\n const evt = { type: 'applyConfig', prefs: prefs, isOpener: window.opener !== null };\n\n if (window.opener) {\n window.opener.postMessage(\n JSON.stringify(evt),\n '*'\n );\n } else {\n window.parent.postMessage(\n JSON.stringify(evt),\n '*'\n );\n }\n }\n}\n","import { Injectable, enableProdMode, isDevMode } from '@angular/core';\nimport { HttpClient } from \"@angular/common/http\";\nimport { ActivatedRoute, Router } from \"@angular/router\";\nimport * as yaml from \"js-yaml\";\nimport * as WebFont from 'webfontloader';\n\nconst isLocal: boolean = /localhost/.test(document.location.host);\n!isLocal && enableProdMode();\n\n\n/** @ignore */\n@Injectable({\n providedIn: 'root'\n})\nexport class AppInitService {\n\n constructor(\n public http: HttpClient,\n private _route: ActivatedRoute,\n private _router: Router) {\n }\n\n init(): Promise<any> {\n\n return new Promise<void>((resolve) => {\n\n this.loadFonts();\n\n if (isDevMode()) {\n console.log(\n '%cRunning in development mode',\n 'background-color:blue; color:yellow;'\n );\n\n /**\n * Shim the shindig prefs functionality for dev mode\n */\n (<any>window).gadgets = {\n\n Prefs: class {\n getString(key: string) { return this.getParameterByName(key) }\n\n getArray(key: string) { return this.getParameterByName(key).split(',') }\n\n getBool(key: string) { return this.getParameterByName(key) === 'true' }\n\n getCountry() { }\n\n getFloat(key: string) { return parseFloat(this.getParameterByName(key)) }\n\n getInt(key: string) { return parseInt(this.getParameterByName(key)) }\n\n getLang() { return this.getParameterByName('lang') === '' ? 'en' : this.getParameterByName('lang'); }\n\n getParameterByName(name: string, search = window.location.href): string {\n\n name = name.replace(/[\\[\\]]/g, '\\\\$&');\n const regex = new RegExp('[?&]' + name + '(=([^&#]*)|&|#|$)'),\n results = regex.exec(search);\n if (!results) return '';\n if (!results[2]) return '';\n return decodeURIComponent(results[2].replace(/\\+/g, ' '));\n }\n }\n };\n\n this.http.get('assets/gadget.yaml', {\n responseType: 'text'\n }).subscribe({\n next: (data) => {\n const doc: any = yaml.load(data);\n const params: any = {}\n for (const val of doc.prefs) {\n params[val.name] = val.default_value\n }\n\n this._router.navigate([], {\n relativeTo: this._route,\n queryParams: params,\n });\n\n console.log(\n `%cUser prefs loaded successfully`,\n 'background-color:blue; color:yellow;'\n );\n },\n error: (err) => {\n console.log(\n `%cUnable to load user preferences YAML definition file: ${err}`,\n 'background-color:blue; color:yellow;'\n );\n console.log(\n `%cPlease see our developer documentation for help with your app configuration: https://developer.reveldigital.com`,\n 'background-color:red; color:yellow;'\n )\n }\n })\n }\n resolve();\n });\n }\n\n\n private getFamilyName(css) {\n\n const FONT_FAMILY_REGEX = /font-family:\\s*(?:[&#39;&#34;])*['\"]*(.+?)['\"]*(?:[&#39;&#34;])*\\s*;/i;\n if (FONT_FAMILY_REGEX.test(css)) {\n const matches = css.match(FONT_FAMILY_REGEX);\n return matches[1].split(',')[0];\n } else {\n return '';\n }\n }\n\n /**\n * Loads the given font from Google Web Fonts.\n */\n private loadFonts(): void {\n\n const parameters = new URLSearchParams(window.location.search);\n parameters.forEach((val, key) => {\n try {\n const fontFamily = this.getFamilyName(val);\n if (fontFamily !== '') {\n WebFont.load({\n google: {\n families: [fontFamily]\n },\n fontactive: (familyName) => {\n console.log(`%cActivating font: ${familyName}`,\n 'background-color:blue; color:yellow;');\n },\n fontinactive: (familyName) => {\n console.log(`%cFont inactive: ${familyName}`,\n 'background-color:red; color:yellow;');\n }\n });\n }\n } catch (e) {\n }\n });\n }\n}\n","import { Pipe, PipeTransform, NgModule } from '@angular/core';\nimport { DomSanitizer } from '@angular/platform-browser';\n\n/**\n * The safe style pipe is used when custom styles are defined for a gadget and must be applied to an Angular\n * component. This pipe will ensure the style can be appied safely by utilizing the DomSanitizer.\n * \n * ```html\n * <h2 [style]=\"style | safeStyle\">Sample Pref: {{ prefs.getString('myStringPref') }}</h2>\n * ```\n */\n@Pipe({\n name: 'safeStyle',\n standalone: true,\n})\nexport class SafeStylePipe implements PipeTransform {\n constructor(private sanitized: DomSanitizer) { }\n\n transform(value: any): unknown {\n return this.sanitized.bypassSecurityTrustStyle(value);\n }\n}\n\n@NgModule({\n imports: [SafeStylePipe],\n exports: [SafeStylePipe],\n})\nexport class NgSafeStylePipeModule { }\n","import { APP_INITIALIZER, LOCALE_ID, NgModule } from '@angular/core';\nimport { PlayerClientService } from './player-client.service';\nimport { AppInitService } from './app-init.service';\nimport { HttpClientModule } from '@angular/common/http';\nimport { RouterModule } from '@angular/router';\nimport { APP_BASE_HREF } from '@angular/common';\nimport { NgSafeStylePipeModule } from './safe-style.pipe';\n\ndeclare const gadgets: any;\n\n\n@NgModule({\n imports: [\n HttpClientModule,\n RouterModule.forRoot([]),\n NgSafeStylePipeModule\n ],\n exports: [\n NgSafeStylePipeModule\n ],\n providers: [{\n provide: APP_INITIALIZER,\n useFactory: initializeApp,\n deps: [AppInitService, PlayerClientService],\n multi: true\n },\n {\n provide: LOCALE_ID,\n useFactory: () => {\n try {\n return new gadgets.Prefs().getLang();\n } catch {\n return 'en';\n }\n }\n },\n { provide: APP_BASE_HREF, useValue: '/gadgets/ifr' }]\n})\nexport class PlayerClientModule { }\n\nfunction initializeApp(appInitService: AppInitService) {\n return async () => {\n PlayerClientService.init({});\n await appInitService.init();\n }\n}\n","/*\n * Public API Surface of player-client\n */\n\nexport * from './lib/player-client.service';\nexport * from './lib/player-client.module';\nexport * from './lib/safe-style.pipe';\nexport * from './lib/datatable-ref';\nexport * from './lib/interfaces/datatable.interface';","/**\n * Generated bundle index. Do not edit.\n */\n\nexport * from './public-api';\n"],"names":["i1"],"mappings":"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"}
1
+ {"version":3,"file":"reveldigital-player-client.mjs","sources":["../../../../projects/reveldigital/player-client/src/lib/datatable-ref.ts","../../../../projects/reveldigital/player-client/src/lib/version.ts","../../../../projects/reveldigital/player-client/src/lib/player-client.service.ts","../../../../projects/reveldigital/player-client/src/lib/app-init.service.ts","../../../../projects/reveldigital/player-client/src/lib/safe-style.pipe.ts","../../../../projects/reveldigital/player-client/src/lib/player-client.module.ts","../../../../projects/reveldigital/player-client/src/public-api.ts","../../../../projects/reveldigital/player-client/src/reveldigital-player-client.ts"],"sourcesContent":["import { NgZone } from '@angular/core';\nimport { Subject } from 'rxjs';\nimport {\n IDataTableChangeEvent,\n IDataTableColumn,\n IDataTableOptions,\n IDataTablePref,\n IDataTableQueryParams,\n IDataTableResult,\n IDataTableSchema\n} from './interfaces/datatable.interface';\n\n/**\n * Angular-friendly wrapper around the global `gadgets.reveldigital.datatable` library.\n *\n * Provides typed Promise-based methods and RxJS Observables for real-time events.\n * All event emissions run inside Angular's zone to ensure change detection triggers.\n *\n * ```typescript\n * const dt = this.client.createDataTable('tbl_menu_items');\n *\n * // Fetch rows\n * const result = await dt.getRows({ sort: 'price', sortDir: 'asc' });\n *\n * // Real-time updates\n * dt.rowUpdated$.subscribe(change => console.log('Updated:', change));\n *\n * // Cleanup\n * dt.dispose();\n * ```\n */\nexport class DataTableRef {\n\n /** Emits when an existing row is modified. */\n public rowUpdated$ = new Subject<IDataTableChangeEvent>();\n\n /** Emits when a new row is added. */\n public rowCreated$ = new Subject<IDataTableChangeEvent>();\n\n /** Emits when a row is removed. */\n public rowDeleted$ = new Subject<IDataTableChangeEvent>();\n\n /** @ignore */\n private _instance: any;\n\n /** @ignore */\n private _zone: NgZone;\n\n /** @ignore */\n private _onRowUpdated!: (change: IDataTableChangeEvent) => void;\n /** @ignore */\n private _onRowCreated!: (change: IDataTableChangeEvent) => void;\n /** @ignore */\n private _onRowDeleted!: (change: IDataTableChangeEvent) => void;\n\n /**\n * Creates a new DataTableRef.\n *\n * @param tableId - The data table ID (e.g. 'tbl_menu_items')\n * @param zone - Angular NgZone for ensuring change detection on callbacks\n * @param options - Optional configuration overrides\n * @throws Error if the global datatable library is not loaded\n */\n constructor(tableId: string, zone: NgZone, options?: IDataTableOptions) {\n\n this._zone = zone;\n\n const lib = (window as any).gadgets?.['reveldigital.datatable'];\n\n if (!lib || typeof lib.create !== 'function') {\n throw new Error(\n 'RevelDigital DataTable library is not available. ' +\n 'Ensure the datatable feature is enabled for this gadget.'\n );\n }\n\n this._instance = lib.create(tableId, options);\n this._wireEvents();\n }\n\n /**\n * Fetches rows from the data table.\n *\n * @param params - Optional query parameters (filter, sort, pagination)\n * @returns Promise resolving to the result set\n *\n * ```typescript\n * const result = await dt.getRows({\n * filter: { category: 'Entree', price: { op: 'lte', value: 25 } },\n * sort: 'itemName',\n * sortDir: 'asc',\n * pageSize: 20\n * });\n * ```\n */\n public getRows(params?: IDataTableQueryParams): Promise<IDataTableResult> {\n return this._instance.getRows(params);\n }\n\n /**\n * Fetches the table schema (column definitions and metadata).\n *\n * @returns Promise resolving to the table schema\n */\n public getSchema(): Promise<IDataTableSchema> {\n return this._instance.getSchema();\n }\n\n /**\n * Gets visible (non-hidden) columns from the table schema.\n *\n * @returns Promise resolving to an array of visible column definitions\n */\n public getVisibleColumns(): Promise<IDataTableColumn[]> {\n return this._instance.getVisibleColumns();\n }\n\n /**\n * Fetches rows with hidden column data stripped.\n *\n * @param params - Optional query parameters (same as getRows)\n * @returns Promise resolving to the result set with hidden fields removed\n */\n public getVisibleRows(params?: IDataTableQueryParams): Promise<IDataTableResult> {\n return this._instance.getVisibleRows(params);\n }\n\n /**\n * Starts polling for changes at the given interval.\n * Emits on `rowUpdated$` when new data is detected.\n *\n * @param intervalMs - Polling interval in milliseconds (default 30000)\n */\n public startPolling(intervalMs?: number): void {\n this._instance.startPolling(intervalMs);\n }\n\n /**\n * Stops polling for changes.\n */\n public stopPolling(): void {\n this._instance.stopPolling();\n }\n\n /**\n * Releases all resources: stops polling, closes the real-time connection,\n * removes event listeners, and completes all RxJS observables.\n */\n public dispose(): void {\n this._instance.off('rowUpdated', this._onRowUpdated);\n this._instance.off('rowCreated', this._onRowCreated);\n this._instance.off('rowDeleted', this._onRowDeleted);\n\n this._instance.dispose();\n\n this.rowUpdated$.complete();\n this.rowCreated$.complete();\n this.rowDeleted$.complete();\n }\n\n /** @ignore */\n static _fromInstance(instance: any, zone: NgZone): DataTableRef {\n const ref = Object.create(DataTableRef.prototype) as DataTableRef;\n ref.rowUpdated$ = new Subject<IDataTableChangeEvent>();\n ref.rowCreated$ = new Subject<IDataTableChangeEvent>();\n ref.rowDeleted$ = new Subject<IDataTableChangeEvent>();\n ref._instance = instance;\n ref._zone = zone;\n ref._wireEvents();\n return ref;\n }\n\n /** @ignore */\n private _wireEvents(): void {\n this._onRowUpdated = (change: IDataTableChangeEvent) => {\n this._zone.run(() => this.rowUpdated$.next(change));\n };\n this._onRowCreated = (change: IDataTableChangeEvent) => {\n this._zone.run(() => this.rowCreated$.next(change));\n };\n this._onRowDeleted = (change: IDataTableChangeEvent) => {\n this._zone.run(() => this.rowDeleted$.next(change));\n };\n\n this._instance.on('rowUpdated', this._onRowUpdated);\n this._instance.on('rowCreated', this._onRowCreated);\n this._instance.on('rowDeleted', this._onRowDeleted);\n }\n}\n\n\n/**\n * Wrapper around a data table created from a gadget preference value.\n *\n * Automatically configures filter and sort settings from the preference,\n * and provides a `getFilteredRows()` convenience method that applies them.\n *\n * ```typescript\n * const cfg = this.client.createDataTableFromPref(prefs.getString('rdDataTable'));\n *\n * // Fetch rows with auto-wired filter + sort from the preference\n * const result = await cfg.getFilteredRows();\n *\n * // Access the underlying DataTableRef for schema, events, etc.\n * cfg.dataTable.rowUpdated$.subscribe(change => console.log(change));\n *\n * // Cleanup\n * cfg.dispose();\n * ```\n */\nexport class DataTablePrefRef {\n\n /** The underlying DataTableRef with full access to schema, events, polling, etc. */\n public readonly dataTable: DataTableRef;\n\n /** The parsed preference object. */\n public readonly pref: IDataTablePref;\n\n /** @ignore */\n private _config: any;\n\n /**\n * Creates a new DataTablePrefRef from a gadget preference JSON string.\n *\n * @param prefValue - The raw gadget preference string (JSON)\n * @param zone - Angular NgZone for ensuring change detection on callbacks\n * @param options - Optional configuration overrides\n * @throws Error if the global datatable library is not loaded\n */\n constructor(prefValue: string, zone: NgZone, options?: IDataTableOptions) {\n\n const lib = (window as any).gadgets?.['reveldigital.datatable'];\n\n if (!lib || typeof lib.createFromPref !== 'function') {\n throw new Error(\n 'RevelDigital DataTable library is not available. ' +\n 'Ensure the datatable feature is enabled for this gadget.'\n );\n }\n\n this._config = lib.createFromPref(prefValue, options);\n this.pref = this._config.pref;\n this.dataTable = DataTableRef._fromInstance(this._config.dt, zone);\n }\n\n /**\n * Fetches rows with the filter and sort settings from the preference automatically applied.\n * Additional query parameters can override or supplement the preference settings.\n *\n * @param params - Optional additional query parameters\n * @returns Promise resolving to the result set\n *\n * ```typescript\n * // Use preference defaults\n * const result = await cfg.getFilteredRows();\n *\n * // Override page size\n * const page = await cfg.getFilteredRows({ pageSize: 10 });\n * ```\n */\n public getFilteredRows(params?: IDataTableQueryParams): Promise<IDataTableResult> {\n return this._config.getFilteredRows(params);\n }\n\n /**\n * Releases all resources held by the underlying DataTableRef.\n */\n public dispose(): void {\n this.dataTable.dispose();\n }\n}\n","// Generated by genversion.\nexport const version = '2.3.0'\n","import { Injectable, NgZone, OnDestroy } from '@angular/core';\nimport { gadgets } from '@reveldigital/gadget-types';\nimport { BehaviorSubject, fromEvent, Subject, Subscription } from 'rxjs';\nimport { filter, map, share, tap } from 'rxjs/operators';\nimport { IClient } from './interfaces/client.interface';\nimport { ICommand } from './interfaces/command.interface';\nimport { IDictionary } from './interfaces/config.interface';\nimport { IDataTableOptions } from './interfaces/datatable.interface';\nimport { IDevice } from './interfaces/device.interface';\nimport { IEventProperties } from './interfaces/event-properties.interface';\nimport { DataTableRef, DataTablePrefRef } from './datatable-ref';\nimport { version } from './version';\n\n//import { version } from './version.js';\n\n// So that TypeScript doesn't complain, we're going to augment the GLOBAL / WINDOW \n// name-space definition to include the Tracker API. This also provides us with a place\n// to actually DOCUMENT the API so that our developers aren't guessing about what's\n// available on the library.\n\n/** @ignore */\ndeclare global {\n let Client: IClient;\n}\n\n/**\n * Service for interacting with the Revel Digital player client.\n * \n * This service provides a comprehensive interface for gadgets and web applications\n * to communicate with the Revel Digital player environment. It handles device\n * information, commands, events, analytics tracking, and configuration management.\n * \n * The service supports both direct API calls and event-based communication patterns,\n * allowing gadgets to respond to player lifecycle events (start, stop, commands) and\n * send data back to the player or remote devices.\n * \n * ```typescript\n * constructor(private client: PlayerClientService) {\n * // Subscribe to player events\n * this.client.onStart$.subscribe(() => {\n * console.log('Gadget started');\n * });\n * \n * this.client.onCommand$.subscribe(command => {\n * console.log('Received command:', command);\n * });\n * }\n * \n * async ngOnInit() {\n * // Get device information\n * const device = await this.client.getDevice();\n * const deviceTime = await this.client.getDeviceTime();\n * \n * // Track analytics\n * this.client.track('gadget_loaded', { version: '1.0' });\n * }\n * ```\n * \n * @since 1.0.0\n */\n@Injectable({\n providedIn: 'root'\n})\nexport class PlayerClientService implements OnDestroy {\n\n /** @ignore */\n private zone: NgZone;\n /** @ignore */\n private clientPromise: Promise<IClient> | null;\n\n /**\n * Observable stream of commands sent to this player from the Revel Digital platform.\n * Subscribe to this to handle custom commands sent from templates, playlists, or remote devices.\n * \n * ```typescript\n * this.client.onCommand$.subscribe(command => {\n * if (command.name === 'customAction') {\n * this.handleCustomAction(command.arg);\n * }\n * });\n * ```\n */\n public onCommand$ = new Subject<ICommand>();\n\n /**\n * Observable that signals when the gadget has been loaded and is ready to start.\n * Emits `true` when ready, `false` when destroyed.\n * \n * ```typescript\n * this.client.onReady$.subscribe(isReady => {\n * if (isReady) {\n * console.log('Client is ready');\n * this.initializeGadget();\n * }\n * });\n * ```\n */\n public onReady$ = new BehaviorSubject(false);\n\n /**\n * Observable that signals when the gadget has been started by the player.\n * This event occurs when the player begins execution of the gadget content.\n * \n * ```typescript\n * this.client.onStart$.subscribe(() => {\n * console.log('Gadget started');\n * this.startAnimation();\n * });\n * ```\n */\n public onStart$ = new Subject();\n\n /**\n * Observable that signals when the gadget has been stopped by the player.\n * This event occurs when the player stops execution, typically when moving\n * to the next item in a playlist or when the content duration expires.\n * \n * ```typescript\n * this.client.onStop$.subscribe(() => {\n * console.log('Gadget stopped');\n * this.cleanup();\n * });\n * ```\n */\n public onStop$ = new Subject();\n\n /**\n * Observable that signals when the gadget should open the configuration window.\n * This allows gadgets to respond to configuration requests from the player.\n * \n * ```typescript\n * this.client.onConfig$.subscribe(() => {\n * this.openConfigurationDialog();\n * });\n * ```\n */\n public onConfig$ = new Subject();\n\n /**\n * Observable that signals when the gadget has received a postMessage event from the player.\n * This handles communication between the gadget and player via the postMessage API.\n * \n * ```typescript\n * this.client.onPostMessage$.subscribe(message => {\n * console.log('Received message:', message);\n * this.handlePlayerMessage(message);\n * });\n * ```\n */\n public onPostMessage$ = new Subject();\n\n //\n // Two methods available for calling into the library:\n //\n // 1) Using dispatchEvent() with the following custom events\n // 2) Using the window scoped RevelDigital object as defined in the constructor\n //\n /** @ignore */\n private onStartSub: Subscription;\n /** @ignore */\n private onStartEvt$ = fromEvent(window, 'RevelDigital.Start').pipe(\n share(),\n tap(this.onStart$)\n );\n /** @ignore */\n private onStopSub: Subscription;\n /** @ignore */\n private onStopEvt$ = fromEvent(window, 'RevelDigital.Stop').pipe(\n share(),\n tap(this.onStop$)\n );\n /** @ignore */\n private onCommandSub: Subscription;\n /** @ignore */\n private onCommandEvt$ = fromEvent<ICommand>(window, 'RevelDigital.Command').pipe(\n map((e: any) => { return { name: e.detail.name, arg: e.detail.arg } as ICommand }),\n share(),\n tap(this.onCommand$)\n );\n /** @ignore */\n // private onConfigSub: Subscription;\n // /** @ignore */\n // private onConfigEvt$ = fromEvent(window, 'RevelDigital.Config').pipe(\n // share(),\n // tap((e: CustomEvent) => {\n // console.log(e);\n\n // if (e.detail.type === 'applyConfig' && e.detail.isOpener) {\n // this.applyConfig(e.detail.config); // propagate config to iframe parent from the popup window\n // } else {\n // this.onConfig$.next(e.detail);\n // }\n // })\n // );\n // private onPostMessageSub: Subscription;\n // private onPostMessageEvt$ = fromEvent(window, 'message').pipe(\n // filter((messageEvent: MessageEvent) =>\n // messageEvent.source !== window.parent &&\n // typeof messageEvent.data === 'string' &&\n // messageEvent.data.startsWith('reveldigital:')),\n // map((e: any) => { return JSON.parse(e.substring(13)) as Command }),\n // share(),\n // tap(this.onCommand$)\n // );\n\n private onPostMessageSub: Subscription;\n private onPostMessageEvt$ = fromEvent(window, 'message').pipe(\n filter((messageEvent: MessageEvent) =>\n //messageEvent.source !== window.parent &&\n typeof messageEvent.data === 'string'),\n map((e: any) => JSON.parse(e.data)),\n share(),\n tap((e: any) => {\n if (e.type === 'applyConfig' && e.isOpener) {\n e.isOpener = false;\n // propagate config to iframe parent from the popup window\n window.parent.postMessage(\n JSON.stringify(e),\n '*'\n );\n } else if (e.type === 'openConfig') {\n this.onConfig$.next(null);\n }\n }),\n tap(e => this.onPostMessage$.next(e))\n );\n\n\n /** @ignore */\n constructor(zone: NgZone) {\n\n this.zone = zone;\n\n (window as any).RevelDigital = {\n Controller: {\n onCommand: (name: string, arg: string) => {\n zone.run(() => {\n this.onCommand$.next({ name: name, arg: arg });\n });\n },\n onStart: () => {\n zone.run(() => {\n this.onStart$.next(null);\n });\n },\n onStop: () => {\n zone.run(() => {\n this.onStop$.next(null);\n });\n }\n }\n }\n\n this.onStartSub = this.onStartEvt$.subscribe(() => { });\n this.onStopSub = this.onStopEvt$.subscribe(() => { });\n this.onCommandSub = this.onCommandEvt$.subscribe(() => { });\n //this.onConfigSub = this.onConfigEvt$.subscribe(() => { });\n this.onPostMessageSub = this.onPostMessageEvt$.subscribe(() => { });\n\n this.clientPromise = null;\n\n this.onReady$.next(true);\n }\n\n /** @ignore */\n ngOnDestroy(): void {\n\n this.onStartSub?.unsubscribe();\n this.onStopSub?.unsubscribe();\n this.onCommandSub?.unsubscribe();\n //this.onConfigSub?.unsubscribe();\n this.onPostMessageSub?.unsubscribe();\n\n this.onReady$.next(false);\n }\n\n /** @ignore */\n public static init(data: any) {\n\n console.log(\n '%c⚙️ Initializing Revel Digital client library',\n 'background-color:blue; color:yellow;'\n );\n }\n\n /**\n * Sends a callback to player scripting with variable arguments.\n * \n * This method allows the gadget to communicate with player scripting.\n * If the appropriate scripting is in place in the currently running template, \n * calling this method will initiate a callback which can be acted upon in player script.\n * \n * @param args - Variable number of arguments to pass to the callback (max 5 arguments)\n * \n * ```typescript\n * // Send a simple callback\n * this.client.callback('test', 'data');\n * \n * // Send multiple parameters\n * this.client.callback('action', 'value1', 'value2', { data: 'object' });\n * ```\n */\n public callback(...args: any[]): void {\n\n this.getClient().then((client) => {\n\n switch (args.length) {\n case 0:\n client.callback();\n break;\n case 1:\n client.callback(args[0]);\n break;\n case 2:\n client.callback(args[1]);\n break;\n case 3:\n client.callback(args[2]);\n break;\n case 4:\n client.callback(args[3]);\n break;\n case 5:\n client.callback(args[4]);\n break;\n }\n })\n }\n\n /**\n * Gets the user preferences interface exposed by the Google Gadgets API.\n * \n * This method provides access to gadget preferences which can be configured\n * in the Revel Digital CMS. Use this to retrieve user-configurable settings\n * for your gadget.\n * \n * @returns The Gadgets API Prefs object for accessing preference values\n * \n * ```typescript\n * constructor(public client: PlayerClientService) {\n * const prefs = client.getPrefs();\n * const myString = prefs.getString('myStringPref');\n * const myNumber = prefs.getInt('myNumberPref');\n * const myBool = prefs.getBool('myBoolPref');\n * }\n * ```\n * \n * @see {@link https://developers.google.com/gadgets/docs/basic} Google Gadgets API documentation\n */\n public getPrefs(): gadgets.Prefs {\n\n return new window['gadgets']['Prefs']();\n }\n\n /**\n * Gets the current device time in ISO8601 format.\n * \n * Current device time is determined by the device timezone assigned to the device in the CMS.\n * This is useful for displaying time-sensitive content or scheduling operations based on\n * the device's local time rather than browser time.\n * \n * @param date - Optional. If supplied, will translate the supplied date/time to device time based on respective timezones\n * @returns Promise resolving to date/time in ISO8601 format\n * \n * ```typescript\n * // Get current device time\n * const currentTime = await this.client.getDeviceTime();\n * console.log('Device time:', currentTime);\n * \n * // Convert a specific date to device time\n * const specificDate = new Date('2023-01-01T12:00:00Z');\n * const deviceTime = await this.client.getDeviceTime(specificDate);\n * ```\n */\n public async getDeviceTime(date?: Date): Promise<string> {\n\n const client = await this.getClient();\n\n if (date !== undefined) {\n return client.getDeviceTime(date);\n }\n return client.getDeviceTime();\n }\n\n /**\n * Gets the timezone name currently assigned to the device.\n * \n * @returns Promise resolving to the timezone name (e.g., \"America/New_York\")\n * \n * ```typescript\n * const timezoneName = await this.client.getDeviceTimeZoneName();\n * console.log('Device timezone:', timezoneName);\n * ```\n */\n public async getDeviceTimeZoneName(): Promise<string> {\n\n const client = await this.getClient();\n\n return client.getDeviceTimeZoneName();\n }\n\n /**\n * Gets the timezone ID currently assigned to the device.\n * \n * @returns Promise resolving to the timezone ID\n * \n * ```typescript\n * const timezoneId = await this.client.getDeviceTimeZoneID();\n * console.log('Device timezone ID:', timezoneId);\n * ```\n */\n public async getDeviceTimeZoneID(): Promise<string> {\n\n const client = await this.getClient();\n\n return client.getDeviceTimeZoneID();\n }\n\n /**\n * Gets the numerical offset from GMT of the timezone currently assigned to the device.\n * \n * @returns Promise resolving to the timezone offset in hours (e.g., -5 for EST)\n * \n * ```typescript\n * const offset = await this.client.getDeviceTimeZoneOffset();\n * console.log('Device timezone offset:', offset, 'hours from GMT');\n * ```\n */\n public async getDeviceTimeZoneOffset(): Promise<number> {\n\n const client = await this.getClient();\n\n return client.getDeviceTimeZoneOffset();\n }\n\n /**\n * Gets the language code of the language currently assigned to the device.\n * \n * @returns Promise resolving to the language code (e.g., \"en-US\", \"fr-FR\")\n * \n * ```typescript\n * const languageCode = await this.client.getLanguageCode();\n * console.log('Device language:', languageCode);\n * // Use for localization\n * this.loadLanguageResources(languageCode);\n * ```\n */\n public async getLanguageCode(): Promise<string> {\n\n const client = await this.getClient();\n\n return client.getLanguageCode();\n }\n\n /**\n * Gets the unique Revel Digital device key associated with the device.\n * \n * The device key is a unique identifier for this specific player device\n * and can be used for device-specific operations or remote commands.\n * \n * @returns Promise resolving to the device key string\n * \n * ```typescript\n * const deviceKey = await this.client.getDeviceKey();\n * console.log('Device key:', deviceKey);\n * ```\n */\n public async getDeviceKey(): Promise<string> {\n\n const client = await this.getClient();\n\n return client.getDeviceKey();\n }\n\n /**\n * Sends a command to the local player device.\n * \n * Commands can be used to control various aspects of the player or trigger\n * specific behaviors. The command is processed by the local player only.\n * \n * @param name - The command name/identifier\n * @param arg - The command argument/payload\n * \n * ```typescript\n * // Send a simple command\n * this.client.sendCommand('refresh', '');\n * \n * // Send a command with data\n * this.client.sendCommand('setVolume', '50');\n * \n * // Send a command with JSON data\n * this.client.sendCommand('configure', JSON.stringify({ setting: 'value' }));\n * ```\n */\n public sendCommand(name: string, arg: string): void {\n\n this.getClient().then((client) => {\n client.sendCommand(name, arg);\n })\n }\n\n /**\n * Sends a command to remote player devices with the specified device keys.\n * \n * Remote commands allow cross-device communication within the same Revel Digital account.\n * This is useful for synchronized displays, device coordination, or remote control scenarios.\n * \n * Note: Remote commands can only be delivered to devices within the same account as the sender device.\n * \n * @param deviceKeys - Array of target device keys to send the command to\n * @param name - The command name/identifier\n * @param arg - The command argument/payload\n * \n * ```typescript\n * // Send command to specific devices\n * const targetDevices = ['device-key-1', 'device-key-2'];\n * this.client.sendRemoteCommand(targetDevices, 'syncAction', 'start');\n * \n * // Broadcast to multiple devices\n * this.client.sendRemoteCommand(\n * ['lobby-display', 'kiosk-1', 'kiosk-2'], \n * 'updateContent', \n * JSON.stringify({ contentId: '12345' })\n * );\n * ```\n */\n public sendRemoteCommand(deviceKeys: string[], name: string, arg: string): void {\n\n this.getClient().then((client) => {\n client.sendRemoteCommand(deviceKeys, name, arg);\n });\n }\n\n /**\n * Logs an analytics event for use with AdHawk analytics and reporting.\n * \n * Events are used for tracking various metrics including usage statistics, \n * player condition, state changes, user interactions, and custom business metrics.\n * These events can be viewed in the Revel Digital analytics dashboard.\n * \n * @param eventName - Unique name for this event (should be descriptive and consistent)\n * @param properties - Optional map of user-defined properties to associate with this event\n * \n * ```typescript\n * // Simple event tracking\n * this.client.track('gadget_loaded');\n * \n * // Event with properties\n * this.client.track('user_interaction', {\n * action: 'button_click',\n * button_id: 'start_button',\n * timestamp: new Date().toISOString()\n * });\n * \n * // Performance tracking\n * this.client.track('content_displayed', {\n * content_type: 'video',\n * duration: 30,\n * quality: 'HD'\n * });\n * ```\n */\n public track(eventName: string, properties?: IEventProperties): void {\n\n this.getClient().then((client) => {\n client.track(eventName, JSON.stringify(properties));\n })\n }\n\n /**\n * Initiates a timed event for duration tracking.\n * \n * Timed events are useful for tracking the duration of operations or user interactions.\n * This method must be followed by a call to track() with the same event name to complete\n * the timing measurement. The duration will be automatically calculated and included\n * in the event properties.\n * \n * @param eventName - Unique name for this timed event (must match the subsequent track() call)\n * \n * ```typescript\n * // Start timing an event\n * this.client.timeEvent('video_playback');\n * \n * // ... video plays for some duration ...\n * \n * // End timing and log the event with duration\n * this.client.track('video_playback', {\n * video_id: 'abc123',\n * quality: 'HD'\n * }); // Duration will be automatically added\n * \n * // Example for user interaction timing\n * this.client.timeEvent('form_completion');\n * // ... user fills out form ...\n * this.client.track('form_completion', { form_type: 'contact' });\n * ```\n */\n public timeEvent(eventName: string): void {\n\n this.getClient().then((client) => {\n client.timeEvent(eventName);\n })\n }\n\n /**\n * Creates a new analytics event session for grouping related events.\n * \n * A session is a way of grouping events together for analysis. Each event tracked\n * after calling this method will have the same session ID until a new session is created.\n * Session IDs are randomly generated unless explicitly provided.\n * \n * This is useful for tracking user journeys, workflow completion, or grouping\n * related interactions within a specific time period.\n * \n * @param id - Optional user-supplied session ID. If not provided, a random session ID will be generated\n * \n * ```typescript\n * // Start a new session with auto-generated ID\n * this.client.newEventSession();\n * this.client.track('session_start');\n * this.client.track('user_action_1');\n * this.client.track('user_action_2');\n * \n * // Start a session with custom ID\n * this.client.newEventSession('user-workflow-12345');\n * this.client.track('workflow_start');\n * this.client.track('step_completed', { step: 1 });\n * \n * // Start a new session for different workflow\n * this.client.newEventSession();\n * this.client.track('different_workflow_start');\n * ```\n */\n public newEventSession(id?: string): void {\n\n this.getClient().then((client) => {\n if (id !== undefined) {\n client.newEventSession();\n } else {\n client.newEventSession(id);\n }\n })\n }\n\n /**\n * Gets the root folder path utilized by this player device.\n * \n * This returns the base directory path where the Revel Digital player\n * stores its files and resources on the local device.\n * \n * @returns Promise resolving to the path of the root folder\n * \n * ```typescript\n * const rootPath = await this.client.getRevelRoot();\n * console.log('Player root directory:', rootPath);\n * // Use for constructing file paths or understanding storage structure\n * ```\n */\n public async getRevelRoot(): Promise<string> {\n\n const client = await this.getClient();\n\n return client.getRevelRoot();\n }\n\n /**\n * Gets a map of commands currently active for this device.\n * \n * This returns the current command configuration that defines how the device\n * responds to various command triggers and remote commands.\n * \n * @returns Promise resolving to a map of currently active commands\n * \n * ```typescript\n * const commandMap = await this.client.getCommandMap();\n * console.log('Active commands:', commandMap);\n * \n * // Check if specific command is available\n * if (commandMap['customCommand']) {\n * console.log('Custom command is available');\n * }\n * ```\n */\n public async getCommandMap(): Promise<any> {\n\n const client = await this.getClient();\n\n return JSON.parse(await client.getCommandMap());\n }\n\n /**\n * Signals to the player that this gadget has completed its visualization.\n * \n * This method notifies the player that the current gadget has finished its\n * content display or interaction cycle. The player can then proceed with\n * the next item in a playlist if applicable, or handle the completion\n * according to its configuration.\n * \n * Call this method when your gadget has completed its intended function,\n * such as finishing an animation, completing a form, or reaching a natural\n * stopping point.\n * \n * ```typescript\n * // After completing an animation\n * private onAnimationComplete(): void {\n * this.client.finish();\n * }\n * \n * // After user interaction is complete\n * private onFormSubmitted(): void {\n * this.saveFormData();\n * this.client.finish();\n * }\n * \n * // After a timed display period\n * setTimeout(() => {\n * this.client.finish();\n * }, 30000); // 30 seconds\n * ```\n */\n public finish(): void {\n\n this.getClient().then((client) => {\n\n client.finish();\n })\n }\n\n /**\n * Checks if the gadget is running in preview mode.\n * \n * Preview mode is enabled when the gadget is being edited in the Revel Digital CMS,\n * tested in the gadget editor, or otherwise not running in a normal player environment.\n * This is useful for providing different behavior during development/testing versus\n * production deployment.\n * \n * @returns Promise resolving to true if running in preview mode, false if running on actual player\n * \n * ```typescript\n * const isPreview = await this.client.isPreviewMode();\n * \n * if (isPreview) {\n * console.log('Running in preview mode - using mock data');\n * this.loadMockData();\n * } else {\n * console.log('Running on player device - using live data');\n * this.loadLiveData();\n * }\n * \n * // Show different UI in preview\n * this.showPreviewIndicator = isPreview;\n * ```\n */\n public async isPreviewMode(): Promise<boolean> {\n\n const client = await this.getClient();\n\n return client instanceof NoopClient;\n }\n\n /**\n * Gets detailed information about the device running the player.\n * \n * Returns comprehensive device details including name, registration key, type,\n * service date, language, timezone, tags, and location information. This data\n * is configured in the Revel Digital CMS for each device.\n * \n * @returns Promise resolving to device details object, or null if not available\n * \n * ```typescript\n * const device = await this.client.getDevice();\n * \n * if (device) {\n * console.log('Device name:', device.name);\n * console.log('Device type:', device.deviceType);\n * console.log('Location:', device.location.city, device.location.state);\n * console.log('Tags:', device.tags);\n * \n * // Use device info for customization\n * if (device.location.country === 'US') {\n * this.loadUSContent();\n * }\n * \n * // Check device capabilities based on type\n * if (device.deviceType === 'android') {\n * this.enableTouchFeatures();\n * }\n * }\n * ```\n */\n public async getDevice(): Promise<IDevice | null> {\n\n const client = await this.getClient();\n\n const obj: any = JSON.parse(<string>await client.getDevice());\n\n const device: IDevice[] = [obj].map((device: any) => {\n\n return {\n name: device.name,\n registrationKey: device.key,\n deviceType: device.devicetype,\n enteredService: new Date(device.enteredservice),\n langCode: device.langcode,\n timeZone: device.timezone,\n tags: device.description?.split('\\n'),\n location: {\n city: device.location?.city,\n state: device.location?.state,\n country: device.location?.country,\n postalCode: device.location?.postalcode,\n address: device.location?.address,\n latitude: device.location?.latitude,\n longitude: device.location?.longitude\n }\n }\n });\n return device[0];\n }\n\n /**\n * Gets the width of the visualization area in pixels.\n * \n * This returns the available width for content display, which may be\n * different from the full screen width depending on player configuration\n * and template layout.\n * \n * @returns Promise resolving to width in pixels, or null if not available\n * \n * ```typescript\n * const width = await this.client.getWidth();\n * \n * if (width) {\n * console.log('Available width:', width, 'pixels');\n * \n * // Adapt content layout based on width\n * if (width < 800) {\n * this.enableMobileLayout();\n * } else {\n * this.enableDesktopLayout();\n * }\n * }\n * ```\n */\n public async getWidth(): Promise<number | null> {\n\n const client = await this.getClient();\n\n return client.getWidth();\n }\n\n /**\n * Gets the height of the visualization area in pixels.\n * \n * This returns the available height for content display, which may be\n * different from the full screen height depending on player configuration\n * and template layout.\n * \n * @returns Promise resolving to height in pixels, or null if not available\n * \n * ```typescript\n * const height = await this.client.getHeight();\n * \n * if (height) {\n * console.log('Available height:', height, 'pixels');\n * \n * // Calculate aspect ratio for responsive design\n * const width = await this.client.getWidth();\n * const aspectRatio = width / height;\n * this.adjustContentForAspectRatio(aspectRatio);\n * }\n * ```\n */\n public async getHeight(): Promise<number | null> {\n\n const client = await this.getClient();\n\n return client.getHeight();\n }\n\n /**\n * Gets the duration of the currently playing content item.\n * \n * This method is only applicable when the gadget is associated with a playlist\n * and returns the duration assigned to the current playlist item. The duration\n * determines how long the content should be displayed before moving to the next item.\n * \n * @returns Promise resolving to duration in milliseconds, or null if not applicable/available\n * \n * ```typescript\n * const duration = await this.client.getDuration();\n * \n * if (duration) {\n * console.log('Content duration:', duration, 'milliseconds');\n * \n * // Set up auto-finish timer\n * setTimeout(() => {\n * this.client.finish();\n * }, duration);\n * \n * // Show progress indicator\n * this.startProgressIndicator(duration);\n * }\n * ```\n */\n public async getDuration(): Promise<number | null> {\n\n const client = await this.getClient();\n\n return client.getDuration();\n }\n\n /**\n * Gets the current version of the Revel Digital SDK.\n * \n * @returns Promise resolving to the SDK version string\n * \n * ```typescript\n * const version = await this.client.getSdkVersion();\n * console.log('SDK Version:', version);\n * \n * // Use for compatibility checks or logging\n * this.client.track('gadget_loaded', { sdkVersion: version });\n * ```\n */\n public async getSdkVersion(): Promise<string> {\n\n return Promise.resolve(version);\n }\n\n /**\n * Applies configuration preferences to the gadget (preview mode only).\n * \n * This method is only available when running in preview mode (typically during\n * gadget development or testing in the CMS). It allows applying configuration\n * changes that would normally come from the gadget's preference settings.\n * \n * @param prefs - Dictionary of preference key-value pairs to apply\n * \n * ```typescript\n * if (await this.client.isPreviewMode()) {\n * // Apply test configuration in preview\n * await this.client.applyConfig({\n * 'title': 'Test Title',\n * 'backgroundColor': '#ff0000',\n * 'showBorder': true,\n * 'refreshInterval': 30\n * });\n * }\n * ```\n */\n public async applyConfig(prefs: IDictionary<any>) {\n\n if (await this.isPreviewMode()) {\n const client = await this.getClient();\n client.applyConfig(prefs);\n } else {\n console.log(\n '%capplyConfig() is only available in preview mode.',\n 'background-color:blue; color:yellow;'\n );\n }\n }\n\n /**\n * Creates a typed wrapper for a Revel Digital data table.\n *\n * The data table feature must be enabled for the gadget. The returned\n * {@link DataTableRef} provides typed Promise-based methods and RxJS\n * Observables for real-time row change events.\n *\n * @param tableId - The data table ID (e.g. 'tbl_menu_items')\n * @param options - Optional configuration overrides\n * @returns A {@link DataTableRef} instance\n * @throws Error if the global datatable library is not loaded\n *\n * ```typescript\n * const dt = this.client.createDataTable('tbl_menu_items');\n *\n * // Fetch rows with filtering and sorting\n * const result = await dt.getRows({\n * filter: { category: 'Entree', price: { op: 'lte', value: 25 } },\n * sort: 'price',\n * sortDir: 'asc'\n * });\n *\n * // Subscribe to real-time updates\n * dt.rowUpdated$.subscribe(change => console.log('Row updated:', change));\n * dt.rowCreated$.subscribe(change => console.log('Row created:', change));\n * dt.rowDeleted$.subscribe(change => console.log('Row deleted:', change));\n *\n * // Cleanup when done\n * dt.dispose();\n * ```\n */\n public createDataTable(tableId: string, options?: IDataTableOptions): DataTableRef {\n return new DataTableRef(tableId, this.zone, options);\n }\n\n /**\n * Creates a typed data table wrapper from a gadget preference value.\n *\n * The preference JSON string (as serialized by the template editor's datatable\n * option) is parsed and used to auto-configure filter, sort, and logic settings.\n * The returned {@link DataTablePrefRef} provides a `getFilteredRows()` convenience\n * method that applies these settings automatically.\n *\n * @param prefValue - The raw gadget preference string (JSON)\n * @param options - Optional configuration overrides\n * @returns A {@link DataTablePrefRef} instance\n * @throws Error if the global datatable library is not loaded\n *\n * ```typescript\n * const cfg = this.client.createDataTableFromPref(prefs.getString('rdDataTable'));\n *\n * // Fetch rows with auto-wired filter + sort\n * const result = await cfg.getFilteredRows();\n *\n * // Access the underlying DataTableRef for events, schema, etc.\n * cfg.dataTable.rowUpdated$.subscribe(change => console.log(change));\n *\n * // Cleanup when done\n * cfg.dispose();\n * ```\n */\n public createDataTableFromPref(prefValue: string, options?: IDataTableOptions): DataTablePrefRef {\n return new DataTablePrefRef(prefValue, this.zone, options);\n }\n\n // ---\n // PRIVATE METHODS.\n // ---\n /** @ignore */\n private getClient(): Promise<IClient> {\n\n if (this.clientPromise) {\n\n return (this.clientPromise);\n }\n\n if ((window as any).Client) {\n\n return (this.clientPromise = Promise.resolve((window as any).Client));\n }\n\n // A \"complete\" status indicates that the \"load\" event has been fired on the\n // window; and, that all sub-resources such as Scripts, Images, and Frames have\n // been loaded.\n if (window.document.readyState === \"complete\") {\n\n // If this event has fired AND the 3rd-party script isn't available (see IF-\n // condition BEFORE this one), it means that the 3rd-party script either\n // failed on the network or was BLOCKED by an ad-blocker. As such, we have to\n // fall-back to using a mock API.\n return (this.clientPromise = Promise.resolve(new NoopClient()));\n }\n\n // ASSERT: If we made it this far, the document has not completed loading (but it\n // may be in an \"interactive\" state which is when I believe that the Angular app\n // gets bootstrapped). As such, we need bind to the LOAD event to wait for our\n // third-party scripts to load (or fail to load, or be blocked).\n this.clientPromise = new Promise<IClient>(\n (resolve) => {\n\n window.addEventListener(\n \"load\",\n function handleWindowLoad() {\n\n // At this point, the 3rd-party library is either available or\n // it's not - there's no further loading to do. If it's not\n // present on the global scope, we're going to fall-back to using\n // a mock API.\n resolve((window as any).Client || new NoopClient());\n }\n );\n\n }\n );\n\n return (this.clientPromise);\n }\n}\n\n\n\n// ----------------------------------------------------------------------------------- //\n// ----------------------------------------------------------------------------------- //\n\n/**\n * Mock implementation of the IClient interface.\n * \n * This class provides a no-operation (NOOP) implementation of the client API\n * that allows consuming code to function normally even when the actual player\n * API is not available. This typically occurs during development, testing,\n * or when the player script is blocked by ad-blockers.\n * \n * All methods in this class either return null/empty values or perform no\n * operations, allowing gadgets to function without errors while providing\n * appropriate fallback behavior.\n * \n * @private\n */\nclass NoopClient implements IClient {\n\n constructor() {\n\n console.log(\n '%cClient API not available, falling back to mock API',\n 'background-color:blue; color:yellow;'\n );\n }\n\n public callback(...args: any[]): void {\n\n // NOOP implement, nothing to do....\n }\n\n public getDeviceTime(date?: Date): Promise<string> {\n\n return Promise.resolve(new Date().toISOString());\n }\n\n public async getDeviceTimeZoneName(): Promise<string> {\n\n return Promise.resolve(null);\n }\n\n public async getDeviceTimeZoneID(): Promise<string> {\n\n return Promise.resolve(null);\n }\n\n public async getDeviceTimeZoneOffset(): Promise<number> {\n\n return Promise.resolve(null);\n }\n\n public async getLanguageCode(): Promise<string> {\n\n return Promise.resolve(null);\n }\n\n public async getDeviceKey(): Promise<string> {\n\n return Promise.resolve(null);\n }\n\n public sendCommand(name: string, arg: string): void {\n\n // NOOP implement, nothing to do....\n }\n\n public sendRemoteCommand(deviceKeys: string[], name: string, arg: string) {\n\n // NOOP implement, nothing to do....\n }\n\n public track(eventName: string, properties?: string): void {\n\n // NOOP implement, nothing to do....\n }\n\n public timeEvent(eventName: string): void {\n\n // NOOP implement, nothing to do....\n }\n\n public newEventSession(id?: string): void {\n\n // NOOP implement, nothing to do....\n }\n\n public async getRevelRoot(): Promise<string> {\n\n return Promise.resolve(null);\n }\n\n public async getCommandMap(): Promise<string> {\n\n return Promise.resolve('{}');\n }\n\n public finish(): void {\n\n // NOOP implement, nothing to do....\n }\n\n public async getDevice(): Promise<string | null> {\n\n return Promise.resolve(null);\n }\n\n public async getWidth(): Promise<number | null> {\n\n return Promise.resolve(null);\n }\n\n public async getHeight(): Promise<number | null> {\n\n return Promise.resolve(null);\n }\n\n public async getDuration(): Promise<number | null> {\n\n return Promise.resolve(null);\n }\n\n public async getSdkVersion(): Promise<string> {\n\n return Promise.resolve(version);\n }\n\n public applyConfig(prefs: IDictionary<any>): void {\n\n const evt = { type: 'applyConfig', prefs: prefs, isOpener: window.opener !== null };\n\n if (window.opener) {\n window.opener.postMessage(\n JSON.stringify(evt),\n '*'\n );\n } else {\n window.parent.postMessage(\n JSON.stringify(evt),\n '*'\n );\n }\n }\n}\n","import { Injectable, enableProdMode, isDevMode } from '@angular/core';\nimport { HttpClient } from \"@angular/common/http\";\nimport { ActivatedRoute, Router } from \"@angular/router\";\nimport * as yaml from \"js-yaml\";\nimport * as WebFont from 'webfontloader';\n\nconst isLocal: boolean = /localhost/.test(document.location.host);\n!isLocal && enableProdMode();\n\n\n/** @ignore */\n@Injectable({\n providedIn: 'root'\n})\nexport class AppInitService {\n\n constructor(\n public http: HttpClient,\n private _route: ActivatedRoute,\n private _router: Router) {\n }\n\n init(): Promise<any> {\n\n return new Promise<void>((resolve) => {\n\n this.loadFonts();\n\n if (isDevMode()) {\n console.log(\n '%cRunning in development mode',\n 'background-color:blue; color:yellow;'\n );\n\n /**\n * Shim the shindig prefs functionality for dev mode\n */\n (<any>window).gadgets = {\n\n Prefs: class {\n getString(key: string) { return this.getParameterByName(key) }\n\n getArray(key: string) { return this.getParameterByName(key).split(',') }\n\n getBool(key: string) { return this.getParameterByName(key) === 'true' }\n\n getCountry() { }\n\n getFloat(key: string) { return parseFloat(this.getParameterByName(key)) }\n\n getInt(key: string) { return parseInt(this.getParameterByName(key)) }\n\n getLang() { return this.getParameterByName('lang') === '' ? 'en' : this.getParameterByName('lang'); }\n\n getParameterByName(name: string, search = window.location.href): string {\n\n name = name.replace(/[\\[\\]]/g, '\\\\$&');\n const regex = new RegExp('[?&]' + name + '(=([^&#]*)|&|#|$)'),\n results = regex.exec(search);\n if (!results) return '';\n if (!results[2]) return '';\n return decodeURIComponent(results[2].replace(/\\+/g, ' '));\n }\n }\n };\n\n this.http.get('assets/gadget.yaml', {\n responseType: 'text'\n }).subscribe({\n next: (data) => {\n const doc: any = yaml.load(data);\n const params: any = {}\n for (const val of doc.prefs) {\n params[val.name] = val.default_value\n }\n\n this._router.navigate([], {\n relativeTo: this._route,\n queryParams: params,\n });\n\n console.log(\n `%cUser prefs loaded successfully`,\n 'background-color:blue; color:yellow;'\n );\n },\n error: (err) => {\n console.log(\n `%cUnable to load user preferences YAML definition file: ${err}`,\n 'background-color:blue; color:yellow;'\n );\n console.log(\n `%cPlease see our developer documentation for help with your app configuration: https://developer.reveldigital.com`,\n 'background-color:red; color:yellow;'\n )\n }\n })\n }\n resolve();\n });\n }\n\n\n private getFamilyName(css) {\n\n const FONT_FAMILY_REGEX = /font-family:\\s*(?:[&#39;&#34;])*['\"]*(.+?)['\"]*(?:[&#39;&#34;])*\\s*;/i;\n if (FONT_FAMILY_REGEX.test(css)) {\n const matches = css.match(FONT_FAMILY_REGEX);\n return matches[1].split(',')[0];\n } else {\n return '';\n }\n }\n\n /**\n * Loads the given font from Google Web Fonts.\n */\n private loadFonts(): void {\n\n const parameters = new URLSearchParams(window.location.search);\n parameters.forEach((val, key) => {\n try {\n const fontFamily = this.getFamilyName(val);\n if (fontFamily !== '') {\n WebFont.load({\n google: {\n families: [fontFamily]\n },\n fontactive: (familyName) => {\n console.log(`%cActivating font: ${familyName}`,\n 'background-color:blue; color:yellow;');\n },\n fontinactive: (familyName) => {\n console.log(`%cFont inactive: ${familyName}`,\n 'background-color:red; color:yellow;');\n }\n });\n }\n } catch (e) {\n }\n });\n }\n}\n","import { Pipe, PipeTransform, NgModule } from '@angular/core';\nimport { DomSanitizer } from '@angular/platform-browser';\n\n/**\n * The safe style pipe is used when custom styles are defined for a gadget and must be applied to an Angular\n * component. This pipe will ensure the style can be appied safely by utilizing the DomSanitizer.\n * \n * ```html\n * <h2 [style]=\"style | safeStyle\">Sample Pref: {{ prefs.getString('myStringPref') }}</h2>\n * ```\n */\n@Pipe({\n name: 'safeStyle',\n standalone: true,\n})\nexport class SafeStylePipe implements PipeTransform {\n constructor(private sanitized: DomSanitizer) { }\n\n transform(value: any): unknown {\n return this.sanitized.bypassSecurityTrustStyle(value);\n }\n}\n\n@NgModule({\n imports: [SafeStylePipe],\n exports: [SafeStylePipe],\n})\nexport class NgSafeStylePipeModule { }\n","import { APP_INITIALIZER, LOCALE_ID, NgModule } from '@angular/core';\nimport { PlayerClientService } from './player-client.service';\nimport { AppInitService } from './app-init.service';\nimport { HttpClientModule } from '@angular/common/http';\nimport { RouterModule } from '@angular/router';\nimport { APP_BASE_HREF } from '@angular/common';\nimport { NgSafeStylePipeModule } from './safe-style.pipe';\n\ndeclare const gadgets: any;\n\n\n@NgModule({\n imports: [\n HttpClientModule,\n RouterModule.forRoot([]),\n NgSafeStylePipeModule\n ],\n exports: [\n NgSafeStylePipeModule\n ],\n providers: [{\n provide: APP_INITIALIZER,\n useFactory: initializeApp,\n deps: [AppInitService, PlayerClientService],\n multi: true\n },\n {\n provide: LOCALE_ID,\n useFactory: () => {\n try {\n return new gadgets.Prefs().getLang();\n } catch {\n return 'en';\n }\n }\n },\n { provide: APP_BASE_HREF, useValue: '/gadgets/ifr' }]\n})\nexport class PlayerClientModule { }\n\nfunction initializeApp(appInitService: AppInitService) {\n return async () => {\n PlayerClientService.init({});\n await appInitService.init();\n }\n}\n","/*\n * Public API Surface of player-client\n */\n\nexport * from './lib/player-client.service';\nexport * from './lib/player-client.module';\nexport * from './lib/safe-style.pipe';\nexport * from './lib/datatable-ref';\nexport * from './lib/interfaces/datatable.interface';","/**\n * Generated bundle index. Do not edit.\n */\n\nexport * from './public-api';\n"],"names":["i1"],"mappings":"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"}
@@ -1,9 +1,11 @@
1
+ import { NgZone } from '@angular/core';
1
2
  import { Subject } from 'rxjs';
2
3
  import { IDataTableChangeEvent, IDataTableColumn, IDataTableOptions, IDataTablePref, IDataTableQueryParams, IDataTableResult, IDataTableSchema } from './interfaces/datatable.interface';
3
4
  /**
4
5
  * Angular-friendly wrapper around the global `gadgets.reveldigital.datatable` library.
5
6
  *
6
7
  * Provides typed Promise-based methods and RxJS Observables for real-time events.
8
+ * All event emissions run inside Angular's zone to ensure change detection triggers.
7
9
  *
8
10
  * ```typescript
9
11
  * const dt = this.client.createDataTable('tbl_menu_items');
@@ -28,6 +30,8 @@ export declare class DataTableRef {
28
30
  /** @ignore */
29
31
  private _instance;
30
32
  /** @ignore */
33
+ private _zone;
34
+ /** @ignore */
31
35
  private _onRowUpdated;
32
36
  /** @ignore */
33
37
  private _onRowCreated;
@@ -37,10 +41,11 @@ export declare class DataTableRef {
37
41
  * Creates a new DataTableRef.
38
42
  *
39
43
  * @param tableId - The data table ID (e.g. 'tbl_menu_items')
44
+ * @param zone - Angular NgZone for ensuring change detection on callbacks
40
45
  * @param options - Optional configuration overrides
41
46
  * @throws Error if the global datatable library is not loaded
42
47
  */
43
- constructor(tableId: string, options?: IDataTableOptions);
48
+ constructor(tableId: string, zone: NgZone, options?: IDataTableOptions);
44
49
  /**
45
50
  * Fetches rows from the data table.
46
51
  *
@@ -93,7 +98,7 @@ export declare class DataTableRef {
93
98
  */
94
99
  dispose(): void;
95
100
  /** @ignore */
96
- static _fromInstance(instance: any): DataTableRef;
101
+ static _fromInstance(instance: any, zone: NgZone): DataTableRef;
97
102
  /** @ignore */
98
103
  private _wireEvents;
99
104
  }
@@ -127,10 +132,11 @@ export declare class DataTablePrefRef {
127
132
  * Creates a new DataTablePrefRef from a gadget preference JSON string.
128
133
  *
129
134
  * @param prefValue - The raw gadget preference string (JSON)
135
+ * @param zone - Angular NgZone for ensuring change detection on callbacks
130
136
  * @param options - Optional configuration overrides
131
137
  * @throws Error if the global datatable library is not loaded
132
138
  */
133
- constructor(prefValue: string, options?: IDataTableOptions);
139
+ constructor(prefValue: string, zone: NgZone, options?: IDataTableOptions);
134
140
  /**
135
141
  * Fetches rows with the filter and sort settings from the preference automatically applied.
136
142
  * Additional query parameters can override or supplement the preference settings.
@@ -1 +1 @@
1
- {"version":3,"file":"datatable-ref.d.ts","sourceRoot":"","sources":["../../../../projects/reveldigital/player-client/src/lib/datatable-ref.ts"],"names":[],"mappings":"AAAA,OAAO,EAAE,OAAO,EAAE,MAAM,MAAM,CAAC;AAC/B,OAAO,EACL,qBAAqB,EACrB,gBAAgB,EAChB,iBAAiB,EACjB,cAAc,EACd,qBAAqB,EACrB,gBAAgB,EAChB,gBAAgB,EACjB,MAAM,kCAAkC,CAAC;AAE1C;;;;;;;;;;;;;;;;;GAiBG;AACH,qBAAa,YAAY;IAEvB,8CAA8C;IACvC,WAAW,iCAAwC;IAE1D,qCAAqC;IAC9B,WAAW,iCAAwC;IAE1D,mCAAmC;IAC5B,WAAW,iCAAwC;IAE1D,cAAc;IACd,OAAO,CAAC,SAAS,CAAM;IAEvB,cAAc;IACd,OAAO,CAAC,aAAa,CAA2C;IAChE,cAAc;IACd,OAAO,CAAC,aAAa,CAA2C;IAChE,cAAc;IACd,OAAO,CAAC,aAAa,CAA2C;IAEhE;;;;;;OAMG;gBACS,OAAO,EAAE,MAAM,EAAE,OAAO,CAAC,EAAE,iBAAiB;IAexD;;;;;;;;;;;;;;OAcG;IACI,OAAO,CAAC,MAAM,CAAC,EAAE,qBAAqB,GAAG,OAAO,CAAC,gBAAgB,CAAC;IAIzE;;;;OAIG;IACI,SAAS,IAAI,OAAO,CAAC,gBAAgB,CAAC;IAI7C;;;;OAIG;IACI,iBAAiB,IAAI,OAAO,CAAC,gBAAgB,EAAE,CAAC;IAIvD;;;;;OAKG;IACI,cAAc,CAAC,MAAM,CAAC,EAAE,qBAAqB,GAAG,OAAO,CAAC,gBAAgB,CAAC;IAIhF;;;;;OAKG;IACI,YAAY,CAAC,UAAU,CAAC,EAAE,MAAM,GAAG,IAAI;IAI9C;;OAEG;IACI,WAAW,IAAI,IAAI;IAI1B;;;OAGG;IACI,OAAO,IAAI,IAAI;IAYtB,cAAc;IACd,MAAM,CAAC,aAAa,CAAC,QAAQ,EAAE,GAAG,GAAG,YAAY;IAUjD,cAAc;IACd,OAAO,CAAC,WAAW;CASpB;AAGD;;;;;;;;;;;;;;;;;;GAkBG;AACH,qBAAa,gBAAgB;IAE3B,oFAAoF;IACpF,SAAgB,SAAS,EAAE,YAAY,CAAC;IAExC,oCAAoC;IACpC,SAAgB,IAAI,EAAE,cAAc,CAAC;IAErC,cAAc;IACd,OAAO,CAAC,OAAO,CAAM;IAErB;;;;;;OAMG;gBACS,SAAS,EAAE,MAAM,EAAE,OAAO,CAAC,EAAE,iBAAiB;IAgB1D;;;;;;;;;;;;;;OAcG;IACI,eAAe,CAAC,MAAM,CAAC,EAAE,qBAAqB,GAAG,OAAO,CAAC,gBAAgB,CAAC;IAIjF;;OAEG;IACI,OAAO,IAAI,IAAI;CAGvB"}
1
+ {"version":3,"file":"datatable-ref.d.ts","sourceRoot":"","sources":["../../../../projects/reveldigital/player-client/src/lib/datatable-ref.ts"],"names":[],"mappings":"AAAA,OAAO,EAAE,MAAM,EAAE,MAAM,eAAe,CAAC;AACvC,OAAO,EAAE,OAAO,EAAE,MAAM,MAAM,CAAC;AAC/B,OAAO,EACL,qBAAqB,EACrB,gBAAgB,EAChB,iBAAiB,EACjB,cAAc,EACd,qBAAqB,EACrB,gBAAgB,EAChB,gBAAgB,EACjB,MAAM,kCAAkC,CAAC;AAE1C;;;;;;;;;;;;;;;;;;GAkBG;AACH,qBAAa,YAAY;IAEvB,8CAA8C;IACvC,WAAW,iCAAwC;IAE1D,qCAAqC;IAC9B,WAAW,iCAAwC;IAE1D,mCAAmC;IAC5B,WAAW,iCAAwC;IAE1D,cAAc;IACd,OAAO,CAAC,SAAS,CAAM;IAEvB,cAAc;IACd,OAAO,CAAC,KAAK,CAAS;IAEtB,cAAc;IACd,OAAO,CAAC,aAAa,CAA2C;IAChE,cAAc;IACd,OAAO,CAAC,aAAa,CAA2C;IAChE,cAAc;IACd,OAAO,CAAC,aAAa,CAA2C;IAEhE;;;;;;;OAOG;gBACS,OAAO,EAAE,MAAM,EAAE,IAAI,EAAE,MAAM,EAAE,OAAO,CAAC,EAAE,iBAAiB;IAiBtE;;;;;;;;;;;;;;OAcG;IACI,OAAO,CAAC,MAAM,CAAC,EAAE,qBAAqB,GAAG,OAAO,CAAC,gBAAgB,CAAC;IAIzE;;;;OAIG;IACI,SAAS,IAAI,OAAO,CAAC,gBAAgB,CAAC;IAI7C;;;;OAIG;IACI,iBAAiB,IAAI,OAAO,CAAC,gBAAgB,EAAE,CAAC;IAIvD;;;;;OAKG;IACI,cAAc,CAAC,MAAM,CAAC,EAAE,qBAAqB,GAAG,OAAO,CAAC,gBAAgB,CAAC;IAIhF;;;;;OAKG;IACI,YAAY,CAAC,UAAU,CAAC,EAAE,MAAM,GAAG,IAAI;IAI9C;;OAEG;IACI,WAAW,IAAI,IAAI;IAI1B;;;OAGG;IACI,OAAO,IAAI,IAAI;IAYtB,cAAc;IACd,MAAM,CAAC,aAAa,CAAC,QAAQ,EAAE,GAAG,EAAE,IAAI,EAAE,MAAM,GAAG,YAAY;IAW/D,cAAc;IACd,OAAO,CAAC,WAAW;CAepB;AAGD;;;;;;;;;;;;;;;;;;GAkBG;AACH,qBAAa,gBAAgB;IAE3B,oFAAoF;IACpF,SAAgB,SAAS,EAAE,YAAY,CAAC;IAExC,oCAAoC;IACpC,SAAgB,IAAI,EAAE,cAAc,CAAC;IAErC,cAAc;IACd,OAAO,CAAC,OAAO,CAAM;IAErB;;;;;;;OAOG;gBACS,SAAS,EAAE,MAAM,EAAE,IAAI,EAAE,MAAM,EAAE,OAAO,CAAC,EAAE,iBAAiB;IAgBxE;;;;;;;;;;;;;;OAcG;IACI,eAAe,CAAC,MAAM,CAAC,EAAE,qBAAqB,GAAG,OAAO,CAAC,gBAAgB,CAAC;IAIjF;;OAEG;IACI,OAAO,IAAI,IAAI;CAGvB"}
@@ -49,6 +49,8 @@ declare global {
49
49
  * @since 1.0.0
50
50
  */
51
51
  export declare class PlayerClientService implements OnDestroy {
52
+ /** @ignore */
53
+ private zone;
52
54
  /** @ignore */
53
55
  private clientPromise;
54
56
  /**
@@ -1 +1 @@
1
- {"version":3,"file":"player-client.service.d.ts","sourceRoot":"","sources":["../../../../projects/reveldigital/player-client/src/lib/player-client.service.ts"],"names":[],"mappings":"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"}
1
+ {"version":3,"file":"player-client.service.d.ts","sourceRoot":"","sources":["../../../../projects/reveldigital/player-client/src/lib/player-client.service.ts"],"names":[],"mappings":"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"}
package/lib/version.d.ts CHANGED
@@ -1,2 +1,2 @@
1
- export declare const version = "2.2.0";
1
+ export declare const version = "2.3.0";
2
2
  //# sourceMappingURL=version.d.ts.map
package/package.json CHANGED
@@ -1,6 +1,6 @@
1
1
  {
2
2
  "name": "@reveldigital/player-client",
3
- "version": "2.2.0",
3
+ "version": "2.3.0",
4
4
  "description": "Helper library for interfacing Angular apps with the Revel Digital gadget framework.",
5
5
  "schematics": "./schematics/collection.json",
6
6
  "ng-add": {