Source: lib/cea/cea708_window.js

  1. /*! @license
  2. * Shaka Player
  3. * Copyright 2016 Google LLC
  4. * SPDX-License-Identifier: Apache-2.0
  5. */
  6. goog.provide('shaka.cea.Cea708Window');
  7. goog.require('shaka.cea.CeaUtils');
  8. goog.require('shaka.cea.CeaUtils.StyledChar');
  9. goog.require('shaka.text.Cue');
  10. goog.require('shaka.text.CueRegion');
  11. /**
  12. * CEA-708 Window. Each CEA-708 service owns 8 of these.
  13. */
  14. shaka.cea.Cea708Window = class {
  15. /**
  16. * @param {number} windowNum
  17. */
  18. constructor(windowNum, parentService) {
  19. /**
  20. * Number for the parent service (1 - 63).
  21. * @private {number}
  22. */
  23. this.parentService_ = parentService;
  24. /**
  25. * A number from 0 - 7 indicating the window number in the
  26. * service that owns this window.
  27. * @private {number}
  28. */
  29. this.windowNum_ = windowNum;
  30. /**
  31. * Indicates whether this window is visible.
  32. * @private {boolean}
  33. */
  34. this.visible_ = false;
  35. /**
  36. * Indicates whether the horizontal and vertical anchors coordinates specify
  37. * a percentage of the screen, or physical coordinates on the screen.
  38. * @private {boolean}
  39. */
  40. this.relativeToggle_ = false;
  41. /**
  42. * Horizontal anchor. Loosely corresponds to a WebVTT viewport X anchor.
  43. * @private {number}
  44. */
  45. this.horizontalAnchor_ = 0;
  46. /**
  47. * Vertical anchor. Loosely corresponds to a WebVTT viewport Y anchor.
  48. * @private {number}
  49. */
  50. this.verticalAnchor_ = 0;
  51. /**
  52. * If valid, ranges from 0 to 8, specifying one of 9 locations on window:
  53. * 0________1________2
  54. * | | |
  55. * 3________4________5
  56. * | | |
  57. * 6________7________8
  58. * Diagram is valid as per CEA-708-E section 8.4.4.
  59. * Each of these locations corresponds to a WebVTT region's "region anchor".
  60. * @private {number}
  61. */
  62. this.anchorId_ = 0;
  63. /**
  64. * Indicates the number of rows in this window's buffer/memory.
  65. * @private {number}
  66. */
  67. this.rowCount_ = 0;
  68. /**
  69. * Indicates the number of columns in this window's buffer/memory.
  70. * @private {number}
  71. */
  72. this.colCount_ = 0;
  73. /**
  74. * Center by default.
  75. * @private {!shaka.cea.Cea708Window.TextJustification}
  76. */
  77. this.justification_ = shaka.cea.Cea708Window.TextJustification.CENTER;
  78. /**
  79. * An array of rows of styled characters, representing the
  80. * current text and styling of text in this window.
  81. * @private {!Array<!Array<?shaka.cea.CeaUtils.StyledChar>>}
  82. */
  83. this.memory_ = [];
  84. /**
  85. * @private {number}
  86. */
  87. this.startTime_ = 0;
  88. /**
  89. * Row that the current pen is pointing at.
  90. * @private {number}
  91. */
  92. this.row_ = 0;
  93. /**
  94. * Column that the current pen is pointing at.
  95. * @private {number}
  96. */
  97. this.col_ = 0;
  98. /**
  99. * Indicates whether the current pen position is italicized.
  100. * @private {boolean}
  101. */
  102. this.italics_ = false;
  103. /**
  104. * Indicates whether the current pen position is underlined.
  105. * @private {boolean}
  106. */
  107. this.underline_ = false;
  108. /**
  109. * Indicates the text color at the current pen position.
  110. * @private {string}
  111. */
  112. this.textColor_ = shaka.cea.CeaUtils.DEFAULT_TXT_COLOR;
  113. /**
  114. * Indicates the background color at the current pen position.
  115. * @private {string}
  116. */
  117. this.backgroundColor_ = shaka.cea.CeaUtils.DEFAULT_BG_COLOR;
  118. this.resetMemory();
  119. }
  120. /**
  121. * @param {boolean} visible
  122. * @param {number} verticalAnchor
  123. * @param {number} horizontalAnchor
  124. * @param {number} anchorId
  125. * @param {boolean} relativeToggle
  126. * @param {number} rowCount
  127. * @param {number} colCount
  128. */
  129. defineWindow(visible, verticalAnchor, horizontalAnchor, anchorId,
  130. relativeToggle, rowCount, colCount) {
  131. this.visible_ = visible;
  132. this.verticalAnchor_ = verticalAnchor;
  133. this.horizontalAnchor_ = horizontalAnchor;
  134. this.anchorId_ = anchorId;
  135. this.relativeToggle_ = relativeToggle;
  136. this.rowCount_ = rowCount;
  137. this.colCount_ = colCount;
  138. }
  139. /**
  140. * Resets the memory buffer.
  141. */
  142. resetMemory() {
  143. this.memory_ = [];
  144. for (let i = 0; i < shaka.cea.Cea708Window.MAX_ROWS; i++) {
  145. this.memory_.push(this.createNewRow_());
  146. }
  147. }
  148. /**
  149. * Allocates and returns a new row.
  150. * @return {!Array<?shaka.cea.CeaUtils.StyledChar>}
  151. * @private
  152. */
  153. createNewRow_() {
  154. const row = [];
  155. for (let j = 0; j < shaka.cea.Cea708Window.MAX_COLS; j++) {
  156. row.push(null);
  157. }
  158. return row;
  159. }
  160. /**
  161. * Sets the unicode value for a char at the current pen location.
  162. * @param {string} char
  163. */
  164. setCharacter(char) {
  165. // Check if the pen is out of bounds.
  166. if (!this.isPenInBounds_()) {
  167. return;
  168. }
  169. const cea708Char = new shaka.cea.CeaUtils.StyledChar(
  170. char, this.underline_, this.italics_,
  171. this.backgroundColor_, this.textColor_);
  172. this.memory_[this.row_][this.col_] = cea708Char;
  173. // Increment column
  174. this.col_ ++;
  175. }
  176. /**
  177. * Erases a character from the buffer and moves the pen back.
  178. */
  179. backspace() {
  180. if (!this.isPenInBounds_()) {
  181. return;
  182. }
  183. // Check if a backspace can be done.
  184. if (this.col_ <= 0 && this.row_ <= 0) {
  185. return;
  186. }
  187. if (this.col_ <= 0) {
  188. // Move pen back a row.
  189. this.col_ = this.colCount_ - 1;
  190. this.row_--;
  191. } else {
  192. // Move pen back a column.
  193. this.col_--;
  194. }
  195. // Erase the character occupied at that position.
  196. this.memory_[this.row_][this.col_] = null;
  197. }
  198. /**
  199. * @private
  200. */
  201. isPenInBounds_() {
  202. const inRowBounds = this.row_ < this.rowCount_ && this.row_ >= 0;
  203. const inColBounds = this.col_ < this.colCount_ && this.col_ >= 0;
  204. return inRowBounds && inColBounds;
  205. }
  206. /**
  207. * @return {boolean}
  208. */
  209. isVisible() {
  210. return this.visible_;
  211. }
  212. /**
  213. * Moves up <count> rows in the buffer.
  214. * @param {number} count
  215. * @private
  216. */
  217. moveUpRows_(count) {
  218. let dst = 0; // Row each row should be moved to.
  219. // Move existing rows up by <count>.
  220. for (let i = count; i < shaka.cea.Cea708Window.MAX_ROWS; i++, dst++) {
  221. this.memory_[dst] = this.memory_[i];
  222. }
  223. // Create <count> new rows at the bottom.
  224. for (let i = 0; i < count; i++, dst++) {
  225. this.memory_[dst] = this.createNewRow_();
  226. }
  227. }
  228. /**
  229. * Handles CR. Increments row - if last row, "roll up" all rows by one.
  230. */
  231. carriageReturn() {
  232. if (this.row_ + 1 >= this.rowCount_) {
  233. this.moveUpRows_(1);
  234. this.col_ = 0;
  235. return;
  236. }
  237. this.row_++;
  238. this.col_ = 0;
  239. }
  240. /**
  241. * Handles HCR. Moves the pen to the beginning of the cur. row and clears it.
  242. */
  243. horizontalCarriageReturn() {
  244. this.memory_[this.row_] = this.createNewRow_();
  245. this.col_ = 0;
  246. }
  247. /**
  248. * @param {number} endTime
  249. * @param {number} serviceNumber Number of the service emitting this caption.
  250. * @return {?shaka.extern.ICaptionDecoder.ClosedCaption}
  251. */
  252. forceEmit(endTime, serviceNumber) {
  253. const stream = `svc${serviceNumber}`;
  254. const TextJustification = shaka.cea.Cea708Window.TextJustification;
  255. const topLevelCue = new shaka.text.Cue(
  256. this.startTime_, endTime, /* payload= */ '');
  257. if (this.justification_ === TextJustification.LEFT) {
  258. // LEFT justified.
  259. topLevelCue.textAlign = shaka.text.Cue.textAlign.LEFT;
  260. } else if (this.justification_ === TextJustification.RIGHT) {
  261. // RIGHT justified.
  262. topLevelCue.textAlign = shaka.text.Cue.textAlign.RIGHT;
  263. } else {
  264. // CENTER justified. Both FULL and CENTER are handled as CENTER justified.
  265. topLevelCue.textAlign = shaka.text.Cue.textAlign.CENTER;
  266. }
  267. this.adjustRegion_(topLevelCue.region);
  268. const caption = shaka.cea.CeaUtils.getParsedCaption(
  269. topLevelCue, stream, this.memory_, this.startTime_, endTime);
  270. if (caption) {
  271. // If a caption is being emitted, then the next caption's start time
  272. // should be no less than this caption's end time.
  273. this.setStartTime(endTime);
  274. }
  275. return caption;
  276. }
  277. /**
  278. * @param {number} row
  279. * @param {number} col
  280. */
  281. setPenLocation(row, col) {
  282. this.row_ = row;
  283. this.col_ = col;
  284. }
  285. /**
  286. * @param {string} backgroundColor
  287. */
  288. setPenBackgroundColor(backgroundColor) {
  289. this.backgroundColor_ = backgroundColor;
  290. }
  291. /**
  292. * @param {string} textColor
  293. */
  294. setPenTextColor(textColor) {
  295. this.textColor_ = textColor;
  296. }
  297. /**
  298. * @param {boolean} underline
  299. */
  300. setPenUnderline(underline) {
  301. this.underline_ = underline;
  302. }
  303. /**
  304. * @param {boolean} italics
  305. */
  306. setPenItalics(italics) {
  307. this.italics_ = italics;
  308. }
  309. /** Reset the pen to 0,0 with default styling. */
  310. resetPen() {
  311. this.row_ = 0;
  312. this.col_ = 0;
  313. this.underline_ = false;
  314. this.italics_ = false;
  315. this.textColor_ = shaka.cea.CeaUtils.DEFAULT_TXT_COLOR;
  316. this.backgroundColor_ = shaka.cea.CeaUtils.DEFAULT_BG_COLOR;
  317. }
  318. /**
  319. * @param {!shaka.cea.Cea708Window.TextJustification} justification
  320. */
  321. setJustification(justification) {
  322. this.justification_ = justification;
  323. }
  324. /**
  325. * Sets the window to visible.
  326. */
  327. display() {
  328. this.visible_ = true;
  329. }
  330. /**
  331. * Sets the window to invisible.
  332. */
  333. hide() {
  334. this.visible_ = false;
  335. }
  336. /**
  337. * Toggles the visibility of the window.
  338. */
  339. toggle() {
  340. this.visible_ = !this.visible_;
  341. }
  342. /**
  343. * Sets the start time for the cue to be emitted.
  344. * @param {number} pts
  345. */
  346. setStartTime(pts) {
  347. this.startTime_ = pts;
  348. }
  349. /**
  350. * Support window positioning by mapping anchor related values to CueRegion.
  351. * https://dvcs.w3.org/hg/text-tracks/raw-file/default/608toVTT/608toVTT.html#positioning-in-cea-708
  352. * @param {shaka.text.CueRegion} region
  353. * @private
  354. */
  355. adjustRegion_(region) {
  356. if (this.parentService_) {
  357. region.id += 'svc' + this.parentService_;
  358. }
  359. region.id += 'win' + this.windowNum_;
  360. region.height = this.rowCount_;
  361. region.width = this.colCount_;
  362. region.heightUnits = shaka.text.CueRegion.units.LINES;
  363. region.widthUnits = shaka.text.CueRegion.units.LINES;
  364. region.viewportAnchorX = this.horizontalAnchor_;
  365. region.viewportAnchorY = this.verticalAnchor_;
  366. // WebVTT's region viewport anchors are technically always in percentages.
  367. // However, we don't know the aspect ratio of the video at this point,
  368. // which determines how we interpret the horizontal anchor.
  369. // So, we expose the additonal flag to reflect whether these viewport anchor
  370. // values can be used be used as is or should be converted to percentages.
  371. region.viewportAnchorUnits = this.relativeToggle_ ?
  372. shaka.text.CueRegion.units.PERCENTAGE : shaka.text.CueRegion.units.LINES;
  373. const AnchorId = shaka.cea.Cea708Window.AnchorId;
  374. switch (this.anchorId_) {
  375. case AnchorId.UPPER_LEFT:
  376. region.regionAnchorX = 0;
  377. region.regionAnchorY = 0;
  378. break;
  379. case AnchorId.UPPER_CENTER:
  380. region.regionAnchorX = 50;
  381. region.regionAnchorY = 0;
  382. break;
  383. case AnchorId.UPPER_RIGHT:
  384. region.regionAnchorX = 100;
  385. region.regionAnchorY = 0;
  386. break;
  387. case AnchorId.MIDDLE_LEFT:
  388. region.regionAnchorX = 0;
  389. region.regionAnchorY = 50;
  390. break;
  391. case AnchorId.MIDDLE_CENTER:
  392. region.regionAnchorX = 50;
  393. region.regionAnchorY = 50;
  394. break;
  395. case AnchorId.MIDDLE_RIGHT:
  396. region.regionAnchorX = 100;
  397. region.regionAnchorY = 50;
  398. break;
  399. case AnchorId.LOWER_LEFT:
  400. region.regionAnchorX = 0;
  401. region.regionAnchorY = 100;
  402. break;
  403. case AnchorId.LOWER_CENTER:
  404. region.regionAnchorX = 50;
  405. region.regionAnchorY = 100;
  406. break;
  407. case AnchorId.LOWER_RIGHT:
  408. region.regionAnchorX = 100;
  409. region.regionAnchorY = 100;
  410. break;
  411. }
  412. }
  413. };
  414. /**
  415. * Caption type.
  416. * @const @enum {number}
  417. */
  418. shaka.cea.Cea708Window.TextJustification = {
  419. LEFT: 0,
  420. RIGHT: 1,
  421. CENTER: 2,
  422. FULL: 3,
  423. };
  424. /**
  425. * Possible AnchorId values.
  426. * @const @enum {number}
  427. */
  428. shaka.cea.Cea708Window.AnchorId = {
  429. UPPER_LEFT: 0,
  430. UPPER_CENTER: 1,
  431. UPPER_RIGHT: 2,
  432. MIDDLE_LEFT: 3,
  433. MIDDLE_CENTER: 4,
  434. MIDDLE_RIGHT: 5,
  435. LOWER_LEFT: 6,
  436. LOWER_CENTER: 7,
  437. LOWER_RIGHT: 8,
  438. };
  439. /**
  440. * Can be indexed 0-31 for 4:3 format, and 0-41 for 16:9 formats.
  441. * Thus the absolute maximum is 42 columns for the 16:9 format.
  442. * @private @const {number}
  443. */
  444. shaka.cea.Cea708Window.MAX_COLS = 42;
  445. /**
  446. * Maximum of 16 rows that can be indexed from 0 to 15.
  447. * @private @const {number}
  448. */
  449. shaka.cea.Cea708Window.MAX_ROWS = 16;