Source: lib/ads/ad_utils.js

  1. /*! @license
  2. * Shaka Player
  3. * Copyright 2016 Google LLC
  4. * SPDX-License-Identifier: Apache-2.0
  5. */
  6. goog.provide('shaka.ads.Utils');
  7. goog.require('shaka.util.TextParser');
  8. goog.require('shaka.util.TXml');
  9. /**
  10. * A class responsible for ad utils.
  11. * @export
  12. */
  13. shaka.ads.Utils = class {
  14. /**
  15. * @param {!shaka.extern.xml.Node} vast
  16. * @param {?number} currentTime
  17. * @return {!Array.<shaka.extern.AdInterstitial>}
  18. */
  19. static parseVastToInterstitials(vast, currentTime) {
  20. const TXml = shaka.util.TXml;
  21. /** @type {!Array.<shaka.extern.AdInterstitial>} */
  22. const interstitials = [];
  23. let startTime = 0;
  24. if (currentTime != null) {
  25. startTime = currentTime;
  26. }
  27. for (const ad of TXml.findChildren(vast, 'Ad')) {
  28. const inline = TXml.findChild(ad, 'InLine');
  29. if (!inline) {
  30. continue;
  31. }
  32. const creatives = TXml.findChild(inline, 'Creatives');
  33. if (!creatives) {
  34. continue;
  35. }
  36. for (const creative of TXml.findChildren(creatives, 'Creative')) {
  37. const linear = TXml.findChild(creative, 'Linear');
  38. if (!linear) {
  39. continue;
  40. }
  41. let skipOffset = null;
  42. if (linear.attributes['skipoffset']) {
  43. skipOffset = shaka.util.TextParser.parseTime(
  44. linear.attributes['skipoffset']);
  45. if (isNaN(skipOffset)) {
  46. skipOffset = null;
  47. }
  48. }
  49. const mediaFiles = TXml.findChild(linear, 'MediaFiles');
  50. if (!mediaFiles) {
  51. continue;
  52. }
  53. const medias = TXml.findChildren(mediaFiles, 'MediaFile');
  54. let checkMedias = medias;
  55. const streamingMedias = medias.filter((media) => {
  56. return media.attributes['delivery'] == 'streaming';
  57. });
  58. if (streamingMedias.length) {
  59. checkMedias = streamingMedias;
  60. }
  61. const sortedMedias = checkMedias.sort((a, b) => {
  62. const aHeight = parseInt(a.attributes['height'], 10) || 0;
  63. const bHeight = parseInt(b.attributes['height'], 10) || 0;
  64. return bHeight - aHeight;
  65. });
  66. for (const media of sortedMedias) {
  67. const adUrl = TXml.getTextContents(media);
  68. if (!adUrl) {
  69. continue;
  70. }
  71. interstitials.push({
  72. id: null,
  73. startTime: startTime,
  74. endTime: null,
  75. uri: adUrl,
  76. isSkippable: skipOffset != null,
  77. skipOffset,
  78. canJump: false,
  79. resumeOffset: 0,
  80. playoutLimit: null,
  81. once: true,
  82. pre: currentTime == null,
  83. post: currentTime == Infinity,
  84. timelineRange: false,
  85. });
  86. break;
  87. }
  88. }
  89. }
  90. return interstitials;
  91. }
  92. /**
  93. * @param {!shaka.extern.xml.Node} vmap
  94. * @return {!Array.<{time: ?number, uri: string}>}
  95. */
  96. static parseVMAP(vmap) {
  97. const TXml = shaka.util.TXml;
  98. /** @type {!Array.<{time: ?number, uri: string}>} */
  99. const ads = [];
  100. for (const adBreak of TXml.findChildren(vmap, 'vmap:AdBreak')) {
  101. const timeOffset = adBreak.attributes['timeOffset'];
  102. if (!timeOffset) {
  103. continue;
  104. }
  105. let time = null;
  106. if (timeOffset == 'start') {
  107. time = 0;
  108. } else if (timeOffset == 'end') {
  109. time = Infinity;
  110. } else {
  111. time = shaka.util.TextParser.parseTime(timeOffset);
  112. }
  113. const adSource = TXml.findChild(adBreak, 'vmap:AdSource');
  114. if (!adSource) {
  115. continue;
  116. }
  117. const adTagURI = TXml.findChild(adSource, 'vmap:AdTagURI');
  118. if (!adTagURI) {
  119. continue;
  120. }
  121. const uri = TXml.getTextContents(adTagURI);
  122. if (!uri) {
  123. continue;
  124. }
  125. ads.push({
  126. time,
  127. uri,
  128. });
  129. }
  130. return ads;
  131. }
  132. };
  133. /**
  134. * The event name for when a sequence of ads has been loaded.
  135. *
  136. * @const {string}
  137. * @export
  138. */
  139. shaka.ads.Utils.ADS_LOADED = 'ads-loaded';
  140. /**
  141. * The event name for when an ad has started playing.
  142. *
  143. * @const {string}
  144. * @export
  145. */
  146. shaka.ads.Utils.AD_STARTED = 'ad-started';
  147. /**
  148. * The event name for when an ad playhead crosses first quartile.
  149. *
  150. * @const {string}
  151. * @export
  152. */
  153. shaka.ads.Utils.AD_FIRST_QUARTILE = 'ad-first-quartile';
  154. /**
  155. * The event name for when an ad playhead crosses midpoint.
  156. *
  157. * @const {string}
  158. * @export
  159. */
  160. shaka.ads.Utils.AD_MIDPOINT = 'ad-midpoint';
  161. /**
  162. * The event name for when an ad playhead crosses third quartile.
  163. *
  164. * @const {string}
  165. * @export
  166. */
  167. shaka.ads.Utils.AD_THIRD_QUARTILE = 'ad-third-quartile';
  168. /**
  169. * The event name for when an ad has completed playing.
  170. *
  171. * @const {string}
  172. * @export
  173. */
  174. shaka.ads.Utils.AD_COMPLETE = 'ad-complete';
  175. /**
  176. * The event name for when an ad has finished playing
  177. * (played all the way through, was skipped, or was unable to proceed
  178. * due to an error).
  179. *
  180. * @const {string}
  181. * @export
  182. */
  183. shaka.ads.Utils.AD_STOPPED = 'ad-stopped';
  184. /**
  185. * The event name for when an ad is skipped by the user..
  186. *
  187. * @const {string}
  188. * @export
  189. */
  190. shaka.ads.Utils.AD_SKIPPED = 'ad-skipped';
  191. /**
  192. * The event name for when the ad volume has changed.
  193. *
  194. * @const {string}
  195. * @export
  196. */
  197. shaka.ads.Utils.AD_VOLUME_CHANGED = 'ad-volume-changed';
  198. /**
  199. * The event name for when the ad was muted.
  200. *
  201. * @const {string}
  202. * @export
  203. */
  204. shaka.ads.Utils.AD_MUTED = 'ad-muted';
  205. /**
  206. * The event name for when the ad was paused.
  207. *
  208. * @const {string}
  209. * @export
  210. */
  211. shaka.ads.Utils.AD_PAUSED = 'ad-paused';
  212. /**
  213. * The event name for when the ad was resumed after a pause.
  214. *
  215. * @const {string}
  216. * @export
  217. */
  218. shaka.ads.Utils.AD_RESUMED = 'ad-resumed';
  219. /**
  220. * The event name for when the ad's skip status changes
  221. * (usually it becomes skippable when it wasn't before).
  222. *
  223. * @const {string}
  224. * @export
  225. */
  226. shaka.ads.Utils.AD_SKIP_STATE_CHANGED = 'ad-skip-state-changed';
  227. /**
  228. * The event name for when the ad's cue points (start/end markers)
  229. * have changed.
  230. *
  231. * @const {string}
  232. * @export
  233. */
  234. shaka.ads.Utils.CUEPOINTS_CHANGED = 'ad-cue-points-changed';
  235. /**
  236. * The event name for when the native IMA ad manager object has
  237. * loaded and become available.
  238. *
  239. * @const {string}
  240. * @export
  241. */
  242. shaka.ads.Utils.IMA_AD_MANAGER_LOADED = 'ima-ad-manager-loaded';
  243. /**
  244. * The event name for when the native IMA stream manager object has
  245. * loaded and become available.
  246. *
  247. * @const {string}
  248. * @export
  249. */
  250. shaka.ads.Utils.IMA_STREAM_MANAGER_LOADED = 'ima-stream-manager-loaded';
  251. /**
  252. * The event name for when the ad was clicked.
  253. *
  254. * @const {string}
  255. * @export
  256. */
  257. shaka.ads.Utils.AD_CLICKED = 'ad-clicked';
  258. /**
  259. * The event name for when there is an update to the current ad's progress.
  260. *
  261. * @const {string}
  262. * @export
  263. */
  264. shaka.ads.Utils.AD_PROGRESS = 'ad-progress';
  265. /**
  266. * The event name for when the ad is buffering.
  267. *
  268. * @const {string}
  269. * @export
  270. */
  271. shaka.ads.Utils.AD_BUFFERING = 'ad-buffering';
  272. /**
  273. * The event name for when the ad's URL was hit.
  274. *
  275. * @const {string}
  276. * @export
  277. */
  278. shaka.ads.Utils.AD_IMPRESSION = 'ad-impression';
  279. /**
  280. * The event name for when the ad's duration changed.
  281. *
  282. * @const {string}
  283. * @export
  284. */
  285. shaka.ads.Utils.AD_DURATION_CHANGED = 'ad-duration-changed';
  286. /**
  287. * The event name for when the ad was closed by the user.
  288. *
  289. * @const {string}
  290. * @export
  291. */
  292. shaka.ads.Utils.AD_CLOSED = 'ad-closed';
  293. /**
  294. * The event name for when the ad data becomes available.
  295. *
  296. * @const {string}
  297. * @export
  298. */
  299. shaka.ads.Utils.AD_LOADED = 'ad-loaded';
  300. /**
  301. * The event name for when all the ads were completed.
  302. *
  303. * @const {string}
  304. * @export
  305. */
  306. shaka.ads.Utils.ALL_ADS_COMPLETED = 'all-ads-completed';
  307. /**
  308. * The event name for when the ad changes from or to linear.
  309. *
  310. * @const {string}
  311. * @export
  312. */
  313. shaka.ads.Utils.AD_LINEAR_CHANGED = 'ad-linear-changed';
  314. /**
  315. * The event name for when the ad's metadata becomes available.
  316. *
  317. * @const {string}
  318. * @export
  319. */
  320. shaka.ads.Utils.AD_METADATA = 'ad-metadata';
  321. /**
  322. * The event name for when the ad display encountered a recoverable
  323. * error.
  324. *
  325. * @const {string}
  326. * @export
  327. */
  328. shaka.ads.Utils.AD_RECOVERABLE_ERROR = 'ad-recoverable-error';
  329. /**
  330. * The event name for when the ad manager dispatch errors.
  331. *
  332. * @const {string}
  333. * @export
  334. */
  335. shaka.ads.Utils.AD_ERROR = 'ad-error';
  336. /**
  337. * The event name for when the client side SDK signalled its readiness
  338. * to play a VPAID ad or an ad rule.
  339. *
  340. * @const {string}
  341. * @export
  342. */
  343. shaka.ads.Utils.AD_BREAK_READY = 'ad-break-ready';
  344. /**
  345. * The event name for when the interaction callback for the ad was
  346. * triggered.
  347. *
  348. * @const {string}
  349. * @export
  350. */
  351. shaka.ads.Utils.AD_INTERACTION = 'ad-interaction';
  352. /**
  353. * The name of the event for when an ad requires the main content to be paused.
  354. * Fired when the platform does not support multiple media elements.
  355. *
  356. * @const {string}
  357. * @export
  358. */
  359. shaka.ads.Utils.AD_CONTENT_PAUSE_REQUESTED = 'ad-content-pause-requested';
  360. /**
  361. * The name of the event for when an ad requires the main content to be resumed.
  362. * Fired when the platform does not support multiple media elements.
  363. *
  364. * @const {string}
  365. * @export
  366. */
  367. shaka.ads.Utils.AD_CONTENT_RESUME_REQUESTED = 'ad-content-resume-requested';
  368. /**
  369. * The name of the event for when an ad requires the video of the main content
  370. * to be attached.
  371. *
  372. * @const {string}
  373. * @export
  374. */
  375. shaka.ads.Utils.AD_CONTENT_ATTACH_REQUESTED = 'ad-content-attach-requested';