stormcloud-video-player 0.2.12 → 0.2.14

This diff represents the content of publicly available package versions that have been released to one of the supported registries. The information contained in this diff is provided for informational purposes only and reflects changes between package versions as they appear in their respective public registries.
Files changed (38) hide show
  1. package/README.md +100 -1
  2. package/dist/stormcloud-vp.min.js +2 -2
  3. package/lib/index.cjs +1206 -151
  4. package/lib/index.cjs.map +1 -1
  5. package/lib/index.d.cts +77 -2
  6. package/lib/index.d.ts +77 -2
  7. package/lib/index.js +1196 -151
  8. package/lib/index.js.map +1 -1
  9. package/lib/player/StormcloudVideoPlayer.cjs +1100 -114
  10. package/lib/player/StormcloudVideoPlayer.cjs.map +1 -1
  11. package/lib/player/StormcloudVideoPlayer.d.cts +2 -2
  12. package/lib/players/FilePlayer.cjs +14 -7
  13. package/lib/players/FilePlayer.cjs.map +1 -1
  14. package/lib/players/HlsPlayer.cjs +1108 -119
  15. package/lib/players/HlsPlayer.cjs.map +1 -1
  16. package/lib/players/HlsPlayer.d.cts +1 -1
  17. package/lib/players/index.cjs +1122 -126
  18. package/lib/players/index.cjs.map +1 -1
  19. package/lib/sdk/hlsAdPlayer.cjs +561 -0
  20. package/lib/sdk/hlsAdPlayer.cjs.map +1 -0
  21. package/lib/sdk/hlsAdPlayer.d.cts +10 -0
  22. package/lib/sdk/ima.cjs +176 -23
  23. package/lib/sdk/ima.cjs.map +1 -1
  24. package/lib/sdk/ima.d.cts +1 -1
  25. package/lib/{types-GpA_hKek.d.cts → types-mVgmKmzM.d.cts} +3 -0
  26. package/lib/ui/StormcloudVideoPlayer.cjs +1107 -119
  27. package/lib/ui/StormcloudVideoPlayer.cjs.map +1 -1
  28. package/lib/ui/StormcloudVideoPlayer.d.cts +1 -1
  29. package/lib/utils/browserCompat.cjs +229 -0
  30. package/lib/utils/browserCompat.cjs.map +1 -0
  31. package/lib/utils/browserCompat.d.cts +36 -0
  32. package/lib/utils/polyfills.cjs +253 -0
  33. package/lib/utils/polyfills.cjs.map +1 -0
  34. package/lib/utils/polyfills.d.cts +11 -0
  35. package/lib/utils/tracking.cjs +10 -9
  36. package/lib/utils/tracking.cjs.map +1 -1
  37. package/lib/utils/tracking.d.cts +1 -1
  38. package/package.json +1 -1
