Source: lib/ads/interstitial_ad_manager.js

  1. /*! @license
  2. * Shaka Player
  3. * Copyright 2016 Google LLC
  4. * SPDX-License-Identifier: Apache-2.0
  5. */
  6. goog.provide('shaka.ads.InterstitialAdManager');
  7. goog.require('goog.asserts');
  8. goog.require('shaka.Player');
  9. goog.require('shaka.ads.InterstitialAd');
  10. goog.require('shaka.ads.Utils');
  11. goog.require('shaka.log');
  12. goog.require('shaka.media.PreloadManager');
  13. goog.require('shaka.net.NetworkingEngine');
  14. goog.require('shaka.util.Dom');
  15. goog.require('shaka.util.Error');
  16. goog.require('shaka.util.EventManager');
  17. goog.require('shaka.util.FakeEvent');
  18. goog.require('shaka.util.IReleasable');
  19. goog.require('shaka.util.Platform');
  20. goog.require('shaka.util.PublicPromise');
  21. goog.require('shaka.util.StringUtils');
  22. goog.require('shaka.util.Timer');
  23. goog.require('shaka.util.TXml');
  24. /**
  25. * A class responsible for Interstitial ad interactions.
  26. *
  27. * @implements {shaka.util.IReleasable}
  28. */
  29. shaka.ads.InterstitialAdManager = class {
  30. /**
  31. * @param {HTMLElement} adContainer
  32. * @param {shaka.Player} basePlayer
  33. * @param {HTMLMediaElement} baseVideo
  34. * @param {function(!shaka.util.FakeEvent)} onEvent
  35. */
  36. constructor(adContainer, basePlayer, baseVideo, onEvent) {
  37. /** @private {?shaka.extern.AdsConfiguration} */
  38. this.config_ = null;
  39. /** @private {HTMLElement} */
  40. this.adContainer_ = adContainer;
  41. /** @private {shaka.Player} */
  42. this.basePlayer_ = basePlayer;
  43. /** @private {HTMLMediaElement} */
  44. this.baseVideo_ = baseVideo;
  45. /** @private {?HTMLMediaElement} */
  46. this.adVideo_ = null;
  47. /** @private {boolean} */
  48. this.usingBaseVideo_ = true;
  49. /** @private {HTMLMediaElement} */
  50. this.video_ = this.baseVideo_;
  51. /** @private {function(!shaka.util.FakeEvent)} */
  52. this.onEvent_ = onEvent;
  53. /** @private {!Set.<string>} */
  54. this.interstitialIds_ = new Set();
  55. /** @private {!Set.<shaka.extern.AdInterstitial>} */
  56. this.interstitials_ = new Set();
  57. /**
  58. * @private {!Map.<shaka.extern.AdInterstitial,
  59. * Promise<?shaka.media.PreloadManager>>}
  60. */
  61. this.preloadManagerInterstitials_ = new Map();
  62. /** @private {shaka.Player} */
  63. this.player_ = new shaka.Player();
  64. this.updatePlayerConfig_();
  65. /** @private {shaka.util.EventManager} */
  66. this.eventManager_ = new shaka.util.EventManager();
  67. /** @private {shaka.util.EventManager} */
  68. this.adEventManager_ = new shaka.util.EventManager();
  69. /** @private {boolean} */
  70. this.playingAd_ = false;
  71. /** @private {?number} */
  72. this.lastTime_ = null;
  73. /** @private {?shaka.extern.AdInterstitial} */
  74. this.lastPlayedAd_ = null;
  75. this.eventManager_.listen(this.baseVideo_, 'timeupdate', () => {
  76. if (this.playingAd_ || this.lastTime_ ||
  77. this.basePlayer_.isRemotePlayback()) {
  78. return;
  79. }
  80. this.lastTime_ = this.baseVideo_.currentTime;
  81. const currentInterstitial = this.getCurrentInterstitial_(
  82. /* needPreRoll= */ true);
  83. if (currentInterstitial) {
  84. this.setupAd_(currentInterstitial, /* sequenceLength= */ 1,
  85. /* adPosition= */ 1, /* initialTime= */ Date.now());
  86. }
  87. });
  88. const checkForInterstitials = () => {
  89. if (this.playingAd_ || !this.lastTime_ ||
  90. this.basePlayer_.isRemotePlayback()) {
  91. return;
  92. }
  93. this.lastTime_ = this.baseVideo_.currentTime;
  94. // Remove last played add when the new time is before to the ad time.
  95. if (this.lastPlayedAd_ &&
  96. !this.lastPlayedAd_.pre && !this.lastPlayedAd_.post &&
  97. this.lastTime_ < this.lastPlayedAd_.startTime) {
  98. this.lastPlayedAd_ = null;
  99. }
  100. const currentInterstitial = this.getCurrentInterstitial_();
  101. if (currentInterstitial) {
  102. this.setupAd_(currentInterstitial, /* sequenceLength= */ 1,
  103. /* adPosition= */ 1, /* initialTime= */ Date.now());
  104. }
  105. };
  106. this.eventManager_.listen(this.baseVideo_, 'ended', () => {
  107. checkForInterstitials();
  108. });
  109. /** @private {shaka.util.Timer} */
  110. this.timeUpdateTimer_ = new shaka.util.Timer(checkForInterstitials);
  111. if ('requestVideoFrameCallback' in this.baseVideo_ &&
  112. !shaka.util.Platform.isSmartTV()) {
  113. const baseVideo = /** @type {!HTMLVideoElement} */ (this.baseVideo_);
  114. const videoFrameCallback = () => {
  115. checkForInterstitials();
  116. baseVideo.requestVideoFrameCallback(videoFrameCallback);
  117. };
  118. baseVideo.requestVideoFrameCallback(videoFrameCallback);
  119. } else {
  120. this.timeUpdateTimer_.tickEvery(/* seconds= */ 0.025);
  121. }
  122. /** @private {shaka.util.Timer} */
  123. this.pollTimer_ = new shaka.util.Timer(async () => {
  124. if (this.interstitials_.size && this.lastTime_ != null) {
  125. const currentLoadMode = this.basePlayer_.getLoadMode();
  126. if (currentLoadMode == shaka.Player.LoadMode.DESTROYED ||
  127. currentLoadMode == shaka.Player.LoadMode.NOT_LOADED) {
  128. return;
  129. }
  130. let cuepointsChanged = false;
  131. const interstitials = Array.from(this.interstitials_);
  132. const seekRange = this.basePlayer_.seekRange();
  133. for (const interstitial of interstitials) {
  134. if (interstitial == this.lastPlayedAd_) {
  135. continue;
  136. }
  137. const comparisonTime = interstitial.endTime || interstitial.startTime;
  138. if ((seekRange.start - comparisonTime) >= 1) {
  139. if (this.preloadManagerInterstitials_.has(interstitial)) {
  140. const preloadManager =
  141. // eslint-disable-next-line no-await-in-loop
  142. await this.preloadManagerInterstitials_.get(interstitial);
  143. if (preloadManager) {
  144. preloadManager.destroy();
  145. }
  146. this.preloadManagerInterstitials_.delete(interstitial);
  147. }
  148. const interstitialId = JSON.stringify(interstitial);
  149. if (this.interstitialIds_.has(interstitialId)) {
  150. this.interstitialIds_.delete(interstitialId);
  151. }
  152. this.interstitials_.delete(interstitial);
  153. cuepointsChanged = true;
  154. } else {
  155. const difference = interstitial.startTime - this.lastTime_;
  156. if (difference > 0 && difference <= 10) {
  157. if (!this.preloadManagerInterstitials_.has(interstitial)) {
  158. this.preloadManagerInterstitials_.set(
  159. interstitial, this.player_.preload(
  160. interstitial.uri,
  161. /* startTime= */ null,
  162. interstitial.mimeType || undefined));
  163. }
  164. }
  165. }
  166. }
  167. if (cuepointsChanged) {
  168. this.cuepointsChanged_();
  169. }
  170. }
  171. }).tickEvery(/* seconds= */ 1);
  172. }
  173. /**
  174. * Called by the AdManager to provide an updated configuration any time it
  175. * changes.
  176. *
  177. * @param {shaka.extern.AdsConfiguration} config
  178. */
  179. configure(config) {
  180. this.config_ = config;
  181. this.determineIfUsingBaseVideo_();
  182. }
  183. /**
  184. * @private
  185. */
  186. determineIfUsingBaseVideo_() {
  187. if (!this.adContainer_ || !this.config_ || this.playingAd_) {
  188. return;
  189. }
  190. let supportsMultipleMediaElements =
  191. this.config_.supportsMultipleMediaElements;
  192. const video = /** @type {HTMLVideoElement} */(this.baseVideo_);
  193. if (video.webkitSupportsFullscreen && video.webkitDisplayingFullscreen) {
  194. supportsMultipleMediaElements = false;
  195. }
  196. if (this.usingBaseVideo_ != supportsMultipleMediaElements) {
  197. return;
  198. }
  199. this.usingBaseVideo_ = !supportsMultipleMediaElements;
  200. if (this.usingBaseVideo_) {
  201. this.video_ = this.baseVideo_;
  202. if (this.adVideo_) {
  203. if (this.adVideo_.parentElement) {
  204. this.adContainer_.removeChild(this.adVideo_);
  205. }
  206. this.adVideo_ = null;
  207. }
  208. } else {
  209. if (!this.adVideo_) {
  210. this.adVideo_ = this.createMediaElement_();
  211. }
  212. this.video_ = this.adVideo_;
  213. }
  214. }
  215. /**
  216. * Resets the Interstitial manager and removes any continuous polling.
  217. */
  218. stop() {
  219. if (this.adEventManager_) {
  220. this.adEventManager_.removeAll();
  221. }
  222. this.interstitialIds_.clear();
  223. this.interstitials_.clear();
  224. this.player_.destroyAllPreloads();
  225. this.preloadManagerInterstitials_.clear();
  226. this.player_.detach();
  227. this.playingAd_ = false;
  228. this.lastTime_ = null;
  229. this.lastPlayedAd_ = null;
  230. }
  231. /** @override */
  232. release() {
  233. this.stop();
  234. if (this.eventManager_) {
  235. this.eventManager_.release();
  236. }
  237. if (this.adEventManager_) {
  238. this.adEventManager_.release();
  239. }
  240. if (this.adContainer_) {
  241. shaka.util.Dom.removeAllChildren(this.adContainer_);
  242. }
  243. if (this.timeUpdateTimer_) {
  244. this.timeUpdateTimer_.stop();
  245. this.timeUpdateTimer_ = null;
  246. }
  247. if (this.pollTimer_) {
  248. this.pollTimer_.stop();
  249. this.pollTimer_ = null;
  250. }
  251. this.player_.destroy();
  252. }
  253. /**
  254. * @param {shaka.extern.HLSInterstitial} hlsInterstitial
  255. */
  256. async addMetadata(hlsInterstitial) {
  257. this.updatePlayerConfig_();
  258. const adInterstitials = await this.getInterstitialsInfo_(hlsInterstitial);
  259. if (adInterstitials.length) {
  260. this.addInterstitials(adInterstitials);
  261. } else {
  262. shaka.log.alwaysWarn('Unsupported HLS interstitial', hlsInterstitial);
  263. }
  264. }
  265. /**
  266. * @param {shaka.extern.TimelineRegionInfo} region
  267. */
  268. addRegion(region) {
  269. let alternativeMPDUri;
  270. let alternativeMPDmode;
  271. for (const node of region.eventNode.children) {
  272. if (node.tagName == 'AlternativeMPD') {
  273. const uri = node.attributes['uri'];
  274. const mode = node.attributes['mode'];
  275. if (uri) {
  276. alternativeMPDUri = uri;
  277. alternativeMPDmode = mode;
  278. break;
  279. }
  280. }
  281. }
  282. if (!alternativeMPDUri) {
  283. shaka.log.alwaysWarn('Unsupported MPD alternate', region);
  284. return;
  285. }
  286. const isReplace = alternativeMPDmode == 'replace';
  287. const isInsert = alternativeMPDmode == 'insert';
  288. if (!isReplace && !isInsert) {
  289. shaka.log.warning('Unsupported MPD alternate', region);
  290. return;
  291. }
  292. /** @type {!shaka.extern.AdInterstitial} */
  293. const interstitial = {
  294. id: region.id,
  295. startTime: region.startTime,
  296. endTime: region.endTime,
  297. uri: alternativeMPDUri,
  298. mimeType: null,
  299. isSkippable: false,
  300. skipOffset: null,
  301. skipFor: null,
  302. canJump: true,
  303. resumeOffset: isInsert ? 0 : null,
  304. playoutLimit: null,
  305. once: false,
  306. pre: false,
  307. post: false,
  308. timelineRange: isReplace && !isInsert,
  309. };
  310. this.addInterstitials([interstitial]);
  311. }
  312. /**
  313. * @param {string} url
  314. * @return {!Promise}
  315. */
  316. async addAdUrlInterstitial(url) {
  317. const NetworkingEngine = shaka.net.NetworkingEngine;
  318. const context = {
  319. type: NetworkingEngine.AdvancedRequestType.INTERSTITIAL_AD_URL,
  320. };
  321. const responseData = await this.makeAdRequest_(url, context);
  322. const data = shaka.util.TXml.parseXml(responseData, 'VAST,vmap:VMAP');
  323. if (!data) {
  324. throw new shaka.util.Error(
  325. shaka.util.Error.Severity.CRITICAL,
  326. shaka.util.Error.Category.ADS,
  327. shaka.util.Error.Code.VAST_INVALID_XML);
  328. }
  329. let interstitials = [];
  330. if (data.tagName == 'VAST') {
  331. interstitials = shaka.ads.Utils.parseVastToInterstitials(
  332. data, this.lastTime_);
  333. } else if (data.tagName == 'vmap:VMAP') {
  334. for (const ad of shaka.ads.Utils.parseVMAP(data)) {
  335. // eslint-disable-next-line no-await-in-loop
  336. const vastResponseData = await this.makeAdRequest_(ad.uri, context);
  337. const vast = shaka.util.TXml.parseXml(vastResponseData, 'VAST');
  338. if (!vast) {
  339. throw new shaka.util.Error(
  340. shaka.util.Error.Severity.CRITICAL,
  341. shaka.util.Error.Category.ADS,
  342. shaka.util.Error.Code.VAST_INVALID_XML);
  343. }
  344. interstitials.push(...shaka.ads.Utils.parseVastToInterstitials(
  345. vast, ad.time));
  346. }
  347. }
  348. this.addInterstitials(interstitials);
  349. }
  350. /**
  351. * @param {!Array.<shaka.extern.AdInterstitial>} interstitials
  352. */
  353. addInterstitials(interstitials) {
  354. let cuepointsChanged = false;
  355. for (const interstitial of interstitials) {
  356. const interstitialId = interstitial.id || JSON.stringify(interstitial);
  357. if (this.interstitialIds_.has(interstitialId)) {
  358. continue;
  359. }
  360. cuepointsChanged = true;
  361. this.interstitialIds_.add(interstitialId);
  362. this.interstitials_.add(interstitial);
  363. let shouldPreload = false;
  364. if (interstitial.pre && this.lastTime_ == null) {
  365. shouldPreload = true;
  366. } else if (interstitial.startTime == 0 && !interstitial.canJump) {
  367. shouldPreload = true;
  368. } else if (this.lastTime_ != null) {
  369. const difference = interstitial.startTime - this.lastTime_;
  370. if (difference > 0 && difference <= 10) {
  371. shouldPreload = true;
  372. }
  373. }
  374. if (shouldPreload) {
  375. if (!this.preloadManagerInterstitials_.has(interstitial)) {
  376. this.preloadManagerInterstitials_.set(
  377. interstitial, this.player_.preload(
  378. interstitial.uri,
  379. /* startTime= */ null,
  380. interstitial.mimeType || undefined));
  381. }
  382. }
  383. }
  384. if (cuepointsChanged) {
  385. this.cuepointsChanged_();
  386. }
  387. }
  388. /**
  389. * @return {!HTMLMediaElement}
  390. * @private
  391. */
  392. createMediaElement_() {
  393. const video = /** @type {!HTMLMediaElement} */(
  394. document.createElement(this.baseVideo_.tagName));
  395. video.autoplay = true;
  396. video.style.position = 'absolute';
  397. video.style.top = '0';
  398. video.style.left = '0';
  399. video.style.width = '100%';
  400. video.style.height = '100%';
  401. video.style.backgroundColor = 'rgb(0, 0, 0)';
  402. video.setAttribute('playsinline', '');
  403. return video;
  404. }
  405. /**
  406. * @param {boolean=} needPreRoll
  407. * @param {?number=} numberToSkip
  408. * @return {?shaka.extern.AdInterstitial}
  409. * @private
  410. */
  411. getCurrentInterstitial_(needPreRoll = false, numberToSkip = null) {
  412. let skipped = 0;
  413. let currentInterstitial = null;
  414. if (this.interstitials_.size && this.lastTime_ != null) {
  415. const isEnded = this.baseVideo_.ended;
  416. const interstitials = Array.from(this.interstitials_).sort((a, b) => {
  417. return b.startTime - a.startTime;
  418. });
  419. const roundDecimals = (number) => {
  420. return Math.round(number * 1000) / 1000;
  421. };
  422. let interstitialsToCheck = interstitials;
  423. if (needPreRoll) {
  424. interstitialsToCheck = interstitials.filter((i) => i.pre);
  425. } else if (isEnded) {
  426. interstitialsToCheck = interstitials.filter((i) => i.post);
  427. } else {
  428. interstitialsToCheck = interstitials.filter((i) => !i.pre && !i.post);
  429. }
  430. for (const interstitial of interstitialsToCheck) {
  431. let isValid = false;
  432. if (needPreRoll) {
  433. isValid = interstitial.pre;
  434. } else if (isEnded) {
  435. isValid = interstitial.post;
  436. } else if (!interstitial.pre && !interstitial.post) {
  437. const difference =
  438. this.lastTime_ - roundDecimals(interstitial.startTime);
  439. if (difference > 0 &&
  440. (difference <= 1 || !interstitial.canJump)) {
  441. if (numberToSkip == null && this.lastPlayedAd_ &&
  442. !this.lastPlayedAd_.pre && !this.lastPlayedAd_.post &&
  443. this.lastPlayedAd_.startTime >= interstitial.startTime) {
  444. isValid = false;
  445. } else {
  446. isValid = true;
  447. }
  448. }
  449. }
  450. if (isValid && (!this.lastPlayedAd_ ||
  451. interstitial.startTime >= this.lastPlayedAd_.startTime)) {
  452. if (skipped == (numberToSkip || 0)) {
  453. currentInterstitial = interstitial;
  454. } else if (currentInterstitial && !interstitial.canJump) {
  455. const currentStartTime =
  456. roundDecimals(currentInterstitial.startTime);
  457. const newStartTime =
  458. roundDecimals(interstitial.startTime);
  459. if (newStartTime - currentStartTime > 0.001) {
  460. currentInterstitial = interstitial;
  461. skipped = 0;
  462. }
  463. }
  464. skipped++;
  465. }
  466. }
  467. }
  468. return currentInterstitial;
  469. }
  470. /**
  471. * @param {shaka.extern.AdInterstitial} interstitial
  472. * @param {number} sequenceLength
  473. * @param {number} adPosition
  474. * @param {number} initialTime the clock time the ad started at
  475. * @param {number=} oncePlayed
  476. * @private
  477. */
  478. async setupAd_(interstitial, sequenceLength, adPosition, initialTime,
  479. oncePlayed = 0) {
  480. this.determineIfUsingBaseVideo_();
  481. goog.asserts.assert(this.video_, 'Must have video');
  482. this.lastPlayedAd_ = interstitial;
  483. shaka.log.info('Starting interstitial',
  484. interstitial.startTime, 'at', this.lastTime_);
  485. const startTime = Date.now();
  486. if (!this.video_.parentElement && this.adContainer_) {
  487. this.adContainer_.appendChild(this.video_);
  488. }
  489. if (adPosition == 1 && sequenceLength == 1) {
  490. sequenceLength = Array.from(this.interstitials_).filter((i) => {
  491. if (interstitial.pre) {
  492. return i.pre == interstitial.pre;
  493. } else if (interstitial.post) {
  494. return i.post == interstitial.post;
  495. }
  496. return Math.abs(i.startTime - interstitial.startTime) < 0.001;
  497. }).length;
  498. }
  499. if (interstitial.once) {
  500. oncePlayed++;
  501. this.interstitials_.delete(interstitial);
  502. this.cuepointsChanged_();
  503. }
  504. this.playingAd_ = true;
  505. if (this.usingBaseVideo_ && adPosition == 1) {
  506. this.onEvent_(new shaka.util.FakeEvent(
  507. shaka.ads.Utils.AD_CONTENT_PAUSE_REQUESTED,
  508. (new Map()).set('saveLivePosition', true)));
  509. const detachBasePlayerPromise = new shaka.util.PublicPromise();
  510. const checkState = async (e) => {
  511. if (e['state'] == 'detach') {
  512. if (shaka.util.Platform.isSmartTV()) {
  513. await new Promise(
  514. (resolve) => new shaka.util.Timer(resolve).tickAfter(0.1));
  515. }
  516. detachBasePlayerPromise.resolve();
  517. this.adEventManager_.unlisten(
  518. this.basePlayer_, 'onstatechange', checkState);
  519. }
  520. };
  521. this.adEventManager_.listen(
  522. this.basePlayer_, 'onstatechange', checkState);
  523. await detachBasePlayerPromise;
  524. }
  525. if (!this.usingBaseVideo_) {
  526. this.baseVideo_.pause();
  527. if (interstitial.resumeOffset != null &&
  528. interstitial.resumeOffset != 0) {
  529. this.baseVideo_.currentTime += interstitial.resumeOffset;
  530. }
  531. }
  532. let unloadingInterstitial = false;
  533. /** @type {?shaka.util.Timer} */
  534. let playoutLimitTimer = null;
  535. const updateBaseVideoTime = () => {
  536. if (!this.usingBaseVideo_) {
  537. if (interstitial.resumeOffset == null) {
  538. if (interstitial.timelineRange && interstitial.endTime &&
  539. interstitial.endTime != Infinity) {
  540. if (this.baseVideo_.currentTime != interstitial.endTime) {
  541. this.baseVideo_.currentTime = interstitial.endTime;
  542. }
  543. } else {
  544. const now = Date.now();
  545. this.baseVideo_.currentTime += (now - initialTime) / 1000;
  546. initialTime = now;
  547. }
  548. }
  549. }
  550. };
  551. const basicTask = async () => {
  552. updateBaseVideoTime();
  553. if (playoutLimitTimer) {
  554. playoutLimitTimer.stop();
  555. }
  556. goog.asserts.assert(typeof(oncePlayed) == 'number',
  557. 'Should be an number!');
  558. // Optimization to avoid returning to main content when there is another
  559. // interstitial below.
  560. const nextCurrentInterstitial = this.getCurrentInterstitial_(
  561. interstitial.pre, adPosition - oncePlayed);
  562. if (nextCurrentInterstitial) {
  563. this.onEvent_(
  564. new shaka.util.FakeEvent(shaka.ads.Utils.AD_STOPPED));
  565. this.adEventManager_.removeAll();
  566. this.setupAd_(nextCurrentInterstitial, sequenceLength,
  567. ++adPosition, initialTime, oncePlayed);
  568. } else {
  569. if (interstitial.post) {
  570. this.lastTime_ = null;
  571. this.lastPlayedAd_ = null;
  572. }
  573. if (this.usingBaseVideo_) {
  574. await this.player_.detach();
  575. } else {
  576. await this.player_.unload();
  577. }
  578. if (this.usingBaseVideo_) {
  579. let offset = interstitial.resumeOffset;
  580. if (offset == null) {
  581. if (interstitial.timelineRange && interstitial.endTime &&
  582. interstitial.endTime != Infinity) {
  583. offset = interstitial.endTime - (this.lastTime_ || 0);
  584. } else {
  585. offset = (Date.now() - initialTime) / 1000;
  586. }
  587. }
  588. this.onEvent_(new shaka.util.FakeEvent(
  589. shaka.ads.Utils.AD_CONTENT_RESUME_REQUESTED,
  590. (new Map()).set('offset', offset)));
  591. }
  592. this.onEvent_(
  593. new shaka.util.FakeEvent(shaka.ads.Utils.AD_STOPPED));
  594. this.adEventManager_.removeAll();
  595. this.playingAd_ = false;
  596. if (!this.usingBaseVideo_) {
  597. updateBaseVideoTime();
  598. if (!this.baseVideo_.ended) {
  599. this.baseVideo_.play();
  600. }
  601. } else {
  602. this.cuepointsChanged_();
  603. }
  604. this.determineIfUsingBaseVideo_();
  605. }
  606. };
  607. const error = async (e) => {
  608. if (unloadingInterstitial) {
  609. return;
  610. }
  611. unloadingInterstitial = true;
  612. this.onEvent_(new shaka.util.FakeEvent(shaka.ads.Utils.AD_ERROR,
  613. (new Map()).set('originalEvent', e)));
  614. await basicTask();
  615. };
  616. const complete = async () => {
  617. if (unloadingInterstitial) {
  618. return;
  619. }
  620. unloadingInterstitial = true;
  621. await basicTask();
  622. this.onEvent_(
  623. new shaka.util.FakeEvent(shaka.ads.Utils.AD_COMPLETE));
  624. };
  625. const onSkip = async () => {
  626. if (unloadingInterstitial) {
  627. return;
  628. }
  629. unloadingInterstitial = true;
  630. this.onEvent_(new shaka.util.FakeEvent(shaka.ads.Utils.AD_SKIPPED));
  631. await basicTask();
  632. };
  633. const ad = new shaka.ads.InterstitialAd(this.video_,
  634. interstitial.isSkippable, interstitial.skipOffset,
  635. interstitial.skipFor, onSkip, sequenceLength, adPosition,
  636. !this.usingBaseVideo_);
  637. if (!this.usingBaseVideo_) {
  638. ad.setMuted(this.baseVideo_.muted);
  639. ad.setVolume(this.baseVideo_.volume);
  640. }
  641. this.onEvent_(
  642. new shaka.util.FakeEvent(shaka.ads.Utils.AD_STARTED,
  643. (new Map()).set('ad', ad)));
  644. let prevCanSkipNow = ad.canSkipNow();
  645. if (prevCanSkipNow) {
  646. this.onEvent_(new shaka.util.FakeEvent(
  647. shaka.ads.Utils.AD_SKIP_STATE_CHANGED));
  648. }
  649. this.adEventManager_.listenOnce(this.player_, 'error', error);
  650. this.adEventManager_.listen(this.video_, 'timeupdate', () => {
  651. const duration = this.video_.duration;
  652. if (!duration) {
  653. return;
  654. }
  655. const currentCanSkipNow = ad.canSkipNow();
  656. if (prevCanSkipNow != currentCanSkipNow &&
  657. ad.getRemainingTime() > 0 && ad.getDuration() > 0) {
  658. this.onEvent_(
  659. new shaka.util.FakeEvent(shaka.ads.Utils.AD_SKIP_STATE_CHANGED));
  660. }
  661. prevCanSkipNow = currentCanSkipNow;
  662. });
  663. this.adEventManager_.listenOnce(this.player_, 'firstquartile', () => {
  664. updateBaseVideoTime();
  665. this.onEvent_(
  666. new shaka.util.FakeEvent(shaka.ads.Utils.AD_FIRST_QUARTILE));
  667. });
  668. this.adEventManager_.listenOnce(this.player_, 'midpoint', () => {
  669. updateBaseVideoTime();
  670. this.onEvent_(
  671. new shaka.util.FakeEvent(shaka.ads.Utils.AD_MIDPOINT));
  672. });
  673. this.adEventManager_.listenOnce(this.player_, 'thirdquartile', () => {
  674. updateBaseVideoTime();
  675. this.onEvent_(
  676. new shaka.util.FakeEvent(shaka.ads.Utils.AD_THIRD_QUARTILE));
  677. });
  678. this.adEventManager_.listenOnce(this.player_, 'complete', complete);
  679. this.adEventManager_.listen(this.video_, 'play', () => {
  680. this.onEvent_(
  681. new shaka.util.FakeEvent(shaka.ads.Utils.AD_RESUMED));
  682. });
  683. this.adEventManager_.listen(this.video_, 'pause', () => {
  684. // playRangeEnd in src= causes the ended event not to be fired when that
  685. // position is reached, instead pause event is fired.
  686. const currentConfig = this.player_.getConfiguration();
  687. if (this.video_.currentTime >= currentConfig.playRangeEnd) {
  688. complete();
  689. return;
  690. }
  691. this.onEvent_(
  692. new shaka.util.FakeEvent(shaka.ads.Utils.AD_PAUSED));
  693. });
  694. this.adEventManager_.listen(this.video_, 'volumechange', () => {
  695. if (this.video_.muted) {
  696. this.onEvent_(
  697. new shaka.util.FakeEvent(shaka.ads.Utils.AD_MUTED));
  698. } else {
  699. this.onEvent_(
  700. new shaka.util.FakeEvent(shaka.ads.Utils.AD_VOLUME_CHANGED));
  701. }
  702. });
  703. try {
  704. this.updatePlayerConfig_();
  705. if (interstitial.startTime && interstitial.endTime &&
  706. interstitial.endTime != Infinity &&
  707. interstitial.startTime != interstitial.endTime) {
  708. const duration = interstitial.endTime - interstitial.startTime;
  709. if (duration > 0) {
  710. this.player_.configure('playRangeEnd', duration);
  711. }
  712. }
  713. if (interstitial.playoutLimit) {
  714. playoutLimitTimer = new shaka.util.Timer(() => {
  715. ad.skip();
  716. }).tickAfter(interstitial.playoutLimit);
  717. this.player_.configure('playRangeEnd', interstitial.playoutLimit);
  718. }
  719. await this.player_.attach(this.video_);
  720. if (this.preloadManagerInterstitials_.has(interstitial)) {
  721. const preloadManager =
  722. await this.preloadManagerInterstitials_.get(interstitial);
  723. this.preloadManagerInterstitials_.delete(interstitial);
  724. if (preloadManager) {
  725. await this.player_.load(preloadManager);
  726. } else {
  727. await this.player_.load(
  728. interstitial.uri,
  729. /* startTime= */ null,
  730. interstitial.mimeType || undefined);
  731. }
  732. } else {
  733. await this.player_.load(
  734. interstitial.uri,
  735. /* startTime= */ null,
  736. interstitial.mimeType || undefined);
  737. }
  738. if (interstitial.playoutLimit) {
  739. if (playoutLimitTimer) {
  740. playoutLimitTimer.stop();
  741. }
  742. playoutLimitTimer = new shaka.util.Timer(() => {
  743. ad.skip();
  744. }).tickAfter(interstitial.playoutLimit);
  745. }
  746. const loadTime = (Date.now() - startTime) / 1000;
  747. this.onEvent_(new shaka.util.FakeEvent(shaka.ads.Utils.ADS_LOADED,
  748. (new Map()).set('loadTime', loadTime)));
  749. if (this.usingBaseVideo_) {
  750. this.baseVideo_.play();
  751. }
  752. } catch (e) {
  753. error(e);
  754. }
  755. }
  756. /**
  757. * @param {shaka.extern.HLSInterstitial} hlsInterstitial
  758. * @return {!Promise.<!Array.<shaka.extern.AdInterstitial>>}
  759. * @private
  760. */
  761. async getInterstitialsInfo_(hlsInterstitial) {
  762. const interstitialsAd = [];
  763. if (!hlsInterstitial) {
  764. return interstitialsAd;
  765. }
  766. const assetUri = hlsInterstitial.values.find((v) => v.key == 'X-ASSET-URI');
  767. const assetList =
  768. hlsInterstitial.values.find((v) => v.key == 'X-ASSET-LIST');
  769. if (!assetUri && !assetList) {
  770. return interstitialsAd;
  771. }
  772. let id = null;
  773. const hlsInterstitialId = hlsInterstitial.values.find((v) => v.key == 'ID');
  774. if (hlsInterstitialId) {
  775. id = /** @type {string} */(hlsInterstitialId.data);
  776. }
  777. const startTime = id == null ?
  778. Math.floor(hlsInterstitial.startTime * 10) / 10:
  779. hlsInterstitial.startTime;
  780. let endTime = hlsInterstitial.endTime;
  781. if (hlsInterstitial.endTime && hlsInterstitial.endTime != Infinity &&
  782. typeof(hlsInterstitial.endTime) == 'number') {
  783. endTime = id == null ?
  784. Math.floor(hlsInterstitial.endTime * 10) / 10:
  785. hlsInterstitial.endTime;
  786. }
  787. const restrict = hlsInterstitial.values.find((v) => v.key == 'X-RESTRICT');
  788. let isSkippable = true;
  789. let canJump = true;
  790. if (restrict && restrict.data) {
  791. const data = /** @type {string} */(restrict.data);
  792. isSkippable = !data.includes('SKIP');
  793. canJump = !data.includes('JUMP');
  794. }
  795. let skipOffset = isSkippable ? 0 : null;
  796. const enableSkipAfter =
  797. hlsInterstitial.values.find((v) => v.key == 'X-ENABLE-SKIP-AFTER');
  798. if (enableSkipAfter) {
  799. const enableSkipAfterString = /** @type {string} */(enableSkipAfter.data);
  800. skipOffset = parseFloat(enableSkipAfterString);
  801. if (isNaN(skipOffset)) {
  802. skipOffset = isSkippable ? 0 : null;
  803. }
  804. }
  805. let skipFor = null;
  806. const enableSkipFor =
  807. hlsInterstitial.values.find((v) => v.key == 'X-ENABLE-SKIP-FOR');
  808. if (enableSkipFor) {
  809. const enableSkipForString = /** @type {string} */(enableSkipFor.data);
  810. skipFor = parseFloat(enableSkipForString);
  811. if (isNaN(skipOffset)) {
  812. skipFor = null;
  813. }
  814. }
  815. let resumeOffset = null;
  816. const resume =
  817. hlsInterstitial.values.find((v) => v.key == 'X-RESUME-OFFSET');
  818. if (resume) {
  819. const resumeOffsetString = /** @type {string} */(resume.data);
  820. resumeOffset = parseFloat(resumeOffsetString);
  821. if (isNaN(resumeOffset)) {
  822. resumeOffset = null;
  823. }
  824. }
  825. let playoutLimit = null;
  826. const playout =
  827. hlsInterstitial.values.find((v) => v.key == 'X-PLAYOUT-LIMIT');
  828. if (playout) {
  829. const playoutLimitString = /** @type {string} */(playout.data);
  830. playoutLimit = parseFloat(playoutLimitString);
  831. if (isNaN(playoutLimit)) {
  832. playoutLimit = null;
  833. }
  834. }
  835. let once = false;
  836. let pre = false;
  837. let post = false;
  838. const cue = hlsInterstitial.values.find((v) => v.key == 'CUE');
  839. if (cue) {
  840. const data = /** @type {string} */(cue.data);
  841. once = data.includes('ONCE');
  842. pre = data.includes('PRE');
  843. post = data.includes('POST');
  844. }
  845. let timelineRange = false;
  846. const timelineOccupies =
  847. hlsInterstitial.values.find((v) => v.key == 'X-TIMELINE-OCCUPIES');
  848. if (timelineOccupies) {
  849. const data = /** @type {string} */(timelineOccupies.data);
  850. timelineRange = data.includes('RANGE');
  851. } else if (!resume && this.basePlayer_.isLive()) {
  852. timelineRange = !pre && !post;
  853. }
  854. if (assetUri) {
  855. const uri = /** @type {string} */(assetUri.data);
  856. if (!uri) {
  857. return interstitialsAd;
  858. }
  859. interstitialsAd.push({
  860. id,
  861. startTime,
  862. endTime,
  863. uri,
  864. mimeType: null,
  865. isSkippable,
  866. skipOffset,
  867. skipFor,
  868. canJump,
  869. resumeOffset,
  870. playoutLimit,
  871. once,
  872. pre,
  873. post,
  874. timelineRange,
  875. });
  876. } else if (assetList) {
  877. const uri = /** @type {string} */(assetList.data);
  878. if (!uri) {
  879. return interstitialsAd;
  880. }
  881. try {
  882. const NetworkingEngine = shaka.net.NetworkingEngine;
  883. const context = {
  884. type: NetworkingEngine.AdvancedRequestType.INTERSTITIAL_ASSET_LIST,
  885. };
  886. const responseData = await this.makeAdRequest_(uri, context);
  887. const data = shaka.util.StringUtils.fromUTF8(responseData);
  888. const dataAsJson =
  889. /** @type {!shaka.ads.InterstitialAdManager.AssetsList} */ (
  890. JSON.parse(data));
  891. const skipControl = dataAsJson['SKIP-CONTROL'];
  892. if (skipControl) {
  893. const enableSkipAfterValue = skipControl['ENABLE-SKIP-AFTER'];
  894. if ((typeof enableSkipAfterValue) == 'number') {
  895. skipOffset = parseFloat(enableSkipAfterValue);
  896. if (isNaN(enableSkipAfterValue)) {
  897. skipOffset = isSkippable ? 0 : null;
  898. }
  899. }
  900. const enableSkipForValue = skipControl['ENABLE-SKIP-FOR'];
  901. if ((typeof enableSkipForValue) == 'number') {
  902. skipFor = parseFloat(enableSkipForValue);
  903. if (isNaN(enableSkipForValue)) {
  904. skipFor = null;
  905. }
  906. }
  907. }
  908. for (let i = 0; i < dataAsJson['ASSETS'].length; i++) {
  909. const asset = dataAsJson['ASSETS'][i];
  910. if (asset['URI']) {
  911. interstitialsAd.push({
  912. id: id + '_asset_' + i,
  913. startTime,
  914. endTime,
  915. uri: asset['URI'],
  916. mimeType: null,
  917. isSkippable,
  918. skipOffset,
  919. skipFor,
  920. canJump,
  921. resumeOffset,
  922. playoutLimit,
  923. once,
  924. pre,
  925. post,
  926. timelineRange,
  927. });
  928. }
  929. }
  930. } catch (e) {
  931. // Ignore errors
  932. }
  933. }
  934. return interstitialsAd;
  935. }
  936. /**
  937. * @private
  938. */
  939. cuepointsChanged_() {
  940. /** @type {!Array.<!shaka.extern.AdCuePoint>} */
  941. const cuePoints = [];
  942. for (const interstitial of this.interstitials_) {
  943. /** @type {shaka.extern.AdCuePoint} */
  944. const shakaCuePoint = {
  945. start: interstitial.startTime,
  946. end: null,
  947. };
  948. if (interstitial.pre) {
  949. shakaCuePoint.start = 0;
  950. shakaCuePoint.end = null;
  951. } else if (interstitial.post) {
  952. shakaCuePoint.start = -1;
  953. shakaCuePoint.end = null;
  954. } else if (interstitial.timelineRange) {
  955. shakaCuePoint.end = interstitial.endTime;
  956. }
  957. const isValid = !cuePoints.find((c) => {
  958. return shakaCuePoint.start == c.start && shakaCuePoint.end == c.end;
  959. });
  960. if (isValid) {
  961. cuePoints.push(shakaCuePoint);
  962. }
  963. }
  964. this.onEvent_(new shaka.util.FakeEvent(
  965. shaka.ads.Utils.CUEPOINTS_CHANGED,
  966. (new Map()).set('cuepoints', cuePoints)));
  967. }
  968. /**
  969. * @private
  970. */
  971. updatePlayerConfig_() {
  972. goog.asserts.assert(this.player_, 'Must have player');
  973. goog.asserts.assert(this.basePlayer_, 'Must have base player');
  974. this.player_.configure(this.basePlayer_.getNonDefaultConfiguration());
  975. this.player_.configure('ads.disableHLSInterstitial', true);
  976. this.player_.configure('ads.disableDASHInterstitial', true);
  977. this.player_.configure('playRangeEnd', Infinity);
  978. const netEngine = this.player_.getNetworkingEngine();
  979. goog.asserts.assert(netEngine, 'Need networking engine');
  980. netEngine.clearAllRequestFilters();
  981. netEngine.clearAllResponseFilters();
  982. this.basePlayer_.getNetworkingEngine().copyFiltersInto(netEngine);
  983. }
  984. /**
  985. * @param {string} url
  986. * @param {shaka.extern.RequestContext=} context
  987. * @return {!Promise.<BufferSource>}
  988. * @private
  989. */
  990. async makeAdRequest_(url, context) {
  991. const type = shaka.net.NetworkingEngine.RequestType.ADS;
  992. const request = shaka.net.NetworkingEngine.makeRequest(
  993. [url],
  994. shaka.net.NetworkingEngine.defaultRetryParameters());
  995. const op = this.basePlayer_.getNetworkingEngine()
  996. .request(type, request, context);
  997. const response = await op.promise;
  998. return response.data;
  999. }
  1000. };
  1001. /**
  1002. * @typedef {{
  1003. * ASSETS: !Array.<shaka.ads.InterstitialAdManager.Asset>,
  1004. * SKIP-CONTROL: ?shaka.ads.InterstitialAdManager.SkipControl
  1005. * }}
  1006. *
  1007. * @property {!Array.<shaka.ads.InterstitialAdManager.Asset>} ASSETS
  1008. * @property {shaka.ads.InterstitialAdManager.SkipControl} SKIP-CONTROL
  1009. */
  1010. shaka.ads.InterstitialAdManager.AssetsList;
  1011. /**
  1012. * @typedef {{
  1013. * URI: string
  1014. * }}
  1015. *
  1016. * @property {string} URI
  1017. */
  1018. shaka.ads.InterstitialAdManager.Asset;
  1019. /**
  1020. * @typedef {{
  1021. * ENABLE-SKIP-AFTER: number,
  1022. * ENABLE-SKIP-FOR: number
  1023. * }}
  1024. *
  1025. * @property {number} ENABLE-SKIP-AFTER
  1026. * @property {number} ENABLE-SKIP-FOR
  1027. */
  1028. shaka.ads.InterstitialAdManager.SkipControl;