UNPKG

shaka-player

Version:
1,582 lines (1,509 loc) 68.9 kB
/*! @license * Shaka Player * Copyright 2016 Google LLC * SPDX-License-Identifier: Apache-2.0 */ describe('TtmlTextParser', () => { const Cue = shaka.text.Cue; const CueRegion = shaka.text.CueRegion; const Util = shaka.test.Util; const anyString = jasmine.any(String); it('supports no cues', () => { verifyHelper([], '<tt></tt>', {periodStart: 0, segmentStart: 0, segmentEnd: 10, vttOffset: 0}, {}); }); it('supports empty text string', () => { verifyHelper([], '', {periodStart: 0, segmentStart: 0, segmentEnd: 10, vttOffset: 0}, {}); }); it('supports div with no cues but whitespace', () => { verifyHelper( [], '<tt><body><div> \r\n </div></body></tt>', {periodStart: 0, segmentStart: 0, segmentEnd: 10, vttOffset: 0}, {}); }); it('supports xml:space', () => { const ttBody = '<body><div>\n' + ' <p begin="01:02.03" end="01:02.05">\n' + ' <span> A B C </span>\n' + ' </p>\n' + '</div></body>\n'; // When xml:space="default", ignore whitespace outside tags. verifyHelper( [ { startTime: 62.03, endTime: 62.05, nestedCues: [{ payload: 'A B C', startTime: 62.03, endTime: 62.05, }], payload: '', }, ], '<tt xml:space="default">' + ttBody + '</tt>', {periodStart: 0, segmentStart: 60, segmentEnd: 70, vttOffset: 0}, {startTime: 62.03, endTime: 62.05}); // When xml:space="preserve", take them into account. verifyHelper( [ { startTime: 62.03, endTime: 62.05, nestedCues: [{ // anonymous span payload: '\n ', startTime: 62.03, endTime: 62.05, }, { payload: ' A B C ', startTime: 62.03, endTime: 62.05, }, { // anonymous span payload: '\n ', startTime: 62.03, endTime: 62.05, }], }, ], '<tt xml:space="preserve">' + ttBody + '</tt>', {periodStart: 0, segmentStart: 60, segmentEnd: 70, vttOffset: 0}, {startTime: 62.03, endTime: 62.05}); // The default value for xml:space is "default". verifyHelper( [ { startTime: 62.03, endTime: 62.05, nestedCues: [{ payload: 'A B C', startTime: 62.03, endTime: 62.05, }], }, ], '<tt>' + ttBody + '</tt>', {periodStart: 0, segmentStart: 60, segmentEnd: 70, vttOffset: 0}, {startTime: 62.03, endTime: 62.05}); // Any other value is rejected as an error. errorHelper(shaka.util.Error.Code.INVALID_XML, '<tt xml:space="invalid">' + ttBody + '</tt>', jasmine.any(String)); }); it('supports xml:space overriding default at span level', () => { const ttBody = '\n' + ' <body><div>\n' + ' <p begin="01:02.03" end="01:02.05">\n' + ' <span xml:space="preserve"> A B C </span>\n' + ' </p>\n' + ' </div></body>\n'; // When xml:space="preserve", take them into account. verifyHelper( [ { startTime: 62.03, endTime: 62.05, nestedCues: [{ payload: ' A B C ', startTime: 62.03, endTime: 62.05, }], }, ], '<tt>' + ttBody + '</tt>', {periodStart: 0, segmentStart: 60, segmentEnd: 70, vttOffset: 0}, {startTime: 62.03, endTime: 62.05}); }); it('rejects invalid ttml', () => { errorHelper(shaka.util.Error.Code.INVALID_XML, '<test></test>', anyString); }); it('rejects ttml with body>p instead of body>div', () => { errorHelper( shaka.util.Error.Code.INVALID_TEXT_CUE, '<tt><body><p></p></body></tt>', anyString); }); it('rejects ttml with div>span instead of div>p', () => { errorHelper( shaka.util.Error.Code.INVALID_TEXT_CUE, '<tt><body><div><span></span></div></body></tt>', anyString); }); it('rejects invalid time format', () => { const wrap = (ttmlCue) => { return `<tt><body><div>${ttmlCue}</div></body></tt>`; }; errorHelper( shaka.util.Error.Code.INVALID_TEXT_CUE, wrap('<p begin="test" end="test">My very own cue</p>'), anyString); errorHelper( shaka.util.Error.Code.INVALID_TEXT_CUE, wrap('<p begin="3.45" end="1a">An invalid cue</p>'), anyString); }); it('supports spans as nestedCues of paragraphs', () => { verifyHelper( [ { startTime: 62.05, endTime: 3723.2, payload: '', nestedCues: [ { payload: 'First cue', startTime: 62.05, endTime: 3723.2, }, { payload: '', lineBreak: true, startTime: 62.05, endTime: 3723.2, }, { payload: 'Second cue', startTime: 62.05, endTime: 3723.2, }, ], }, ], '<tt><body><div>' + '<p begin="01:02.05" end="01:02:03.200">' + '<span>First cue</span><br /><span>Second cue</span>' + '</p>' + '</div></body></tt>', {periodStart: 0, segmentStart: 60, segmentEnd: 3730, vttOffset: 0}, {startTime: 62.05, endTime: 3723.2}); }); it('supports anonymous spans as nestedCues of paragraphs', () => { verifyHelper( [ { startTime: 62.05, endTime: 3723.2, payload: '', nestedCues: [ { payload: 'First cue', startTime: 62.05, endTime: 3723.2, }, { payload: '', lineBreak: true, startTime: 62.05, endTime: 3723.2, }, { payload: 'Second cue', startTime: 62.05, endTime: 3723.2, }, ], }, ], '<tt><body><div>' + '<p begin="01:02.05" end="01:02:03.200">' + 'First cue<br />Second cue' + '</p>' + '</div></body></tt>', {periodStart: 0, segmentStart: 60, segmentEnd: 3730, vttOffset: 0}, {startTime: 62.05, endTime: 3723.2}); }); it('supports multiple levels of nestedCues', () => { verifyHelper( [ { startTime: 62.05, endTime: 3723.2, payload: '', color: 'black', // cues in p container. nestedCues: [ // anonymous span in p container. { payload: 'First cue', startTime: 62.05, endTime: 3723.2, }, // container for Second cue and Third cue. { payload: '', startTime: 62.05, endTime: 3723.2, color: 'blue', nestedCues: [ { payload: 'Second cue', startTime: 62.05, endTime: 3723.2, color: '', }, { payload: 'Third cue', startTime: 62.05, endTime: 3723.2, color: 'green', }, ], }, ], }, ], '<tt xmlns:tts="http://www.w3.org/ns/ttml#styling">' + '<styling>' + '<style xml:id="black" tts:color="black" />' + '<style xml:id="blue" tts:color="blue" />' + '<style xml:id="green" tts:color="green" />' + '</styling>' + '<body><div>' + '<p begin="01:02.05" end="01:02:03.200" style="black">' + 'First cue' + '<span style="blue">Second cue' + '<span style="green">Third cue</span>' + '</span>' + '</p>' + '</div></body></tt>', {periodStart: 0, segmentStart: 60, segmentEnd: 3730, vttOffset: 0}, {startTime: 62.05, endTime: 3723.2}); }); it('inherits timing information of nested cues if unprovided', () => { verifyHelper( [ { startTime: 62.05, endTime: 3723.2, nestedCues: [ {startTime: 62.05, endTime: 3723.2, payload: 'Test'}, ], }, ], '<tt><body>' + '<div><p begin="01:02.05" end="01:02:03.200">' + '<span>Test</span></p></div>' + '</body></tt>', {periodStart: 0, segmentStart: 60, segmentEnd: 3730, vttOffset: 0}, {startTime: 62.05, endTime: 3723.2}); }); it('does not discard cues with image subcues', () => { verifyHelper( [ { startTime: 62.05, endTime: 3723.2, nestedCues: [ { startTime: 62.05, endTime: 3723.2, payload: '', backgroundImage: 'data:image/png;base64,base64EncodedImage', }, ], }, ], '<tt ' + 'xmlns:ttm="http://www.w3.org/ns/ttml#metadata" ' + 'xmlns:smpte="http://www.smpte-ra.org/schemas/2052-1/2010/smpte-tt">' + '<metadata>' + '<smpte:image imageType="PNG" encoding="Base64" xml:id="img_0">' + 'base64EncodedImage</smpte:image>' + '</metadata><body><div>' + '<p><div begin="01:02.05" end="01:02:03.200" ' + 'smpte:backgroundImage="#img_0"></div></p>' + '</div></body></tt>', {periodStart: 0, segmentStart: 60, segmentEnd: 3730, vttOffset: 0}, {startTime: 62.05, endTime: 3723.2}); }); it('supports colon formatted time', () => { verifyHelper( [ {startTime: 62.05, endTime: 3723.2, payload: 'Test'}, ], '<tt><body><div>' + '<p begin="01:02.05" end="01:02:03.200">Test</p>' + '</div></body></tt>', {periodStart: 0, segmentStart: 60, segmentEnd: 3730, vttOffset: 0}, {startTime: 62.05, endTime: 3723.2}); }); it('accounts for offset', () => { verifyHelper( [ {startTime: 69.05, endTime: 3730.2, payload: 'Test'}, ], '<tt><body><div>' + '<p begin="01:02.05" end="01:02:03.200">Test</p>' + '</div></body></tt>', {periodStart: 7, segmentStart: 60, segmentEnd: 3740, vttOffset: 0}, {startTime: 69.05, endTime: 3730.2}); }); it('supports nested cues with an offset', () => { verifyHelper( [ { startTime: 69.05, endTime: 3730.2, payload: '', nestedCues: [ { payload: 'Nested cue', startTime: 69.05, endTime: 3730.2, }, ], }, ], '<tt><body><div>' + '<p begin="01:02.05" end="01:02:03.200"><span>Nested cue</span></p>' + '</div></body></tt>', {periodStart: 7, segmentStart: 60, segmentEnd: 3740, vttOffset: 0}, {startTime: 69.05, endTime: 3730.2}); }); it('supports time in 0.00h 0.00m 0.00s format', () => { verifyHelper( [ {startTime: 3567.03, endTime: 5402.3, payload: 'Test'}, ], '<tt><body><div>' + '<p begin="59.45m30ms" end="1.5h2.3s">Test</p>' + '</div></body></tt>', {periodStart: 0, segmentStart: 3560, segmentEnd: 5410, vttOffset: 0}, {startTime: 3567.03, endTime: 5402.3}); }); it('supports time with frame rate', () => { verifyHelper( [ {startTime: 615.5, endTime: 663, payload: 'Test'}, ], '<tt xmlns:ttp="http://www.w3.org/ns/ttml#parameter" ' + 'ttp:frameRate="30"> ' + '<body><div>' + '<p begin="00:10:15:15" end="00:11:02:30">Test</p>' + '</div></body></tt>', {periodStart: 0, segmentStart: 610, segmentEnd: 670, vttOffset: 0}, {startTime: 615.5, endTime: 663}); }); it('supports time with frame rate multiplier', () => { verifyHelper( [ {startTime: 615.5, endTime: 663, payload: 'Test'}, ], '<tt xmlns:ttp="http://www.w3.org/ns/ttml#parameter" ' + 'ttp:frameRate="60" ttp:frameRateMultiplier="1 2"> ' + '<body><div>' + '<p begin="00:10:15:15" end="00:11:02:30">Test</p>' + '</div></body></tt>', {periodStart: 0, segmentStart: 610, segmentEnd: 670, vttOffset: 0}, {startTime: 615.5, endTime: 663}); }); it('supports time with subframes', () => { verifyHelper( [ { startTime: Util.closeTo(615.5 + 1 / 60), endTime: 663, payload: 'Test', }, ], '<tt xmlns:ttp="http://www.w3.org/ns/ttml#parameter" ' + 'ttp:frameRate="30" ttp:subFrameRate="2"> ' + '<body><div>' + '<p begin="00:10:15:15.1" end="00:11:02:29.2">Test</p>' + '</div></body></tt>', {periodStart: 0, segmentStart: 610, segmentEnd: 670, vttOffset: 0}, {startTime: Util.closeTo(615.5 + 1 / 60), endTime: 663}); }); it('supports time in frame format', () => { verifyHelper( [ {startTime: 2.5, endTime: Util.closeTo(10.01), payload: 'Test'}, ], '<tt xmlns:ttp="http://www.w3.org/ns/ttml#parameter" ' + 'ttp:frameRate="60" ttp:frameRateMultiplier="1 2">' + '<body><div>' + '<p begin="75f" end="300.3f">Test</p>' + '</div></body></tt>', {periodStart: 0, segmentStart: 0, segmentEnd: 20, vttOffset: 0}, {startTime: 2.5, endTime: Util.closeTo(10.01)}); }); it('supports time in tick format', () => { verifyHelper( [ {startTime: 5, endTime: Util.closeTo(6.02), payload: 'Test'}, ], '<tt xmlns:ttp="http://www.w3.org/ns/ttml#parameter" ' + 'ttp:frameRate="60" ttp:tickRate="10">' + '<body><div>' + '<p begin="50t" end="60.2t">Test</p>' + '</div></body></tt>', {periodStart: 0, segmentStart: 0, segmentEnd: 10, vttOffset: 0}, {startTime: 5, endTime: Util.closeTo(6.02)}); }); it('supports time with duration', () => { verifyHelper( [ {startTime: 62.05, endTime: 67.05, payload: 'Test'}, ], '<tt><body><div>' + '<p begin="01:02.05" dur="5s">Test</p>' + '</div></body></tt>', {periodStart: 0, segmentStart: 60, segmentEnd: 70, vttOffset: 0}, {startTime: 62.05, endTime: 67.05}); }); it('supports comments in the body', () => { verifyHelper( [], '<tt><body><div>' + '<!-- text-based TTML -->' + '</div></body></tt>', {periodStart: 0, segmentStart: 0, segmentEnd: 10, vttOffset: 0}, {}); }); it('does not inherit regions', () => { verifyHelper( [ { startTime: 62.05, endTime: 3723.2, payload: 'Test', }, ], '<tt xmlns:tts="http://www.w3.org/ns/ttml#styling">' + '<layout>' + '<region xml:id="subtitleArea" />' + '</layout>' + '<body region="subtitleArea"><div>' + '<p begin="01:02.05" end="01:02:03.200">Test</p>' + '</div></body></tt>', {periodStart: 0, segmentStart: 60, segmentEnd: 3730, vttOffset: 0}, {startTime: 62.05, endTime: 3723.2, region: {id: 'subtitleArea'}}, {startTime: 62.05, endTime: 3723.2}); }); it('parses alignment from textAlign attribute of a region', () => { verifyHelper( [ { startTime: 62.05, endTime: 3723.2, payload: 'Test', lineAlign: Cue.textAlign.START, }, ], '<tt xmlns:tts="http://www.w3.org/ns/ttml#styling">' + '<layout>' + '<region xml:id="subtitleArea" tts:textAlign="start" />' + '</layout>' + '<body region="subtitleArea"><div>' + '<p begin="01:02.05" end="01:02:03.200">Test</p>' + '</div></body></tt>', {periodStart: 0, segmentStart: 60, segmentEnd: 3730, vttOffset: 0}, {startTime: 62.05, endTime: 3723.2, region: {id: 'subtitleArea'}}, {startTime: 62.05, endTime: 3723.2}); }); it('allows non-standard namespace names', () => { verifyHelper( [ { startTime: 62.05, endTime: 3723.2, payload: 'Test', lineAlign: Cue.textAlign.START, }, ], '<tt xmlns:p1="http://www.w3.org/ns/ttml#styling">' + '<layout>' + '<region xml:id="subtitleArea" p1:textAlign="start" />' + '</layout>' + '<body region="subtitleArea"><div>' + '<p begin="01:02.05" end="01:02:03.200">Test</p>' + '</div></body></tt>', {periodStart: 0, segmentStart: 60, segmentEnd: 3730, vttOffset: 0}, {startTime: 62.05, endTime: 3723.2, region: {id: 'subtitleArea'}}, {startTime: 62.05, endTime: 3723.2}); }); it('parses alignment from <style> block with id on region', () => { verifyHelper( [ { startTime: 62.05, endTime: 3723.2, payload: 'Test', lineAlign: Cue.textAlign.END, }, ], '<tt xmlns:tts="http://www.w3.org/ns/ttml#styling">' + '<styling>' + '<style xml:id="s1" tts:textAlign="end"/>' + '</styling>' + '<layout>' + '<region xml:id="subtitleArea" style="s1" />' + '</layout>' + '<body region="subtitleArea"><div>' + '<p begin="01:02.05" end="01:02:03.200">Test</p>' + '</div></body></tt>', {periodStart: 0, segmentStart: 60, segmentEnd: 3730, vttOffset: 0}, {startTime: 62.05, endTime: 3723.2, region: {id: 'subtitleArea'}}, {startTime: 62.05, endTime: 3723.2}); }); it('parses alignment from <style> block with id on p', () => { verifyHelper( [ { startTime: 62.05, endTime: 3723.2, payload: 'Test', lineAlign: Cue.textAlign.END, }, ], '<tt xmlns:tts="http://www.w3.org/ns/ttml#styling">' + '<styling>' + '<style xml:id="s1" tts:textAlign="end"/>' + '</styling>' + '<layout>' + '<region xml:id="subtitleArea" />' + '</layout>' + '<body region="subtitleArea"><div>' + '<p begin="01:02.05" end="01:02:03.200" style="s1">Test</p>' + '</div></body></tt>', {periodStart: 0, segmentStart: 60, segmentEnd: 3730, vttOffset: 0}, {startTime: 62.05, endTime: 3723.2, region: {id: 'subtitleArea'}}, {startTime: 62.05, endTime: 3723.2}); }); it('supports region settings for horizontal text', () => { verifyHelper( [ { startTime: 62.05, endTime: 3723.2, payload: 'Test', }, ], '<tt xmlns:tts="http://www.w3.org/ns/ttml#styling">' + '<layout>' + '<region xml:id="subtitleArea" tts:origin="50% 16%"/>' + '</layout>' + '<body region="subtitleArea"><div>' + '<p begin="01:02.05" end="01:02:03.200">Test</p>' + '</div></body></tt>', {periodStart: 0, segmentStart: 60, segmentEnd: 3730, vttOffset: 0}, { region: { id: 'subtitleArea', viewportAnchorX: 50, viewportAnchorY: 16, width: 100, height: 100, }, startTime: 62.05, endTime: 3723.2, }, {startTime: 62.05, endTime: 3723.2}); verifyHelper( [ { startTime: 62.05, endTime: 3723.2, payload: 'Test', }, ], '<tt xmlns:tts="http://www.w3.org/ns/ttml#styling">' + '<layout>' + '<region xml:id="subtitleArea" tts:origin="50% 16%" ' + 'tts:writingMode="lrtb" />' + '</layout>' + '<body region="subtitleArea"><div>' + '<p begin="01:02.05" end="01:02:03.200">Test</p>' + '</div></body></tt>', {periodStart: 0, segmentStart: 60, segmentEnd: 3730, vttOffset: 0}, { region: { viewportAnchorX: 50, viewportAnchorY: 16, width: 100, height: 100, }, startTime: 62.05, endTime: 3723.2, }, {startTime: 62.05, endTime: 3723.2}); verifyHelper( [ { startTime: 62.05, endTime: 3723.2, payload: 'Test', }, ], '<tt xmlns:tts="http://www.w3.org/ns/ttml#styling">' + '<layout>' + '<region xml:id="subtitleArea" tts:origin="50% 16%" ' + 'tts:writingMode="lr" />' + '</layout>' + '<body region="subtitleArea"><div>' + '<p begin="01:02.05" end="01:02:03.200">Test</p>' + '</div></body></tt>', {periodStart: 0, segmentStart: 60, segmentEnd: 3730, vttOffset: 0}, { region: { viewportAnchorX: 50, viewportAnchorY: 16, width: 100, height: 100, }, startTime: 62.05, endTime: 3723.2, }, {startTime: 62.05, endTime: 3723.2}); }); it('supports region settings in pixels (origin)', () => { verifyHelper( [ { startTime: 62.05, endTime: 3723.2, payload: 'Test', }, ], '<tt xmlns:tts="http://www.w3.org/ns/ttml#styling">' + '<layout>' + '<region xml:id="subtitleArea" tts:origin="50px 16px"/>' + '</layout>' + '<body region="subtitleArea"><div>' + '<p begin="01:02.05" end="01:02:03.200">Test</p>' + '</div></body></tt>', {periodStart: 0, segmentStart: 60, segmentEnd: 3730, vttOffset: 0}, { region: { id: 'subtitleArea', viewportAnchorX: 50, viewportAnchorY: 16, regionAnchorX: 0, regionAnchorY: 0, width: 100, height: 100, heightUnits: CueRegion.units.PERCENTAGE, widthUnits: CueRegion.units.PERCENTAGE, viewportAnchorUnits: CueRegion.units.PX, scroll: CueRegion.scrollMode.NONE, }, startTime: 62.05, endTime: 3723.2, }, {startTime: 62.05, endTime: 3723.2}); }); it('supports region settings in pixels (extent)', () => { verifyHelper( [ { startTime: 62.05, endTime: 3723.2, payload: 'Test', }, ], '<tt xmlns:tts="http://www.w3.org/ns/ttml#styling">' + '<layout>' + '<region xml:id="subtitleArea" tts:extent="50px 16px"/>' + '</layout>' + '<body region="subtitleArea"><div>' + '<p begin="01:02.05" end="01:02:03.200">Test</p>' + '</div></body></tt>', {periodStart: 0, segmentStart: 60, segmentEnd: 3730, vttOffset: 0}, { region: { id: 'subtitleArea', viewportAnchorX: 0, viewportAnchorY: 0, regionAnchorX: 0, regionAnchorY: 0, width: 50, height: 16, heightUnits: CueRegion.units.PX, widthUnits: CueRegion.units.PX, viewportAnchorUnits: CueRegion.units.PERCENTAGE, scroll: CueRegion.scrollMode.NONE, }, startTime: 62.05, endTime: 3723.2, }, {startTime: 62.05, endTime: 3723.2}); }); it('supports region settings in pixels: origin (with global extent)', () => { verifyHelper( [ { startTime: 62.05, endTime: 3723.2, payload: 'Test', }, ], '<tt xmlns:tts="http://www.w3.org/ns/ttml#styling" tts:extent="1920px 1080px">' + '<layout>' + '<region xml:id="subtitleArea" tts:origin="192px 108px"/>' + '</layout>' + '<body region="subtitleArea"><div>' + '<p begin="01:02.05" end="01:02:03.200">Test</p>' + '</div></body></tt>', {periodStart: 0, segmentStart: 60, segmentEnd: 3730, vttOffset: 0}, { region: { id: 'subtitleArea', viewportAnchorX: 10, viewportAnchorY: 10, regionAnchorX: 0, regionAnchorY: 0, width: 100, height: 100, heightUnits: CueRegion.units.PERCENTAGE, widthUnits: CueRegion.units.PERCENTAGE, viewportAnchorUnits: CueRegion.units.PERCENTAGE, scroll: CueRegion.scrollMode.NONE, }, startTime: 62.05, endTime: 3723.2, }, {startTime: 62.05, endTime: 3723.2}); }); it('supports region settings in pixels: extent (with global extent)', () => { verifyHelper( [ { startTime: 62.05, endTime: 3723.2, payload: 'Test', }, ], '<tt xmlns:tts="http://www.w3.org/ns/ttml#styling" tts:extent="1920px 1080px">' + '<layout>' + '<region xml:id="subtitleArea" tts:extent="576px 324px" ' + 'tts:writingMode="lrtb" />' + '</layout>' + '<body region="subtitleArea"><div>' + '<p begin="01:02.05" end="01:02:03.200">Test</p>' + '</div></body></tt>', {periodStart: 0, segmentStart: 60, segmentEnd: 3730, vttOffset: 0}, { region: { id: 'subtitleArea', viewportAnchorX: 0, viewportAnchorY: 0, regionAnchorX: 0, regionAnchorY: 0, width: 30, height: 30, heightUnits: CueRegion.units.PERCENTAGE, widthUnits: CueRegion.units.PERCENTAGE, viewportAnchorUnits: CueRegion.units.PERCENTAGE, scroll: CueRegion.scrollMode.NONE, }, startTime: 62.05, endTime: 3723.2, }, {startTime: 62.05, endTime: 3723.2}); }); it('supports region settings in percentage (origin)', () => { verifyHelper( [ { startTime: 62.05, endTime: 3723.2, payload: 'Test', }, ], '<tt xmlns:tts="http://www.w3.org/ns/ttml#styling" ' + 'tts:extent="1920px 1080px">' + '<layout>' + '<region xml:id="subtitleArea" tts:origin="50% 16%"/>' + '</layout>' + '<body region="subtitleArea"><div>' + '<p begin="01:02.05" end="01:02:03.200">Test</p>' + '</div></body></tt>', {periodStart: 0, segmentStart: 60, segmentEnd: 3730, vttOffset: 0}, { region: { id: 'subtitleArea', viewportAnchorX: 50, viewportAnchorY: 16, regionAnchorX: 0, regionAnchorY: 0, width: 100, height: 100, heightUnits: CueRegion.units.PERCENTAGE, widthUnits: CueRegion.units.PERCENTAGE, viewportAnchorUnits: CueRegion.units.PERCENTAGE, scroll: CueRegion.scrollMode.NONE, }, startTime: 62.05, endTime: 3723.2, }, {startTime: 62.05, endTime: 3723.2}); }); it('supports region settings in percentage (extent)', () => { verifyHelper( [ { startTime: 62.05, endTime: 3723.2, payload: 'Test', }, ], '<tt xmlns:tts="http://www.w3.org/ns/ttml#styling" ' + 'tts:extent="1920px 1080px">' + '<layout>' + '<region xml:id="subtitleArea" tts:extent="50% 16%" ' + 'tts:writingMode="lrtb" />' + '</layout>' + '<body region="subtitleArea"><div>' + '<p begin="01:02.05" end="01:02:03.200">Test</p>' + '</div></body></tt>', {periodStart: 0, segmentStart: 60, segmentEnd: 3730, vttOffset: 0}, { region: { id: 'subtitleArea', viewportAnchorX: 0, viewportAnchorY: 0, regionAnchorX: 0, regionAnchorY: 0, width: 50, height: 16, heightUnits: CueRegion.units.PERCENTAGE, widthUnits: CueRegion.units.PERCENTAGE, viewportAnchorUnits: CueRegion.units.PERCENTAGE, scroll: CueRegion.scrollMode.NONE, }, startTime: 62.05, endTime: 3723.2, }, {startTime: 62.05, endTime: 3723.2}); }); it('supports region settings for vertical text', () => { verifyHelper( [ { startTime: 62.05, endTime: 3723.2, payload: 'Test', }, ], '<tt xmlns:tts="http://www.w3.org/ns/ttml#styling">' + '<layout>' + '<region xml:id="subtitleArea" tts:origin="50% 16%" ' + 'tts:writingMode="tb" />' + '</layout>' + '<body region="subtitleArea"><div>' + '<p begin="01:02.05" end="01:02:03.200">Test</p>' + '</div></body></tt>', {periodStart: 0, segmentStart: 60, segmentEnd: 3730, vttOffset: 0}, { region: { id: 'subtitleArea', viewportAnchorX: 50, viewportAnchorY: 16, width: 100, height: 100, }, startTime: 62.05, endTime: 3723.2, }, {startTime: 62.05, endTime: 3723.2}); verifyHelper( [ { startTime: 62.05, endTime: 3723.2, payload: 'Test', }, ], '<tt xmlns:tts="http://www.w3.org/ns/ttml#styling">' + '<layout>' + '<region xml:id="subtitleArea" tts:origin="50% 16%" ' + 'tts:writingMode="tblr" />' + '</layout>' + '<body region="subtitleArea"><div>' + '<p begin="01:02.05" end="01:02:03.200">Test</p>' + '</div></body></tt>', {periodStart: 0, segmentStart: 60, segmentEnd: 3730, vttOffset: 0}, { region: { viewportAnchorX: 50, viewportAnchorY: 16, width: 100, height: 100, }, startTime: 62.05, endTime: 3723.2, }, {startTime: 62.05, endTime: 3723.2}); verifyHelper( [ { startTime: 62.05, endTime: 3723.2, payload: 'Test', }, ], '<tt xmlns:tts="http://www.w3.org/ns/ttml#styling">' + '<layout>' + '<region xml:id="subtitleArea" tts:origin="50% 16%" ' + 'tts:writingMode="tbrl" />' + '</layout>' + '<body region="subtitleArea"><div>' + '<p begin="01:02.05" end="01:02:03.200">Test</p>' + '</div></body></tt>', {periodStart: 0, segmentStart: 60, segmentEnd: 3730, vttOffset: 0}, { region: { viewportAnchorX: 50, viewportAnchorY: 16, width: 100, height: 100, }, startTime: 62.05, endTime: 3723.2, }, {startTime: 62.05, endTime: 3723.2}); }); it('supports percentages containing decimals', () => { verifyHelper( [ { startTime: 62.05, endTime: 3723.2, payload: 'Test', }, ], '<tt xmlns:tts="http://www.w3.org/ns/ttml#styling">' + '<layout>' + '<region xml:id="subtitleArea" tts:origin="12.2% 50.005%" ' + 'tts:writingMode="tb" />' + '</layout>' + '<body region="subtitleArea"><div>' + '<p begin="01:02.05" end="01:02:03.200">Test</p>' + '</div></body></tt>', {periodStart: 0, segmentStart: 60, segmentEnd: 3730, vttOffset: 0}, { region: { id: 'subtitleArea', viewportAnchorX: 12.2, viewportAnchorY: 50.005, width: 100, height: 100, }, startTime: 62.05, endTime: 3723.2, }, {startTime: 62.05, endTime: 3723.2}); }); it('supports writingMode setting', () => { verifyHelper( [ { startTime: 62.05, endTime: 3723.2, payload: 'Test', writingMode: Cue.writingMode.VERTICAL_LEFT_TO_RIGHT, }, ], '<tt xmlns:tts="http://www.w3.org/ns/ttml#styling">' + '<layout>' + '<region xml:id="subtitleArea" ' + 'tts:writingMode="tb" />' + '</layout>' + '<body region="subtitleArea"><div>' + '<p begin="01:02.05" end="01:02:03.200">Test</p>' + '</div></body></tt>', {periodStart: 0, segmentStart: 60, segmentEnd: 3730, vttOffset: 0}, {startTime: 62.05, endTime: 3723.2, region: {id: 'subtitleArea'}}, {startTime: 62.05, endTime: 3723.2}); verifyHelper( [ { startTime: 62.05, endTime: 3723.2, payload: 'Test', writingMode: Cue.writingMode.VERTICAL_RIGHT_TO_LEFT, }, ], '<tt xmlns:tts="http://www.w3.org/ns/ttml#styling">' + '<layout>' + '<region xml:id="subtitleArea" ' + 'tts:writingMode="tbrl" />' + '</layout>' + '<body region="subtitleArea"><div>' + '<p begin="01:02.05" end="01:02:03.200">Test</p>' + '</div></body></tt>', {periodStart: 0, segmentStart: 60, segmentEnd: 3730, vttOffset: 0}, {startTime: 62.05, endTime: 3723.2, region: {id: 'subtitleArea'}}, {startTime: 62.05, endTime: 3723.2}); verifyHelper( [ { startTime: 62.05, endTime: 3723.2, payload: 'Test', writingMode: Cue.writingMode.VERTICAL_LEFT_TO_RIGHT, }, ], '<tt xmlns:tts="http://www.w3.org/ns/ttml#styling">' + '<layout>' + '<region xml:id="subtitleArea" ' + 'tts:writingMode="tblr" />' + '</layout>' + '<body region="subtitleArea"><div>' + '<p begin="01:02.05" end="01:02:03.200">Test</p>' + '</div></body></tt>', {periodStart: 0, segmentStart: 60, segmentEnd: 3730, vttOffset: 0}, {startTime: 62.05, endTime: 3723.2, region: {id: 'subtitleArea'}}, {startTime: 62.05, endTime: 3723.2}); verifyHelper( [ { startTime: 62.05, endTime: 3723.2, payload: 'Test', direction: Cue.direction.HORIZONTAL_RIGHT_TO_LEFT, }, ], '<tt xmlns:tts="http://www.w3.org/ns/ttml#styling">' + '<layout>' + '<region xml:id="subtitleArea" ' + 'tts:direction="rtl" />' + '</layout>' + '<body region="subtitleArea"><div>' + '<p begin="01:02.05" end="01:02:03.200">Test</p>' + '</div></body></tt>', {periodStart: 0, segmentStart: 60, segmentEnd: 3730, vttOffset: 0}, {startTime: 62.05, endTime: 3723.2, region: {id: 'subtitleArea'}}, {startTime: 62.05, endTime: 3723.2}); verifyHelper( [ { startTime: 62.05, endTime: 3723.2, payload: 'Test', direction: Cue.direction.HORIZONTAL_LEFT_TO_RIGHT, }, ], '<tt xmlns:tts="http://www.w3.org/ns/ttml#styling">' + '<layout>' + '<region xml:id="subtitleArea" ' + 'tts:direction="rtl" tts:writingMode="lrtb"/>' + '</layout>' + '<body region="subtitleArea"><div>' + '<p begin="01:02.05" end="01:02:03.200">Test</p>' + '</div></body></tt>', {periodStart: 0, segmentStart: 60, segmentEnd: 3730, vttOffset: 0}, {startTime: 62.05, endTime: 3723.2, region: {id: 'subtitleArea'}}, {startTime: 62.05, endTime: 3723.2}); }); it('supports textCombine setting', () => { verifyHelper( [ { startTime: 62.05, endTime: 3723.2, payload: 'Test', textCombineUpright: 'all', writingMode: Cue.writingMode.VERTICAL_LEFT_TO_RIGHT, }, ], '<tt xmlns:tts="http://www.w3.org/ns/ttml#styling">' + '<layout>' + '<region xml:id="subtitleArea" ' + 'tts:writingMode="tb" ' + 'tts:textCombine="all"/>' + '</layout>' + '<body region="subtitleArea"><div>' + '<p begin="01:02.05" end="01:02:03.200">Test</p>' + '</div></body></tt>', {periodStart: 0, segmentStart: 60, segmentEnd: 3730, vttOffset: 0}, {startTime: 62.05, endTime: 3723.2, region: {id: 'subtitleArea'}}, {startTime: 62.05, endTime: 3723.2}); }); it('disregards empty divs and ps', () => { verifyHelper( [ {startTime: 62.05, endTime: 3723.2, payload: 'Test'}, ], '<tt>' + '<body>' + '<div>' + '<p begin="01:02.05" end="01:02:03.200">Test</p>' + '</div>' + '<div></div>' + '</body>' + '</tt>', {periodStart: 0, segmentStart: 60, segmentEnd: 3730, vttOffset: 0}, {startTime: 62.05, endTime: 3723.2}); verifyHelper( [ {startTime: 62.05, endTime: 3723.2, payload: 'Test'}, ], '<tt>' + '<body>' + '<div>' + '<p begin="01:02.05" end="01:02:03.200">Test</p>' + '<p></p>' + '</div>' + '</body>' + '</tt>', {periodStart: 0, segmentStart: 60, segmentEnd: 3730, vttOffset: 0}, {startTime: 62.05, endTime: 3723.2}); verifyHelper( [], '<tt>' + '<body>' + '<div>' + '<p></p>' + '</div>' + '<div></div>' + '</body>' + '</tt>', {periodStart: 0, segmentStart: 60, segmentEnd: 3730, vttOffset: 0}, {}); }); it('should let empty paragraphs with begin or end attributes through', () => { verifyHelper( [ {startTime: 62.05, endTime: 3723.2, payload: ''}, ], '<tt>' + '<body>' + '<div>' + '<p begin="01:02.05" end="01:02:03.200" />' + '<p></p>' + '</div>' + '</body>' + '</tt>', {periodStart: 0, segmentStart: 60, segmentEnd: 3730, vttOffset: 0}, {startTime: 62.05, endTime: 3723.2}); }); it('supports smpte:backgroundImage attribute', () => { verifyHelper( [ { startTime: 62.05, endTime: 3723.2, payload: '', }, ], '<tt ' + 'xmlns:ttm="http://www.w3.org/ns/ttml#metadata" ' + 'xmlns:smpte="http://www.smpte-ra.org/schemas/2052-1/2010/smpte-tt">' + '<metadata>' + '<smpte:image imageType="PNG" encoding="Base64" xml:id="img_0">' + 'base64EncodedImage</smpte:image>' + '</metadata>' + '<body><div smpte:backgroundImage="#img_0">' + '<p begin="01:02.05" end="01:02:03.200"></p>' + '</div></body></tt>', {periodStart: 0, segmentStart: 60, segmentEnd: 3730, vttOffset: 0}, {startTime: 62.05, endTime: 3723.2}, { startTime: 62.05, endTime: 3723.2, backgroundImage: 'data:image/png;base64,base64EncodedImage', isContainer: false, }); }); it('supports smpte:backgroundImage attribute in div element', () => { verifyHelper( [], '<tt ' + 'xmlns:ttm="http://www.w3.org/ns/ttml#metadata" ' + 'xmlns:smpte="http://www.smpte-ra.org/schemas/2052-1/2010/smpte-tt">' + '<metadata>' + '<smpte:image imageType="PNG" encoding="Base64" xml:id="img_0">' + 'base64EncodedImage</smpte:image>' + '</metadata>' + '<body><div begin="00:00.00" end="01:02.05" '+ 'smpte:backgroundImage="#img_0"></div>' + '</body></tt>', {periodStart: 0, segmentStart: 0, segmentEnd: 70, vttOffset: 0}, {startTime: 0, endTime: 62.05}, { startTime: 0, endTime: 62.05, backgroundImage: 'data:image/png;base64,base64EncodedImage', isContainer: false, }); }); it('supports smpte:backgroundImage attribute alt namespace', () => { verifyHelper( [ { startTime: 62.05, endTime: 3723.2, payload: '', backgroundImage: 'data:image/png;base64,base64EncodedImage', }, ], '<tt ' + 'xmlns:ttm="http://www.w3.org/ns/ttml#metadata" ' + 'xmlns:smpte="http://www.smpte-ra.org/schemas/2052-1/2013/smpte-tt">' + '<metadata>' + '<smpte:image imageType="PNG" encoding="Base64" xml:id="img_0">' + 'base64EncodedImage</smpte:image>' + '</metadata>' + '<body><div>' + '<p begin="01:02.05" end="01:02:03.200" ' + 'smpte:backgroundImage="#img_0" />' + '</div></body></tt>', {periodStart: 0, segmentStart: 60, segmentEnd: 3730, vttOffset: 0}, {startTime: 62.05, endTime: 3723.2}); }); it('supports tts:ruby', () => { verifyHelper( [{ startTime: 62.05, endTime: 3723.2, payload: '', nestedCues: [{ startTime: 62.05, endTime: 3723.2, payload: 'Line1', }, { startTime: 62.05, endTime: 3723.2, payload: 'Line2', rubyTag: 'rt', }], rubyTag: 'ruby', }], // With anonymous spans '<tt xmlns:tts="http://www.w3.org/ns/ttml#styling"><body><div>' + '<p begin="01:02.05" end="01:02:03.200" tts:ruby="container">' + '<span tts:ruby="base">Line1</span><span tts:ruby="text">Line2' + '</span></p></div></body></tt>', {periodStart: 0, segmentStart: 60, segmentEnd: 3730, vttOffset: 0}, {startTime: 62.05, endTime: 3723.2}); }); it('inserts line breaks for <br> tags', () => { verifyHelper( [{ startTime: 62.05, endTime: 3723.2, payload: '', nestedCues: [{ startTime: 62.05, endTime: 3723.2, payload: 'Line1', }, { startTime: 62.05, endTime: 3723.2, payload: '', lineBreak: true, }, { startTime: 62.05, endTime: 3723.2, payload: 'Line2', }], }], // With anonymous spans '<tt><body><div>' + '<p begin="01:02.05" end="01:02:03.200">Line1<br/>Line2</p>' + '</div></body></tt>', {periodStart: 0, segmentStart: 60, segmentEnd: 3730, vttOffset: 0}, {startTime: 62.05, endTime: 3723.2}); verifyHelper( [{ startTime: 62.05, endTime: 3723.2, payload: '', nestedCues: [{ startTime: 62.05, endTime: 3723.2, payload: '', nestedCues: [{ startTime: 62.05, endTime: 3723.2, payload: 'Line1', }, { startTime: 62.05, endTime: 3723.2, payload: '', lineBreak: true, }, { startTime: 62.05, endTime: 3723.2, payload: 'Line2', }], }], }], // With explicit spans '<tt><body><div>' + '<p begin="01:02.05" end="01:02:03.200">' + '<span>Line1<br/>Line2</span>' + '</p>' + '</div></body></tt>', {periodStart: 0, segmentStart: 60, segmentEnd: 3730, vttOffset: 0}, {startTime: 62.05, endTime: 3723.2}); }); it('allows old-standard namespace', () => { verifyHelper( [ { startTime: 1, endTime: 2, payload: 'Test', cellResolution: { columns: 60, rows: 20, }, fontSize: '67%', }, ], '<tt ' + 'xmlns:ttp="http://www.w3.org/2006/10/ttaf1#parameter" ' + 'xmlns:tts="http://www.w3.org/2006/10/ttaf1#styling" ' + 'ttp:cellResolution="60 20">' + '<styling>' + '<style xml:id="s1" tts:fontSize="67%"/>' + '</styling>' + '<body><div>' + '<p begin="00:01.00" end="00:02.00" style="s1">Test</p>' + '</div></body></tt>', {periodStart: 0, segmentStart: 0, segmentEnd: 10, vttOffset: 0}, {startTime: 1, endTime: 2}); }); it('parses cue alignment from textAlign attribute', () => { verifyHelper( [ { startTime: 62.05, endTime: 3723.2, payload: 'Test', lineAlign: Cue.lineAlign.START, textAlign: Cue.textAlign.LEFT, positionAlign: Cue.positionAlign.LEFT, }, ], '<tt xmlns:tts="http://www.w3.org/ns/ttml#styling">' + '<styling>' + '<style xml:id="s1" tts:textAlign="left"/>' + '</styling>' + '<layout>' + '<region xml:id="subtitleArea" />' + '</layout>' + '<body region="subtitleArea"><div>' + '<p begin="01:02.05" end="01:02:03.200" style="s1">Test</p>' + '</div></body></tt>', {periodStart: 0, segmentStart: 60, segmentEnd: 3730, vttOffset: 0}, {startTime: 62.05, endTime: 3723.2, region: {id: 'subtitleArea'}}, {startTime: 62.05, endTime: 3723.2}); }); it('parses text style information', () => { verifyHelper( [ { startTime: 1, endTime: 2, payload: 'Test', color: 'red', backgroundColor: 'blue', fontWeight: Cue.fontWeight.BOLD, fontFamily: 'Times New Roman', fontStyle: Cue.fontStyle.ITALIC, lineHeight: '20px', fontSize: '10em', textStrokeColor: 'blue', textStrokeWidth: '3px', }, { startTime: 2, endTime: 4, payload: 'Test 2', fontSize: '0.80c', }, ], '<tt xmlns:tts="http://www.w3.org/ns/ttml#styling">' + '<styling>' + '<style xml:id="s1" tts:color="red" ' + 'tts:textOutline="blue 3px" ' + 'tts:backgroundColor="blue" ' + 'tts:fontWeight="bold" ' + 'tts:fontFamily="Times New Roman" ' + 'tts:fontStyle="italic" ' + 'tts:lineHeight="20px" ' + 'tts:fontSize="10em"/>' + '<style xml:id="s2" tts:fontSize="0.80c" />' + '</styling>' + '<layout>' + '<region xml:id="subtitleArea" />' + '</layout>' + '<body region="subtitleArea"><div>' + '<p begin="00:01.00" end="00:02.00" style="s1">Test</p>' + '<p begin="00:02.00" end="00:04.00" style="s2">Test 2</p>' + '</div></body></tt>', {periodStart: 0, segmentStart: 0, segmentEnd: 10, vttOffset: 0}, {startTime: 1, endTime: 4, region: {id: 'subtitleArea'}}, {startTime: 1, endTime: 4}); }); it('uses text color if tts:textOutline does not specify color', () => { verifyHelper( [ { startTime: 1, endTime: 2, payload: 'Test', color: 'red', textStrokeColor: 'red', textStrokeWidth: '3px', }, ], '<tt xmlns:tts="http://www.w3.org/ns/ttml#styling">' + '<styling>' + '<style xml:id="s1" tts:color="red" ' + 'tts:textOutline="3px" />' + '</styling>' + '<body region="subtitleArea"><div>' + '<p begin="00:01.00" end="00:02.00" style="s1">Test</p>' + '</div></body></tt>', {periodStart: 0, segmentStart: 0, segmentEnd: 10, vttOffset: 0}, {startTime: 1, endTime: 2}); }); it('does not add an outline if tts:textOutline only contains color', () => { verifyHelper( [ { startTime: 1, endTime: 2, payload: 'Test', color: 'red', }, ], '<tt xmlns:tts="http://www.w3.org/ns/ttml#styling">' + '<styling>' + '<style xml:id="s1" tts:color="red" ' + 'tts:textOutline="blue" />' + '</styling>' + '<body region="subtitleArea"><div>' + '<p begin="00:01.00" end="00:02.00" style="s1">Test</p>' + '</div></body></tt>', {periodStart: 0, segmentStart: 0, segmentEnd: 10, vttOffset: 0}, {startTime: 1, endTime: 2}); }); // Regression test for #2623 it('does not apply background colors to containers', () => { verifyHelper( [ { startTime: 1, endTime: 2, payload: '', // background color should not be set on the p container. backgroundColor: '', nestedCues: [{ payload: 'Test', color: 'red', backgroundColor: 'blue', }], }, ], '<tt xmlns:tts="http://www.w3.org/ns/ttml#styling">' + '<head>' + '<styling>' + '<style xml:id="s1" tts:color="red" ' + 'tts:backgroundColor="blue" />' + '</styling>' + '<layout>' + '<region xml:id="r1" />' + '</layout>' + '</head>' + '<body><div>' + '<p begin="00:01.00"