Source: lib/cea/cea608_data_channel.js

  1. /*! @license
  2. * Shaka Player
  3. * Copyright 2016 Google LLC
  4. * SPDX-License-Identifier: Apache-2.0
  5. */
  6. goog.provide('shaka.cea.Cea608DataChannel');
  7. goog.require('shaka.cea.Cea608Memory');
  8. goog.require('shaka.cea.CeaUtils');
  9. goog.require('shaka.log');
  10. /**
  11. * 608 closed captions channel.
  12. */
  13. shaka.cea.Cea608DataChannel = class {
  14. /**
  15. * @param {number} fieldNum Field number.
  16. * @param {number} channelNum Channel number.
  17. */
  18. constructor(fieldNum, channelNum) {
  19. /**
  20. * Current Caption Type.
  21. * @public {!shaka.cea.Cea608DataChannel.CaptionType}
  22. */
  23. this.type_ = shaka.cea.Cea608DataChannel.CaptionType.NONE;
  24. /**
  25. * Text buffer for CEA-608 "text mode". Although, we don't emit text mode.
  26. * So, this buffer serves as a no-op placeholder, just in case we receive
  27. * captions that toggle text mode.
  28. * @private @const {!shaka.cea.Cea608Memory}
  29. */
  30. this.text_ =
  31. new shaka.cea.Cea608Memory(fieldNum, channelNum);
  32. /**
  33. * Displayed memory.
  34. * @private {!shaka.cea.Cea608Memory}
  35. */
  36. this.displayedMemory_ =
  37. new shaka.cea.Cea608Memory(fieldNum, channelNum);
  38. /**
  39. * Non-displayed memory.
  40. * @private {!shaka.cea.Cea608Memory}
  41. */
  42. this.nonDisplayedMemory_ =
  43. new shaka.cea.Cea608Memory(fieldNum, channelNum);
  44. /**
  45. * Points to current buffer.
  46. * @private {!shaka.cea.Cea608Memory}
  47. */
  48. this.curBuf_ = this.nonDisplayedMemory_;
  49. /**
  50. * End time of the previous caption, serves as start time of next caption.
  51. * @private {number}
  52. */
  53. this.prevEndTime_ = 0;
  54. /**
  55. * Last control pair, 16 bits representing byte 1 and byte 2
  56. * @private {?number}
  57. */
  58. this.lastCp_ = null;
  59. }
  60. /**
  61. * Resets channel state.
  62. */
  63. reset() {
  64. this.type_ = shaka.cea.Cea608DataChannel.CaptionType.NONE;
  65. this.curBuf_ = this.nonDisplayedMemory_;
  66. this.lastCp_ = null;
  67. this.displayedMemory_.reset();
  68. this.nonDisplayedMemory_.reset();
  69. this.text_.reset();
  70. }
  71. /**
  72. * Set the initial PTS, which may not be 0 if we start decoding at a later
  73. * point in the stream. Without this, the first cue's startTime can be way
  74. * off.
  75. *
  76. * @param {number} pts
  77. */
  78. firstPts(pts) {
  79. this.prevEndTime_ = pts;
  80. }
  81. /**
  82. * Gets the row index from a Preamble Address Code byte pair.
  83. * @param {number} b1 Byte 1.
  84. * @param {number} b2 Byte 2.
  85. * @return {number} Row index.
  86. * @private
  87. */
  88. pacToRow_(b1, b2) {
  89. const ccRowTab = [
  90. 11, 11, // 0x00 or 0x01
  91. 1, 2, // 0x02 -> 0x03
  92. 3, 4, // 0x04 -> 0x05
  93. 12, 13, // 0x06 -> 0x07
  94. 14, 15, // 0x08 -> 0x09
  95. 5, 6, // 0x0A -> 0x0B
  96. 7, 8, // 0x0C -> 0x0D
  97. 9, 10, // 0x0E -> 0x0F
  98. ];
  99. return ccRowTab[((b1 & 0x07) << 1) | ((b2 >> 5) & 0x01)];
  100. }
  101. /**
  102. * PAC - Preamble Address Code.
  103. * b1 is of the form |P|0|0|1|C|0|ROW|
  104. * b2 is of the form |P|1|N|ATTRIBUTE|U|
  105. * @param {number} b1 Byte 1.
  106. * @param {number} b2 Byte 2.
  107. * @private
  108. */
  109. controlPac_(b1, b2) {
  110. const row = this.pacToRow_(b1, b2);
  111. // Set up the defaults.
  112. let textColor = shaka.cea.CeaUtils.DEFAULT_TXT_COLOR;
  113. let italics = false;
  114. let indent = null;
  115. // Get PAC index;
  116. let pacIndex;
  117. if (b2 > 0x5f) {
  118. pacIndex = b2 - 0x60;
  119. } else {
  120. pacIndex = b2 - 0x40;
  121. }
  122. if (pacIndex <= 0xd) {
  123. const colorIndex = Math.floor(pacIndex / 2);
  124. textColor = shaka.cea.Cea608DataChannel.TEXT_COLORS[colorIndex];
  125. } else if (pacIndex <= 0xf) {
  126. italics = true; // color stays white
  127. } else {
  128. indent = Math.floor((pacIndex - 0x10) / 2);
  129. }
  130. // PACs toggle underline on the last bit of b2.
  131. const underline = (b2 & 0x01) === 0x01;
  132. if (this.type_ === shaka.cea.Cea608DataChannel.CaptionType.TEXT) {
  133. // Don't execute the PAC if in text mode.
  134. return;
  135. }
  136. // Execute the PAC.
  137. const buf = this.curBuf_;
  138. // Move entire scroll window to a new base in rollup mode.
  139. if (this.type_ === shaka.cea.Cea608DataChannel.CaptionType.ROLLUP &&
  140. row !== buf.getRow()) {
  141. const oldTopRow = 1 + buf.getRow() - buf.getScrollSize();
  142. const newTopRow = 1 + row - buf.getScrollSize();
  143. // Shift up the scroll window.
  144. buf.moveRows(newTopRow, oldTopRow, buf.getScrollSize());
  145. // Clear everything outside of the new scroll window.
  146. buf.resetRows(0, newTopRow - 1);
  147. buf.resetRows(row + 1,
  148. shaka.cea.Cea608Memory.CC_ROWS - row);
  149. }
  150. buf.setRow(row);
  151. buf.setUnderline(underline);
  152. buf.setItalics(italics);
  153. buf.setTextColor(textColor);
  154. buf.setIndent(indent);
  155. // Clear the background color, since new row (PAC) should reset ALL styles.
  156. buf.setBackgroundColor(shaka.cea.CeaUtils.DEFAULT_BG_COLOR);
  157. }
  158. /**
  159. * Mid-Row control code handler.
  160. * @param {number} b2 Byte #2.
  161. * @private
  162. */
  163. controlMidrow_(b2) {
  164. // Clear all pre-existing midrow style attributes.
  165. this.curBuf_.setUnderline(false);
  166. this.curBuf_.setItalics(false);
  167. this.curBuf_.setTextColor(shaka.cea.CeaUtils.DEFAULT_TXT_COLOR);
  168. // Mid-row attrs use a space.
  169. this.curBuf_.addChar(
  170. shaka.cea.Cea608Memory.CharSet.BASIC_NORTH_AMERICAN, ' '.charCodeAt(0));
  171. let textColor = shaka.cea.CeaUtils.DEFAULT_TXT_COLOR;
  172. let italics = false;
  173. // Midrow codes set underline on last (LSB) bit.
  174. const underline = (b2 & 0x01) === 0x01;
  175. // b2 has the form |P|0|1|0|STYLE|U|
  176. textColor = shaka.cea.Cea608DataChannel.TEXT_COLORS[(b2 & 0xe) >> 1];
  177. if (textColor === 'white_italics') {
  178. textColor = 'white';
  179. italics = true;
  180. }
  181. this.curBuf_.setUnderline(underline);
  182. this.curBuf_.setItalics(italics);
  183. this.curBuf_.setTextColor(textColor);
  184. }
  185. /**
  186. * Background attribute control code handler.
  187. * @param {number} b1 Byte #1
  188. * @param {number} b2 Byte #2.
  189. * @private
  190. */
  191. controlBackgroundAttribute_(b1, b2) {
  192. let backgroundColor = shaka.cea.CeaUtils.DEFAULT_BG_COLOR;
  193. if ((b1 & 0x07) === 0x0) {
  194. // If background provided, last 3 bits of b1 are |0|0|0|. Color is in b2.
  195. backgroundColor = shaka.cea.Cea608DataChannel.BG_COLORS[(b2 & 0xe) >> 1];
  196. }
  197. this.curBuf_.setBackgroundColor(backgroundColor);
  198. }
  199. /**
  200. * The Cea608DataChannel control methods implement all CC control operations.
  201. * @param {!shaka.cea.Cea608DataChannel.Cea608Packet} ccPacket
  202. * @return {?shaka.extern.ICaptionDecoder.ClosedCaption}
  203. * @private
  204. */
  205. controlMiscellaneous_(ccPacket) {
  206. const MiscCmd = shaka.cea.Cea608DataChannel.MiscCmd_;
  207. const b2 = ccPacket.ccData2;
  208. const pts = ccPacket.pts;
  209. let parsedClosedCaption = null;
  210. switch (b2) {
  211. case MiscCmd.RCL:
  212. this.controlRcl_();
  213. break;
  214. case MiscCmd.BS:
  215. this.controlBs_();
  216. break;
  217. // unused (alarm off and alarm on)
  218. case MiscCmd.AOD:
  219. case MiscCmd.AON:
  220. break;
  221. case MiscCmd.DER:
  222. // Delete to End of Row. Not implemented since position not supported.
  223. break;
  224. case MiscCmd.RU2:
  225. parsedClosedCaption = this.controlRu_(2, pts);
  226. break;
  227. case MiscCmd.RU3:
  228. parsedClosedCaption = this.controlRu_(3, pts);
  229. break;
  230. case MiscCmd.RU4:
  231. parsedClosedCaption = this.controlRu_(4, pts);
  232. break;
  233. case MiscCmd.FON:
  234. this.controlFon_();
  235. break;
  236. case MiscCmd.RDC:
  237. this.controlRdc_(pts);
  238. break;
  239. case MiscCmd.TR:
  240. this.controlTr_();
  241. break;
  242. case MiscCmd.RTD:
  243. this.controlRtd_();
  244. break;
  245. case MiscCmd.EDM:
  246. parsedClosedCaption = this.controlEdm_(pts);
  247. break;
  248. case MiscCmd.CR:
  249. parsedClosedCaption = this.controlCr_(pts);
  250. break;
  251. case MiscCmd.ENM:
  252. this.controlEnm_();
  253. break;
  254. case MiscCmd.EOC:
  255. parsedClosedCaption = this.controlEoc_(pts);
  256. break;
  257. }
  258. return parsedClosedCaption;
  259. }
  260. /**
  261. * Handles CR - Carriage Return (Start new row).
  262. * CR only affects scroll windows (Rollup and Text modes).
  263. * Any currently buffered line needs to be emitted, along
  264. * with a window scroll action.
  265. * @param {number} pts in seconds.
  266. * @return {?shaka.extern.ICaptionDecoder.ClosedCaption}
  267. * @private
  268. */
  269. controlCr_(pts) {
  270. const buf = this.curBuf_;
  271. // Only rollup and text mode is affected, but we don't emit text mode.
  272. if (this.type_ !== shaka.cea.Cea608DataChannel.CaptionType.ROLLUP) {
  273. return null;
  274. }
  275. // Force out the scroll window since the top row will cleared.
  276. const parsedClosedCaption = buf.forceEmit(this.prevEndTime_, pts);
  277. // Calculate the top of the scroll window.
  278. const topRow = (buf.getRow() - buf.getScrollSize()) + 1;
  279. // Shift up the window one row higher.
  280. buf.moveRows(topRow - 1, topRow, buf.getScrollSize());
  281. // Clear out anything that's outside of our current scroll window.
  282. buf.resetRows(0, topRow - 1);
  283. buf.resetRows(buf.getRow(), shaka.cea.Cea608Memory.CC_ROWS - buf.getRow());
  284. // Update the end time so the next caption emits starting at this time.
  285. this.prevEndTime_ = pts;
  286. return parsedClosedCaption;
  287. }
  288. /**
  289. * Handles RU2, RU3, RU4 - Roll-Up, N rows.
  290. * If in TEXT, POPON or PAINTON, any displayed captions are erased.
  291. * This means must force emit entire display buffer.
  292. * @param {number} scrollSize New scroll window size.
  293. * @param {number} pts
  294. * @return {?shaka.extern.ICaptionDecoder.ClosedCaption}
  295. * @private
  296. */
  297. controlRu_(scrollSize, pts) {
  298. this.curBuf_ = this.displayedMemory_; // Point to displayed memory
  299. const buf = this.curBuf_;
  300. let parsedClosedCaption = null;
  301. // For any type except rollup and text mode, it should be emitted,
  302. // and memories cleared.
  303. if (this.type_ !== shaka.cea.Cea608DataChannel.CaptionType.ROLLUP &&
  304. this.type_ !== shaka.cea.Cea608DataChannel.CaptionType.TEXT) {
  305. parsedClosedCaption = buf.forceEmit(this.prevEndTime_, pts);
  306. // Clear both memories.
  307. this.displayedMemory_.eraseBuffer();
  308. this.nonDisplayedMemory_.eraseBuffer();
  309. // Rollup base row defaults to the last row (15).
  310. buf.setRow(shaka.cea.Cea608Memory.CC_ROWS);
  311. }
  312. this.type_ = shaka.cea.Cea608DataChannel.CaptionType.ROLLUP;
  313. // Set the new rollup window size.
  314. buf.setScrollSize(scrollSize);
  315. return parsedClosedCaption;
  316. }
  317. /**
  318. * Handles flash on.
  319. * @private
  320. */
  321. controlFon_() {
  322. this.curBuf_.addChar(
  323. shaka.cea.Cea608Memory.CharSet.BASIC_NORTH_AMERICAN,
  324. ' '.charCodeAt(0));
  325. }
  326. /**
  327. * Handles EDM - Erase Displayed Mem
  328. * Mode check:
  329. * EDM affects all captioning modes (but not Text mode);
  330. * @param {number} pts
  331. * @return {?shaka.extern.ICaptionDecoder.ClosedCaption}
  332. * @private
  333. */
  334. controlEdm_(pts) {
  335. const buf = this.displayedMemory_;
  336. let parsedClosedCaption = null;
  337. if (this.type_ !== shaka.cea.Cea608DataChannel.CaptionType.TEXT) {
  338. // Clearing displayed memory means we now know how long
  339. // its contents were displayed, so force it out.
  340. parsedClosedCaption = buf.forceEmit(this.prevEndTime_, pts);
  341. }
  342. buf.resetAllRows();
  343. return parsedClosedCaption;
  344. }
  345. /**
  346. * Handles RDC - Resume Direct Captions. Initiates Paint-On captioning mode.
  347. * RDC does not affect current display, so nothing needs to be forced out yet.
  348. * @param {number} pts in seconds
  349. * @private
  350. */
  351. controlRdc_(pts) {
  352. this.type_ = shaka.cea.Cea608DataChannel.CaptionType.PAINTON;
  353. // Point to displayed memory.
  354. this.curBuf_ = this.displayedMemory_;
  355. // No scroll window now.
  356. this.curBuf_.setScrollSize(0);
  357. // The next paint-on caption needs this time as the start time.
  358. this.prevEndTime_ = pts;
  359. }
  360. /**
  361. * Handles ENM - Erase Nondisplayed Mem
  362. * @private
  363. */
  364. controlEnm_() {
  365. this.nonDisplayedMemory_.resetAllRows();
  366. }
  367. /**
  368. * Handles EOC - End Of Caption (flip mem)
  369. * This forces Pop-On mode, and swaps the displayed and nondisplayed memories.
  370. * @private
  371. * @param {number} pts
  372. * @return {?shaka.extern.ICaptionDecoder.ClosedCaption}
  373. */
  374. controlEoc_(pts) {
  375. let parsedClosedCaption = null;
  376. if (this.type_ !== shaka.cea.Cea608DataChannel.CaptionType.TEXT) {
  377. parsedClosedCaption =
  378. this.displayedMemory_.forceEmit(this.prevEndTime_, pts);
  379. }
  380. // Swap memories
  381. const buf = this.nonDisplayedMemory_;
  382. this.nonDisplayedMemory_ = this.displayedMemory_; // Swap buffers
  383. this.displayedMemory_ = buf;
  384. // Enter Pop-On mode.
  385. this.controlRcl_();
  386. // The caption ended, and so the previous end time should be updated.
  387. this.prevEndTime_ = pts;
  388. return parsedClosedCaption;
  389. }
  390. /**
  391. * Handles RCL - Resume Caption Loading
  392. * Initiates Pop-On style captioning. No need to force anything out upon
  393. * entering Pop-On mode because it does not affect the current display.
  394. * @private
  395. */
  396. controlRcl_() {
  397. this.type_ = shaka.cea.Cea608DataChannel.CaptionType.POPON;
  398. this.curBuf_ = this.nonDisplayedMemory_;
  399. // No scroll window now
  400. this.curBuf_.setScrollSize(0);
  401. }
  402. /**
  403. * Handles BS - BackSpace.
  404. * @private
  405. */
  406. controlBs_() {
  407. this.curBuf_.eraseChar();
  408. }
  409. /**
  410. * Handles TR - Text Restart.
  411. * Clears text buffer and resumes Text Mode.
  412. * @private
  413. */
  414. controlTr_() {
  415. this.text_.reset();
  416. this.controlRtd_(); // Put into text mode.
  417. }
  418. /**
  419. * Handles RTD - Resume Text Display.
  420. * Resumes text mode. No need to force anything out, because Text Mode doesn't
  421. * affect current display. Also, this decoder does not emit Text Mode anyway.
  422. * @private
  423. */
  424. controlRtd_() {
  425. shaka.log.warnOnce('Cea608DataChannel',
  426. 'CEA-608 text mode entered, but is unsupported');
  427. this.curBuf_ = this.text_;
  428. this.type_ = shaka.cea.Cea608DataChannel.CaptionType.TEXT;
  429. }
  430. /**
  431. * Handles a Basic North American byte pair.
  432. * @param {number} b1 Byte 1.
  433. * @param {number} b2 Byte 2.
  434. */
  435. handleBasicNorthAmericanChar(b1, b2) {
  436. this.curBuf_.addChar(
  437. shaka.cea.Cea608Memory.CharSet.BASIC_NORTH_AMERICAN, b1);
  438. this.curBuf_.addChar(
  439. shaka.cea.Cea608Memory.CharSet.BASIC_NORTH_AMERICAN, b2);
  440. }
  441. /**
  442. * Handles an Extended Western European byte pair.
  443. * @param {number} b1 Byte 1.
  444. * @param {number} b2 Byte 2.
  445. * @private
  446. */
  447. handleExtendedWesternEuropeanChar_(b1, b2) {
  448. // Get the char set from the LSB, which is the char set toggle bit.
  449. const charSet = b1 & 0x01 ?
  450. shaka.cea.Cea608Memory.CharSet.PORTUGUESE_GERMAN:
  451. shaka.cea.Cea608Memory.CharSet.SPANISH_FRENCH;
  452. this.curBuf_.addChar(charSet, b2);
  453. }
  454. /**
  455. * Handles a tab offset.
  456. *
  457. * @param {number} offset
  458. * @private
  459. */
  460. handleOffset_(offset) {
  461. this.curBuf_.setOffset(offset);
  462. }
  463. /**
  464. * Decodes control code.
  465. * Three types of control codes:
  466. * Preamble Address Codes, Mid-Row Codes, and Miscellaneous Control Codes.
  467. * @param {!shaka.cea.Cea608DataChannel.Cea608Packet} ccPacket
  468. * @return {?shaka.extern.ICaptionDecoder.ClosedCaption}
  469. */
  470. handleControlCode(ccPacket) {
  471. const b1 = ccPacket.ccData1;
  472. const b2 = ccPacket.ccData2;
  473. // FCC wants control codes transmitted twice, and that will often be
  474. // seen in broadcast captures. If the very next frame has a duplicate
  475. // control code, that duplicate is ignored. Note that this only applies
  476. // to the very next frame, and only for one match.
  477. if (this.lastCp_ === ((b1 << 8) | b2)) {
  478. this.lastCp_ = null;
  479. return null;
  480. }
  481. // Remember valid control code for checking in next frame!
  482. this.lastCp_ = (b1 << 8) | b2;
  483. if (this.isPAC_(b1, b2)) {
  484. this.controlPac_(b1, b2);
  485. } else if (this.isMidrowStyleChange_(b1, b2)) {
  486. this.controlMidrow_(b2);
  487. } else if (this.isBackgroundAttribute_(b1, b2)) {
  488. this.controlBackgroundAttribute_(b1, b2);
  489. } else if (this.isSpecialNorthAmericanChar_(b1, b2)) {
  490. this.curBuf_.addChar(
  491. shaka.cea.Cea608Memory.CharSet.SPECIAL_NORTH_AMERICAN, b2);
  492. } else if (this.isExtendedWesternEuropeanChar_(b1, b2)) {
  493. this.handleExtendedWesternEuropeanChar_(b1, b2);
  494. } else if (this.isMiscellaneous_(b1, b2)) {
  495. return this.controlMiscellaneous_(ccPacket);
  496. } else if (this.isOffset_(b1, b2)) {
  497. const offset = b2 - 0x20;
  498. this.handleOffset_(offset);
  499. }
  500. return null;
  501. }
  502. /**
  503. * Checks if this is a Miscellaneous control code.
  504. * @param {number} b1 Byte 1.
  505. * @param {number} b2 Byte 2.
  506. * @return {boolean}
  507. * @private
  508. */
  509. isMiscellaneous_(b1, b2) {
  510. // For Miscellaneous Control Codes, the bytes take the following form:
  511. // b1 -> |0|0|0|1|C|1|0|F|
  512. // b2 -> |0|0|1|0|X|X|X|X|
  513. return ((b1 & 0xf6) === 0x14) && ((b2 & 0xf0) === 0x20);
  514. }
  515. /**
  516. * Checks if this is a offset control code.
  517. * @param {number} b1 Byte 1.
  518. * @param {number} b2 Byte 2.
  519. * @return {boolean}
  520. * @private
  521. */
  522. isOffset_(b1, b2) {
  523. return (b1 == 0x17 || b1 == 0x1f) && b2 >= 0x21 && b2 <= 0x23;
  524. }
  525. /**
  526. * Checks if this is a PAC control code.
  527. * @param {number} b1 Byte 1.
  528. * @param {number} b2 Byte 2.
  529. * @return {boolean}
  530. * @private
  531. */
  532. isPAC_(b1, b2) {
  533. // For Preamble Address Codes, the bytes take the following form:
  534. // b1 -> |0|0|0|1|X|X|X|X|
  535. // b2 -> |0|1|X|X|X|X|X|X|
  536. return ((b1 & 0xf0) === 0x10) && ((b2 & 0xc0) === 0x40);
  537. }
  538. /**
  539. * Checks if this is a Midrow style change control code.
  540. * @param {number} b1 Byte 1.
  541. * @param {number} b2 Byte 2.
  542. * @return {boolean}
  543. * @private
  544. */
  545. isMidrowStyleChange_(b1, b2) {
  546. // For Midrow Control Codes, the bytes take the following form:
  547. // b1 -> |0|0|0|1|C|0|0|1|
  548. // b2 -> |0|0|1|0|X|X|X|X|
  549. return ((b1 & 0xf7) === 0x11) && ((b2 & 0xf0) === 0x20);
  550. }
  551. /**
  552. * Checks if this is a background attribute control code.
  553. * @param {number} b1 Byte 1.
  554. * @param {number} b2 Byte 2.
  555. * @return {boolean}
  556. * @private
  557. */
  558. isBackgroundAttribute_(b1, b2) {
  559. // For Background Attribute Codes, the bytes take the following form:
  560. // Bg provided: b1 -> |0|0|0|1|C|0|0|0| b2 -> |0|0|1|0|COLOR|T|
  561. // No Bg: b1 -> |0|0|0|1|C|1|1|1| b2 -> |0|0|1|0|1|1|0|1|
  562. return (((b1 & 0xf7) === 0x10) && ((b2 & 0xf0) === 0x20)) ||
  563. (((b1 & 0xf7) === 0x17) && ((b2 & 0xff) === 0x2D));
  564. }
  565. /**
  566. * Checks if the character is in the Special North American char. set.
  567. * @param {number} b1 Byte 1.
  568. * @param {number} b2 Byte 2.
  569. * @return {boolean}
  570. * @private
  571. */
  572. isSpecialNorthAmericanChar_(b1, b2) {
  573. // The bytes take the following form:
  574. // b1 -> |0|0|0|1|C|0|0|1|
  575. // b2 -> |0|0|1|1| CHAR |
  576. return ((b1 & 0xf7) === 0x11) && ((b2 & 0xf0) === 0x30);
  577. }
  578. /**
  579. * Checks if the character is in the Extended Western European char. set.
  580. * @param {number} b1 Byte 1.
  581. * @param {number} b2 Byte 2.
  582. * @return {boolean}
  583. * @private
  584. */
  585. isExtendedWesternEuropeanChar_(b1, b2) {
  586. // The bytes take the following form:
  587. // b1 -> |0|0|0|1|C|0|1|S|
  588. // b2 -> |0|0|1|CHARACTER|
  589. return ((b1 & 0xf6) === 0x12) && ((b2 & 0xe0) === 0x20);
  590. }
  591. /**
  592. * Checks if the data contains a control code.
  593. * @param {number} b1 Byte 1.
  594. * @return {boolean}
  595. */
  596. static isControlCode(b1) {
  597. // For control codes, the first byte takes the following form:
  598. // b1 -> |P|0|0|1|X|X|X|X|
  599. return (b1 & 0x70) === 0x10;
  600. }
  601. /**
  602. * Checks if the data contains a XDS control code.
  603. * @param {number} b1 Byte 1.
  604. * @return {boolean}
  605. */
  606. static isXdsControlCode(b1) {
  607. return b1 >= 0x01 && b1 <= 0x0F;
  608. }
  609. };
  610. /**
  611. * Command codes.
  612. * @enum {number}
  613. * @private
  614. */
  615. shaka.cea.Cea608DataChannel.MiscCmd_ = {
  616. // "RCL - Resume Caption Loading"
  617. RCL: 0x20,
  618. // "BS - BackSpace"
  619. BS: 0x21,
  620. // "AOD - Unused (alarm off)"
  621. AOD: 0x22,
  622. // "AON - Unused (alarm on)"
  623. AON: 0x23,
  624. // "DER - Delete to End of Row"
  625. DER: 0x24,
  626. // "RU2 - Roll-Up, 2 rows"
  627. RU2: 0x25,
  628. // "RU3 - Roll-Up, 3 rows"
  629. RU3: 0x26,
  630. // "RU4 - Roll-Up, 4 rows"
  631. RU4: 0x27,
  632. // "FON - Flash On"
  633. FON: 0x28,
  634. // "RDC - Resume Direct Captions"
  635. RDC: 0x29,
  636. // "TR - Text Restart"
  637. TR: 0x2a,
  638. // "RTD - Resume Text Display"
  639. RTD: 0x2b,
  640. // "EDM - Erase Displayed Mem"
  641. EDM: 0x2c,
  642. // "CR - Carriage return"
  643. CR: 0x2d,
  644. // "ENM - Erase Nondisplayed Mem"
  645. ENM: 0x2e,
  646. // "EOC - End Of Caption (flip mem)"
  647. EOC: 0x2f,
  648. };
  649. /**
  650. * Caption type.
  651. * @private @const @enum {number}
  652. */
  653. shaka.cea.Cea608DataChannel.CaptionType = {
  654. NONE: 0,
  655. POPON: 1,
  656. PAINTON: 2,
  657. ROLLUP: 3,
  658. TEXT: 4,
  659. };
  660. /**
  661. * @const {!Array<string>}
  662. */
  663. shaka.cea.Cea608DataChannel.BG_COLORS = [
  664. 'black',
  665. 'green',
  666. 'blue',
  667. 'cyan',
  668. 'red',
  669. 'yellow',
  670. 'magenta',
  671. 'black',
  672. ];
  673. /**
  674. * @const {!Array<string>}
  675. */
  676. shaka.cea.Cea608DataChannel.TEXT_COLORS = [
  677. 'white',
  678. 'green',
  679. 'blue',
  680. 'cyan',
  681. 'red',
  682. 'yellow',
  683. 'magenta',
  684. 'white_italics',
  685. ];
  686. /**
  687. * Style associated with a cue.
  688. * @typedef {{
  689. * textColor: ?string,
  690. * backgroundColor: ?string,
  691. * italics: ?boolean,
  692. * underline: ?boolean
  693. * }}
  694. */
  695. shaka.cea.Cea608DataChannel.Style;
  696. /**
  697. * CEA closed captions packet.
  698. * @typedef {{
  699. * pts: number,
  700. * type: number,
  701. * ccData1: number,
  702. * ccData2: number,
  703. * order: number
  704. * }}
  705. *
  706. * @property {number} pts
  707. * Presentation timestamp (in second) at which this packet was received.
  708. * @property {number} type
  709. * Type of the packet. Either 0 or 1, representing the CEA-608 field.
  710. * @property {number} ccData1 CEA-608 byte 1.
  711. * @property {number} ccData2 CEA-608 byte 2.
  712. * @property {number} order
  713. * A number indicating the order this packet was received in a sequence
  714. * of packets. Used to break ties in a stable sorting algorithm
  715. */
  716. shaka.cea.Cea608DataChannel.Cea608Packet;