Source: lib/text/simple_text_displayer.js

  1. /*! @license
  2. * Shaka Player
  3. * Copyright 2016 Google LLC
  4. * SPDX-License-Identifier: Apache-2.0
  5. */
  6. /**
  7. * @fileoverview
  8. */
  9. goog.provide('shaka.text.SimpleTextDisplayer');
  10. goog.require('goog.asserts');
  11. goog.require('shaka.log');
  12. goog.require('shaka.text.Cue');
  13. goog.require('shaka.text.Utils');
  14. /**
  15. * A text displayer plugin using the browser's native VTTCue interface.
  16. *
  17. * @implements {shaka.extern.TextDisplayer}
  18. * @export
  19. */
  20. shaka.text.SimpleTextDisplayer = class {
  21. /**
  22. * @param {HTMLMediaElement} video
  23. * @param {string} label
  24. */
  25. constructor(video, label) {
  26. /** @private {TextTrack} */
  27. this.textTrack_ = null;
  28. // TODO: Test that in all cases, the built-in CC controls in the video
  29. // element are toggling our TextTrack.
  30. // If the video element has TextTracks, disable them. If we see one that
  31. // was created by a previous instance of Shaka Player, reuse it.
  32. for (const track of Array.from(video.textTracks)) {
  33. // NOTE: There is no API available to remove a TextTrack from a video
  34. // element.
  35. track.mode = 'disabled';
  36. if (track.label == label) {
  37. this.textTrack_ = track;
  38. }
  39. }
  40. if (!this.textTrack_) {
  41. // As far as I can tell, there is no observable difference between setting
  42. // kind to 'subtitles' or 'captions' when creating the TextTrack object.
  43. // The individual text tracks from the manifest will still have their own
  44. // kinds which can be displayed in the app's UI.
  45. this.textTrack_ = video.addTextTrack('subtitles', label);
  46. }
  47. this.textTrack_.mode = 'hidden';
  48. }
  49. /**
  50. * @override
  51. * @export
  52. */
  53. configure(config) {
  54. // Unused.
  55. }
  56. /**
  57. * @override
  58. * @export
  59. */
  60. remove(start, end) {
  61. // Check that the displayer hasn't been destroyed.
  62. if (!this.textTrack_) {
  63. return false;
  64. }
  65. const removeInRange = (cue) => {
  66. const inside = cue.startTime < end && cue.endTime > start;
  67. return inside;
  68. };
  69. shaka.text.SimpleTextDisplayer.removeWhere_(this.textTrack_, removeInRange);
  70. return true;
  71. }
  72. /**
  73. * @override
  74. * @export
  75. */
  76. append(cues) {
  77. const flattenedCues = shaka.text.Utils.getCuesToFlatten(cues);
  78. // Convert cues.
  79. const textTrackCues = [];
  80. const cuesInTextTrack = this.textTrack_.cues ?
  81. Array.from(this.textTrack_.cues) : [];
  82. for (const inCue of flattenedCues) {
  83. // When a VTT cue spans a segment boundary, the cue will be duplicated
  84. // into two segments.
  85. // To avoid displaying duplicate cues, if the current textTrack cues
  86. // list already contains the cue, skip it.
  87. const containsCue = cuesInTextTrack.some((cueInTextTrack) => {
  88. if (cueInTextTrack.startTime == inCue.startTime &&
  89. cueInTextTrack.endTime == inCue.endTime &&
  90. cueInTextTrack.text == inCue.payload) {
  91. return true;
  92. }
  93. return false;
  94. });
  95. if (!containsCue) {
  96. const cue =
  97. shaka.text.SimpleTextDisplayer.convertToTextTrackCue_(inCue);
  98. if (cue) {
  99. textTrackCues.push(cue);
  100. }
  101. }
  102. }
  103. // Sort the cues based on start/end times. Make a copy of the array so
  104. // we can get the index in the original ordering. Out of order cues are
  105. // rejected by Edge. See https://bit.ly/2K9VX3s
  106. const sortedCues = textTrackCues.slice().sort((a, b) => {
  107. if (a.startTime != b.startTime) {
  108. return a.startTime - b.startTime;
  109. } else if (a.endTime != b.endTime) {
  110. return a.endTime - b.startTime;
  111. } else {
  112. // The browser will display cues with identical time ranges from the
  113. // bottom up. Reversing the order of equal cues means the first one
  114. // parsed will be at the top, as you would expect.
  115. // See https://github.com/shaka-project/shaka-player/issues/848 for
  116. // more info.
  117. // However, this ordering behavior is part of VTTCue's "line" field.
  118. // Some platforms don't have a real VTTCue and use a polyfill instead.
  119. // When VTTCue is polyfilled or does not support "line", we should _not_
  120. // reverse the order. This occurs on legacy Edge.
  121. // eslint-disable-next-line no-restricted-syntax
  122. if ('line' in VTTCue.prototype) {
  123. // Native VTTCue
  124. return textTrackCues.indexOf(b) - textTrackCues.indexOf(a);
  125. } else {
  126. // Polyfilled VTTCue
  127. return textTrackCues.indexOf(a) - textTrackCues.indexOf(b);
  128. }
  129. }
  130. });
  131. for (const cue of sortedCues) {
  132. this.textTrack_.addCue(cue);
  133. }
  134. }
  135. /**
  136. * @override
  137. * @export
  138. */
  139. destroy() {
  140. if (this.textTrack_) {
  141. const removeIt = (cue) => true;
  142. shaka.text.SimpleTextDisplayer.removeWhere_(this.textTrack_, removeIt);
  143. // NOTE: There is no API available to remove a TextTrack from a video
  144. // element.
  145. this.textTrack_.mode = 'disabled';
  146. }
  147. this.textTrack_ = null;
  148. return Promise.resolve();
  149. }
  150. /**
  151. * @override
  152. * @export
  153. */
  154. isTextVisible() {
  155. return this.textTrack_.mode == 'showing';
  156. }
  157. /**
  158. * @override
  159. * @export
  160. */
  161. setTextVisibility(on) {
  162. this.textTrack_.mode = on ? 'showing' : 'hidden';
  163. }
  164. /**
  165. * @param {!shaka.text.Cue} shakaCue
  166. * @return {TextTrackCue}
  167. * @private
  168. */
  169. static convertToTextTrackCue_(shakaCue) {
  170. if (shakaCue.startTime >= shakaCue.endTime) {
  171. // Edge will throw in this case.
  172. // See issue #501
  173. shaka.log.warning('Invalid cue times: ' + shakaCue.startTime +
  174. ' - ' + shakaCue.endTime);
  175. return null;
  176. }
  177. const Cue = shaka.text.Cue;
  178. /** @type {VTTCue} */
  179. const vttCue = new VTTCue(
  180. shakaCue.startTime,
  181. shakaCue.endTime,
  182. shakaCue.payload);
  183. // NOTE: positionAlign and lineAlign settings are not supported by Chrome
  184. // at the moment, so setting them will have no effect.
  185. // The bug on chromium to implement them:
  186. // https://bugs.chromium.org/p/chromium/issues/detail?id=633690
  187. vttCue.lineAlign = shakaCue.lineAlign;
  188. vttCue.positionAlign = shakaCue.positionAlign;
  189. if (shakaCue.size) {
  190. vttCue.size = shakaCue.size;
  191. }
  192. try {
  193. // Safari 10 seems to throw on align='center'.
  194. vttCue.align = shakaCue.textAlign;
  195. } catch (exception) {}
  196. if (shakaCue.textAlign == 'center' && vttCue.align != 'center') {
  197. // We want vttCue.position = 'auto'. By default, |position| is set to
  198. // "auto". If we set it to "auto" safari will throw an exception, so we
  199. // must rely on the default value.
  200. vttCue.align = 'middle';
  201. }
  202. if (shakaCue.writingMode ==
  203. Cue.writingMode.VERTICAL_LEFT_TO_RIGHT) {
  204. vttCue.vertical = 'lr';
  205. } else if (shakaCue.writingMode ==
  206. Cue.writingMode.VERTICAL_RIGHT_TO_LEFT) {
  207. vttCue.vertical = 'rl';
  208. }
  209. // snapToLines flag is true by default
  210. if (shakaCue.lineInterpretation == Cue.lineInterpretation.PERCENTAGE) {
  211. vttCue.snapToLines = false;
  212. }
  213. if (shakaCue.line != null) {
  214. vttCue.line = shakaCue.line;
  215. }
  216. if (shakaCue.position != null) {
  217. vttCue.position = shakaCue.position;
  218. }
  219. return vttCue;
  220. }
  221. /**
  222. * Iterate over all the cues in a text track and remove all those for which
  223. * |predicate(cue)| returns true.
  224. *
  225. * @param {!TextTrack} track
  226. * @param {function(!TextTrackCue):boolean} predicate
  227. * @private
  228. */
  229. static removeWhere_(track, predicate) {
  230. // Since |track.cues| can be null if |track.mode| is "disabled", force it to
  231. // something other than "disabled".
  232. //
  233. // If the track is already showing, then we should keep it as showing. But
  234. // if it something else, we will use hidden so that we don't "flash" cues on
  235. // the screen.
  236. const oldState = track.mode;
  237. const tempState = oldState == 'showing' ? 'showing' : 'hidden';
  238. track.mode = tempState;
  239. goog.asserts.assert(
  240. track.cues,
  241. 'Cues should be accessible when mode is set to "' + tempState + '".');
  242. // Create a copy of the list to avoid errors while iterating.
  243. for (const cue of Array.from(track.cues)) {
  244. if (cue && predicate(cue)) {
  245. track.removeCue(cue);
  246. }
  247. }
  248. track.mode = oldState;
  249. }
  250. };