UNPKG

shaka-player

Version:
471 lines (386 loc) 16.5 kB
/*! @license * Shaka Player * Copyright 2016 Google LLC * SPDX-License-Identifier: Apache-2.0 */ // TODO: Move this suite to the text/ folder where it belongs const supportsScreenshots = () => shaka.test.Util.supportsScreenshots(); filterDescribe('TextDisplayer layout', supportsScreenshots, () => { const UiUtils = shaka.test.UiUtils; const Util = shaka.test.Util; /** @type {!shaka.extern.TextDisplayer} */ let textDisplayer; /** * The video container or other element which is used for the screenshot. * @type {!HTMLElement} */ let videoContainer; /** * Awaited before each screenshot, and can vary by test suite. * @type {function(number):!Promise} */ let beforeScreenshot; // A minimum similarity score for screenshots, between 0 and 1. const minSimilarity = 0.95; const originalCast = window.chrome && window.chrome.cast; describe('using UI', () => { /** @type {!HTMLLinkElement} */ let cssLink; /** @type {!shaka.test.FakeVideo} */ let mockVideo; function createTextDisplayer() { textDisplayer = new shaka.text.UITextDisplayer( /** @type {!HTMLMediaElement} */(mockVideo), videoContainer); textDisplayer.setTextVisibility(true); } beforeAll(async () => { // Disable cast so the UI controls don't create cast sessions. if (window.chrome) { window.chrome['cast'] = null; } // Add css file cssLink = /** @type {!HTMLLinkElement} */(document.createElement('link')); await UiUtils.setupCSS(cssLink); // There's no actual video inside this container, but subtitles will be // positioned within this space. videoContainer = /** @type {!HTMLElement} */( document.createElement('div')); document.body.appendChild(videoContainer); positionElementForScreenshot(videoContainer); // Some of the styles in our CSS are only applied within this class. Add // this explicitly, since we don't instantiate controls in all of the // tests. videoContainer.classList.add('shaka-video-container'); await Util.waitForFont('Roboto'); // eslint-disable-next-line require-await beforeScreenshot = async (time) => { // Set the faked time. mockVideo.currentTime = time; // Trigger the display update logic to notice the time change by // appending an empty array. textDisplayer.append([]); }; }); beforeEach(() => { mockVideo = new shaka.test.FakeVideo(); createTextDisplayer(); }); afterEach(async () => { await textDisplayer.destroy(); }); afterAll(() => { document.body.removeChild(videoContainer); document.head.removeChild(cssLink); if (window.chrome) { window.chrome['cast'] = originalCast; } }); defineTests('ui'); // This test is unique to the UI. it('moves cues to avoid controls', async () => { let ui; try { // Set up UI controls. The video element is in a paused state by // default, so the controls should be shown. The video is not in the // DOM and is purely temporary. const player = new shaka.Player(null); ui = new shaka.ui.Overlay( player, videoContainer, shaka.test.UiUtils.createVideoElement()); // Turn off every part of the UI that we can, so that the screenshot is // less likey to change because of something unrelated to text // rendering. ui.configure('controlPanelElements', []); ui.configure('addSeekBar', false); ui.configure('addBigPlayButton', false); ui.configure('enableFullscreenOnRotation', false); // Recreate the text displayer so that the text container comes after // the controls (as it does in production). This is important for the // CSS that moves the cues above the controls when they are shown. await textDisplayer.destroy(); createTextDisplayer(); const cue = new shaka.text.Cue( 0, 1, 'Captain\'s log, stardate 41636.9'); cue.region.id = '1'; // Position the cue *explicitly* at the bottom of the screen. cue.region.viewportAnchorX = 0; // % cue.region.viewportAnchorY = 90; // % cue.region.width = 100; // % cue.region.height = 10; // % textDisplayer.append([cue]); await checkScreenshot('ui', 'cue-with-controls'); } finally { await ui.destroy(); } }); }); describe('using browser-native rendering', () => { /** @type {!HTMLVideoElement} */ let video; beforeAll(async () => { video = shaka.test.UiUtils.createVideoElement(); // On some platforms, such as Chrome on Android, we may see a "cast" // button overlayed if this isn't set. video.disableRemotePlayback = true; document.body.appendChild(video); positionElementForScreenshot(video); const eventManager = new shaka.util.EventManager(); const waiter = new shaka.test.Waiter(eventManager); const canPlay = waiter.failOnTimeout(false).timeoutAfter(10) .waitForEvent(video, 'canplay'); // Video content is required to show native subtitles. This is a small // green frame. video.src = '/base/test/test/assets/green-pixel.mp4'; await canPlay; expect(video.duration).toBeGreaterThan(0); expect(video.videoWidth).toBeGreaterThan(0); // There is no actual container. Screenshots will be taken of the video // element itself. videoContainer = video; // On Firefox, Safari, and legacy Edge, the video must be played a little // _after_ appending cues in order to consistently show subtitles // natively on the video element. beforeScreenshot = async (time) => { // Seek to the beginning so that we can reasonably wait for movement // after playing below. If somehow the playhead ends up at the end of // the video, we should seek back before we play. video.currentTime = 0; // The video must be played a little now, after the cues were appended, // but before the screenshot. video.playbackRate = 1; video.play(); await waiter.failOnTimeout(false).timeoutAfter(5) .waitForMovement(video); video.pause(); // Seek to a time when cues should be showing. video.currentTime = time; // Get into a playing state, but without movement. video.playbackRate = 0; video.play(); // Add a short delay to ensure that the system has caught up and that // native text displayers have been updated by the browser. await Util.delay(0.1); }; }); beforeEach(() => { textDisplayer = new shaka.text.SimpleTextDisplayer(video); textDisplayer.setTextVisibility(true); }); afterEach(async () => { await textDisplayer.destroy(); }); afterAll(() => { document.body.removeChild(video); }); defineTests('native'); }); /** @param {string} prefix Prepended to screenshot names */ function defineTests(prefix) { it('basic cue', async () => { textDisplayer.append([ new shaka.text.Cue(0, 1, 'Captain\'s log, stardate 41636.9'), ]); await checkScreenshot(prefix, 'basic-cue'); }); it('cue with newline', async () => { textDisplayer.append([ new shaka.text.Cue(0, 1, 'Captain\'s log,\nstardate 41636.9'), ]); await checkScreenshot(prefix, 'cue-with-newline'); }); it('two basic cues', async () => { textDisplayer.append([ new shaka.text.Cue(0, 1, 'Captain\'s log,'), new shaka.text.Cue(0, 1, 'stardate 41636.9'), ]); await checkScreenshot(prefix, 'two-basic-cues'); }); // Regression test for #2497 // Only one cue should be displayed. it('duplicate cues', async () => { // In reality, this occurs when a VTT cue crossed a segment boundary and // appears in more than one segment. So we must simulate this with two // calls to append(). textDisplayer.append([ new shaka.text.Cue(0, 1, 'Captain\'s log, stardate 41636.9'), ]); textDisplayer.append([ new shaka.text.Cue(0, 1, 'Captain\'s log, stardate 41636.9'), ]); await checkScreenshot(prefix, 'duplicate-cues'); }); // Regression test for #3151 // Only one cue should be displayed. Note, however, that we don't control // this in a browser's native display. As of Feb 2021, only Firefox does // the right thing in native text display. Chrome, Edge, and Safari all // show both cues in this edge case. When we control the display of text // through the UI & DOM, we can always get the timing right. it('cues ending exactly now', async () => { // At time exactly 1, this cue should _not_ be displayed any more. textDisplayer.append([ new shaka.text.Cue(0, 1, 'This cue is over and gone.'), ]); // At time exactly 1, this cue should _just_ be starting. textDisplayer.append([ new shaka.text.Cue(1, 2, 'This cue is just starting.'), ]); await checkScreenshot(prefix, 'end-time-edge-case', /* time= */ 1); }); // Regression test for #2524 it('two nested cues', async () => { const cue = new shaka.text.Cue(0, 1, ''); cue.nestedCues = [ new shaka.text.Cue(0, 1, 'Captain\'s log, '), new shaka.text.Cue(0, 1, 'stardate 41636.9'), ]; textDisplayer.append([cue]); await checkScreenshot(prefix, 'two-nested-cues'); }); // Distinct from "newline" test above, which has a literal \n character in // the text payload. This uses a nested "lineBreak" cue, which is what you // get with <br> in TTML. it('nested cues with linebreak', async () => { const cue = new shaka.text.Cue(0, 1, ''); cue.nestedCues = [ new shaka.text.Cue(0, 1, 'Captain\'s log,'), shaka.text.Cue.lineBreak(0, 1), new shaka.text.Cue(0, 1, 'stardate 41636.9'), ]; textDisplayer.append([cue]); await checkScreenshot(prefix, 'nested-cues-with-linebreak'); }); // Regression test for #3600 it('cue positioning', async () => { const cue = new shaka.text.Cue(0, 1, 'Text'); cue.line = 10; cue.lineInterpretation = shaka.text.Cue.lineInterpretation.PERCENTAGE; textDisplayer.append([cue]); await checkScreenshot(prefix, 'cue-position'); }); // Regression test for #2157 and #2584 it('region positioning', async () => { const nestedCue = new shaka.text.Cue( 0, 1, 'Captain\'s log, stardate 41636.9'); const cue = new shaka.text.Cue(0, 1, ''); cue.region.id = '1'; cue.region.viewportAnchorX = 70; // % cue.region.viewportAnchorY = 35; // % cue.region.width = 30; // % cue.region.height = 65; // % cue.nestedCues = [nestedCue]; textDisplayer.append([cue]); await checkScreenshot(prefix, 'region-position'); }); // Regression test for #3379, in which the displayAlign was not respected, // placing text at the top of the region instead of the bottom. it('region with display alignment', async () => { const cue = new shaka.text.Cue(0, 1, ''); cue.region.id = '1'; cue.region.viewportAnchorX = 10; cue.region.viewportAnchorY = 10; cue.region.width = 80; cue.region.height = 80; cue.positionAlign = shaka.text.Cue.positionAlign.CENTER; cue.lineAlign = shaka.text.Cue.lineAlign.CENTER; cue.displayAlign = shaka.text.Cue.displayAlign.AFTER; cue.nestedCues = [ // For those who don't speak Unicode, \xbf is an upside down "?". new shaka.text.Cue(0, 1, '\xbfBien?'), ]; textDisplayer.append([cue]); await checkScreenshot(prefix, 'region-with-display-alignment'); }); // Regression test for #2188 it('bitmap-based cues', async () => { const cue = new shaka.text.Cue(0, 1, ''); cue.region.id = '1'; cue.region.height = 15; // % cue.region.viewportAnchorX = 20; // % cue.region.viewportAnchorY = 10; // % // eslint-disable-next-line max-len cue.backgroundImage = ''; textDisplayer.append([cue]); await checkScreenshot(prefix, 'bitmap-cue'); }); // Used to be a regression test for #2623, but that should have been fixed // in the TTML parser instead. Now we ensure that backgroundColor is // honored at all levels, making this a regression test for the fix to our // original fix. :-) it('honors background settings at all levels', async () => { const cue = new shaka.text.Cue(0, 1, ''); cue.nestedCues = [ new shaka.text.Cue(0, 1, 'Captain\'s '), new shaka.text.Cue(0, 1, 'log, '), new shaka.text.Cue(0, 1, 'stardate 41636.9'), ]; // These middle nested cue will show through to the parent cue's // background. cue.nestedCues[0].backgroundColor = 'blue'; cue.nestedCues[1].backgroundColor = 'transparent'; cue.nestedCues[2].backgroundColor = 'yellow'; cue.backgroundColor = 'purple'; textDisplayer.append([cue]); await checkScreenshot(prefix, 'nested-cue-bg'); }); it('colors background for flat cues', async () => { const cue = new shaka.text.Cue(0, 1, 'Captain\'s log, stardate 41636.9'); // This is shown. cue.backgroundColor = 'purple'; textDisplayer.append([cue]); await checkScreenshot(prefix, 'flat-cue-bg'); }); // https://github.com/shaka-project/shaka-player/issues/2761 it('deeply-nested cues', async () => { const makeCue = (text, fg = '', bg = '', nestedCues = []) => { const cue = new shaka.text.Cue(0, 1, text); cue.color = fg; cue.backgroundColor = bg; cue.nestedCues = nestedCues; return cue; }; const cue = makeCue('', '', '', [ makeCue('', '', 'black', [ makeCue('Captain\'s '), makeCue('log, ', 'red'), makeCue('stardate ', 'green', 'blue', [ makeCue('41636.9', 'purple'), ]), ]), ]); textDisplayer.append([cue]); await checkScreenshot(prefix, 'deeply-nested-cues'); }); } /** @param {!HTMLElement} element */ function positionElementForScreenshot(element) { // The element we screenshot will be 16:9 and small. element.style.width = '320px'; element.style.height = '180px'; // The background is green so we can better see the background color of // the text spans within the subtitles, and so it is easier to identify // cropping issues. element.style.backgroundColor = 'green'; // Make sure the element is in the top-left corner of the iframe that // contains the tests. element.style.top = '0'; element.style.left = '0'; element.style.position = 'fixed'; element.style.margin = '0'; element.style.padding = '0'; } /** * @param {string} prefix A prefix added to the screenshot name with a hyphen * to provide context. * @param {string} baseName The base name of the screenshot. * @param {number=} time The time to seek to in the screenshot. Defaults to * 0.1, when most of our tests will be showing cues (timed 0-1). * @return {!Promise} */ async function checkScreenshot(prefix, baseName, time=0.1) { await beforeScreenshot(time); return Util.checkScreenshot( /* element= */ videoContainer, prefix + '-' + baseName, minSimilarity); } });