Source: lib/text/ttml_text_parser.js

  1. /*! @license
  2. * Shaka Player
  3. * Copyright 2016 Google LLC
  4. * SPDX-License-Identifier: Apache-2.0
  5. */
  6. goog.provide('shaka.text.TtmlTextParser');
  7. goog.require('goog.asserts');
  8. goog.require('goog.Uri');
  9. goog.require('shaka.log');
  10. goog.require('shaka.text.Cue');
  11. goog.require('shaka.text.CueRegion');
  12. goog.require('shaka.text.TextEngine');
  13. goog.require('shaka.util.ArrayUtils');
  14. goog.require('shaka.util.Error');
  15. goog.require('shaka.util.StringUtils');
  16. goog.require('shaka.util.TXml');
  17. /**
  18. * @implements {shaka.extern.TextParser}
  19. * @export
  20. */
  21. shaka.text.TtmlTextParser = class {
  22. /**
  23. * @override
  24. * @export
  25. */
  26. parseInit(data) {
  27. goog.asserts.assert(false, 'TTML does not have init segments');
  28. }
  29. /**
  30. * @override
  31. * @export
  32. */
  33. setSequenceMode(sequenceMode) {
  34. // Unused.
  35. }
  36. /**
  37. * @override
  38. * @export
  39. */
  40. setManifestType(manifestType) {
  41. // Unused.
  42. }
  43. /**
  44. * @override
  45. * @export
  46. */
  47. parseMedia(data, time, uri) {
  48. const TtmlTextParser = shaka.text.TtmlTextParser;
  49. const TXml = shaka.util.TXml;
  50. const ttpNs = TtmlTextParser.parameterNs_;
  51. const ttsNs = TtmlTextParser.styleNs_;
  52. const str = shaka.util.StringUtils.fromUTF8(data);
  53. const cues = [];
  54. // dont try to parse empty string as
  55. // DOMParser will not throw error but return an errored xml
  56. if (str == '') {
  57. return cues;
  58. }
  59. const tt = TXml.parseXmlString(str, 'tt');
  60. if (!tt) {
  61. throw new shaka.util.Error(
  62. shaka.util.Error.Severity.CRITICAL,
  63. shaka.util.Error.Category.TEXT,
  64. shaka.util.Error.Code.INVALID_XML,
  65. 'Failed to parse TTML.');
  66. }
  67. const body = TXml.getElementsByTagName(tt, 'body')[0];
  68. if (!body) {
  69. return [];
  70. }
  71. // Get the framerate, subFrameRate and frameRateMultiplier if applicable.
  72. const frameRate = TXml.getAttributeNSList(tt, ttpNs, 'frameRate');
  73. const subFrameRate = TXml.getAttributeNSList(
  74. tt, ttpNs, 'subFrameRate');
  75. const frameRateMultiplier =
  76. TXml.getAttributeNSList(tt, ttpNs, 'frameRateMultiplier');
  77. const tickRate = TXml.getAttributeNSList(tt, ttpNs, 'tickRate');
  78. const cellResolution = TXml.getAttributeNSList(
  79. tt, ttpNs, 'cellResolution');
  80. const spaceStyle = tt.attributes['xml:space'] || 'default';
  81. const extent = TXml.getAttributeNSList(tt, ttsNs, 'extent');
  82. if (spaceStyle != 'default' && spaceStyle != 'preserve') {
  83. throw new shaka.util.Error(
  84. shaka.util.Error.Severity.CRITICAL,
  85. shaka.util.Error.Category.TEXT,
  86. shaka.util.Error.Code.INVALID_XML,
  87. 'Invalid xml:space value: ' + spaceStyle);
  88. }
  89. const collapseMultipleSpaces = spaceStyle == 'default';
  90. const rateInfo = new TtmlTextParser.RateInfo_(
  91. frameRate, subFrameRate, frameRateMultiplier, tickRate);
  92. const cellResolutionInfo =
  93. TtmlTextParser.getCellResolution_(cellResolution);
  94. const metadata = TXml.getElementsByTagName(tt, 'metadata')[0];
  95. const metadataElements = metadata ? metadata.children : [];
  96. const styles = TXml.getElementsByTagName(tt, 'style');
  97. const regionElements = TXml.getElementsByTagName(tt, 'region');
  98. const cueRegions = [];
  99. for (const region of regionElements) {
  100. const cueRegion =
  101. TtmlTextParser.parseCueRegion_(region, styles, extent);
  102. if (cueRegion) {
  103. cueRegions.push(cueRegion);
  104. }
  105. }
  106. // A <body> element should only contain <div> elements, not <p> or <span>
  107. // elements. We used to allow this, but it is non-compliant, and the
  108. // loose nature of our previous parser made it difficult to implement TTML
  109. // nesting more fully.
  110. if (TXml.findChildren(body, 'p').length) {
  111. throw new shaka.util.Error(
  112. shaka.util.Error.Severity.CRITICAL,
  113. shaka.util.Error.Category.TEXT,
  114. shaka.util.Error.Code.INVALID_TEXT_CUE,
  115. '<p> can only be inside <div> in TTML');
  116. }
  117. for (const div of TXml.findChildren(body, 'div')) {
  118. // A <div> element should only contain <p>, not <span>.
  119. if (TXml.findChildren(div, 'span').length) {
  120. throw new shaka.util.Error(
  121. shaka.util.Error.Severity.CRITICAL,
  122. shaka.util.Error.Category.TEXT,
  123. shaka.util.Error.Code.INVALID_TEXT_CUE,
  124. '<span> can only be inside <p> in TTML');
  125. }
  126. }
  127. const cue = TtmlTextParser.parseCue_(
  128. body, time, rateInfo, metadataElements, styles,
  129. regionElements, cueRegions, collapseMultipleSpaces,
  130. cellResolutionInfo, /* parentCueElement= */ null,
  131. /* isContent= */ false, uri);
  132. if (cue) {
  133. // According to the TTML spec, backgrounds default to transparent.
  134. // So default the background of the top-level element to transparent.
  135. // Nested elements may override that background color already.
  136. if (!cue.backgroundColor) {
  137. cue.backgroundColor = 'transparent';
  138. }
  139. cues.push(cue);
  140. }
  141. return cues;
  142. }
  143. /**
  144. * Parses a TTML node into a Cue.
  145. *
  146. * @param {!shaka.extern.xml.Node} cueNode
  147. * @param {shaka.extern.TextParser.TimeContext} timeContext
  148. * @param {!shaka.text.TtmlTextParser.RateInfo_} rateInfo
  149. * @param {!Array.<!shaka.extern.xml.Node>} metadataElements
  150. * @param {!Array.<!shaka.extern.xml.Node>} styles
  151. * @param {!Array.<!shaka.extern.xml.Node>} regionElements
  152. * @param {!Array.<!shaka.text.CueRegion>} cueRegions
  153. * @param {boolean} collapseMultipleSpaces
  154. * @param {?{columns: number, rows: number}} cellResolution
  155. * @param {?shaka.extern.xml.Node} parentCueElement
  156. * @param {boolean} isContent
  157. * @param {?(string|undefined)} uri
  158. * @return {shaka.text.Cue}
  159. * @private
  160. */
  161. static parseCue_(
  162. cueNode, timeContext, rateInfo, metadataElements, styles, regionElements,
  163. cueRegions, collapseMultipleSpaces, cellResolution, parentCueElement,
  164. isContent, uri) {
  165. const TXml = shaka.util.TXml;
  166. const StringUtils = shaka.util.StringUtils;
  167. /** @type {shaka.extern.xml.Node} */
  168. let cueElement;
  169. /** @type {?shaka.extern.xml.Node} */
  170. let parentElement = parentCueElement;
  171. if (TXml.isText(cueNode)) {
  172. if (!isContent) {
  173. // Ignore text elements outside the content. For example, whitespace
  174. // on the same lexical level as the <p> elements, in a document with
  175. // xml:space="preserve", should not be renderer.
  176. return null;
  177. }
  178. // This should generate an "anonymous span" according to the TTML spec.
  179. // So pretend the element was a <span>. parentElement was set above, so
  180. // we should still be able to correctly traverse up for timing
  181. // information later.
  182. /** @type {shaka.extern.xml.Node} */
  183. const span = {
  184. tagName: 'span',
  185. children: [TXml.getTextContents(cueNode)],
  186. attributes: {},
  187. parent: null,
  188. };
  189. cueElement = span;
  190. } else {
  191. cueElement = cueNode;
  192. }
  193. goog.asserts.assert(cueElement, 'cueElement should be non-null!');
  194. let imageElement = null;
  195. for (const nameSpace of shaka.text.TtmlTextParser.smpteNsList_) {
  196. imageElement = shaka.text.TtmlTextParser.getElementsFromCollection_(
  197. cueElement, 'backgroundImage', metadataElements, '#',
  198. nameSpace)[0];
  199. if (imageElement) {
  200. break;
  201. }
  202. }
  203. let imageUri = null;
  204. const backgroundImage = TXml.getAttributeNSList(
  205. cueElement,
  206. shaka.text.TtmlTextParser.smpteNsList_,
  207. 'backgroundImage');
  208. if (uri && backgroundImage && !backgroundImage.startsWith('#')) {
  209. const baseUri = new goog.Uri(uri);
  210. const relativeUri = new goog.Uri(backgroundImage);
  211. const newUri = baseUri.resolve(relativeUri).toString();
  212. if (newUri) {
  213. imageUri = newUri;
  214. }
  215. }
  216. if (cueNode.tagName == 'p' || imageElement || imageUri) {
  217. isContent = true;
  218. }
  219. const parentIsContent = isContent;
  220. const spaceStyle = cueElement.attributes['xml:space'] ||
  221. (collapseMultipleSpaces ? 'default' : 'preserve');
  222. const localCollapseMultipleSpaces = spaceStyle == 'default';
  223. // Parse any nested cues first.
  224. const isLeafNode = cueElement.children.every(TXml.isText);
  225. const nestedCues = [];
  226. if (!isLeafNode) {
  227. // Otherwise, recurse into the children. Text nodes will convert into
  228. // anonymous spans, which will then be leaf nodes.
  229. for (const childNode of cueElement.children) {
  230. const nestedCue = shaka.text.TtmlTextParser.parseCue_(
  231. childNode,
  232. timeContext,
  233. rateInfo,
  234. metadataElements,
  235. styles,
  236. regionElements,
  237. cueRegions,
  238. localCollapseMultipleSpaces,
  239. cellResolution,
  240. cueElement,
  241. isContent,
  242. uri,
  243. );
  244. // This node may or may not generate a nested cue.
  245. if (nestedCue) {
  246. nestedCues.push(nestedCue);
  247. }
  248. }
  249. }
  250. const isNested = /** @type {boolean} */ (parentCueElement != null);
  251. const textContent = TXml.getTextContents(cueElement);
  252. // In this regex, "\S" means "non-whitespace character".
  253. const hasTextContent = cueElement.children.length &&
  254. textContent &&
  255. /\S/.test(textContent);
  256. const hasTimeAttributes =
  257. cueElement.attributes['begin'] ||
  258. cueElement.attributes['end'] ||
  259. cueElement.attributes['dur'];
  260. if (!hasTimeAttributes && !hasTextContent && cueElement.tagName != 'br' &&
  261. nestedCues.length == 0) {
  262. if (!isNested) {
  263. // Disregards empty <p> elements without time attributes nor content.
  264. // <p begin="..." smpte:backgroundImage="..." /> will go through,
  265. // as some information could be held by its attributes.
  266. // <p /> won't, as it would not be displayed.
  267. return null;
  268. } else if (localCollapseMultipleSpaces) {
  269. // Disregards empty anonymous spans when (local) trim is true.
  270. return null;
  271. }
  272. }
  273. // Get local time attributes.
  274. let {start, end} = shaka.text.TtmlTextParser.parseTime_(
  275. cueElement, rateInfo);
  276. // Resolve local time relative to parent elements. Time elements can appear
  277. // all the way up to 'body', but not 'tt'.
  278. while (parentElement && TXml.isNode(parentElement) &&
  279. parentElement.tagName != 'tt') {
  280. ({start, end} = shaka.text.TtmlTextParser.resolveTime_(
  281. parentElement, rateInfo, start, end));
  282. parentElement =
  283. /** @type {shaka.extern.xml.Node} */ (parentElement.parent);
  284. }
  285. if (start == null) {
  286. start = 0;
  287. }
  288. start += timeContext.periodStart;
  289. // If end is null, that means the duration is effectively infinite.
  290. if (end == null) {
  291. end = Infinity;
  292. } else {
  293. end += timeContext.periodStart;
  294. }
  295. // Clip times to segment boundaries.
  296. // https://github.com/shaka-project/shaka-player/issues/4631
  297. start = Math.max(start, timeContext.segmentStart);
  298. end = Math.min(end, timeContext.segmentEnd);
  299. if (!hasTimeAttributes && nestedCues.length > 0) {
  300. // If no time is defined for this cue, base the timing information on
  301. // the time of the nested cues. In the case of multiple nested cues with
  302. // different start times, it is the text displayer's responsibility to
  303. // make sure that only the appropriate nested cue is drawn at any given
  304. // time.
  305. start = Infinity;
  306. end = 0;
  307. for (const cue of nestedCues) {
  308. start = Math.min(start, cue.startTime);
  309. end = Math.max(end, cue.endTime);
  310. }
  311. }
  312. if (cueElement.tagName == 'br') {
  313. const cue = new shaka.text.Cue(start, end, '');
  314. cue.lineBreak = true;
  315. return cue;
  316. }
  317. let payload = '';
  318. if (isLeafNode) {
  319. // If the childNodes are all text, this is a leaf node. Get the payload.
  320. payload = StringUtils.htmlUnescape(
  321. shaka.util.TXml.getTextContents(cueElement) || '');
  322. if (localCollapseMultipleSpaces) {
  323. // Collapse multiple spaces into one.
  324. payload = payload.replace(/\s+/g, ' ');
  325. }
  326. }
  327. const cue = new shaka.text.Cue(start, end, payload);
  328. cue.nestedCues = nestedCues;
  329. if (!isContent) {
  330. // If this is not a <p> element or a <div> with images, and it has no
  331. // parent that was a <p> element, then it's part of the outer containers
  332. // (e.g. the <body> or a normal <div> element within it).
  333. cue.isContainer = true;
  334. }
  335. if (cellResolution) {
  336. cue.cellResolution = cellResolution;
  337. }
  338. // Get other properties if available.
  339. const regionElement = shaka.text.TtmlTextParser.getElementsFromCollection_(
  340. cueElement, 'region', regionElements, /* prefix= */ '')[0];
  341. // Do not actually apply that region unless it is non-inherited, though.
  342. // This makes it so that, if a parent element has a region, the children
  343. // don't also all independently apply the positioning of that region.
  344. if (cueElement.attributes['region']) {
  345. if (regionElement && regionElement.attributes['xml:id']) {
  346. const regionId = regionElement.attributes['xml:id'];
  347. cue.region = cueRegions.filter((region) => region.id == regionId)[0];
  348. }
  349. }
  350. let regionElementForStyle = regionElement;
  351. if (parentCueElement && isNested && !cueElement.attributes['region'] &&
  352. !cueElement.attributes['style']) {
  353. regionElementForStyle =
  354. shaka.text.TtmlTextParser.getElementsFromCollection_(
  355. parentCueElement, 'region', regionElements, /* prefix= */ '')[0];
  356. }
  357. shaka.text.TtmlTextParser.addStyle_(
  358. cue,
  359. cueElement,
  360. regionElementForStyle,
  361. /** @type {!shaka.extern.xml.Node} */(imageElement),
  362. imageUri,
  363. styles,
  364. /** isNested= */ parentIsContent, // "nested in a <div>" doesn't count.
  365. /** isLeaf= */ (nestedCues.length == 0));
  366. return cue;
  367. }
  368. /**
  369. * Parses an Element into a TextTrackCue or VTTCue.
  370. *
  371. * @param {!shaka.extern.xml.Node} regionElement
  372. * @param {!Array.<!shaka.extern.xml.Node>} styles
  373. * Defined in the top of tt element and used principally for images.
  374. * @param {?string} globalExtent
  375. * @return {shaka.text.CueRegion}
  376. * @private
  377. */
  378. static parseCueRegion_(regionElement, styles, globalExtent) {
  379. const TtmlTextParser = shaka.text.TtmlTextParser;
  380. const region = new shaka.text.CueRegion();
  381. const id = regionElement.attributes['xml:id'];
  382. if (!id) {
  383. shaka.log.warning('TtmlTextParser parser encountered a region with ' +
  384. 'no id. Region will be ignored.');
  385. return null;
  386. }
  387. region.id = id;
  388. let globalResults = null;
  389. if (globalExtent) {
  390. globalResults = TtmlTextParser.percentValues_.exec(globalExtent) ||
  391. TtmlTextParser.pixelValues_.exec(globalExtent);
  392. }
  393. const globalWidth = globalResults ? Number(globalResults[1]) : null;
  394. const globalHeight = globalResults ? Number(globalResults[2]) : null;
  395. let results = null;
  396. let percentage = null;
  397. const extent = TtmlTextParser.getStyleAttributeFromRegion_(
  398. regionElement, styles, 'extent');
  399. if (extent) {
  400. percentage = TtmlTextParser.percentValues_.exec(extent);
  401. results = percentage || TtmlTextParser.pixelValues_.exec(extent);
  402. if (results != null) {
  403. region.width = Number(results[1]);
  404. region.height = Number(results[2]);
  405. if (!percentage) {
  406. if (globalWidth != null) {
  407. region.width = region.width * 100 / globalWidth;
  408. }
  409. if (globalHeight != null) {
  410. region.height = region.height * 100 / globalHeight;
  411. }
  412. }
  413. region.widthUnits = percentage || globalWidth != null ?
  414. shaka.text.CueRegion.units.PERCENTAGE :
  415. shaka.text.CueRegion.units.PX;
  416. region.heightUnits = percentage || globalHeight != null ?
  417. shaka.text.CueRegion.units.PERCENTAGE :
  418. shaka.text.CueRegion.units.PX;
  419. }
  420. }
  421. const origin = TtmlTextParser.getStyleAttributeFromRegion_(
  422. regionElement, styles, 'origin');
  423. if (origin) {
  424. percentage = TtmlTextParser.percentValues_.exec(origin);
  425. results = percentage || TtmlTextParser.pixelValues_.exec(origin);
  426. if (results != null) {
  427. region.viewportAnchorX = Number(results[1]);
  428. region.viewportAnchorY = Number(results[2]);
  429. if (!percentage) {
  430. if (globalHeight != null) {
  431. region.viewportAnchorY = region.viewportAnchorY * 100 /
  432. globalHeight;
  433. }
  434. if (globalWidth != null) {
  435. region.viewportAnchorX = region.viewportAnchorX * 100 /
  436. globalWidth;
  437. }
  438. } else if (!extent) {
  439. region.width = 100 - region.viewportAnchorX;
  440. region.widthUnits = shaka.text.CueRegion.units.PERCENTAGE;
  441. region.height = 100 - region.viewportAnchorY;
  442. region.heightUnits = shaka.text.CueRegion.units.PERCENTAGE;
  443. }
  444. region.viewportAnchorUnits = percentage || globalWidth != null ?
  445. shaka.text.CueRegion.units.PERCENTAGE :
  446. shaka.text.CueRegion.units.PX;
  447. }
  448. }
  449. return region;
  450. }
  451. /**
  452. * Adds applicable style properties to a cue.
  453. *
  454. * @param {!shaka.text.Cue} cue
  455. * @param {!shaka.extern.xml.Node} cueElement
  456. * @param {shaka.extern.xml.Node} region
  457. * @param {shaka.extern.xml.Node} imageElement
  458. * @param {?string} imageUri
  459. * @param {!Array.<!shaka.extern.xml.Node>} styles
  460. * @param {boolean} isNested
  461. * @param {boolean} isLeaf
  462. * @private
  463. */
  464. static addStyle_(
  465. cue, cueElement, region, imageElement, imageUri, styles,
  466. isNested, isLeaf) {
  467. const TtmlTextParser = shaka.text.TtmlTextParser;
  468. const TXml = shaka.util.TXml;
  469. const Cue = shaka.text.Cue;
  470. // Styles should be inherited from regions, if a style property is not
  471. // associated with a Content element (or an anonymous span).
  472. const shouldInheritRegionStyles = isNested || isLeaf;
  473. const direction = TtmlTextParser.getStyleAttribute_(
  474. cueElement, region, styles, 'direction', shouldInheritRegionStyles);
  475. if (direction == 'rtl') {
  476. cue.direction = Cue.direction.HORIZONTAL_RIGHT_TO_LEFT;
  477. }
  478. // Direction attribute specifies one-dimentional writing direction
  479. // (left to right or right to left). Writing mode specifies that
  480. // plus whether text is vertical or horizontal.
  481. // They should not contradict each other. If they do, we give
  482. // preference to writing mode.
  483. const writingMode = TtmlTextParser.getStyleAttribute_(
  484. cueElement, region, styles, 'writingMode', shouldInheritRegionStyles);
  485. // Set cue's direction if the text is horizontal, and cue's writingMode if
  486. // it's vertical.
  487. if (writingMode == 'tb' || writingMode == 'tblr') {
  488. cue.writingMode = Cue.writingMode.VERTICAL_LEFT_TO_RIGHT;
  489. } else if (writingMode == 'tbrl') {
  490. cue.writingMode = Cue.writingMode.VERTICAL_RIGHT_TO_LEFT;
  491. } else if (writingMode == 'rltb' || writingMode == 'rl') {
  492. cue.direction = Cue.direction.HORIZONTAL_RIGHT_TO_LEFT;
  493. } else if (writingMode) {
  494. cue.direction = Cue.direction.HORIZONTAL_LEFT_TO_RIGHT;
  495. }
  496. const align = TtmlTextParser.getStyleAttribute_(
  497. cueElement, region, styles, 'textAlign', true);
  498. if (align) {
  499. cue.positionAlign = TtmlTextParser.textAlignToPositionAlign_[align];
  500. cue.lineAlign = TtmlTextParser.textAlignToLineAlign_[align];
  501. goog.asserts.assert(align.toUpperCase() in Cue.textAlign,
  502. align.toUpperCase() + ' Should be in Cue.textAlign values!');
  503. cue.textAlign = Cue.textAlign[align.toUpperCase()];
  504. } else {
  505. // Default value is START in the TTML spec: https://bit.ly/32OGmvo
  506. // But to make the subtitle render consitent with other players and the
  507. // shaka.text.Cue we use CENTER
  508. cue.textAlign = Cue.textAlign.CENTER;
  509. }
  510. const displayAlign = TtmlTextParser.getStyleAttribute_(
  511. cueElement, region, styles, 'displayAlign', true);
  512. if (displayAlign) {
  513. goog.asserts.assert(displayAlign.toUpperCase() in Cue.displayAlign,
  514. displayAlign.toUpperCase() +
  515. ' Should be in Cue.displayAlign values!');
  516. cue.displayAlign = Cue.displayAlign[displayAlign.toUpperCase()];
  517. }
  518. const color = TtmlTextParser.getStyleAttribute_(
  519. cueElement, region, styles, 'color', shouldInheritRegionStyles);
  520. if (color) {
  521. cue.color = color;
  522. }
  523. // Background color should not be set on a container. If this is a nested
  524. // cue, you can set the background. If it's a top-level that happens to
  525. // also be a leaf, you can set the background.
  526. // See https://github.com/shaka-project/shaka-player/issues/2623
  527. // This used to be handled in the displayer, but that is confusing. The Cue
  528. // structure should reflect what you want to happen in the displayer, and
  529. // the displayer shouldn't have to know about TTML.
  530. const backgroundColor = TtmlTextParser.getStyleAttribute_(
  531. cueElement, region, styles, 'backgroundColor',
  532. shouldInheritRegionStyles);
  533. if (backgroundColor) {
  534. cue.backgroundColor = backgroundColor;
  535. }
  536. const border = TtmlTextParser.getStyleAttribute_(
  537. cueElement, region, styles, 'border', shouldInheritRegionStyles);
  538. if (border) {
  539. cue.border = border;
  540. }
  541. const fontFamily = TtmlTextParser.getStyleAttribute_(
  542. cueElement, region, styles, 'fontFamily', shouldInheritRegionStyles);
  543. // See https://github.com/sandflow/imscJS/blob/1.1.3/src/main/js/html.js#L1384
  544. if (fontFamily) {
  545. switch (fontFamily) {
  546. case 'monospaceSerif':
  547. cue.fontFamily = 'Courier New,Liberation Mono,Courier,monospace';
  548. break;
  549. case 'proportionalSansSerif':
  550. cue.fontFamily = 'Arial,Helvetica,Liberation Sans,sans-serif';
  551. break;
  552. case 'sansSerif':
  553. cue.fontFamily = 'sans-serif';
  554. break;
  555. case 'monospaceSansSerif':
  556. cue.fontFamily = 'Consolas,monospace';
  557. break;
  558. case 'proportionalSerif':
  559. cue.fontFamily = 'serif';
  560. break;
  561. default:
  562. cue.fontFamily = fontFamily;
  563. break;
  564. }
  565. }
  566. const fontWeight = TtmlTextParser.getStyleAttribute_(
  567. cueElement, region, styles, 'fontWeight', shouldInheritRegionStyles);
  568. if (fontWeight && fontWeight == 'bold') {
  569. cue.fontWeight = Cue.fontWeight.BOLD;
  570. }
  571. const wrapOption = TtmlTextParser.getStyleAttribute_(
  572. cueElement, region, styles, 'wrapOption', shouldInheritRegionStyles);
  573. if (wrapOption && wrapOption == 'noWrap') {
  574. cue.wrapLine = false;
  575. } else {
  576. cue.wrapLine = true;
  577. }
  578. const lineHeight = TtmlTextParser.getStyleAttribute_(
  579. cueElement, region, styles, 'lineHeight', shouldInheritRegionStyles);
  580. if (lineHeight && lineHeight.match(TtmlTextParser.unitValues_)) {
  581. cue.lineHeight = lineHeight;
  582. }
  583. const fontSize = TtmlTextParser.getStyleAttribute_(
  584. cueElement, region, styles, 'fontSize', shouldInheritRegionStyles);
  585. if (fontSize) {
  586. const isValidFontSizeUnit =
  587. fontSize.match(TtmlTextParser.unitValues_) ||
  588. fontSize.match(TtmlTextParser.percentValue_);
  589. if (isValidFontSizeUnit) {
  590. cue.fontSize = fontSize;
  591. }
  592. }
  593. const fontStyle = TtmlTextParser.getStyleAttribute_(
  594. cueElement, region, styles, 'fontStyle', shouldInheritRegionStyles);
  595. if (fontStyle) {
  596. goog.asserts.assert(fontStyle.toUpperCase() in Cue.fontStyle,
  597. fontStyle.toUpperCase() +
  598. ' Should be in Cue.fontStyle values!');
  599. cue.fontStyle = Cue.fontStyle[fontStyle.toUpperCase()];
  600. }
  601. if (imageElement) {
  602. // According to the spec, we should use imageType (camelCase), but
  603. // historically we have checked for imagetype (lowercase).
  604. // This was the case since background image support was first introduced
  605. // in PR #1859, in April 2019, and first released in v2.5.0.
  606. // Now we check for both, although only imageType (camelCase) is to spec.
  607. const backgroundImageType =
  608. imageElement.attributes['imageType'] ||
  609. imageElement.attributes['imagetype'];
  610. const backgroundImageEncoding = imageElement.attributes['encoding'];
  611. const backgroundImageData = (TXml.getTextContents(imageElement)).trim();
  612. if (backgroundImageType == 'PNG' &&
  613. backgroundImageEncoding == 'Base64' &&
  614. backgroundImageData) {
  615. cue.backgroundImage = 'data:image/png;base64,' + backgroundImageData;
  616. }
  617. } else if (imageUri) {
  618. cue.backgroundImage = imageUri;
  619. }
  620. const textOutline = TtmlTextParser.getStyleAttribute_(
  621. cueElement, region, styles, 'textOutline', shouldInheritRegionStyles);
  622. if (textOutline) {
  623. // tts:textOutline isn't natively supported by browsers, but it can be
  624. // mostly replicated using the non-standard -webkit-text-stroke-width and
  625. // -webkit-text-stroke-color properties.
  626. const split = textOutline.split(' ');
  627. if (split[0].match(TtmlTextParser.unitValues_)) {
  628. // There is no defined color, so default to the text color.
  629. cue.textStrokeColor = cue.color;
  630. } else {
  631. cue.textStrokeColor = split[0];
  632. split.shift();
  633. }
  634. if (split[0] && split[0].match(TtmlTextParser.unitValues_)) {
  635. cue.textStrokeWidth = split[0];
  636. } else {
  637. // If there is no width, or the width is not a number, don't draw a
  638. // border.
  639. cue.textStrokeColor = '';
  640. }
  641. // There is an optional blur radius also, but we have no way of
  642. // replicating that, so ignore it.
  643. }
  644. const letterSpacing = TtmlTextParser.getStyleAttribute_(
  645. cueElement, region, styles, 'letterSpacing', shouldInheritRegionStyles);
  646. if (letterSpacing && letterSpacing.match(TtmlTextParser.unitValues_)) {
  647. cue.letterSpacing = letterSpacing;
  648. }
  649. const linePadding = TtmlTextParser.getStyleAttribute_(
  650. cueElement, region, styles, 'linePadding', shouldInheritRegionStyles);
  651. if (linePadding && linePadding.match(TtmlTextParser.unitValues_)) {
  652. cue.linePadding = linePadding;
  653. }
  654. const opacity = TtmlTextParser.getStyleAttribute_(
  655. cueElement, region, styles, 'opacity', shouldInheritRegionStyles);
  656. if (opacity) {
  657. cue.opacity = parseFloat(opacity);
  658. }
  659. // Text decoration is an array of values which can come both from the
  660. // element's style or be inherited from elements' parent nodes. All of those
  661. // values should be applied as long as they don't contradict each other. If
  662. // they do, elements' own style gets preference.
  663. const textDecorationRegion = TtmlTextParser.getStyleAttributeFromRegion_(
  664. region, styles, 'textDecoration');
  665. if (textDecorationRegion) {
  666. TtmlTextParser.addTextDecoration_(cue, textDecorationRegion);
  667. }
  668. const textDecorationElement = TtmlTextParser.getStyleAttributeFromElement_(
  669. cueElement, styles, 'textDecoration');
  670. if (textDecorationElement) {
  671. TtmlTextParser.addTextDecoration_(cue, textDecorationElement);
  672. }
  673. const textCombine = TtmlTextParser.getStyleAttribute_(
  674. cueElement, region, styles, 'textCombine', shouldInheritRegionStyles);
  675. if (textCombine) {
  676. cue.textCombineUpright = textCombine;
  677. }
  678. const ruby = TtmlTextParser.getStyleAttribute_(
  679. cueElement, region, styles, 'ruby', shouldInheritRegionStyles);
  680. switch (ruby) {
  681. case 'container':
  682. cue.rubyTag = 'ruby';
  683. break;
  684. case 'text':
  685. cue.rubyTag = 'rt';
  686. break;
  687. }
  688. }
  689. /**
  690. * Parses text decoration values and adds/removes them to/from the cue.
  691. *
  692. * @param {!shaka.text.Cue} cue
  693. * @param {string} decoration
  694. * @private
  695. */
  696. static addTextDecoration_(cue, decoration) {
  697. const Cue = shaka.text.Cue;
  698. for (const value of decoration.split(' ')) {
  699. switch (value) {
  700. case 'underline':
  701. if (!cue.textDecoration.includes(Cue.textDecoration.UNDERLINE)) {
  702. cue.textDecoration.push(Cue.textDecoration.UNDERLINE);
  703. }
  704. break;
  705. case 'noUnderline':
  706. if (cue.textDecoration.includes(Cue.textDecoration.UNDERLINE)) {
  707. shaka.util.ArrayUtils.remove(cue.textDecoration,
  708. Cue.textDecoration.UNDERLINE);
  709. }
  710. break;
  711. case 'lineThrough':
  712. if (!cue.textDecoration.includes(Cue.textDecoration.LINE_THROUGH)) {
  713. cue.textDecoration.push(Cue.textDecoration.LINE_THROUGH);
  714. }
  715. break;
  716. case 'noLineThrough':
  717. if (cue.textDecoration.includes(Cue.textDecoration.LINE_THROUGH)) {
  718. shaka.util.ArrayUtils.remove(cue.textDecoration,
  719. Cue.textDecoration.LINE_THROUGH);
  720. }
  721. break;
  722. case 'overline':
  723. if (!cue.textDecoration.includes(Cue.textDecoration.OVERLINE)) {
  724. cue.textDecoration.push(Cue.textDecoration.OVERLINE);
  725. }
  726. break;
  727. case 'noOverline':
  728. if (cue.textDecoration.includes(Cue.textDecoration.OVERLINE)) {
  729. shaka.util.ArrayUtils.remove(cue.textDecoration,
  730. Cue.textDecoration.OVERLINE);
  731. }
  732. break;
  733. }
  734. }
  735. }
  736. /**
  737. * Finds a specified attribute on either the original cue element or its
  738. * associated region and returns the value if the attribute was found.
  739. *
  740. * @param {!shaka.extern.xml.Node} cueElement
  741. * @param {shaka.extern.xml.Node} region
  742. * @param {!Array.<!shaka.extern.xml.Node>} styles
  743. * @param {string} attribute
  744. * @param {boolean=} shouldInheritRegionStyles
  745. * @return {?string}
  746. * @private
  747. */
  748. static getStyleAttribute_(cueElement, region, styles, attribute,
  749. shouldInheritRegionStyles=true) {
  750. // An attribute can be specified on region level or in a styling block
  751. // associated with the region or original element.
  752. const TtmlTextParser = shaka.text.TtmlTextParser;
  753. const attr = TtmlTextParser.getStyleAttributeFromElement_(
  754. cueElement, styles, attribute);
  755. if (attr) {
  756. return attr;
  757. }
  758. if (shouldInheritRegionStyles) {
  759. return TtmlTextParser.getStyleAttributeFromRegion_(
  760. region, styles, attribute);
  761. }
  762. return null;
  763. }
  764. /**
  765. * Finds a specified attribute on the element's associated region
  766. * and returns the value if the attribute was found.
  767. *
  768. * @param {shaka.extern.xml.Node} region
  769. * @param {!Array.<!shaka.extern.xml.Node>} styles
  770. * @param {string} attribute
  771. * @return {?string}
  772. * @private
  773. */
  774. static getStyleAttributeFromRegion_(region, styles, attribute) {
  775. const TXml = shaka.util.TXml;
  776. const ttsNs = shaka.text.TtmlTextParser.styleNs_;
  777. if (!region) {
  778. return null;
  779. }
  780. const attr = TXml.getAttributeNSList(region, ttsNs, attribute);
  781. if (attr) {
  782. return attr;
  783. }
  784. return shaka.text.TtmlTextParser.getInheritedStyleAttribute_(
  785. region, styles, attribute);
  786. }
  787. /**
  788. * Finds a specified attribute on the cue element and returns the value
  789. * if the attribute was found.
  790. *
  791. * @param {!shaka.extern.xml.Node} cueElement
  792. * @param {!Array.<!shaka.extern.xml.Node>} styles
  793. * @param {string} attribute
  794. * @return {?string}
  795. * @private
  796. */
  797. static getStyleAttributeFromElement_(cueElement, styles, attribute) {
  798. const TXml = shaka.util.TXml;
  799. const ttsNs = shaka.text.TtmlTextParser.styleNs_;
  800. // Styling on elements should take precedence
  801. // over the main styling attributes
  802. const elementAttribute = TXml.getAttributeNSList(
  803. cueElement,
  804. ttsNs,
  805. attribute);
  806. if (elementAttribute) {
  807. return elementAttribute;
  808. }
  809. return shaka.text.TtmlTextParser.getInheritedStyleAttribute_(
  810. cueElement, styles, attribute);
  811. }
  812. /**
  813. * Finds a specified attribute on an element's styles and the styles those
  814. * styles inherit from.
  815. *
  816. * @param {!shaka.extern.xml.Node} element
  817. * @param {!Array.<!shaka.extern.xml.Node>} styles
  818. * @param {string} attribute
  819. * @return {?string}
  820. * @private
  821. */
  822. static getInheritedStyleAttribute_(element, styles, attribute) {
  823. const TXml = shaka.util.TXml;
  824. const ttsNs = shaka.text.TtmlTextParser.styleNs_;
  825. const ebuttsNs = shaka.text.TtmlTextParser.styleEbuttsNs_;
  826. const inheritedStyles =
  827. shaka.text.TtmlTextParser.getElementsFromCollection_(
  828. element, 'style', styles, /* prefix= */ '');
  829. let styleValue = null;
  830. // The last value in our styles stack takes the precedence over the others
  831. for (let i = 0; i < inheritedStyles.length; i++) {
  832. // Check ebu namespace first.
  833. let styleAttributeValue = TXml.getAttributeNS(
  834. inheritedStyles[i],
  835. ebuttsNs,
  836. attribute);
  837. if (!styleAttributeValue) {
  838. // Fall back to tts namespace.
  839. styleAttributeValue = TXml.getAttributeNSList(
  840. inheritedStyles[i],
  841. ttsNs,
  842. attribute);
  843. }
  844. if (!styleAttributeValue) {
  845. // Next, check inheritance.
  846. // Styles can inherit from other styles, so traverse up that chain.
  847. styleAttributeValue =
  848. shaka.text.TtmlTextParser.getStyleAttributeFromElement_(
  849. inheritedStyles[i], styles, attribute);
  850. }
  851. if (styleAttributeValue) {
  852. styleValue = styleAttributeValue;
  853. }
  854. }
  855. return styleValue;
  856. }
  857. /**
  858. * Selects items from |collection| whose id matches |attributeName|
  859. * from |element|.
  860. *
  861. * @param {shaka.extern.xml.Node} element
  862. * @param {string} attributeName
  863. * @param {!Array.<shaka.extern.xml.Node>} collection
  864. * @param {string} prefixName
  865. * @param {string=} nsName
  866. * @return {!Array.<!shaka.extern.xml.Node>}
  867. * @private
  868. */
  869. static getElementsFromCollection_(
  870. element, attributeName, collection, prefixName, nsName) {
  871. const items = [];
  872. if (!element || collection.length < 1) {
  873. return items;
  874. }
  875. const attributeValue = shaka.text.TtmlTextParser.getInheritedAttribute_(
  876. element, attributeName, nsName);
  877. if (attributeValue) {
  878. // There could be multiple items in one attribute
  879. // <span style="style1 style2">A cue</span>
  880. const itemNames = attributeValue.split(' ');
  881. for (const name of itemNames) {
  882. for (const item of collection) {
  883. if ((prefixName + item.attributes['xml:id']) == name) {
  884. items.push(item);
  885. break;
  886. }
  887. }
  888. }
  889. }
  890. return items;
  891. }
  892. /**
  893. * Traverses upwards from a given node until a given attribute is found.
  894. *
  895. * @param {!shaka.extern.xml.Node} element
  896. * @param {string} attributeName
  897. * @param {string=} nsName
  898. * @return {?string}
  899. * @private
  900. */
  901. static getInheritedAttribute_(element, attributeName, nsName) {
  902. let ret = null;
  903. const TXml = shaka.util.TXml;
  904. while (!ret) {
  905. ret = nsName ?
  906. TXml.getAttributeNS(element, nsName, attributeName) :
  907. element.attributes[attributeName];
  908. if (ret) {
  909. break;
  910. }
  911. // Element.parentNode can lead to XMLDocument, which is not an Element and
  912. // has no getAttribute().
  913. const parentNode = element.parent;
  914. if (parentNode) {
  915. element = parentNode;
  916. } else {
  917. break;
  918. }
  919. }
  920. return ret;
  921. }
  922. /**
  923. * Factor parent/ancestor time attributes into the parsed time of a
  924. * child/descendent.
  925. *
  926. * @param {!shaka.extern.xml.Node} parentElement
  927. * @param {!shaka.text.TtmlTextParser.RateInfo_} rateInfo
  928. * @param {?number} start The child's start time
  929. * @param {?number} end The child's end time
  930. * @return {{start: ?number, end: ?number}}
  931. * @private
  932. */
  933. static resolveTime_(parentElement, rateInfo, start, end) {
  934. const parentTime = shaka.text.TtmlTextParser.parseTime_(
  935. parentElement, rateInfo);
  936. if (start == null) {
  937. // No start time of your own? Inherit from the parent.
  938. start = parentTime.start;
  939. } else {
  940. // Otherwise, the start time is relative to the parent's start time.
  941. if (parentTime.start != null) {
  942. start += parentTime.start;
  943. }
  944. }
  945. if (end == null) {
  946. // No end time of your own? Inherit from the parent.
  947. end = parentTime.end;
  948. } else {
  949. // Otherwise, the end time is relative to the parent's _start_ time.
  950. // This is not a typo. Both times are relative to the parent's _start_.
  951. if (parentTime.start != null) {
  952. end += parentTime.start;
  953. }
  954. }
  955. return {start, end};
  956. }
  957. /**
  958. * Parse TTML time attributes from the given element.
  959. *
  960. * @param {!shaka.extern.xml.Node} element
  961. * @param {!shaka.text.TtmlTextParser.RateInfo_} rateInfo
  962. * @return {{start: ?number, end: ?number}}
  963. * @private
  964. */
  965. static parseTime_(element, rateInfo) {
  966. const start = shaka.text.TtmlTextParser.parseTimeAttribute_(
  967. element.attributes['begin'], rateInfo);
  968. let end = shaka.text.TtmlTextParser.parseTimeAttribute_(
  969. element.attributes['end'], rateInfo);
  970. const duration = shaka.text.TtmlTextParser.parseTimeAttribute_(
  971. element.attributes['dur'], rateInfo);
  972. if (end == null && duration != null) {
  973. end = start + duration;
  974. }
  975. return {start, end};
  976. }
  977. /**
  978. * Parses a TTML time from the given attribute text.
  979. *
  980. * @param {string} text
  981. * @param {!shaka.text.TtmlTextParser.RateInfo_} rateInfo
  982. * @return {?number}
  983. * @private
  984. */
  985. static parseTimeAttribute_(text, rateInfo) {
  986. let ret = null;
  987. const TtmlTextParser = shaka.text.TtmlTextParser;
  988. if (TtmlTextParser.timeColonFormatFrames_.test(text)) {
  989. ret = TtmlTextParser.parseColonTimeWithFrames_(rateInfo, text);
  990. } else if (TtmlTextParser.timeColonFormat_.test(text)) {
  991. ret = TtmlTextParser.parseTimeFromRegex_(
  992. TtmlTextParser.timeColonFormat_, text);
  993. } else if (TtmlTextParser.timeColonFormatMilliseconds_.test(text)) {
  994. ret = TtmlTextParser.parseTimeFromRegex_(
  995. TtmlTextParser.timeColonFormatMilliseconds_, text);
  996. } else if (TtmlTextParser.timeFramesFormat_.test(text)) {
  997. ret = TtmlTextParser.parseFramesTime_(rateInfo, text);
  998. } else if (TtmlTextParser.timeTickFormat_.test(text)) {
  999. ret = TtmlTextParser.parseTickTime_(rateInfo, text);
  1000. } else if (TtmlTextParser.timeHMSFormat_.test(text)) {
  1001. ret = TtmlTextParser.parseTimeFromRegex_(
  1002. TtmlTextParser.timeHMSFormat_, text);
  1003. } else if (text) {
  1004. // It's not empty or null, but it doesn't match a known format.
  1005. throw new shaka.util.Error(
  1006. shaka.util.Error.Severity.CRITICAL,
  1007. shaka.util.Error.Category.TEXT,
  1008. shaka.util.Error.Code.INVALID_TEXT_CUE,
  1009. 'Could not parse cue time range in TTML');
  1010. }
  1011. return ret;
  1012. }
  1013. /**
  1014. * Parses a TTML time in frame format.
  1015. *
  1016. * @param {!shaka.text.TtmlTextParser.RateInfo_} rateInfo
  1017. * @param {string} text
  1018. * @return {?number}
  1019. * @private
  1020. */
  1021. static parseFramesTime_(rateInfo, text) {
  1022. // 75f or 75.5f
  1023. const results = shaka.text.TtmlTextParser.timeFramesFormat_.exec(text);
  1024. const frames = Number(results[1]);
  1025. return frames / rateInfo.frameRate;
  1026. }
  1027. /**
  1028. * Parses a TTML time in tick format.
  1029. *
  1030. * @param {!shaka.text.TtmlTextParser.RateInfo_} rateInfo
  1031. * @param {string} text
  1032. * @return {?number}
  1033. * @private
  1034. */
  1035. static parseTickTime_(rateInfo, text) {
  1036. // 50t or 50.5t
  1037. const results = shaka.text.TtmlTextParser.timeTickFormat_.exec(text);
  1038. const ticks = Number(results[1]);
  1039. return ticks / rateInfo.tickRate;
  1040. }
  1041. /**
  1042. * Parses a TTML colon formatted time containing frames.
  1043. *
  1044. * @param {!shaka.text.TtmlTextParser.RateInfo_} rateInfo
  1045. * @param {string} text
  1046. * @return {?number}
  1047. * @private
  1048. */
  1049. static parseColonTimeWithFrames_(rateInfo, text) {
  1050. // 01:02:43:07 ('07' is frames) or 01:02:43:07.1 (subframes)
  1051. const results = shaka.text.TtmlTextParser.timeColonFormatFrames_.exec(text);
  1052. const hours = Number(results[1]);
  1053. const minutes = Number(results[2]);
  1054. let seconds = Number(results[3]);
  1055. let frames = Number(results[4]);
  1056. const subframes = Number(results[5]) || 0;
  1057. frames += subframes / rateInfo.subFrameRate;
  1058. seconds += frames / rateInfo.frameRate;
  1059. return seconds + (minutes * 60) + (hours * 3600);
  1060. }
  1061. /**
  1062. * Parses a TTML time with a given regex. Expects regex to be some
  1063. * sort of a time-matcher to match hours, minutes, seconds and milliseconds
  1064. *
  1065. * @param {!RegExp} regex
  1066. * @param {string} text
  1067. * @return {?number}
  1068. * @private
  1069. */
  1070. static parseTimeFromRegex_(regex, text) {
  1071. const results = regex.exec(text);
  1072. if (results == null || results[0] == '') {
  1073. return null;
  1074. }
  1075. // This capture is optional, but will still be in the array as undefined,
  1076. // in which case it is 0.
  1077. const hours = Number(results[1]) || 0;
  1078. const minutes = Number(results[2]) || 0;
  1079. const seconds = Number(results[3]) || 0;
  1080. const milliseconds = Number(results[4]) || 0;
  1081. return (milliseconds / 1000) + seconds + (minutes * 60) + (hours * 3600);
  1082. }
  1083. /**
  1084. * If ttp:cellResolution provided returns cell resolution info
  1085. * with number of columns and rows into which the Root Container
  1086. * Region area is divided
  1087. *
  1088. * @param {?string} cellResolution
  1089. * @return {?{columns: number, rows: number}}
  1090. * @private
  1091. */
  1092. static getCellResolution_(cellResolution) {
  1093. if (!cellResolution) {
  1094. return null;
  1095. }
  1096. const matches = /^(\d+) (\d+)$/.exec(cellResolution);
  1097. if (!matches) {
  1098. return null;
  1099. }
  1100. const columns = parseInt(matches[1], 10);
  1101. const rows = parseInt(matches[2], 10);
  1102. return {columns, rows};
  1103. }
  1104. };
  1105. /**
  1106. * @summary
  1107. * Contains information about frame/subframe rate
  1108. * and frame rate multiplier for time in frame format.
  1109. *
  1110. * @example 01:02:03:04(4 frames) or 01:02:03:04.1(4 frames, 1 subframe)
  1111. * @private
  1112. */
  1113. shaka.text.TtmlTextParser.RateInfo_ = class {
  1114. /**
  1115. * @param {?string} frameRate
  1116. * @param {?string} subFrameRate
  1117. * @param {?string} frameRateMultiplier
  1118. * @param {?string} tickRate
  1119. */
  1120. constructor(frameRate, subFrameRate, frameRateMultiplier, tickRate) {
  1121. /**
  1122. * @type {number}
  1123. */
  1124. this.frameRate = Number(frameRate) || 30;
  1125. /**
  1126. * @type {number}
  1127. */
  1128. this.subFrameRate = Number(subFrameRate) || 1;
  1129. /**
  1130. * @type {number}
  1131. */
  1132. this.tickRate = Number(tickRate);
  1133. if (this.tickRate == 0) {
  1134. if (frameRate) {
  1135. this.tickRate = this.frameRate * this.subFrameRate;
  1136. } else {
  1137. this.tickRate = 1;
  1138. }
  1139. }
  1140. if (frameRateMultiplier) {
  1141. const multiplierResults = /^(\d+) (\d+)$/g.exec(frameRateMultiplier);
  1142. if (multiplierResults) {
  1143. const numerator = Number(multiplierResults[1]);
  1144. const denominator = Number(multiplierResults[2]);
  1145. const multiplierNum = numerator / denominator;
  1146. this.frameRate *= multiplierNum;
  1147. }
  1148. }
  1149. }
  1150. };
  1151. /**
  1152. * @const
  1153. * @private {!RegExp}
  1154. * @example 50.17% 10%
  1155. */
  1156. shaka.text.TtmlTextParser.percentValues_ =
  1157. /^(\d{1,2}(?:\.\d+)?|100(?:\.0+)?)% (\d{1,2}(?:\.\d+)?|100(?:\.0+)?)%$/;
  1158. /**
  1159. * @const
  1160. * @private {!RegExp}
  1161. * @example 0.6% 90%
  1162. */
  1163. shaka.text.TtmlTextParser.percentValue_ = /^(\d{1,2}(?:\.\d+)?|100)%$/;
  1164. /**
  1165. * @const
  1166. * @private {!RegExp}
  1167. * @example 100px, 8em, 0.80c
  1168. */
  1169. shaka.text.TtmlTextParser.unitValues_ = /^(\d+px|\d+em|\d*\.?\d+c)$/;
  1170. /**
  1171. * @const
  1172. * @private {!RegExp}
  1173. * @example 100px
  1174. */
  1175. shaka.text.TtmlTextParser.pixelValues_ = /^(\d+)px (\d+)px$/;
  1176. /**
  1177. * @const
  1178. * @private {!RegExp}
  1179. * @example 00:00:40:07 (7 frames) or 00:00:40:07.1 (7 frames, 1 subframe)
  1180. */
  1181. shaka.text.TtmlTextParser.timeColonFormatFrames_ =
  1182. /^(\d{2,}):(\d{2}):(\d{2}):(\d{2})\.?(\d+)?$/;
  1183. /**
  1184. * @const
  1185. * @private {!RegExp}
  1186. * @example 00:00:40 or 00:40
  1187. */
  1188. shaka.text.TtmlTextParser.timeColonFormat_ = /^(?:(\d{2,}):)?(\d{2}):(\d{2})$/;
  1189. /**
  1190. * @const
  1191. * @private {!RegExp}
  1192. * @example 01:02:43.0345555 or 02:43.03
  1193. */
  1194. shaka.text.TtmlTextParser.timeColonFormatMilliseconds_ =
  1195. /^(?:(\d{2,}):)?(\d{2}):(\d{2}\.\d{2,})$/;
  1196. /**
  1197. * @const
  1198. * @private {!RegExp}
  1199. * @example 75f or 75.5f
  1200. */
  1201. shaka.text.TtmlTextParser.timeFramesFormat_ = /^(\d*(?:\.\d*)?)f$/;
  1202. /**
  1203. * @const
  1204. * @private {!RegExp}
  1205. * @example 50t or 50.5t
  1206. */
  1207. shaka.text.TtmlTextParser.timeTickFormat_ = /^(\d*(?:\.\d*)?)t$/;
  1208. /**
  1209. * @const
  1210. * @private {!RegExp}
  1211. * @example 3.45h, 3m or 4.20s
  1212. */
  1213. shaka.text.TtmlTextParser.timeHMSFormat_ =
  1214. new RegExp(['^(?:(\\d*(?:\\.\\d*)?)h)?',
  1215. '(?:(\\d*(?:\\.\\d*)?)m)?',
  1216. '(?:(\\d*(?:\\.\\d*)?)s)?',
  1217. '(?:(\\d*(?:\\.\\d*)?)ms)?$'].join(''));
  1218. /**
  1219. * @const
  1220. * @private {!Object.<string, shaka.text.Cue.lineAlign>}
  1221. */
  1222. shaka.text.TtmlTextParser.textAlignToLineAlign_ = {
  1223. 'left': shaka.text.Cue.lineAlign.START,
  1224. 'center': shaka.text.Cue.lineAlign.CENTER,
  1225. 'right': shaka.text.Cue.lineAlign.END,
  1226. 'start': shaka.text.Cue.lineAlign.START,
  1227. 'end': shaka.text.Cue.lineAlign.END,
  1228. };
  1229. /**
  1230. * @const
  1231. * @private {!Object.<string, shaka.text.Cue.positionAlign>}
  1232. */
  1233. shaka.text.TtmlTextParser.textAlignToPositionAlign_ = {
  1234. 'left': shaka.text.Cue.positionAlign.LEFT,
  1235. 'center': shaka.text.Cue.positionAlign.CENTER,
  1236. 'right': shaka.text.Cue.positionAlign.RIGHT,
  1237. };
  1238. /**
  1239. * The namespace URL for TTML parameters. Can be assigned any name in the TTML
  1240. * document, not just "ttp:", so we use this with getAttributeNS() to ensure
  1241. * that we support arbitrary namespace names.
  1242. *
  1243. * @const {!Array.<string>}
  1244. * @private
  1245. */
  1246. shaka.text.TtmlTextParser.parameterNs_ = [
  1247. 'http://www.w3.org/ns/ttml#parameter',
  1248. 'http://www.w3.org/2006/10/ttaf1#parameter',
  1249. ];
  1250. /**
  1251. * The namespace URL for TTML styles. Can be assigned any name in the TTML
  1252. * document, not just "tts:", so we use this with getAttributeNS() to ensure
  1253. * that we support arbitrary namespace names.
  1254. *
  1255. * @const {!Array.<string>}
  1256. * @private
  1257. */
  1258. shaka.text.TtmlTextParser.styleNs_ = [
  1259. 'http://www.w3.org/ns/ttml#styling',
  1260. 'http://www.w3.org/2006/10/ttaf1#styling',
  1261. ];
  1262. /**
  1263. * The namespace URL for EBU TTML styles. Can be assigned any name in the TTML
  1264. * document, not just "ebutts:", so we use this with getAttributeNS() to ensure
  1265. * that we support arbitrary namespace names.
  1266. *
  1267. * @const {string}
  1268. * @private
  1269. */
  1270. shaka.text.TtmlTextParser.styleEbuttsNs_ = 'urn:ebu:tt:style';
  1271. /**
  1272. * The supported namespace URLs for SMPTE fields.
  1273. * @const {!Array.<string>}
  1274. * @private
  1275. */
  1276. shaka.text.TtmlTextParser.smpteNsList_ = [
  1277. 'http://www.smpte-ra.org/schemas/2052-1/2010/smpte-tt',
  1278. 'http://www.smpte-ra.org/schemas/2052-1/2013/smpte-tt',
  1279. ];
  1280. shaka.text.TextEngine.registerParser(
  1281. 'application/ttml+xml', () => new shaka.text.TtmlTextParser());