Source: lib/cea/cea708_service.js

  1. /*! @license
  2. * Shaka Player
  3. * Copyright 2016 Google LLC
  4. * SPDX-License-Identifier: Apache-2.0
  5. */
  6. goog.provide('shaka.cea.Cea708Service');
  7. goog.require('shaka.cea.Cea708Window');
  8. goog.require('shaka.cea.DtvccPacket');
  9. /**
  10. * CEA-708 closed captions service as defined by CEA-708-E. A decoder can own up
  11. * to 63 services. Each service owns eight windows.
  12. */
  13. shaka.cea.Cea708Service = class {
  14. /**
  15. * @param {number} serviceNumber
  16. */
  17. constructor(serviceNumber) {
  18. /**
  19. * Number for this specific service (1 - 63).
  20. * @private {number}
  21. */
  22. this.serviceNumber_ = serviceNumber;
  23. /**
  24. * Eight Cea708 Windows, as defined by the spec.
  25. * @private {!Array<?shaka.cea.Cea708Window>}
  26. */
  27. this.windows_ = [
  28. null, null, null, null, null, null, null, null,
  29. ];
  30. /**
  31. * The current window for which window command operate on.
  32. * @private {?shaka.cea.Cea708Window}
  33. */
  34. this.currentWindow_ = null;
  35. }
  36. /**
  37. * Processes a CEA-708 control code.
  38. * @param {!shaka.cea.DtvccPacket} dtvccPacket
  39. * @return {?shaka.extern.ICaptionDecoder.ClosedCaption}
  40. * @throws {!shaka.util.Error}
  41. */
  42. handleCea708ControlCode(dtvccPacket) {
  43. const blockData = dtvccPacket.readByte();
  44. let controlCode = blockData.value;
  45. const pts = blockData.pts;
  46. // Read extended control code if needed.
  47. if (controlCode === shaka.cea.Cea708Service.EXT_CEA708_CTRL_CODE_BYTE1) {
  48. const extendedControlCodeBlock = dtvccPacket.readByte();
  49. controlCode = (controlCode << 16) | extendedControlCodeBlock.value;
  50. }
  51. // Control codes are in 1 of 4 logical groups:
  52. // CL (C0, C2), CR (C1, C3), GL (G0, G2), GR (G1, G2).
  53. if (controlCode >= 0x00 && controlCode <= 0x1f) {
  54. return this.handleC0_(controlCode, pts);
  55. } else if (controlCode >= 0x80 && controlCode <= 0x9f) {
  56. return this.handleC1_(dtvccPacket, controlCode, pts);
  57. } else if (controlCode >= 0x1000 && controlCode <= 0x101f) {
  58. this.handleC2_(dtvccPacket, controlCode & 0xff);
  59. } else if (controlCode >= 0x1080 && controlCode <= 0x109f) {
  60. this.handleC3_(dtvccPacket, controlCode & 0xff);
  61. } else if (controlCode >= 0x20 && controlCode <= 0x7f) {
  62. this.handleG0_(controlCode);
  63. } else if (controlCode >= 0xa0 && controlCode <= 0xff) {
  64. this.handleG1_(controlCode);
  65. } else if (controlCode >= 0x1020 && controlCode <= 0x107f) {
  66. this.handleG2_(controlCode & 0xff);
  67. } else if (controlCode >= 0x10a0 && controlCode <= 0x10ff) {
  68. this.handleG3_(controlCode & 0xff);
  69. }
  70. return null;
  71. }
  72. /**
  73. * Handles G0 group data.
  74. * @param {number} controlCode
  75. * @private
  76. */
  77. handleG0_(controlCode) {
  78. if (!this.currentWindow_) {
  79. return;
  80. }
  81. // G0 contains ASCII from 0x20 to 0x7f, with the exception that 0x7f
  82. // is replaced by a musical note.
  83. if (controlCode === 0x7f) {
  84. this.currentWindow_.setCharacter('♪');
  85. return;
  86. }
  87. this.currentWindow_.setCharacter(String.fromCharCode(controlCode));
  88. }
  89. /**
  90. * Handles G1 group data.
  91. * @param {number} controlCode
  92. * @private
  93. */
  94. handleG1_(controlCode) {
  95. if (!this.currentWindow_) {
  96. return;
  97. }
  98. // G1 is the Latin-1 Character Set from 0xa0 to 0xff.
  99. this.currentWindow_.setCharacter(String.fromCharCode(controlCode));
  100. }
  101. /**
  102. * Handles G2 group data.
  103. * @param {number} controlCode
  104. * @private
  105. */
  106. handleG2_(controlCode) {
  107. if (!this.currentWindow_) {
  108. return;
  109. }
  110. if (!shaka.cea.Cea708Service.G2Charset.has(controlCode)) {
  111. // If the character is unsupported, the spec says to put an underline.
  112. this.currentWindow_.setCharacter('_');
  113. return;
  114. }
  115. const char = shaka.cea.Cea708Service.G2Charset.get(controlCode);
  116. this.currentWindow_.setCharacter(char);
  117. }
  118. /**
  119. * Handles G3 group data.
  120. * @param {number} controlCode
  121. * @private
  122. */
  123. handleG3_(controlCode) {
  124. if (!this.currentWindow_) {
  125. return;
  126. }
  127. // As of CEA-708-E, the G3 group only contains 1 character. It's a
  128. // [CC] character which has no unicode value on 0xa0.
  129. if (controlCode != 0xa0) {
  130. // Similar to G2, the spec decrees an underline if char is unsupported.
  131. this.currentWindow_.setCharacter('_');
  132. return;
  133. }
  134. this.currentWindow_.setCharacter('[CC]');
  135. }
  136. /**
  137. * Handles C0 group data.
  138. * @param {number} controlCode
  139. * @param {number} pts
  140. * @return {?shaka.extern.ICaptionDecoder.ClosedCaption}
  141. * @private
  142. */
  143. handleC0_(controlCode, pts) {
  144. // All these commands pertain to the current window, so ensure it exists.
  145. if (!this.currentWindow_) {
  146. return null;
  147. }
  148. const window = this.currentWindow_;
  149. let parsedClosedCaption = null;
  150. // Note: This decoder ignores the "ETX" (end of text) control code. Since
  151. // this is JavaScript, a '\0' is not needed to terminate a string.
  152. switch (controlCode) {
  153. case shaka.cea.Cea708Service.ASCII_BACKSPACE:
  154. window.backspace();
  155. break;
  156. case shaka.cea.Cea708Service.ASCII_CARRIAGE_RETURN:
  157. // Force out the buffer, since the top row could be lost.
  158. if (window.isVisible()) {
  159. parsedClosedCaption = window.forceEmit(pts, this.serviceNumber_);
  160. }
  161. window.carriageReturn();
  162. break;
  163. case shaka.cea.Cea708Service.ASCII_HOR_CARRIAGE_RETURN:
  164. // Force out the buffer, a row will be erased.
  165. if (window.isVisible()) {
  166. parsedClosedCaption = window.forceEmit(pts, this.serviceNumber_);
  167. }
  168. window.horizontalCarriageReturn();
  169. break;
  170. case shaka.cea.Cea708Service.ASCII_FORM_FEED:
  171. // Clear window and move pen to (0,0).
  172. // Force emit if the window is visible.
  173. if (window.isVisible()) {
  174. parsedClosedCaption = window.forceEmit(pts, this.serviceNumber_);
  175. }
  176. window.resetMemory();
  177. window.setPenLocation(0, 0);
  178. break;
  179. }
  180. return parsedClosedCaption;
  181. }
  182. /**
  183. * Processes C1 group data.
  184. * These are caption commands.
  185. * @param {!shaka.cea.DtvccPacket} dtvccPacket
  186. * @param {number} captionCommand
  187. * @param {number} pts in seconds
  188. * @return {?shaka.extern.ICaptionDecoder.ClosedCaption}
  189. * @throws {!shaka.util.Error} a possible out-of-range buffer read.
  190. * @private
  191. */
  192. handleC1_(dtvccPacket, captionCommand, pts) {
  193. // Note: This decoder ignores delay and delayCancel control codes in the C1.
  194. // group. These control codes delay processing of data for a set amount of
  195. // time, however this decoder processes that data immediately.
  196. if (captionCommand >= 0x80 && captionCommand <= 0x87) {
  197. const windowNum = captionCommand & 0x07;
  198. this.setCurrentWindow_(windowNum);
  199. } else if (captionCommand === 0x88) {
  200. const bitmap = dtvccPacket.readByte().value;
  201. return this.clearWindows_(bitmap, pts);
  202. } else if (captionCommand === 0x89) {
  203. const bitmap = dtvccPacket.readByte().value;
  204. this.displayWindows_(bitmap, pts);
  205. } else if (captionCommand === 0x8a) {
  206. const bitmap = dtvccPacket.readByte().value;
  207. return this.hideWindows_(bitmap, pts);
  208. } else if (captionCommand === 0x8b) {
  209. const bitmap = dtvccPacket.readByte().value;
  210. return this.toggleWindows_(bitmap, pts);
  211. } else if (captionCommand === 0x8c) {
  212. const bitmap = dtvccPacket.readByte().value;
  213. return this.deleteWindows_(bitmap, pts);
  214. } else if (captionCommand === 0x8f) {
  215. return this.reset_(pts);
  216. } else if (captionCommand === 0x90) {
  217. this.setPenAttributes_(dtvccPacket);
  218. } else if (captionCommand === 0x91) {
  219. this.setPenColor_(dtvccPacket);
  220. } else if (captionCommand === 0x92) {
  221. this.setPenLocation_(dtvccPacket);
  222. } else if (captionCommand === 0x97) {
  223. this.setWindowAttributes_(dtvccPacket);
  224. } else if (captionCommand >= 0x98 && captionCommand <= 0x9f) {
  225. const windowNum = (captionCommand & 0x0f) - 8;
  226. this.defineWindow_(dtvccPacket, windowNum, pts);
  227. }
  228. return null;
  229. }
  230. /**
  231. * Handles C2 group data.
  232. * @param {!shaka.cea.DtvccPacket} dtvccPacket
  233. * @param {number} controlCode
  234. * @private
  235. */
  236. handleC2_(dtvccPacket, controlCode) {
  237. // As of the CEA-708-E spec there are no commands on the C2 table, but if
  238. // seen, then the appropriate number of bytes must be skipped as per spec.
  239. if (controlCode >= 0x08 && controlCode <= 0x0f) {
  240. dtvccPacket.skip(1);
  241. } else if (controlCode >= 0x10 && controlCode <= 0x17) {
  242. dtvccPacket.skip(2);
  243. } else if (controlCode >= 0x18 && controlCode <= 0x1f) {
  244. dtvccPacket.skip(3);
  245. }
  246. }
  247. /**
  248. * Handles C3 group data.
  249. * @param {!shaka.cea.DtvccPacket} dtvccPacket
  250. * @param {number} controlCode
  251. * @private
  252. */
  253. handleC3_(dtvccPacket, controlCode) {
  254. // As of the CEA-708-E spec there are no commands on the C3 table, but if
  255. // seen, then the appropriate number of bytes must be skipped as per spec.
  256. if (controlCode >= 0x80 && controlCode <= 0x87) {
  257. dtvccPacket.skip(4);
  258. } else if (controlCode >= 0x88 && controlCode <= 0x8f) {
  259. dtvccPacket.skip(5);
  260. }
  261. }
  262. /**
  263. * @param {number} windowNum
  264. * @private
  265. */
  266. setCurrentWindow_(windowNum) {
  267. // If the window isn't created, ignore the command.
  268. if (!this.windows_[windowNum]) {
  269. return;
  270. }
  271. this.currentWindow_ = this.windows_[windowNum];
  272. }
  273. /**
  274. * Yields each non-null window specified in the 8-bit bitmap.
  275. * @param {number} bitmap 8 bits corresponding to each of the 8 windows.
  276. * @return {!Array.<number>}
  277. * @private
  278. */
  279. getSpecifiedWindowIds_(bitmap) {
  280. const ids = [];
  281. for (let i = 0; i < 8; i++) {
  282. const windowSpecified = (bitmap & 0x01) === 0x01;
  283. if (windowSpecified && this.windows_[i]) {
  284. ids.push(i);
  285. }
  286. bitmap >>= 1;
  287. }
  288. return ids;
  289. }
  290. /**
  291. * @param {number} windowsBitmap
  292. * @param {number} pts
  293. * @return {?shaka.extern.ICaptionDecoder.ClosedCaption}
  294. * @private
  295. */
  296. clearWindows_(windowsBitmap, pts) {
  297. let parsedClosedCaption = null;
  298. // Clears windows from the 8 bit bitmap.
  299. for (const windowId of this.getSpecifiedWindowIds_(windowsBitmap)) {
  300. // If window visible and being cleared, emit buffer and reset start time!
  301. const window = this.windows_[windowId];
  302. if (window.isVisible()) {
  303. parsedClosedCaption = window.forceEmit(pts, this.serviceNumber_);
  304. }
  305. window.resetMemory();
  306. }
  307. return parsedClosedCaption;
  308. }
  309. /**
  310. * @param {number} windowsBitmap
  311. * @param {number} pts
  312. * @private
  313. */
  314. displayWindows_(windowsBitmap, pts) {
  315. // Displays windows from the 8 bit bitmap.
  316. for (const windowId of this.getSpecifiedWindowIds_(windowsBitmap)) {
  317. const window = this.windows_[windowId];
  318. if (!window.isVisible()) {
  319. // We are turning on the visibility, set the start time.
  320. window.setStartTime(pts);
  321. }
  322. window.display();
  323. }
  324. }
  325. /**
  326. * @param {number} windowsBitmap
  327. * @param {number} pts
  328. * @return {?shaka.extern.ICaptionDecoder.ClosedCaption}
  329. * @private
  330. */
  331. hideWindows_(windowsBitmap, pts) {
  332. let parsedClosedCaption = null;
  333. // Hides windows from the 8 bit bitmap.
  334. for (const windowId of this.getSpecifiedWindowIds_(windowsBitmap)) {
  335. const window = this.windows_[windowId];
  336. if (window.isVisible()) {
  337. // We are turning off the visibility, emit!
  338. parsedClosedCaption = window.forceEmit(pts, this.serviceNumber_);
  339. }
  340. window.hide();
  341. }
  342. return parsedClosedCaption;
  343. }
  344. /**
  345. * @param {number} windowsBitmap
  346. * @param {number} pts
  347. * @return {?shaka.extern.ICaptionDecoder.ClosedCaption}
  348. * @private
  349. */
  350. toggleWindows_(windowsBitmap, pts) {
  351. let parsedClosedCaption = null;
  352. // Toggles windows from the 8 bit bitmap.
  353. for (const windowId of this.getSpecifiedWindowIds_(windowsBitmap)) {
  354. const window = this.windows_[windowId];
  355. if (window.isVisible()) {
  356. // We are turning off the visibility, emit!
  357. parsedClosedCaption = window.forceEmit(pts, this.serviceNumber_);
  358. } else {
  359. // We are turning on visibility, set the start time.
  360. window.setStartTime(pts);
  361. }
  362. window.toggle();
  363. }
  364. return parsedClosedCaption;
  365. }
  366. /**
  367. * @param {number} windowsBitmap
  368. * @param {number} pts
  369. * @return {?shaka.extern.ICaptionDecoder.ClosedCaption}
  370. * @private
  371. */
  372. deleteWindows_(windowsBitmap, pts) {
  373. let parsedClosedCaption = null;
  374. // Deletes windows from the 8 bit bitmap.
  375. for (const windowId of this.getSpecifiedWindowIds_(windowsBitmap)) {
  376. const window = this.windows_[windowId];
  377. if (window.isVisible()) {
  378. // We are turning off the visibility, emit!
  379. parsedClosedCaption = window.forceEmit(pts, this.serviceNumber_);
  380. }
  381. // Delete the window from the list of windows
  382. this.windows_[windowId] = null;
  383. }
  384. return parsedClosedCaption;
  385. }
  386. /**
  387. * Emits anything currently present in any of the windows, and then
  388. * deletes all windows, cancels all delays, reinitializes the service.
  389. * @param {number} pts
  390. * @return {?shaka.extern.ICaptionDecoder.ClosedCaption}
  391. * @private
  392. */
  393. reset_(pts) {
  394. const allWindowsBitmap = 0xff; // All windows should be deleted.
  395. const caption = this.deleteWindows_(allWindowsBitmap, pts);
  396. this.clear();
  397. return caption;
  398. }
  399. /**
  400. * Clears the state of the service completely.
  401. */
  402. clear() {
  403. this.currentWindow_ = null;
  404. this.windows_ = [null, null, null, null, null, null, null, null];
  405. }
  406. /**
  407. * @param {!shaka.cea.DtvccPacket} dtvccPacket
  408. * @throws {!shaka.util.Error}
  409. * @private
  410. */
  411. setPenAttributes_(dtvccPacket) {
  412. // Two bytes follow. For the purpose of this decoder, we are only concerned
  413. // with byte 2, which is of the form |I|U|EDTYP|FNTAG|.
  414. // I (1 bit): Italics toggle.
  415. // U (1 bit): Underline toggle.
  416. // EDTYP (3 bits): Edge type (unused in this decoder).
  417. // FNTAG (3 bits): Font tag (unused in this decoder).
  418. // More info at https://en.wikipedia.org/wiki/CEA-708#SetPenAttributes_(0x90_+_2_bytes)
  419. dtvccPacket.skip(1); // Skip first byte
  420. const attrByte2 = dtvccPacket.readByte().value;
  421. if (!this.currentWindow_) {
  422. return;
  423. }
  424. const italics = (attrByte2 & 0x80) > 0;
  425. const underline = (attrByte2 & 0x40) > 0;
  426. this.currentWindow_.setPenItalics(italics);
  427. this.currentWindow_.setPenUnderline(underline);
  428. }
  429. /**
  430. * @param {!shaka.cea.DtvccPacket} dtvccPacket
  431. * @throws {!shaka.util.Error}
  432. * @private
  433. */
  434. setPenColor_(dtvccPacket) {
  435. // Read foreground and background properties.
  436. const foregroundByte = dtvccPacket.readByte().value;
  437. const backgroundByte = dtvccPacket.readByte().value;
  438. dtvccPacket.skip(1); // Edge color not supported, skip it.
  439. if (!this.currentWindow_) {
  440. return;
  441. }
  442. // Byte semantics are described at the following link:
  443. // https://en.wikipedia.org/wiki/CEA-708#SetPenColor_(0x91_+_3_bytes)
  444. // Foreground color properties: |FOP|F_R|F_G|F_B|.
  445. const foregroundBlue = foregroundByte & 0x03;
  446. const foregroundGreen = (foregroundByte & 0x0c) >> 2;
  447. const foregroundRed = (foregroundByte & 0x30) >> 4;
  448. // Background color properties: |BOP|B_R|B_G|B_B|.
  449. const backgroundBlue = backgroundByte & 0x03;
  450. const backgroundGreen = (backgroundByte & 0x0c) >> 2;
  451. const backgroundRed = (backgroundByte & 0x30) >> 4;
  452. const foregroundColor = this.rgbColorToHex_(
  453. foregroundRed, foregroundGreen, foregroundBlue);
  454. const backgroundColor = this.rgbColorToHex_(
  455. backgroundRed, backgroundGreen, backgroundBlue);
  456. this.currentWindow_.setPenTextColor(foregroundColor);
  457. this.currentWindow_.setPenBackgroundColor(backgroundColor);
  458. }
  459. /**
  460. * @param {!shaka.cea.DtvccPacket} dtvccPacket
  461. * @throws {!shaka.util.Error}
  462. * @private
  463. */
  464. setPenLocation_(dtvccPacket) {
  465. // Following 2 bytes take the following form:
  466. // b1 = |0|0|0|0|ROW| and b2 = |0|0|COLUMN|
  467. const locationByte1 = dtvccPacket.readByte().value;
  468. const locationByte2 = dtvccPacket.readByte().value;
  469. if (!this.currentWindow_) {
  470. return;
  471. }
  472. const row = locationByte1 & 0x0f;
  473. const col = locationByte2 & 0x3f;
  474. this.currentWindow_.setPenLocation(row, col);
  475. }
  476. /**
  477. * @param {!shaka.cea.DtvccPacket} dtvccPacket
  478. * @throws {!shaka.util.Error}
  479. * @private
  480. */
  481. setWindowAttributes_(dtvccPacket) {
  482. // 4 bytes follow, with the following form:
  483. // Byte 1 contains fill-color information. Unused in this decoder.
  484. // Byte 2 contains border color information. Unused in this decoder.
  485. // Byte 3 contains justification information. In this decoder, we only use
  486. // the last 2 bits, which specifies text justification on the screen.
  487. // Byte 4 is special effects. Unused in this decoder.
  488. // More info at https://en.wikipedia.org/wiki/CEA-708#SetWindowAttributes_(0x97_+_4_bytes)
  489. dtvccPacket.skip(1); // Fill color not supported, skip.
  490. dtvccPacket.skip(1); // Border colors not supported, skip.
  491. const b3 = dtvccPacket.readByte().value;
  492. dtvccPacket.skip(1); // Effects not supported, skip.
  493. if (!this.currentWindow_) {
  494. return;
  495. }
  496. // Word wrap is outdated as of CEA-708-E, so we ignore those bits.
  497. // Extract the text justification and set it on the window.
  498. const justification =
  499. /** @type {!shaka.cea.Cea708Window.TextJustification} */ (b3 & 0x03);
  500. this.currentWindow_.setJustification(justification);
  501. }
  502. /**
  503. * @param {!shaka.cea.DtvccPacket} dtvccPacket
  504. * @param {number} windowNum
  505. * @param {number} pts
  506. * @throws {!shaka.util.Error}
  507. * @private
  508. */
  509. defineWindow_(dtvccPacket, windowNum, pts) {
  510. // Create the window if it doesn't exist.
  511. const windowAlreadyExists = this.windows_[windowNum] !== null;
  512. if (!windowAlreadyExists) {
  513. const window = new shaka.cea.Cea708Window(windowNum, this.serviceNumber_);
  514. window.setStartTime(pts);
  515. this.windows_[windowNum] = window;
  516. }
  517. // 6 Bytes follow, with the following form:
  518. // b1 = |0|0|V|R|C|PRIOR| , b2 = |P|VERT_ANCHOR| , b3 = |HOR_ANCHOR|
  519. // b4 = |ANC_ID|ROW_CNT| , b5 = |0|0|COL_COUNT| , b6 = |0|0|WNSTY|PNSTY|
  520. // Semantics of these bytes at https://en.wikipedia.org/wiki/CEA-708#DefineWindow07_(0x98-0x9F,_+_6_bytes)
  521. const b1 = dtvccPacket.readByte().value;
  522. const b2 = dtvccPacket.readByte().value;
  523. const b3 = dtvccPacket.readByte().value;
  524. const b4 = dtvccPacket.readByte().value;
  525. const b5 = dtvccPacket.readByte().value;
  526. const b6 = dtvccPacket.readByte().value;
  527. // As per 8.4.7 of CEA-708-E, row locks and column locks are to be ignored.
  528. // So this decoder will ignore these values.
  529. const visible = (b1 & 0x20) > 0;
  530. const verticalAnchor = b2 & 0x7f;
  531. const relativeToggle = (b2 & 0x80) > 0;
  532. const horAnchor = b3;
  533. const rowCount = (b4 & 0x0f) + 1; // Spec says to add 1.
  534. const anchorId = (b4 & 0xf0) >> 4;
  535. const colCount = (b5 & 0x3f) + 1; // Spec says to add 1.
  536. // If pen style = 0 AND window previously existed, keep its pen style.
  537. // Otherwise, change the pen style (For now, just reset to the default pen).
  538. // TODO add support for predefined pen styles and fonts.
  539. const penStyle = b6 & 0x07;
  540. if (!windowAlreadyExists || penStyle !== 0) {
  541. this.windows_[windowNum].resetPen();
  542. }
  543. this.windows_[windowNum].defineWindow(visible, verticalAnchor,
  544. horAnchor, anchorId, relativeToggle, rowCount, colCount);
  545. // Set the current window to the newly defined window.
  546. this.currentWindow_ = this.windows_[windowNum];
  547. }
  548. /**
  549. * Maps 64 possible CEA-708 colors to 8 CSS colors.
  550. * @param {number} red value from 0-3
  551. * @param {number} green value from 0-3
  552. * @param {number} blue value from 0-3
  553. * @return {string}
  554. * @private
  555. */
  556. rgbColorToHex_(red, green, blue) {
  557. // Rather than supporting 64 colors, this decoder supports 8 colors and
  558. // gets the closest color, as per 9.19 of CEA-708-E. This is because some
  559. // colors on television such as white, are often sent with lower intensity
  560. // and often appear dull/greyish on the browser, making them hard to read.
  561. // As per CEA-708-E 9.19, these mappings will map 64 colors to 8 colors.
  562. const colorMapping = {0: 0, 1: 0, 2: 1, 3: 1};
  563. red = colorMapping[red];
  564. green = colorMapping[green];
  565. blue = colorMapping[blue];
  566. const colorCode = (red << 2) | (green << 1) | blue;
  567. return shaka.cea.Cea708Service.Colors[colorCode];
  568. }
  569. };
  570. /**
  571. * @private @const {number}
  572. */
  573. shaka.cea.Cea708Service.ASCII_BACKSPACE = 0x08;
  574. /**
  575. * @private @const {number}
  576. */
  577. shaka.cea.Cea708Service.ASCII_FORM_FEED = 0x0c;
  578. /**
  579. * @private @const {number}
  580. */
  581. shaka.cea.Cea708Service.ASCII_CARRIAGE_RETURN = 0x0d;
  582. /**
  583. * @private @const {number}
  584. */
  585. shaka.cea.Cea708Service.ASCII_HOR_CARRIAGE_RETURN = 0x0e;
  586. /**
  587. * For extended control codes in block_data on CEA-708, byte 1 is 0x10.
  588. * @private @const {number}
  589. */
  590. shaka.cea.Cea708Service.EXT_CEA708_CTRL_CODE_BYTE1 = 0x10;
  591. /**
  592. * Holds characters mapping for bytes that are G2 control codes.
  593. * @private @const {!Map<number, string>}
  594. */
  595. shaka.cea.Cea708Service.G2Charset = new Map([
  596. [0x20, ' '], [0x21, '\xa0'], [0x25, '…'], [0x2a, 'Š'], [0x2c, 'Œ'],
  597. [0x30, '█'], [0x31, '‘'], [0x32, '’'], [0x33, '“'], [0x34, '”'],
  598. [0x35, '•'], [0x39, '™'], [0x3a, 'š'], [0x3c, 'œ'], [0x3d, '℠'],
  599. [0x3f, 'Ÿ'], [0x76, '⅛'], [0x77, '⅜'], [0x78, '⅝'], [0x79, '⅞'],
  600. [0x7a, '│'], [0x7b, '┐'], [0x7c, '└'], [0x7d, '─'], [0x7e, '┘'], [0x7f, '┌'],
  601. ]);
  602. /**
  603. * An array of 8 colors that 64 colors can be quantized to. Order here matters.
  604. * @private @const {!Array<string>}
  605. */
  606. shaka.cea.Cea708Service.Colors = [
  607. 'black', 'blue', 'green', 'cyan',
  608. 'red', 'magenta', 'yellow', 'white',
  609. ];
  610. /**
  611. * CEA-708 closed captions byte.
  612. * @typedef {{
  613. * pts: number,
  614. * type: number,
  615. * value: number,
  616. * order: number
  617. * }}
  618. *
  619. * @property {number} pts
  620. * Presentation timestamp (in second) at which this packet was received.
  621. * @property {number} type
  622. * Type of the byte. Either 2 or 3, DTVCC Packet Data or a DTVCC Packet Start.
  623. * @property {number} value The byte containing data relevant to the packet.
  624. * @property {number} order
  625. * A number indicating the order this packet was received in a sequence
  626. * of packets. Used to break ties in a stable sorting algorithm
  627. */
  628. shaka.cea.Cea708Service.Cea708Byte;