Source: lib/media/closed_caption_parser.js

  1. /*! @license
  2. * Shaka Player
  3. * Copyright 2016 Google LLC
  4. * SPDX-License-Identifier: Apache-2.0
  5. */
  6. goog.provide('shaka.media.ClosedCaptionParser');
  7. goog.provide('shaka.media.IClosedCaptionParser');
  8. goog.require('shaka.cea.DummyCaptionDecoder');
  9. goog.require('shaka.cea.DummyCeaParser');
  10. goog.require('shaka.log');
  11. goog.require('shaka.util.BufferUtils');
  12. /**
  13. * The IClosedCaptionParser defines the interface to provide all operations for
  14. * parsing the closed captions embedded in Dash videos streams.
  15. * TODO: Remove this interface and move method definitions
  16. * directly to ClosedCaptionParser.
  17. * @interface
  18. * @export
  19. */
  20. shaka.media.IClosedCaptionParser = class {
  21. /**
  22. * Initialize the caption parser. This should be called whenever new init
  23. * segment arrives.
  24. * @param {BufferSource} initSegment
  25. * @param {boolean=} adaptation True if we just automatically switched active
  26. * variant(s).
  27. */
  28. init(initSegment, adaptation = false) {}
  29. /**
  30. * Parses embedded CEA closed captions and interacts with the underlying
  31. * CaptionStream, and calls the callback function when there are closed
  32. * captions.
  33. *
  34. * @param {BufferSource} mediaFragment
  35. * @return {!Array<!shaka.extern.ICaptionDecoder.ClosedCaption>}
  36. * An array of parsed closed captions.
  37. */
  38. parseFrom(mediaFragment) {}
  39. /**
  40. * Resets the CaptionStream.
  41. */
  42. reset() {}
  43. /**
  44. * Returns the streams that the CEA decoder found.
  45. * @return {!Array<string>}
  46. */
  47. getStreams() {}
  48. };
  49. /**
  50. * Closed Caption Parser provides all operations for parsing the closed captions
  51. * embedded in Dash videos streams.
  52. *
  53. * @implements {shaka.media.IClosedCaptionParser}
  54. * @final
  55. * @export
  56. */
  57. shaka.media.ClosedCaptionParser = class {
  58. /**
  59. * @param {string} mimeType
  60. */
  61. constructor(mimeType) {
  62. /** @private {!shaka.extern.ICeaParser} */
  63. this.ceaParser_ = new shaka.cea.DummyCeaParser();
  64. const parserFactory =
  65. shaka.media.ClosedCaptionParser.findParser(mimeType.toLowerCase());
  66. if (parserFactory) {
  67. this.ceaParser_ = parserFactory();
  68. }
  69. /**
  70. * Decoder for decoding CEA-X08 data from closed caption packets.
  71. * @private {!shaka.extern.ICaptionDecoder}
  72. */
  73. this.ceaDecoder_ = new shaka.cea.DummyCaptionDecoder();
  74. const decoderFactory = shaka.media.ClosedCaptionParser.findDecoder();
  75. if (decoderFactory) {
  76. this.ceaDecoder_ = decoderFactory();
  77. }
  78. }
  79. /**
  80. * @override
  81. */
  82. init(initSegment, adaptation = false) {
  83. shaka.log.debug('Passing new init segment to CEA parser');
  84. if (!adaptation) {
  85. // Reset underlying decoder when new init segment arrives
  86. // to clear stored pts values.
  87. // This is necessary when a new Period comes in DASH or a discontinuity
  88. // in HLS.
  89. this.reset();
  90. }
  91. this.ceaParser_.init(initSegment);
  92. }
  93. /**
  94. * @override
  95. */
  96. parseFrom(mediaFragment) {
  97. // Parse the fragment.
  98. const captionPackets = this.ceaParser_.parse(mediaFragment);
  99. // Extract the caption packets for decoding.
  100. for (const captionPacket of captionPackets) {
  101. const uint8ArrayData =
  102. shaka.util.BufferUtils.toUint8(captionPacket.packet);
  103. if (uint8ArrayData.length > 0) {
  104. this.ceaDecoder_.extract(uint8ArrayData, captionPacket.pts);
  105. }
  106. }
  107. // Decode and return the parsed captions.
  108. return this.ceaDecoder_.decode();
  109. }
  110. /**
  111. * @override
  112. */
  113. reset() {
  114. this.ceaDecoder_.clear();
  115. }
  116. /**
  117. * @override
  118. */
  119. getStreams() {
  120. return this.ceaDecoder_.getStreams();
  121. }
  122. /**
  123. * @param {string} mimeType
  124. * @param {!shaka.extern.CeaParserPlugin} plugin
  125. * @export
  126. */
  127. static registerParser(mimeType, plugin) {
  128. shaka.media.ClosedCaptionParser.parserMap_.set(mimeType, plugin);
  129. }
  130. /**
  131. * @param {string} mimeType
  132. * @export
  133. */
  134. static unregisterParser(mimeType) {
  135. shaka.media.ClosedCaptionParser.parserMap_.delete(mimeType);
  136. }
  137. /**
  138. * @param {string} mimeType
  139. * @return {?shaka.extern.CeaParserPlugin}
  140. * @export
  141. */
  142. static findParser(mimeType) {
  143. return shaka.media.ClosedCaptionParser.parserMap_.get(mimeType);
  144. }
  145. /**
  146. * @param {!shaka.extern.CaptionDecoderPlugin} plugin
  147. * @export
  148. */
  149. static registerDecoder(plugin) {
  150. shaka.media.ClosedCaptionParser.decoderFactory_ = plugin;
  151. }
  152. /**
  153. * @export
  154. */
  155. static unregisterDecoder() {
  156. shaka.media.ClosedCaptionParser.decoderFactory_ = null;
  157. }
  158. /**
  159. * @return {?shaka.extern.CaptionDecoderPlugin}
  160. * @export
  161. */
  162. static findDecoder() {
  163. return shaka.media.ClosedCaptionParser.decoderFactory_;
  164. }
  165. };
  166. /** @private {!Map<string, shaka.extern.CeaParserPlugin>} */
  167. shaka.media.ClosedCaptionParser.parserMap_ = new Map();
  168. /** @private {?shaka.extern.CaptionDecoderPlugin} */
  169. shaka.media.ClosedCaptionParser.decoderFactory_ = null;