@@ -0,0 +1,561 @@
1
+ "use strict";
2
+ var __create = Object.create;
3
+ var __defProp = Object.defineProperty;
4
+ var __getOwnPropDesc = Object.getOwnPropertyDescriptor;
5
+ var __getOwnPropNames = Object.getOwnPropertyNames;
6
+ var __getProtoOf = Object.getPrototypeOf;
7
+ var __hasOwnProp = Object.prototype.hasOwnProperty;
8
+ var __export = (target, all) => {
9
+ for (var name in all)
10
+ __defProp(target, name, { get: all[name], enumerable: true });
11
+ };
12
+ var __copyProps = (to, from, except, desc) => {
13
+ if (from && typeof from === "object" || typeof from === "function") {
14
+ for (let key of __getOwnPropNames(from))
15
+ if (!__hasOwnProp.call(to, key) && key !== except)
16
+ __defProp(to, key, { get: () => from[key], enumerable: !(desc = __getOwnPropDesc(from, key)) || desc.enumerable });
17
+ }
18
+ return to;
19
+ };
20
+ var __toESM = (mod, isNodeMode, target) => (target = mod != null ? __create(__getProtoOf(mod)) : {}, __copyProps(
21
+ // If the importer is in node compatibility mode or this is not an ESM
22
+ // file that has been converted to a CommonJS file using a Babel-
23
+ // compatible transform (i.e. "__esModule" has not been set), then set
24
+ // "default" to the CommonJS "module.exports" for node compatibility.
25
+ isNodeMode || !mod || !mod.__esModule ? __defProp(target, "default", { value: mod, enumerable: true }) : target,
26
+ mod
27
+ ));
28
+ var __toCommonJS = (mod) => __copyProps(__defProp({}, "__esModule", { value: true }), mod);
29
+
30
+ // src/sdk/hlsAdPlayer.ts
31
+ var hlsAdPlayer_exports = {};
32
+ __export(hlsAdPlayer_exports, {
33
+ createHlsAdPlayer: () => createHlsAdPlayer
34
+ });
35
+ module.exports = __toCommonJS(hlsAdPlayer_exports);
36
+ var import_hls = __toESM(require("hls.js"), 1);
37
+ function createHlsAdPlayer(contentVideo, options) {
38
+ let adPlaying = false;
39
+ let originalMutedState = false;
40
+ const listeners = /* @__PURE__ */ new Map();
41
+ const licenseKey = options == null ? void 0 : options.licenseKey;
42
+ const mainHlsInstance = options == null ? void 0 : options.mainHlsInstance;
43
+ let adVideoElement;
44
+ let adHls;
45
+ let adContainerEl;
46
+ let currentAd;
47
+ let sessionId;
48
+ let trackingFired = {
49
+ impression: false,
50
+ start: false,
51
+ firstQuartile: false,
52
+ midpoint: false,
53
+ thirdQuartile: false,
54
+ complete: false
55
+ };
56
+ function emit(event, payload) {
57
+ const set = listeners.get(event);
58
+ if (!set) return;
59
+ for (const fn of Array.from(set)) {
60
+ try {
61
+ fn(payload);
62
+ } catch (error) {
63
+ console.warn(`[HlsAdPlayer] Error in event listener for ${event}:`, error);
64
+ }
65
+ }
66
+ }
67
+ function generateSessionId() {
68
+ return `session-${Date.now()}-${Math.random().toString(36).substr(2, 9)}`;
69
+ }
70
+ function fireTrackingPixels(urls) {
71
+ if (!urls || urls.length === 0) return;
72
+ urls.forEach((url) => {
73
+ try {
74
+ let trackingUrl = url;
75
+ if (sessionId) {
76
+ trackingUrl = `${trackingUrl}${trackingUrl.includes("?") ? "&" : "?"}session_id=${sessionId}`;
77
+ }
78
+ if (licenseKey) {
79
+ trackingUrl = `${trackingUrl}${trackingUrl.includes("?") ? "&" : "?"}license_key=${licenseKey}`;
80
+ }
81
+ const img = new Image(1, 1);
82
+ img.src = trackingUrl;
83
+ console.log(`[HlsAdPlayer] Fired tracking pixel: ${trackingUrl}`);
84
+ } catch (error) {
85
+ console.warn(`[HlsAdPlayer] Error firing tracking pixel:`, error);
86
+ }
87
+ });
88
+ }
89
+ function getMainStreamQuality() {
90
+ if (!mainHlsInstance || !mainHlsInstance.levels) {
91
+ return null;
92
+ }
93
+ const currentLevel = mainHlsInstance.currentLevel;
94
+ if (currentLevel === -1 || !mainHlsInstance.levels[currentLevel]) {
95
+ const autoLevel = mainHlsInstance.loadLevel;
96
+ if (autoLevel !== -1 && mainHlsInstance.levels[autoLevel]) {
97
+ const level2 = mainHlsInstance.levels[autoLevel];
98
+ return {
99
+ width: level2.width || 1920,
100
+ height: level2.height || 1080,
101
+ bitrate: level2.bitrate || 5e6
102
+ };
103
+ }
104
+ return null;
105
+ }
106
+ const level = mainHlsInstance.levels[currentLevel];
107
+ return {
108
+ width: level.width || 1920,
109
+ height: level.height || 1080,
110
+ bitrate: level.bitrate || 5e6
111
+ };
112
+ }
113
+ function selectBestMediaFile(mediaFiles) {
114
+ if (mediaFiles.length === 0) {
115
+ throw new Error("No media files available");
116
+ }
117
+ const firstFile = mediaFiles[0];
118
+ if (!firstFile) {
119
+ throw new Error("No media files available");
120
+ }
121
+ if (mediaFiles.length === 1) {
122
+ return firstFile;
123
+ }
124
+ const mainQuality = getMainStreamQuality();
125
+ if (!mainQuality) {
126
+ console.log("[HlsAdPlayer] No main stream quality info, using first media file");
127
+ return firstFile;
128
+ }
129
+ console.log("[HlsAdPlayer] Main stream quality:", mainQuality);
130
+ const scoredFiles = mediaFiles.map((file) => {
131
+ const widthDiff = Math.abs(file.width - mainQuality.width);
132
+ const heightDiff = Math.abs(file.height - mainQuality.height);
133
+ const resolutionDiff = widthDiff + heightDiff;
134
+ const fileBitrate = (file.bitrate || 5e3) * 1e3;
135
+ const bitrateDiff = Math.abs(fileBitrate - mainQuality.bitrate);
136
+ const score = resolutionDiff * 2 + bitrateDiff / 1e3;
137
+ return { file, score, resolutionDiff, bitrateDiff };
138
+ });
139
+ scoredFiles.sort((a, b) => a.score - b.score);
140
+ const bestMatch = scoredFiles[0];
141
+ if (!bestMatch) {
142
+ console.log("[HlsAdPlayer] No best match found, using first media file");
143
+ return firstFile;
144
+ }
145
+ console.log("[HlsAdPlayer] Selected media file:", {
146
+ url: bestMatch.file.url,
147
+ resolution: `${bestMatch.file.width}x${bestMatch.file.height}`,
148
+ bitrate: bestMatch.file.bitrate,
149
+ score: bestMatch.score,
150
+ resolutionDiff: bestMatch.resolutionDiff,
151
+ bitrateDiff: bestMatch.bitrateDiff
152
+ });
153
+ return bestMatch.file;
154
+ }
155
+ function parseVastXml(xmlString) {
156
+ var _a, _b, _c, _d;
157
+ try {
158
+ const parser = new DOMParser();
159
+ const xmlDoc = parser.parseFromString(xmlString, "text/xml");
160
+ const parserError = xmlDoc.querySelector("parsererror");
161
+ if (parserError) {
162
+ console.error("[HlsAdPlayer] XML parsing error:", parserError.textContent);
163
+ return null;
164
+ }
165
+ const adElement = xmlDoc.querySelector("Ad");
166
+ if (!adElement) {
167
+ console.warn("[HlsAdPlayer] No Ad element found in VAST XML");
168
+ return null;
169
+ }
170
+ const adId = adElement.getAttribute("id") || "unknown";
171
+ const title = ((_a = xmlDoc.querySelector("AdTitle")) == null ? void 0 : _a.textContent) || "Ad";
172
+ const durationText = ((_b = xmlDoc.querySelector("Duration")) == null ? void 0 : _b.textContent) || "00:00:30";
173
+ const durationParts = durationText.split(":");
174
+ const duration = parseInt(durationParts[0] || "0", 10) * 3600 + parseInt(durationParts[1] || "0", 10) * 60 + parseInt(durationParts[2] || "0", 10);
175
+ const mediaFileElements = xmlDoc.querySelectorAll("MediaFile");
176
+ const mediaFiles = [];
177
+ mediaFileElements.forEach((mf) => {
178
+ var _a2;
179
+ const type = mf.getAttribute("type") || "";
180
+ if (type === "application/x-mpegURL" || type.includes("m3u8")) {
181
+ const bitrateAttr = mf.getAttribute("bitrate");
182
+ const bitrateValue = bitrateAttr ? parseInt(bitrateAttr, 10) : void 0;
183
+ mediaFiles.push({
184
+ url: ((_a2 = mf.textContent) == null ? void 0 : _a2.trim()) || "",
185
+ type,
186
+ width: parseInt(mf.getAttribute("width") || "1920", 10),
187
+ height: parseInt(mf.getAttribute("height") || "1080", 10),
188
+ bitrate: bitrateValue && bitrateValue > 0 ? bitrateValue : void 0
189
+ });
190
+ }
191
+ });
192
+ if (mediaFiles.length === 0) {
193
+ console.warn("[HlsAdPlayer] No HLS media files found in VAST XML");
194
+ return null;
195
+ }
196
+ const trackingUrls = {
197
+ impression: [],
198
+ start: [],
199
+ firstQuartile: [],
200
+ midpoint: [],
201
+ thirdQuartile: [],
202
+ complete: [],
203
+ mute: [],
204
+ unmute: [],
205
+ pause: [],
206
+ resume: [],
207
+ fullscreen: [],
208
+ exitFullscreen: [],
209
+ skip: [],
210
+ error: []
211
+ };
212
+ xmlDoc.querySelectorAll("Impression").forEach((el) => {
213
+ var _a2;
214
+ const url = (_a2 = el.textContent) == null ? void 0 : _a2.trim();
215
+ if (url) trackingUrls.impression.push(url);
216
+ });
217
+ xmlDoc.querySelectorAll("Tracking").forEach((el) => {
218
+ var _a2;
219
+ const event = el.getAttribute("event");
220
+ const url = (_a2 = el.textContent) == null ? void 0 : _a2.trim();
221
+ if (event && url) {
222
+ const eventKey = event;
223
+ if (trackingUrls[eventKey]) {
224
+ trackingUrls[eventKey].push(url);
225
+ }
226
+ }
227
+ });
228
+ const clickThrough = (_d = (_c = xmlDoc.querySelector("ClickThrough")) == null ? void 0 : _c.textContent) == null ? void 0 : _d.trim();
229
+ return {
230
+ id: adId,
231
+ title,
232
+ duration,
233
+ mediaFiles,
234
+ trackingUrls,
235
+ clickThrough
236
+ };
237
+ } catch (error) {
238
+ console.error("[HlsAdPlayer] Error parsing VAST XML:", error);
239
+ return null;
240
+ }
241
+ }
242
+ function createAdVideoElement() {
243
+ const video = document.createElement("video");
244
+ video.style.position = "absolute";
245
+ video.style.left = "0";
246
+ video.style.top = "0";
247
+ video.style.width = "100%";
248
+ video.style.height = "100%";
249
+ video.style.objectFit = "contain";
250
+ video.style.backgroundColor = "#000";
251
+ video.playsInline = true;
252
+ video.muted = false;
253
+ return video;
254
+ }
255
+ function setupAdEventListeners() {
256
+ if (!adVideoElement || !currentAd) return;
257
+ adVideoElement.addEventListener("timeupdate", () => {
258
+ if (!currentAd || !adVideoElement) return;
259
+ const progress = adVideoElement.currentTime / currentAd.duration;
260
+ if (progress >= 0.25 && !trackingFired.firstQuartile) {
261
+ trackingFired.firstQuartile = true;
262
+ fireTrackingPixels(currentAd.trackingUrls.firstQuartile);
263
+ }
264
+ if (progress >= 0.5 && !trackingFired.midpoint) {
265
+ trackingFired.midpoint = true;
266
+ fireTrackingPixels(currentAd.trackingUrls.midpoint);
267
+ }
268
+ if (progress >= 0.75 && !trackingFired.thirdQuartile) {
269
+ trackingFired.thirdQuartile = true;
270
+ fireTrackingPixels(currentAd.trackingUrls.thirdQuartile);
271
+ }
272
+ });
273
+ adVideoElement.addEventListener("playing", () => {
274
+ if (!currentAd || trackingFired.start) return;
275
+ trackingFired.start = true;
276
+ fireTrackingPixels(currentAd.trackingUrls.start);
277
+ console.log("[HlsAdPlayer] Ad started playing");
278
+ });
279
+ adVideoElement.addEventListener("ended", () => {
280
+ if (!currentAd || trackingFired.complete) return;
281
+ trackingFired.complete = true;
282
+ fireTrackingPixels(currentAd.trackingUrls.complete);
283
+ console.log("[HlsAdPlayer] Ad completed");
284
+ handleAdComplete();
285
+ });
286
+ adVideoElement.addEventListener("error", (e) => {
287
+ console.error("[HlsAdPlayer] Ad video error:", e);
288
+ if (currentAd) {
289
+ fireTrackingPixels(currentAd.trackingUrls.error);
290
+ }
291
+ handleAdError();
292
+ });
293
+ adVideoElement.addEventListener("volumechange", () => {
294
+ if (!currentAd) return;
295
+ if (adVideoElement.muted) {
296
+ fireTrackingPixels(currentAd.trackingUrls.mute);
297
+ } else {
298
+ fireTrackingPixels(currentAd.trackingUrls.unmute);
299
+ }
300
+ });
301
+ adVideoElement.addEventListener("pause", () => {
302
+ if (currentAd && !adVideoElement.ended) {
303
+ fireTrackingPixels(currentAd.trackingUrls.pause);
304
+ }
305
+ });
306
+ adVideoElement.addEventListener("play", () => {
307
+ if (currentAd && adVideoElement.currentTime > 0) {
308
+ fireTrackingPixels(currentAd.trackingUrls.resume);
309
+ }
310
+ });
311
+ }
312
+ function handleAdComplete() {
313
+ console.log("[HlsAdPlayer] Handling ad completion");
314
+ adPlaying = false;
315
+ contentVideo.muted = originalMutedState;
316
+ if (adContainerEl) {
317
+ adContainerEl.style.display = "none";
318
+ adContainerEl.style.pointerEvents = "none";
319
+ }
320
+ if (!(options == null ? void 0 : options.continueLiveStreamDuringAds)) {
321
+ contentVideo.play().catch(() => {
322
+ });
323
+ console.log("[HlsAdPlayer] Content resumed (VOD mode)");
324
+ } else {
325
+ console.log("[HlsAdPlayer] Content unmuted (Live mode)");
326
+ }
327
+ emit("content_resume");
328
+ emit("all_ads_completed");
329
+ }
330
+ function handleAdError() {
331
+ console.log("[HlsAdPlayer] Handling ad error");
332
+ adPlaying = false;
333
+ contentVideo.muted = originalMutedState;
334
+ if (adContainerEl) {
335
+ adContainerEl.style.display = "none";
336
+ adContainerEl.style.pointerEvents = "none";
337
+ }
338
+ if (!(options == null ? void 0 : options.continueLiveStreamDuringAds)) {
339
+ if (contentVideo.paused) {
340
+ contentVideo.play().catch(() => {
341
+ });
342
+ }
343
+ }
344
+ emit("ad_error");
345
+ }
346
+ return {
347
+ initialize() {
348
+ var _a;
349
+ console.log("[HlsAdPlayer] Initializing");
350
+ if (!adContainerEl) {
351
+ const container = document.createElement("div");
352
+ container.style.position = "absolute";
353
+ container.style.left = "0";
354
+ container.style.top = "0";
355
+ container.style.right = "0";
356
+ container.style.bottom = "0";
357
+ container.style.display = "none";
358
+ container.style.alignItems = "center";
359
+ container.style.justifyContent = "center";
360
+ container.style.pointerEvents = "none";
361
+ container.style.zIndex = "2";
362
+ container.style.backgroundColor = "#000";
363
+ (_a = contentVideo.parentElement) == null ? void 0 : _a.appendChild(container);
364
+ adContainerEl = container;
365
+ }
366
+ },
367
+ async requestAds(vastTagUrl) {
368
+ console.log("[HlsAdPlayer] Requesting ads:", vastTagUrl);
369
+ if (adPlaying) {
370
+ console.warn("[HlsAdPlayer] Cannot request new ads while an ad is playing");
371
+ return Promise.reject(new Error("Ad already playing"));
372
+ }
373
+ try {
374
+ sessionId = generateSessionId();
375
+ const response = await fetch(vastTagUrl);
376
+ if (!response.ok) {
377
+ throw new Error(`Failed to fetch VAST: ${response.statusText}`);
378
+ }
379
+ const vastXml = await response.text();
380
+ console.log("[HlsAdPlayer] VAST XML received");
381
+ const ad = parseVastXml(vastXml);
382
+ if (!ad) {
383
+ throw new Error("Failed to parse VAST XML or no ads available");
384
+ }
385
+ currentAd = ad;
386
+ console.log(`[HlsAdPlayer] Ad parsed: ${ad.title}, duration: ${ad.duration}s`);
387
+ fireTrackingPixels(ad.trackingUrls.impression);
388
+ trackingFired.impression = true;
389
+ return Promise.resolve();
390
+ } catch (error) {
391
+ console.error("[HlsAdPlayer] Error requesting ads:", error);
392
+ emit("ad_error");
393
+ return Promise.reject(error);
394
+ }
395
+ },
396
+ async play() {
397
+ if (!currentAd) {
398
+ console.warn("[HlsAdPlayer] Cannot play: No ad loaded");
399
+ return Promise.reject(new Error("No ad loaded"));
400
+ }
401
+ console.log("[HlsAdPlayer] Starting ad playback");
402
+ try {
403
+ if (!adVideoElement) {
404
+ adVideoElement = createAdVideoElement();
405
+ adContainerEl == null ? void 0 : adContainerEl.appendChild(adVideoElement);
406
+ setupAdEventListeners();
407
+ }
408
+ trackingFired = {
409
+ impression: trackingFired.impression,
410
+ start: false,
411
+ firstQuartile: false,
412
+ midpoint: false,
413
+ thirdQuartile: false,
414
+ complete: false
415
+ };
416
+ if (!(options == null ? void 0 : options.continueLiveStreamDuringAds)) {
417
+ contentVideo.pause();
418
+ console.log("[HlsAdPlayer] Content paused (VOD mode)");
419
+ } else {
420
+ console.log("[HlsAdPlayer] Content continues (Live mode)");
421
+ }
422
+ contentVideo.muted = true;
423
+ adPlaying = true;
424
+ if (adContainerEl) {
425
+ adContainerEl.style.display = "flex";
426
+ adContainerEl.style.pointerEvents = "auto";
427
+ }
428
+ emit("content_pause");
429
+ const mediaFile = selectBestMediaFile(currentAd.mediaFiles);
430
+ if (!mediaFile) {
431
+ throw new Error("No media file available for ad");
432
+ }
433
+ console.log(`[HlsAdPlayer] Loading ad from: ${mediaFile.url}`);
434
+ if (import_hls.default.isSupported()) {
435
+ if (adHls) {
436
+ adHls.destroy();
437
+ }
438
+ adHls = new import_hls.default({
439
+ enableWorker: true,
440
+ lowLatencyMode: false
441
+ });
442
+ adHls.loadSource(mediaFile.url);
443
+ adHls.attachMedia(adVideoElement);
444
+ adHls.on(import_hls.default.Events.MANIFEST_PARSED, () => {
445
+ console.log("[HlsAdPlayer] HLS manifest parsed, starting playback");
446
+ adVideoElement.play().catch((error) => {
447
+ console.error("[HlsAdPlayer] Error starting ad playback:", error);
448
+ handleAdError();
449
+ });
450
+ });
451
+ adHls.on(import_hls.default.Events.ERROR, (event, data) => {
452
+ console.error("[HlsAdPlayer] HLS error:", data);
453
+ if (data.fatal) {
454
+ handleAdError();
455
+ }
456
+ });
457
+ } else if (adVideoElement.canPlayType("application/vnd.apple.mpegurl")) {
458
+ adVideoElement.src = mediaFile.url;
459
+ adVideoElement.play().catch((error) => {
460
+ console.error("[HlsAdPlayer] Error starting ad playback:", error);
461
+ handleAdError();
462
+ });
463
+ } else {
464
+ throw new Error("HLS not supported");
465
+ }
466
+ return Promise.resolve();
467
+ } catch (error) {
468
+ console.error("[HlsAdPlayer] Error playing ad:", error);
469
+ handleAdError();
470
+ return Promise.reject(error);
471
+ }
472
+ },
473
+ async stop() {
474
+ console.log("[HlsAdPlayer] Stopping ad");
475
+ adPlaying = false;
476
+ contentVideo.muted = originalMutedState;
477
+ if (adContainerEl) {
478
+ adContainerEl.style.display = "none";
479
+ adContainerEl.style.pointerEvents = "none";
480
+ }
481
+ if (adHls) {
482
+ adHls.destroy();
483
+ adHls = void 0;
484
+ }
485
+ if (adVideoElement) {
486
+ adVideoElement.pause();
487
+ adVideoElement.src = "";
488
+ }
489
+ if (!(options == null ? void 0 : options.continueLiveStreamDuringAds)) {
490
+ contentVideo.play().catch(() => {
491
+ });
492
+ }
493
+ currentAd = void 0;
494
+ },
495
+ destroy() {
496
+ console.log("[HlsAdPlayer] Destroying");
497
+ adPlaying = false;
498
+ contentVideo.muted = originalMutedState;
499
+ if (adHls) {
500
+ adHls.destroy();
501
+ adHls = void 0;
502
+ }
503
+ if (adVideoElement) {
504
+ adVideoElement.pause();
505
+ adVideoElement.src = "";
506
+ adVideoElement.remove();
507
+ adVideoElement = void 0;
508
+ }
509
+ if (adContainerEl == null ? void 0 : adContainerEl.parentElement) {
510
+ adContainerEl.parentElement.removeChild(adContainerEl);
511
+ }
512
+ adContainerEl = void 0;
513
+ currentAd = void 0;
514
+ listeners.clear();
515
+ },
516
+ isAdPlaying() {
517
+ return adPlaying;
518
+ },
519
+ resize(width, height) {
520
+ console.log(`[HlsAdPlayer] Resizing to ${width}x${height}`);
521
+ if (adContainerEl) {
522
+ adContainerEl.style.width = `${width}px`;
523
+ adContainerEl.style.height = `${height}px`;
524
+ }
525
+ if (adVideoElement) {
526
+ adVideoElement.style.width = `${width}px`;
527
+ adVideoElement.style.height = `${height}px`;
528
+ }
529
+ },
530
+ on(event, listener) {
531
+ if (!listeners.has(event)) listeners.set(event, /* @__PURE__ */ new Set());
532
+ listeners.get(event).add(listener);
533
+ },
534
+ off(event, listener) {
535
+ var _a;
536
+ (_a = listeners.get(event)) == null ? void 0 : _a.delete(listener);
537
+ },
538
+ updateOriginalMutedState(muted) {
539
+ originalMutedState = muted;
540
+ },
541
+ getOriginalMutedState() {
542
+ return originalMutedState;
543
+ },
544
+ setAdVolume(volume) {
545
+ if (adVideoElement && adPlaying) {
546
+ adVideoElement.volume = Math.max(0, Math.min(1, volume));
547
+ }
548
+ },
549
+ getAdVolume() {
550
+ if (adVideoElement && adPlaying) {
551
+ return adVideoElement.volume;
552
+ }
553
+ return 1;
554
+ }
555
+ };
556
+ }
557
+ // Annotate the CommonJS export names for ESM import in node:
558
+ 0 && (module.exports = {
559
+ createHlsAdPlayer
560
+ });
561
+ //# sourceMappingURL=hlsAdPlayer.cjs.map
@@ -0,0 +1 @@
1
+ {"version":3,"sources":["../../src/sdk/hlsAdPlayer.ts"],"sourcesContent":["import type { ImaController } from \"../types\";\nimport Hls from \"hls.js\";\n\ninterface VastMediaFile {\n url: string;\n type: string;\n width: number;\n height: number;\n bitrate?: number | undefined;\n}\n\ninterface VastTrackingUrls {\n impression: string[];\n start: string[];\n firstQuartile: string[];\n midpoint: string[];\n thirdQuartile: string[];\n complete: string[];\n mute: string[];\n unmute: string[];\n pause: string[];\n resume: string[];\n fullscreen: string[];\n exitFullscreen: string[];\n skip: string[];\n error: string[];\n}\n\ninterface VastAd {\n id: string;\n title: string;\n duration: number;\n mediaFiles: VastMediaFile[];\n trackingUrls: VastTrackingUrls;\n clickThrough?: string | undefined;\n}\n\nexport function createHlsAdPlayer(\n contentVideo: HTMLVideoElement,\n options?: {\n continueLiveStreamDuringAds?: boolean;\n licenseKey?: string;\n mainHlsInstance?: Hls;\n }\n): ImaController {\n let adPlaying = false;\n let originalMutedState = false;\n const listeners = new Map<string, Set<(payload?: any) => void>>();\n const licenseKey = options?.licenseKey;\n const mainHlsInstance = options?.mainHlsInstance;\n\n let adVideoElement: HTMLVideoElement | undefined;\n let adHls: Hls | undefined;\n let adContainerEl: HTMLDivElement | undefined;\n let currentAd: VastAd | undefined;\n let sessionId: string | undefined;\n\n let trackingFired = {\n impression: false,\n start: false,\n firstQuartile: false,\n midpoint: false,\n thirdQuartile: false,\n complete: false,\n };\n\n function emit(event: string, payload?: any): void {\n const set = listeners.get(event);\n if (!set) return;\n for (const fn of Array.from(set)) {\n try {\n fn(payload);\n } catch (error) {\n console.warn(`[HlsAdPlayer] Error in event listener for ${event}:`, error);\n }\n }\n }\n\n function generateSessionId(): string {\n return `session-${Date.now()}-${Math.random().toString(36).substr(2, 9)}`;\n }\n\n function fireTrackingPixels(urls: string[]): void {\n if (!urls || urls.length === 0) return;\n\n urls.forEach((url) => {\n try {\n let trackingUrl = url;\n\n if (sessionId) {\n trackingUrl = `${trackingUrl}${trackingUrl.includes(\"?\") ? \"&\" : \"?\"}session_id=${sessionId}`;\n }\n\n if (licenseKey) {\n trackingUrl = `${trackingUrl}${trackingUrl.includes(\"?\") ? \"&\" : \"?\"}license_key=${licenseKey}`;\n }\n\n const img = new Image(1, 1);\n img.src = trackingUrl;\n console.log(`[HlsAdPlayer] Fired tracking pixel: ${trackingUrl}`);\n } catch (error) {\n console.warn(`[HlsAdPlayer] Error firing tracking pixel:`, error);\n }\n });\n }\n\n function getMainStreamQuality(): { width: number; height: number; bitrate: number } | null {\n if (!mainHlsInstance || !mainHlsInstance.levels) {\n return null;\n }\n\n const currentLevel = mainHlsInstance.currentLevel;\n if (currentLevel === -1 || !mainHlsInstance.levels[currentLevel]) {\n const autoLevel = mainHlsInstance.loadLevel;\n if (autoLevel !== -1 && mainHlsInstance.levels[autoLevel]) {\n const level = mainHlsInstance.levels[autoLevel];\n return {\n width: level.width || 1920,\n height: level.height || 1080,\n bitrate: level.bitrate || 5000000,\n };\n }\n return null;\n }\n\n const level = mainHlsInstance.levels[currentLevel];\n return {\n width: level.width || 1920,\n height: level.height || 1080,\n bitrate: level.bitrate || 5000000,\n };\n }\n\n function selectBestMediaFile(mediaFiles: VastMediaFile[]): VastMediaFile {\n if (mediaFiles.length === 0) {\n throw new Error(\"No media files available\");\n }\n\n const firstFile = mediaFiles[0];\n if (!firstFile) {\n throw new Error(\"No media files available\");\n }\n\n if (mediaFiles.length === 1) {\n return firstFile;\n }\n\n const mainQuality = getMainStreamQuality();\n if (!mainQuality) {\n console.log(\"[HlsAdPlayer] No main stream quality info, using first media file\");\n return firstFile;\n }\n\n console.log(\"[HlsAdPlayer] Main stream quality:\", mainQuality);\n\n const scoredFiles = mediaFiles.map((file) => {\n const widthDiff = Math.abs(file.width - mainQuality.width);\n const heightDiff = Math.abs(file.height - mainQuality.height);\n const resolutionDiff = widthDiff + heightDiff;\n\n const fileBitrate = (file.bitrate || 5000) * 1000;\n const bitrateDiff = Math.abs(fileBitrate - mainQuality.bitrate);\n\n const score = resolutionDiff * 2 + bitrateDiff / 1000;\n\n return { file, score, resolutionDiff, bitrateDiff };\n });\n\n scoredFiles.sort((a, b) => a.score - b.score);\n\n const bestMatch = scoredFiles[0];\n if (!bestMatch) {\n console.log(\"[HlsAdPlayer] No best match found, using first media file\");\n return firstFile;\n }\n\n console.log(\"[HlsAdPlayer] Selected media file:\", {\n url: bestMatch.file.url,\n resolution: `${bestMatch.file.width}x${bestMatch.file.height}`,\n bitrate: bestMatch.file.bitrate,\n score: bestMatch.score,\n resolutionDiff: bestMatch.resolutionDiff,\n bitrateDiff: bestMatch.bitrateDiff,\n });\n\n return bestMatch.file;\n }\n\n function parseVastXml(xmlString: string): VastAd | null {\n try {\n const parser = new DOMParser();\n const xmlDoc = parser.parseFromString(xmlString, \"text/xml\");\n\n const parserError = xmlDoc.querySelector(\"parsererror\");\n if (parserError) {\n console.error(\"[HlsAdPlayer] XML parsing error:\", parserError.textContent);\n return null;\n }\n\n const adElement = xmlDoc.querySelector(\"Ad\");\n if (!adElement) {\n console.warn(\"[HlsAdPlayer] No Ad element found in VAST XML\");\n return null;\n }\n\n const adId = adElement.getAttribute(\"id\") || \"unknown\";\n const title = xmlDoc.querySelector(\"AdTitle\")?.textContent || \"Ad\";\n\n const durationText = xmlDoc.querySelector(\"Duration\")?.textContent || \"00:00:30\";\n const durationParts = durationText.split(\":\");\n const duration =\n parseInt(durationParts[0] || \"0\", 10) * 3600 +\n parseInt(durationParts[1] || \"0\", 10) * 60 +\n parseInt(durationParts[2] || \"0\", 10);\n\n const mediaFileElements = xmlDoc.querySelectorAll(\"MediaFile\");\n const mediaFiles: VastMediaFile[] = [];\n\n mediaFileElements.forEach((mf) => {\n const type = mf.getAttribute(\"type\") || \"\";\n if (type === \"application/x-mpegURL\" || type.includes(\"m3u8\")) {\n const bitrateAttr = mf.getAttribute(\"bitrate\");\n const bitrateValue = bitrateAttr ? parseInt(bitrateAttr, 10) : undefined;\n\n mediaFiles.push({\n url: mf.textContent?.trim() || \"\",\n type,\n width: parseInt(mf.getAttribute(\"width\") || \"1920\", 10),\n height: parseInt(mf.getAttribute(\"height\") || \"1080\", 10),\n bitrate: bitrateValue && bitrateValue > 0 ? bitrateValue : undefined,\n });\n }\n });\n\n if (mediaFiles.length === 0) {\n console.warn(\"[HlsAdPlayer] No HLS media files found in VAST XML\");\n return null;\n }\n\n const trackingUrls: VastTrackingUrls = {\n impression: [],\n start: [],\n firstQuartile: [],\n midpoint: [],\n thirdQuartile: [],\n complete: [],\n mute: [],\n unmute: [],\n pause: [],\n resume: [],\n fullscreen: [],\n exitFullscreen: [],\n skip: [],\n error: [],\n };\n\n xmlDoc.querySelectorAll(\"Impression\").forEach((el) => {\n const url = el.textContent?.trim();\n if (url) trackingUrls.impression.push(url);\n });\n\n xmlDoc.querySelectorAll(\"Tracking\").forEach((el) => {\n const event = el.getAttribute(\"event\");\n const url = el.textContent?.trim();\n if (event && url) {\n const eventKey = event as keyof VastTrackingUrls;\n if (trackingUrls[eventKey]) {\n trackingUrls[eventKey].push(url);\n }\n }\n });\n\n const clickThrough = xmlDoc.querySelector(\"ClickThrough\")?.textContent?.trim();\n\n return {\n id: adId,\n title,\n duration,\n mediaFiles,\n trackingUrls,\n clickThrough,\n };\n } catch (error) {\n console.error(\"[HlsAdPlayer] Error parsing VAST XML:\", error);\n return null;\n }\n }\n\n function createAdVideoElement(): HTMLVideoElement {\n const video = document.createElement(\"video\");\n video.style.position = \"absolute\";\n video.style.left = \"0\";\n video.style.top = \"0\";\n video.style.width = \"100%\";\n video.style.height = \"100%\";\n video.style.objectFit = \"contain\";\n video.style.backgroundColor = \"#000\";\n video.playsInline = true;\n video.muted = false;\n return video;\n }\n\n function setupAdEventListeners(): void {\n if (!adVideoElement || !currentAd) return;\n\n adVideoElement.addEventListener(\"timeupdate\", () => {\n if (!currentAd || !adVideoElement) return;\n\n const progress = adVideoElement.currentTime / currentAd.duration;\n\n if (progress >= 0.25 && !trackingFired.firstQuartile) {\n trackingFired.firstQuartile = true;\n fireTrackingPixels(currentAd.trackingUrls.firstQuartile);\n }\n\n if (progress >= 0.5 && !trackingFired.midpoint) {\n trackingFired.midpoint = true;\n fireTrackingPixels(currentAd.trackingUrls.midpoint);\n }\n\n if (progress >= 0.75 && !trackingFired.thirdQuartile) {\n trackingFired.thirdQuartile = true;\n fireTrackingPixels(currentAd.trackingUrls.thirdQuartile);\n }\n });\n\n adVideoElement.addEventListener(\"playing\", () => {\n if (!currentAd || trackingFired.start) return;\n trackingFired.start = true;\n fireTrackingPixels(currentAd.trackingUrls.start);\n console.log(\"[HlsAdPlayer] Ad started playing\");\n });\n\n adVideoElement.addEventListener(\"ended\", () => {\n if (!currentAd || trackingFired.complete) return;\n trackingFired.complete = true;\n fireTrackingPixels(currentAd.trackingUrls.complete);\n console.log(\"[HlsAdPlayer] Ad completed\");\n\n handleAdComplete();\n });\n\n adVideoElement.addEventListener(\"error\", (e) => {\n console.error(\"[HlsAdPlayer] Ad video error:\", e);\n if (currentAd) {\n fireTrackingPixels(currentAd.trackingUrls.error);\n }\n handleAdError();\n });\n\n adVideoElement.addEventListener(\"volumechange\", () => {\n if (!currentAd) return;\n if (adVideoElement!.muted) {\n fireTrackingPixels(currentAd.trackingUrls.mute);\n } else {\n fireTrackingPixels(currentAd.trackingUrls.unmute);\n }\n });\n\n adVideoElement.addEventListener(\"pause\", () => {\n if (currentAd && !adVideoElement!.ended) {\n fireTrackingPixels(currentAd.trackingUrls.pause);\n }\n });\n\n adVideoElement.addEventListener(\"play\", () => {\n if (currentAd && adVideoElement!.currentTime > 0) {\n fireTrackingPixels(currentAd.trackingUrls.resume);\n }\n });\n }\n\n function handleAdComplete(): void {\n console.log(\"[HlsAdPlayer] Handling ad completion\");\n adPlaying = false;\n contentVideo.muted = originalMutedState;\n\n if (adContainerEl) {\n adContainerEl.style.display = \"none\";\n adContainerEl.style.pointerEvents = \"none\";\n }\n\n if (!options?.continueLiveStreamDuringAds) {\n contentVideo.play().catch(() => {});\n console.log(\"[HlsAdPlayer] Content resumed (VOD mode)\");\n } else {\n console.log(\"[HlsAdPlayer] Content unmuted (Live mode)\");\n }\n\n emit(\"content_resume\");\n emit(\"all_ads_completed\");\n }\n\n function handleAdError(): void {\n console.log(\"[HlsAdPlayer] Handling ad error\");\n adPlaying = false;\n contentVideo.muted = originalMutedState;\n\n if (adContainerEl) {\n adContainerEl.style.display = \"none\";\n adContainerEl.style.pointerEvents = \"none\";\n }\n\n if (!options?.continueLiveStreamDuringAds) {\n if (contentVideo.paused) {\n contentVideo.play().catch(() => {});\n }\n }\n\n emit(\"ad_error\");\n }\n\n return {\n initialize() {\n console.log(\"[HlsAdPlayer] Initializing\");\n\n if (!adContainerEl) {\n const container = document.createElement(\"div\");\n container.style.position = \"absolute\";\n container.style.left = \"0\";\n container.style.top = \"0\";\n container.style.right = \"0\";\n container.style.bottom = \"0\";\n container.style.display = \"none\";\n container.style.alignItems = \"center\";\n container.style.justifyContent = \"center\";\n container.style.pointerEvents = \"none\";\n container.style.zIndex = \"2\";\n container.style.backgroundColor = \"#000\";\n\n contentVideo.parentElement?.appendChild(container);\n adContainerEl = container;\n }\n },\n\n async requestAds(vastTagUrl: string) {\n console.log(\"[HlsAdPlayer] Requesting ads:\", vastTagUrl);\n\n if (adPlaying) {\n console.warn(\"[HlsAdPlayer] Cannot request new ads while an ad is playing\");\n return Promise.reject(new Error(\"Ad already playing\"));\n }\n\n try {\n sessionId = generateSessionId();\n\n const response = await fetch(vastTagUrl);\n if (!response.ok) {\n throw new Error(`Failed to fetch VAST: ${response.statusText}`);\n }\n\n const vastXml = await response.text();\n console.log(\"[HlsAdPlayer] VAST XML received\");\n\n const ad = parseVastXml(vastXml);\n if (!ad) {\n throw new Error(\"Failed to parse VAST XML or no ads available\");\n }\n\n currentAd = ad;\n console.log(`[HlsAdPlayer] Ad parsed: ${ad.title}, duration: ${ad.duration}s`);\n\n fireTrackingPixels(ad.trackingUrls.impression);\n trackingFired.impression = true;\n\n return Promise.resolve();\n } catch (error) {\n console.error(\"[HlsAdPlayer] Error requesting ads:\", error);\n emit(\"ad_error\");\n return Promise.reject(error);\n }\n },\n\n async play() {\n if (!currentAd) {\n console.warn(\"[HlsAdPlayer] Cannot play: No ad loaded\");\n return Promise.reject(new Error(\"No ad loaded\"));\n }\n\n console.log(\"[HlsAdPlayer] Starting ad playback\");\n\n try {\n if (!adVideoElement) {\n adVideoElement = createAdVideoElement();\n adContainerEl?.appendChild(adVideoElement);\n setupAdEventListeners();\n }\n\n trackingFired = {\n impression: trackingFired.impression,\n start: false,\n firstQuartile: false,\n midpoint: false,\n thirdQuartile: false,\n complete: false,\n };\n\n if (!options?.continueLiveStreamDuringAds) {\n contentVideo.pause();\n console.log(\"[HlsAdPlayer] Content paused (VOD mode)\");\n } else {\n console.log(\"[HlsAdPlayer] Content continues (Live mode)\");\n }\n\n contentVideo.muted = true;\n adPlaying = true;\n\n if (adContainerEl) {\n adContainerEl.style.display = \"flex\";\n adContainerEl.style.pointerEvents = \"auto\";\n }\n\n emit(\"content_pause\");\n\n const mediaFile = selectBestMediaFile(currentAd.mediaFiles);\n if (!mediaFile) {\n throw new Error(\"No media file available for ad\");\n }\n\n console.log(`[HlsAdPlayer] Loading ad from: ${mediaFile.url}`);\n\n if (Hls.isSupported()) {\n if (adHls) {\n adHls.destroy();\n }\n\n adHls = new Hls({\n enableWorker: true,\n lowLatencyMode: false,\n });\n\n adHls.loadSource(mediaFile.url);\n adHls.attachMedia(adVideoElement);\n\n adHls.on(Hls.Events.MANIFEST_PARSED, () => {\n console.log(\"[HlsAdPlayer] HLS manifest parsed, starting playback\");\n adVideoElement!.play().catch((error) => {\n console.error(\"[HlsAdPlayer] Error starting ad playback:\", error);\n handleAdError();\n });\n });\n\n adHls.on(Hls.Events.ERROR, (event, data) => {\n console.error(\"[HlsAdPlayer] HLS error:\", data);\n if (data.fatal) {\n handleAdError();\n }\n });\n } else if (adVideoElement.canPlayType(\"application/vnd.apple.mpegurl\")) {\n adVideoElement.src = mediaFile.url;\n adVideoElement.play().catch((error) => {\n console.error(\"[HlsAdPlayer] Error starting ad playback:\", error);\n handleAdError();\n });\n } else {\n throw new Error(\"HLS not supported\");\n }\n\n return Promise.resolve();\n } catch (error) {\n console.error(\"[HlsAdPlayer] Error playing ad:\", error);\n handleAdError();\n return Promise.reject(error);\n }\n },\n\n async stop() {\n console.log(\"[HlsAdPlayer] Stopping ad\");\n adPlaying = false;\n contentVideo.muted = originalMutedState;\n\n if (adContainerEl) {\n adContainerEl.style.display = \"none\";\n adContainerEl.style.pointerEvents = \"none\";\n }\n\n if (adHls) {\n adHls.destroy();\n adHls = undefined;\n }\n\n if (adVideoElement) {\n adVideoElement.pause();\n adVideoElement.src = \"\";\n }\n\n if (!options?.continueLiveStreamDuringAds) {\n contentVideo.play().catch(() => {});\n }\n\n currentAd = undefined;\n },\n\n destroy() {\n console.log(\"[HlsAdPlayer] Destroying\");\n adPlaying = false;\n contentVideo.muted = originalMutedState;\n\n if (adHls) {\n adHls.destroy();\n adHls = undefined;\n }\n\n if (adVideoElement) {\n adVideoElement.pause();\n adVideoElement.src = \"\";\n adVideoElement.remove();\n adVideoElement = undefined;\n }\n\n if (adContainerEl?.parentElement) {\n adContainerEl.parentElement.removeChild(adContainerEl);\n }\n\n adContainerEl = undefined;\n currentAd = undefined;\n listeners.clear();\n },\n\n isAdPlaying() {\n return adPlaying;\n },\n\n resize(width: number, height: number) {\n console.log(`[HlsAdPlayer] Resizing to ${width}x${height}`);\n\n if (adContainerEl) {\n adContainerEl.style.width = `${width}px`;\n adContainerEl.style.height = `${height}px`;\n }\n\n if (adVideoElement) {\n adVideoElement.style.width = `${width}px`;\n adVideoElement.style.height = `${height}px`;\n }\n },\n\n on(event: string, listener: (payload?: any) => void) {\n if (!listeners.has(event)) listeners.set(event, new Set());\n listeners.get(event)!.add(listener);\n },\n\n off(event: string, listener: (payload?: any) => void) {\n listeners.get(event)?.delete(listener);\n },\n\n updateOriginalMutedState(muted: boolean) {\n originalMutedState = muted;\n },\n\n getOriginalMutedState() {\n return originalMutedState;\n },\n\n setAdVolume(volume: number) {\n if (adVideoElement && adPlaying) {\n adVideoElement.volume = Math.max(0, Math.min(1, volume));\n }\n },\n\n getAdVolume(): number {\n if (adVideoElement && adPlaying) {\n return adVideoElement.volume;\n }\n return 1;\n },\n };\n}\n\n"],"mappings":"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","names":["level","_a","Hls"]}
@@ -0,0 +1,10 @@
1
+ import { I as ImaController } from '../types-mVgmKmzM.cjs';
2
+ import Hls from 'hls.js';
3
+
4
+ declare function createHlsAdPlayer(contentVideo: HTMLVideoElement, options?: {
5
+ continueLiveStreamDuringAds?: boolean;
6
+ licenseKey?: string;
7
+ mainHlsInstance?: Hls;
8
+ }): ImaController;
9
+
10
+ export { createHlsAdPlayer };