/* eslint-env qunit */ import Plugin from '../../src/js/plugin'; import Player from '../../src/js/player.js'; import videojs from '../../src/js/video.js'; import * as Dom from '../../src/js/utils/dom.js'; import * as browser from '../../src/js/utils/browser.js'; import log from '../../src/js/utils/log.js'; import MediaError from '../../src/js/media-error.js'; import Html5 from '../../src/js/tech/html5.js'; import Tech from '../../src/js/tech/tech.js'; import TestHelpers from './test-helpers.js'; import document from 'global/document'; import sinon from 'sinon'; import window from 'global/window'; import * as middleware from '../../src/js/tech/middleware.js'; import * as Events from '../../src/js/utils/events.js'; import pkg from '../../package.json'; import * as Guid from '../../src/js/utils/guid.js'; QUnit.module('Player', { beforeEach() { this.clock = sinon.useFakeTimers(); // reset players storage for (const playerId in Player.players) { if (Player.players[playerId] !== null) { Player.players[playerId].dispose(); } delete Player.players[playerId]; } }, afterEach() { this.clock.restore(); } }); QUnit.test('the default ID of the first player remains "vjs_video_3"', function(assert) { Guid.resetGuidInTestsOnly(); const tag = document.createElement('video'); tag.className = 'video-js'; const player = TestHelpers.makePlayer({}, tag); assert.strictEqual(player.id(), 'vjs_video_3', 'id is correct'); }); QUnit.test('should create player instance that inherits from component and dispose it', function(assert) { const player = TestHelpers.makePlayer(); assert.ok(player.el().nodeName === 'DIV'); assert.ok(player.on, 'component function exists'); player.dispose(); assert.ok(player.el() === null, 'element disposed'); }); QUnit.test('dispose should not throw if styleEl is missing', function(assert) { const player = TestHelpers.makePlayer(); player.styleEl_.parentNode.removeChild(player.styleEl_); player.dispose(); assert.ok(player.el() === null, 'element disposed'); }); QUnit.test('dispose should not throw if techEl is missing', function(assert) { const videoTag = TestHelpers.makeTag(); const fixture = document.getElementById('qunit-fixture'); fixture.appendChild(videoTag); const player = new Player(videoTag); player.tech_.el_.parentNode.removeChild(player.tech_.el_); player.tech_.el_ = null; let error; try { player.dispose(); } catch (e) { error = e; } assert.notOk(error, 'Function did not throw an error on dispose'); }); QUnit.test('dispose should not throw if playerEl is missing', function(assert) { const videoTag = TestHelpers.makeTag(); const fixture = document.getElementById('qunit-fixture'); fixture.appendChild(videoTag); const player = new Player(videoTag); player.el_.parentNode.removeChild(player.el_); player.el_ = null; let error; try { player.dispose(); } catch (e) { error = e; } assert.notOk(error, 'Function did not throw an error on dispose'); }); // technically, all uses of videojs.options should be replaced with // Player.prototype.options_ in this file and a equivalent test using // videojs.options should be made in video.test.js. Keeping this here // until we make that move. QUnit.test('should accept options from multiple sources and override in correct order', function(assert) { // Set a global option videojs.options.attr = 1; const tag0 = TestHelpers.makeTag(); const player0 = new Player(tag0, { techOrder: ['techFaker'] }); assert.equal(player0.options_.attr, 1, 'global option was set'); player0.dispose(); // Set a tag level option const tag2 = TestHelpers.makeTag(); // Attributes must be set as strings tag2.setAttribute('attr', 'asdf'); const player2 = new Player(tag2, { techOrder: ['techFaker'] }); assert.equal(player2.options_.attr, 'asdf', 'Tag options overrode global options'); player2.dispose(); // Set a tag level option const tag3 = TestHelpers.makeTag(); tag3.setAttribute('attr', 'asdf'); const player3 = new Player(tag3, { techOrder: ['techFaker'], attr: 'fdsa' }); assert.equal(player3.options_.attr, 'fdsa', 'Init options overrode tag and global options'); player3.dispose(); }); QUnit.test('should get tag, source, and track settings', function(assert) { // Partially tested in lib->getAttributes const fixture = document.getElementById('qunit-fixture'); let html = ''; fixture.innerHTML += html; const tag = document.getElementById('example_1'); const player = TestHelpers.makePlayer({}, tag); assert.equal(player.options_.autoplay, true, 'autoplay is set to true'); assert.equal(player.options_.preload, 'none', 'preload is set to none'); assert.equal(player.options_.id, 'example_1', 'id is set to example_1'); assert.equal(player.options_.sources.length, 2, 'we have two sources'); assert.equal(player.options_.sources[0].src, 'http://google.com', 'first source is google.com'); assert.equal(player.options_.sources[0].type, 'video/mp4', 'first type is video/mp4'); assert.equal(player.options_.sources[1].type, 'video/webm', 'second type is video/webm'); assert.equal(player.options_.tracks.length, 1, 'we have one text track'); assert.equal(player.options_.tracks[0].kind, 'captions', 'the text track is a captions file'); assert.equal(player.options_.tracks[0].attrtest, '', 'we have an empty attribute called attrtest'); assert.notEqual(player.el().className.indexOf('video-js'), -1, 'transferred class from tag to player div'); assert.equal(player.el().id, 'example_1', 'transferred id from tag to player div'); assert.equal(Player.players[player.id()], player, 'player referenceable from global list'); assert.notEqual(tag.id, player.id, 'tag ID no longer is the same as player ID'); assert.notEqual(tag.className, player.el().className, 'tag classname updated'); player.dispose(); assert.notEqual(tag.player, player, 'tag player ref killed'); assert.ok(!Player.players.example_1, 'global player ref killed'); assert.equal(player.el(), null, 'player el killed'); }); QUnit.test('should get current source from source tag', function(assert) { const fixture = document.getElementById('qunit-fixture'); const html = [ '' ].join(''); fixture.innerHTML += html; const tag = document.getElementById('example_1'); const player = TestHelpers.makePlayer({}, tag); assert.ok(player.currentSource().src === 'http://google.com'); assert.ok(player.currentSource().type === 'video/mp4'); player.dispose(); }); QUnit.test('should get current sources from source tag', function(assert) { const fixture = document.getElementById('qunit-fixture'); const html = [ '' ].join(''); fixture.innerHTML += html; const tag = document.getElementById('example_1'); const player = TestHelpers.makePlayer({}, tag); assert.ok(player.currentSources()[0].src === 'http://google.com'); assert.ok(player.currentSources()[0].type === 'video/mp4'); assert.ok(player.currentSources()[1].src === 'http://hugo.com'); assert.ok(player.currentSources()[1].type === 'video/webm'); // when redefining src expect sources to update accordingly player.src('http://google.com'); assert.ok(player.currentSources()[0].src === 'http://google.com'); assert.ok(player.currentSources()[0].type === undefined); assert.ok(player.currentSources()[1] === undefined); player.dispose(); }); QUnit.test('should get current source from src set', function(assert) { const fixture = document.getElementById('qunit-fixture'); const html = ''; fixture.innerHTML += html; const tag = document.getElementById('example_1'); const player = TestHelpers.makePlayer({}, tag); player.loadTech_('Html5'); // check for matching undefined src assert.deepEqual(player.currentSource(), {}); assert.equal(player.src(), ''); player.src('http://google.com'); assert.ok(player.currentSource().src === 'http://google.com'); assert.ok(player.currentSource().type === undefined); player.src({ src: 'http://google.com' }); assert.ok(player.currentSource().src === 'http://google.com'); assert.ok(player.currentSource().type === undefined); player.src({ src: 'http://google.com', type: 'video/mp4' }); assert.ok(player.currentSource().src === 'http://google.com'); assert.ok(player.currentSource().type === 'video/mp4'); player.dispose(); }); QUnit.test('should get current sources from src set', function(assert) { const fixture = document.getElementById('qunit-fixture'); const html = ''; fixture.innerHTML += html; const tag = document.getElementById('example_1'); const player = TestHelpers.makePlayer({}, tag); player.loadTech_('Html5'); // check for matching undefined src assert.ok(player.currentSources(), []); player.src([{ src: 'http://google.com' }, { src: 'http://hugo.com' }]); assert.ok(player.currentSources()[0].src === 'http://google.com'); assert.ok(player.currentSources()[0].type === undefined); assert.ok(player.currentSources()[1].src === 'http://hugo.com'); assert.ok(player.currentSources()[1].type === undefined); player.src([{ src: 'http://google.com', type: 'video/mp4' }, { src: 'http://hugo.com', type: 'video/webm' }]); assert.ok(player.currentSources()[0].src === 'http://google.com'); assert.ok(player.currentSources()[0].type === 'video/mp4'); assert.ok(player.currentSources()[1].src === 'http://hugo.com'); assert.ok(player.currentSources()[1].type === 'video/webm'); // when redefining src expect sources to update accordingly player.src('http://hugo.com'); assert.ok(player.currentSources()[0].src === 'http://hugo.com'); assert.ok(player.currentSources()[0].type === undefined); assert.ok(player.currentSources()[1] === undefined); player.dispose(); }); QUnit.test('should asynchronously fire error events during source selection', function(assert) { assert.expect(2); sinon.stub(log, 'error'); const player = TestHelpers.makePlayer({ techOrder: ['foo'], sources: [ { src: 'http://vjs.zencdn.net/v/oceans.mp4', type: 'video/mp4' } ] }); assert.ok(player.options_.techOrder[0] === 'foo', 'Foo listed as the only tech'); player.on('error', function(e) { assert.ok(player.error().code === 4, 'Source could not be played error thrown'); }); // The first one is for player initialization // The second one is the setTimeout for triggering the error this.clock.tick(1); this.clock.tick(1); player.dispose(); log.error.restore(); }); QUnit.test('should suppress source error messages', function(assert) { sinon.stub(log, 'error'); const clock = sinon.useFakeTimers(); const player = TestHelpers.makePlayer({ techOrder: ['foo'], suppressNotSupportedError: true }); let errors = 0; player.on('error', function(e) { errors++; }); player.src({src: 'http://example.com', type: 'video/mp4'}); clock.tick(10); assert.strictEqual(errors, 0, 'no error on bad source load'); player.trigger('click'); clock.tick(10); assert.strictEqual(errors, 1, 'error after click'); player.dispose(); assert.strictEqual(log.error.callCount, 2, 'two stubbed errors'); log.error.restore(); }); QUnit.test('should cancel a suppressed error message on loadstart', function(assert) { sinon.stub(log, 'error'); const clock = sinon.useFakeTimers(); const player = TestHelpers.makePlayer({ techOrder: ['foo'], suppressNotSupportedError: true }); let errors = 0; player.on('error', function(e) { errors++; }); player.src({src: 'http://example.com', type: 'video/mp4'}); clock.tick(10); assert.strictEqual(errors, 0, 'no error on bad source load'); assert.strictEqual( player.options_.suppressNotSupportedError, false, 'option was unset when error was suppressed' ); player.trigger('loadstart'); clock.tick(10); player.trigger('click'); clock.tick(10); assert.strictEqual(errors, 0, 'no error after click after loadstart'); assert.strictEqual(log.error.callCount, 3, 'one stubbed errors'); player.dispose(); log.error.restore(); }); QUnit.test('should set the width, height, and aspect ratio via a css class', function(assert) { const player = TestHelpers.makePlayer(); const getStyleText = function(styleEl) { return (styleEl.styleSheet && styleEl.styleSheet.cssText) || styleEl.innerHTML; }; // NOTE: was using npm/css to parse the actual CSS ast // but the css module doesn't support ie8 const confirmSetting = function(prop, val) { let styleText = getStyleText(player.styleEl_); const re = new RegExp(prop + ':\\s?' + val); // Lowercase string for IE8 styleText = styleText.toLowerCase(); return !!re.test(styleText); }; // Initial state assert.ok(!getStyleText(player.styleEl_), 'style element should be empty when the player is given no dimensions'); // Set only the width player.width(100); assert.ok(confirmSetting('width', '100px'), 'style width should equal the supplied width in pixels'); assert.ok(confirmSetting('height', '56.25px'), 'style height should match the default aspect ratio of the width'); // Set the height player.height(200); assert.ok(confirmSetting('height', '200px'), 'style height should match the supplied height in pixels'); // Reset the width and height to defaults player.width(''); player.height(''); assert.ok(confirmSetting('width', '300px'), 'supplying an empty string should reset the width'); assert.ok(confirmSetting('height', '168.75px'), 'supplying an empty string should reset the height'); // Switch to fluid mode player.fluid(true); assert.ok(player.hasClass('vjs-fluid'), 'the vjs-fluid class should be added to the player'); assert.ok(confirmSetting('padding-top', '56.25%'), 'fluid aspect ratio should match the default aspect ratio'); // Change the aspect ratio player.aspectRatio('4:1'); assert.ok(confirmSetting('padding-top', '25%'), 'aspect ratio percent should match the newly set aspect ratio'); player.dispose(); }); QUnit.test('should default to 16:9 when fluid', function(assert) { const player = TestHelpers.makePlayer({fluid: true}); const ratio = player.currentHeight() / player.currentWidth(); // account for some rounding of 0.5625 up to 0.563 assert.ok(((ratio >= 0.562) && (ratio <= 0.563)), 'fluid player without dimensions defaults to 16:9'); player.dispose(); }); QUnit.test('should resize fluid player on resize', function(assert) { const player = TestHelpers.makePlayer({fluid: true}); let ratio = player.currentHeight() / player.currentWidth(); // account for some rounding of 0.5625 up to 0.563 assert.ok(((ratio >= 0.562) && (ratio <= 0.563)), 'fluid player without dimensions defaults to 16:9'); player.tech_.videoWidth = () => 100; player.tech_.videoHeight = () => 50; player.trigger('resize'); this.clock.tick(1); ratio = player.currentHeight() / player.currentWidth(); assert.ok(ratio === 0.5, 'player aspect ratio changed on resize event'); player.dispose(); }); QUnit.test('should resize fluid player on resize if fluid enabled post initialisation', function(assert) { const player = TestHelpers.makePlayer({fluid: false}); player.tech_.videoWidth = () => 100; player.tech_.videoHeight = () => 30; player.fluid(true); player.trigger('resize'); this.clock.tick(1); const ratio = player.currentHeight() / player.currentWidth(); assert.ok(ratio === 0.3, 'player aspect ratio changed on resize event'); player.dispose(); }); QUnit.test('should set fluid to true if element has vjs-fluid class', function(assert) { const tag = TestHelpers.makeTag(); tag.className += ' vjs-fluid'; const player = TestHelpers.makePlayer({}, tag); assert.ok(player.fluid(), 'fluid is true with vjs-fluid class'); player.dispose(); }); QUnit.test('should set fill to true if element has vjs-fill class', function(assert) { const tag = TestHelpers.makeTag(); tag.className += ' vjs-fill'; const player = TestHelpers.makePlayer({}, tag); assert.ok(player.fill(), 'fill is true with vjs-fill class'); player.dispose(); }); QUnit.test('fill turns off fluid', function(assert) { const tag = TestHelpers.makeTag(); tag.className += ' vjs-fluid'; const player = TestHelpers.makePlayer({}, tag); assert.notOk(player.fill(), 'fill is false'); assert.ok(player.fluid(), 'fluid is true'); player.fill(true); assert.ok(player.fill(), 'fill is true'); assert.notOk(player.fluid(), 'fluid is false'); player.dispose(); }); QUnit.test('fluid turns off fill', function(assert) { const tag = TestHelpers.makeTag(); tag.className += ' vjs-fill'; const player = TestHelpers.makePlayer({}, tag); assert.ok(player.fill(), 'fill is true'); assert.notOk(player.fluid(), 'fluid is false'); player.fluid(true); assert.notOk(player.fill(), 'fill is false'); assert.ok(player.fluid(), 'fluid is true'); player.dispose(); }); QUnit.test('should use an class name that begins with an alpha character', function(assert) { const alphaPlayer = TestHelpers.makePlayer({ id: 'alpha1' }); const numericPlayer = TestHelpers.makePlayer({ id: '1numeric' }); const getStyleText = function(styleEl) { return (styleEl.styleSheet && styleEl.styleSheet.cssText) || styleEl.innerHTML; }; alphaPlayer.width(100); numericPlayer.width(100); assert.ok(/\s*\.alpha1-dimensions\s*\{/.test(getStyleText(alphaPlayer.styleEl_)), 'appends -dimensions to an alpha player ID'); assert.ok(/\s*\.dimensions-1numeric\s*\{/.test(getStyleText(numericPlayer.styleEl_)), 'prepends dimensions- to a numeric player ID'); alphaPlayer.dispose(); numericPlayer.dispose(); }); QUnit.test('should wrap the original tag in the player div', function(assert) { const tag = TestHelpers.makeTag(); const container = document.createElement('div'); const fixture = document.getElementById('qunit-fixture'); container.appendChild(tag); fixture.appendChild(container); const player = new Player(tag, { techOrder: ['techFaker'] }); const el = player.el(); assert.ok(el.parentNode === container, 'player placed at same level as tag'); // Tag may be placed inside the player element or it may be removed from the DOM assert.ok(tag.parentNode !== container, 'tag removed from original place'); player.dispose(); }); QUnit.test('should set and update the poster value', function(assert) { const poster = '#'; const updatedPoster = 'http://example.com/updated-poster.jpg'; const tag = TestHelpers.makeTag(); tag.setAttribute('poster', poster); const player = TestHelpers.makePlayer({}, tag); assert.equal(player.poster(), poster, 'the poster property should equal the tag attribute'); let pcEmitted = false; player.on('posterchange', function() { pcEmitted = true; }); player.poster(updatedPoster); assert.ok(pcEmitted, 'posterchange event was emitted'); assert.equal(player.poster(), updatedPoster, 'the updated poster is returned'); player.dispose(); }); // hasStarted() is equivalent to the "show poster flag" in the // standard, for the purpose of displaying the poster image // https://html.spec.whatwg.org/multipage/embedded-content.html#dom-media-play QUnit.test('should hide the poster when play is called', function(assert) { const player = TestHelpers.makePlayer({ poster: 'https://example.com/poster.jpg' }); assert.equal(player.hasStarted(), false, 'the show poster flag is true before play'); player.tech_.trigger('play'); assert.equal(player.hasStarted(), true, 'the show poster flag is false after play'); player.tech_.trigger('loadstart'); assert.equal(player.hasStarted(), false, 'the resource selection algorithm sets the show poster flag to true'); player.tech_.trigger('play'); assert.equal(player.hasStarted(), true, 'the show poster flag is false after play'); player.dispose(); }); QUnit.test('should load a media controller', function(assert) { const player = TestHelpers.makePlayer({ preload: 'none', sources: [ { src: 'http://google.com', type: 'video/mp4' }, { src: 'http://google.com', type: 'video/webm' } ] }); assert.ok(player.el().children[0].className.indexOf('vjs-tech') !== -1, 'media controller loaded'); player.dispose(); }); QUnit.test('should be able to initialize player twice on the same tag using string reference', function(assert) { let videoTag = TestHelpers.makeTag(); const id = videoTag.id; const fixture = document.getElementById('qunit-fixture'); fixture.appendChild(videoTag); let player = videojs(videoTag.id, { techOrder: ['techFaker'] }); assert.ok(player, 'player is created'); player.dispose(); assert.ok(!document.getElementById(id), 'element is removed'); videoTag = TestHelpers.makeTag(); fixture.appendChild(videoTag); // here we receive cached version instead of real player = videojs(videoTag.id, { techOrder: ['techFaker'] }); // here it triggers error, because player was destroyed already after first dispose player.dispose(); }); QUnit.test('should set controls and trigger events', function(assert) { const player = TestHelpers.makePlayer({ controls: false }); assert.ok(player.controls() === false, 'controls set through options'); const hasDisabledClass = player.el().className.indexOf('vjs-controls-disabled'); assert.ok(hasDisabledClass !== -1, 'Disabled class added to player'); player.controls(true); assert.ok(player.controls() === true, 'controls updated'); const hasEnabledClass = player.el().className.indexOf('vjs-controls-enabled'); assert.ok(hasEnabledClass !== -1, 'Disabled class added to player'); player.on('controlsenabled', function() { assert.ok(true, 'enabled fired once'); }); player.on('controlsdisabled', function() { assert.ok(true, 'disabled fired once'); }); player.controls(false); player.dispose(); }); QUnit.test('should toggle user the user state between active and inactive', function(assert) { const player = TestHelpers.makePlayer({}); assert.expect(9); assert.ok(player.userActive(), 'User should be active at player init'); player.on('userinactive', function() { assert.ok(true, 'userinactive event triggered'); }); player.on('useractive', function() { assert.ok(true, 'useractive event triggered'); }); player.userActive(false); assert.ok(player.userActive() === false, 'Player state changed to inactive'); assert.ok(player.el().className.indexOf('vjs-user-active') === -1, 'Active class removed'); assert.ok(player.el().className.indexOf('vjs-user-inactive') !== -1, 'Inactive class added'); player.userActive(true); assert.ok(player.userActive() === true, 'Player state changed to active'); assert.ok(player.el().className.indexOf('vjs-user-inactive') === -1, 'Inactive class removed'); assert.ok(player.el().className.indexOf('vjs-user-active') !== -1, 'Active class added'); player.dispose(); }); QUnit.test('should add a touch-enabled classname when touch is supported', function(assert) { assert.expect(1); // Fake touch support. Real touch support isn't needed for this test. const origTouch = browser.TOUCH_ENABLED; browser.stub_TOUCH_ENABLED(true); const player = TestHelpers.makePlayer({}); assert.notEqual(player.el().className.indexOf('vjs-touch-enabled'), -1, 'touch-enabled classname added'); browser.stub_TOUCH_ENABLED(origTouch); player.dispose(); }); QUnit.test('should not add a touch-enabled classname when touch is not supported', function(assert) { assert.expect(1); // Fake not having touch support in case that the browser running the test supports it const origTouch = browser.TOUCH_ENABLED; browser.stub_TOUCH_ENABLED(false); const player = TestHelpers.makePlayer({}); assert.equal(player.el().className.indexOf('vjs-touch-enabled'), -1, 'touch-enabled classname not added'); browser.stub_TOUCH_ENABLED(origTouch); player.dispose(); }); QUnit.test('should allow for tracking when native controls are used', function(assert) { const player = TestHelpers.makePlayer({}); assert.expect(6); // Make sure native controls is false before starting test player.usingNativeControls(false); player.on('usingnativecontrols', function() { assert.ok(true, 'usingnativecontrols event triggered'); }); player.on('usingcustomcontrols', function() { assert.ok(true, 'usingcustomcontrols event triggered'); }); player.usingNativeControls(true); assert.ok(player.usingNativeControls() === true, 'Using native controls is true'); assert.ok(player.el().className.indexOf('vjs-using-native-controls') !== -1, 'Native controls class added'); player.usingNativeControls(false); assert.ok(player.usingNativeControls() === false, 'Using native controls is false'); assert.ok(player.el().className.indexOf('vjs-using-native-controls') === -1, 'Native controls class removed'); player.dispose(); }); QUnit.test('make sure that controls listeners do not get added too many times', function(assert) { const player = TestHelpers.makePlayer({}); let listeners = 0; player.addTechControlsListeners_ = function() { listeners++; }; // Make sure native controls is false before starting test player.usingNativeControls(false); player.usingNativeControls(true); player.controls(true); assert.equal(listeners, 0, 'addTechControlsListeners_ should not have gotten called yet'); player.usingNativeControls(false); player.controls(false); player.controls(true); assert.equal(listeners, 1, 'addTechControlsListeners_ should have gotten called once'); player.dispose(); }); QUnit.test('should register players with generated ids', function(assert) { const fixture = document.getElementById('qunit-fixture'); const video = document.createElement('video'); video.className = 'vjs-default-skin video-js'; fixture.appendChild(video); const player = new Player(video, { techOrder: ['techFaker'] }); const id = player.el().id; assert.equal(player.el().id, player.id(), 'the player and element ids are equal'); assert.ok(Player.players[id], 'the generated id is registered'); player.dispose(); }); QUnit.test('should not add multiple first play events despite subsequent loads', function(assert) { assert.expect(1); const player = TestHelpers.makePlayer({}); player.on('firstplay', function() { assert.ok(true, 'First play should fire once.'); }); // Checking to make sure onLoadStart removes first play listener before adding a new one. player.tech_.trigger('loadstart'); player.tech_.trigger('loadstart'); player.tech_.trigger('play'); player.dispose(); }); QUnit.test('should fire firstplay after resetting the player', function(assert) { const player = TestHelpers.makePlayer({}); let fpFired = false; player.on('firstplay', function() { fpFired = true; }); // init firstplay listeners player.tech_.trigger('loadstart'); player.tech_.trigger('play'); assert.ok(fpFired, 'First firstplay fired'); // reset the player player.tech_.trigger('loadstart'); fpFired = false; player.tech_.trigger('play'); assert.ok(fpFired, 'Second firstplay fired'); // the play event can fire before the loadstart event. // in that case we still want the firstplay even to fire. player.tech_.paused = function() { return false; }; fpFired = false; // reset the player player.tech_.trigger('loadstart'); // player.tech_.trigger('play'); assert.ok(fpFired, 'Third firstplay fired'); player.dispose(); }); QUnit.test('should remove vjs-has-started class', function(assert) { assert.expect(3); const player = TestHelpers.makePlayer({}); player.tech_.trigger('loadstart'); player.tech_.trigger('play'); assert.ok(player.el().className.indexOf('vjs-has-started') !== -1, 'vjs-has-started class added'); player.tech_.trigger('loadstart'); assert.ok(player.el().className.indexOf('vjs-has-started') === -1, 'vjs-has-started class removed'); player.tech_.trigger('play'); assert.ok(player.el().className.indexOf('vjs-has-started') !== -1, 'vjs-has-started class added again'); player.dispose(); }); QUnit.test('should add and remove vjs-ended class', function(assert) { assert.expect(4); const player = TestHelpers.makePlayer({}); player.tech_.trigger('loadstart'); player.tech_.trigger('play'); player.tech_.trigger('ended'); assert.ok(player.el().className.indexOf('vjs-ended') !== -1, 'vjs-ended class added'); player.tech_.trigger('play'); assert.ok(player.el().className.indexOf('vjs-ended') === -1, 'vjs-ended class removed'); player.tech_.trigger('ended'); assert.ok(player.el().className.indexOf('vjs-ended') !== -1, 'vjs-ended class re-added'); player.tech_.trigger('loadstart'); assert.ok(player.el().className.indexOf('vjs-ended') === -1, 'vjs-ended class removed'); player.dispose(); }); QUnit.test('player should handle different error types', function(assert) { assert.expect(8); const player = TestHelpers.makePlayer({}); const testMsg = 'test message'; // prevent error log messages in the console sinon.stub(log, 'error'); // error code supplied function errCode() { assert.equal(player.error().code, 1, 'error code is correct'); } player.on('error', errCode); player.error(1); player.off('error', errCode); // error instance supplied function errInst() { assert.equal(player.error().code, 2, 'MediaError code is correct'); assert.equal(player.error().message, testMsg, 'MediaError message is correct'); } player.on('error', errInst); player.error(new MediaError({ code: 2, message: testMsg })); player.off('error', errInst); // error message supplied function errMsg() { assert.equal(player.error().code, 0, 'error message code is correct'); assert.equal(player.error().message, testMsg, 'error message is correct'); } player.on('error', errMsg); player.error(testMsg); player.off('error', errMsg); // error config supplied function errConfig() { assert.equal(player.error().code, 3, 'error config code is correct'); assert.equal(player.error().message, testMsg, 'error config message is correct'); } player.on('error', errConfig); player.error({ code: 3, message: testMsg }); player.off('error', errConfig); // check for vjs-error classname assert.ok(player.el().className.indexOf('vjs-error') >= 0, 'player does not have vjs-error classname'); // restore error logging log.error.restore(); player.dispose(); }); QUnit.test('Data attributes on the video element should persist in the new wrapper element', function(assert) { const dataId = 123; const tag = TestHelpers.makeTag(); tag.setAttribute('data-id', dataId); const player = TestHelpers.makePlayer({}, tag); assert.equal(player.el().getAttribute('data-id'), dataId, 'data-id should be available on the new player element after creation'); player.dispose(); }); QUnit.test('should restore attributes from the original video tag when creating a new element', function(assert) { // simulate attributes stored from the original tag const tag = Dom.createEl('video'); tag.setAttribute('preload', 'auto'); tag.setAttribute('autoplay', ''); tag.setAttribute('webkit-playsinline', ''); const html5Mock = { options_: {tag} }; // set options that should override tag attributes html5Mock.options_.preload = 'none'; // create the element const el = Html5.prototype.createEl.call(html5Mock); assert.equal(el.getAttribute('preload'), 'none', 'attribute was successful overridden by an option'); assert.equal(el.getAttribute('autoplay'), '', 'autoplay attribute was set properly'); assert.equal(el.getAttribute('webkit-playsinline'), '', 'webkit-playsinline attribute was set properly'); }); if (Html5.isSupported()) { QUnit.test('player.playsinline() should be able to get/set playsinline attribute', function(assert) { assert.expect(5); const video = document.createElement('video'); const player = TestHelpers.makePlayer({techOrder: ['html5']}, video); // test setter assert.ok(!player.tech_.el().hasAttribute('playsinline'), 'playsinline has not yet been added'); player.playsinline(true); assert.ok(player.tech_.el().hasAttribute('playsinline'), 'playsinline attribute added'); player.playsinline(false); assert.ok(!player.tech_.el().hasAttribute('playsinline'), 'playsinline attribute removed'); // test getter player.tech_.el().setAttribute('playsinline', 'playsinline'); assert.ok(player.playsinline(), 'correctly detects playsinline attribute'); player.tech_.el().removeAttribute('playsinline'); assert.ok(!player.playsinline(), 'correctly detects absence of playsinline attribute'); }); } QUnit.test('if tag exists and movingMediaElementInDOM, re-use the tag', function(assert) { // simulate attributes stored from the original tag const tag = Dom.createEl('video'); tag.setAttribute('preload', 'auto'); tag.setAttribute('autoplay', ''); tag.setAttribute('webkit-playsinline', ''); const html5Mock = { options_: { tag, playerElIngest: false }, movingMediaElementInDOM: true }; // set options that should override tag attributes html5Mock.options_.preload = 'none'; // create the element const el = Html5.prototype.createEl.call(html5Mock); assert.equal(el.getAttribute('preload'), 'none', 'attribute was successful overridden by an option'); assert.equal(el.getAttribute('autoplay'), '', 'autoplay attribute was set properly'); assert.equal(el.getAttribute('webkit-playsinline'), '', 'webkit-playsinline attribute was set properly'); assert.equal(el, tag, 'we have re-used the tag as expected'); }); QUnit.test('if tag exists and *not* movingMediaElementInDOM, create a new tag', function(assert) { // simulate attributes stored from the original tag const tag = Dom.createEl('video'); tag.setAttribute('preload', 'auto'); tag.setAttribute('autoplay', ''); tag.setAttribute('webkit-playsinline', ''); const html5Mock = { options_: { tag, playerElIngest: false }, movingMediaElementInDOM: false }; // set options that should override tag attributes html5Mock.options_.preload = 'none'; // create the element const el = Html5.prototype.createEl.call(html5Mock); assert.equal(el.getAttribute('preload'), 'none', 'attribute was successful overridden by an option'); assert.equal(el.getAttribute('autoplay'), '', 'autoplay attribute was set properly'); assert.equal(el.getAttribute('webkit-playsinline'), '', 'webkit-playsinline attribute was set properly'); assert.notEqual(el, tag, 'we have not re-used the tag as expected'); }); QUnit.test('if tag exists and *not* movingMediaElementInDOM, but playerElIngest re-use tag', function(assert) { // simulate attributes stored from the original tag const tag = Dom.createEl('video'); tag.setAttribute('preload', 'auto'); tag.setAttribute('autoplay', ''); tag.setAttribute('webkit-playsinline', ''); const html5Mock = { options_: { tag, playerElIngest: true }, movingMediaElementInDOM: false }; // set options that should override tag attributes html5Mock.options_.preload = 'none'; // create the element const el = Html5.prototype.createEl.call(html5Mock); assert.equal(el.getAttribute('preload'), 'none', 'attribute was successful overridden by an option'); assert.equal(el.getAttribute('autoplay'), '', 'autoplay attribute was set properly'); assert.equal(el.getAttribute('webkit-playsinline'), '', 'webkit-playsinline attribute was set properly'); assert.equal(el, tag, 'we have re-used the tag as expected'); }); QUnit.test('should honor default inactivity timeout', function(assert) { const clock = sinon.useFakeTimers(); // default timeout is 2000ms const player = TestHelpers.makePlayer({}); player.trigger('play'); assert.equal(player.userActive(), true, 'User is active on creation'); clock.tick(1800); assert.equal(player.userActive(), true, 'User is still active'); clock.tick(500); assert.equal(player.userActive(), false, 'User is inactive after timeout expired'); clock.restore(); player.dispose(); }); QUnit.test('should honor configured inactivity timeout', function(assert) { const clock = sinon.useFakeTimers(); // default timeout is 2000ms, set to shorter 200ms const player = TestHelpers.makePlayer({ inactivityTimeout: 200 }); player.trigger('play'); assert.equal(player.userActive(), true, 'User is active on creation'); clock.tick(150); assert.equal(player.userActive(), true, 'User is still active'); clock.tick(350); // make sure user is now inactive after 500ms assert.equal(player.userActive(), false, 'User is inactive after timeout expired'); clock.restore(); player.dispose(); }); QUnit.test('should honor disabled inactivity timeout', function(assert) { const clock = sinon.useFakeTimers(); // default timeout is 2000ms, disable by setting to zero const player = TestHelpers.makePlayer({ inactivityTimeout: 0 }); assert.equal(player.userActive(), true, 'User is active on creation'); clock.tick(5000); assert.equal(player.userActive(), true, 'User is still active'); clock.restore(); player.dispose(); }); QUnit.test('should clear pending errors on disposal', function(assert) { const clock = sinon.useFakeTimers(); const player = TestHelpers.makePlayer(); clock.tick(1); player.src({ src: 'http://example.com/movie.unsupported-format', type: 'video/unsupported-format' }); clock.tick(1); player.dispose(); try { clock.tick(5000); } catch (e) { return assert.ok(!e, 'threw an error: ' + e.message); } assert.ok(true, 'did not throw an error after disposal'); }); QUnit.test('pause is called when player ended event is fired and player is not paused', function(assert) { const video = document.createElement('video'); const player = TestHelpers.makePlayer({}, video); let pauses = 0; player.paused = function() { return false; }; player.pause = function() { pauses++; }; player.tech_.trigger('ended'); assert.equal(pauses, 1, 'pause was called'); player.dispose(); }); QUnit.test('pause is not called if the player is paused and ended is fired', function(assert) { const video = document.createElement('video'); const player = TestHelpers.makePlayer({}, video); let pauses = 0; player.paused = function() { return true; }; player.pause = function() { pauses++; }; player.tech_.trigger('ended'); assert.equal(pauses, 0, 'pause was not called when ended fired'); player.dispose(); }); QUnit.test('should add an audio class if an audio el is used', function(assert) { const audio = document.createElement('audio'); const player = TestHelpers.makePlayer({}, audio); const audioClass = 'vjs-audio'; assert.ok(player.el().className.indexOf(audioClass) !== -1, 'added ' + audioClass + ' css class'); player.dispose(); }); QUnit.test('should add a video player region if a video el is used', function(assert) { const video = document.createElement('video'); const player = TestHelpers.makePlayer({}, video); assert.ok(player.el().getAttribute('role') === 'region', 'region role is present'); assert.ok(player.el().getAttribute('aria-label') === 'Video Player', 'Video Player label present'); player.dispose(); }); QUnit.test('should add an audio player region if an audio el is used', function(assert) { const audio = document.createElement('audio'); const player = TestHelpers.makePlayer({}, audio); assert.ok(player.el().getAttribute('role') === 'region', 'region role is present'); assert.ok(player.el().getAttribute('aria-label') === 'Audio Player', 'Audio Player label present'); player.dispose(); }); QUnit.test('should setScrubbing when seeking or not seeking', function(assert) { const player = TestHelpers.makePlayer(); let isScrubbing; player.tech_.setScrubbing = (_isScrubbing) => { isScrubbing = _isScrubbing; }; assert.equal(player.scrubbing(), false, 'player is not scrubbing'); player.scrubbing(true); assert.ok(isScrubbing, "tech's setScrubbing was called with true"); player.scrubbing(false); assert.notOk(isScrubbing, "tech's setScrubbing was called with false"); }); QUnit.test('should not be scrubbing while not seeking', function(assert) { const player = TestHelpers.makePlayer(); assert.equal(player.scrubbing(), false, 'player is not scrubbing'); assert.ok(player.el().className.indexOf('scrubbing') === -1, 'scrubbing class is not present'); player.scrubbing(false); assert.equal(player.scrubbing(), false, 'player is not scrubbing'); player.dispose(); }); QUnit.test('should be scrubbing while seeking', function(assert) { const player = TestHelpers.makePlayer(); player.scrubbing(true); assert.equal(player.scrubbing(), true, 'player is scrubbing'); assert.ok(player.el().className.indexOf('scrubbing') !== -1, 'scrubbing class is present'); player.dispose(); }); QUnit.test('should throw on startup no techs are specified', function(assert) { const techOrder = videojs.options.techOrder; const fixture = document.getElementById('qunit-fixture'); videojs.options.techOrder = null; assert.throws(function() { const tag = TestHelpers.makeTag(); fixture.appendChild(tag); videojs(tag); }, 'a falsey techOrder should throw'); videojs.options.techOrder = techOrder; }); QUnit.test('should have a sensible toJSON that is equivalent to player.options', function(assert) { const playerOptions = { html5: { nativeTextTracks: false } }; const player = TestHelpers.makePlayer(playerOptions); assert.deepEqual(player.toJSON(), player.options_, 'simple player options toJSON produces output equivalent to player.options_'); const playerOptions2 = { tracks: [{ label: 'English', srclang: 'en', src: '../docs/examples/shared/example-captions.vtt', kind: 'captions' }] }; const player2 = TestHelpers.makePlayer(playerOptions2); playerOptions2.tracks[0].player = player2; const popts = player2.options_; popts.tracks[0].player = undefined; assert.deepEqual(player2.toJSON(), popts, 'no circular references'); player.dispose(); player2.dispose(); }); QUnit.test('should ignore case in language codes and try primary code', function(assert) { assert.expect(3); const player = TestHelpers.makePlayer({ languages: { 'en-gb': { Good: 'Brilliant' }, 'EN': { Good: 'Awesome', Error: 'Problem' } } }); player.language('en-gb'); assert.strictEqual(player.localize('Good'), 'Brilliant', 'Used subcode specific localisation'); assert.strictEqual(player.localize('Error'), 'Problem', 'Used primary code localisation'); player.language('en-GB'); assert.strictEqual(player.localize('Good'), 'Brilliant', 'Ignored case'); player.dispose(); }); QUnit.test('inherits language from parent element', function(assert) { const fixture = document.getElementById('qunit-fixture'); const oldLang = fixture.getAttribute('lang'); fixture.setAttribute('lang', 'x-test'); const player = TestHelpers.makePlayer(); assert.equal(player.language(), 'x-test', 'player inherits parent element language'); player.dispose(); if (oldLang) { fixture.setAttribute('lang', oldLang); } else { fixture.removeAttribute('lang'); } }); QUnit.test('sets lang attribute on player el', function(assert) { const fixture = document.getElementById('qunit-fixture'); const oldLang = fixture.getAttribute('lang'); fixture.setAttribute('lang', 'x-attr-test'); const player = TestHelpers.makePlayer(); assert.equal(player.el().getAttribute('lang'), 'x-attr-test', 'player sets lang attribute on self'); player.dispose(); if (oldLang) { fixture.setAttribute('lang', oldLang); } else { fixture.removeAttribute('lang'); } }); QUnit.test('language changed should trigger languagechange event', function(assert) { const player = TestHelpers.makePlayer({}); assert.expect(1); player.on('languagechange', function() { assert.ok(true, 'languagechange event triggered'); }); player.language('es-MX'); player.dispose(); }); QUnit.test('language changed should not trigger languagechange event if language is the same', function(assert) { const player = TestHelpers.makePlayer({}); assert.expect(1); let triggered = false; player.language('es-MX'); player.on('languagechange', function() { triggered = true; }); player.language('es-MX'); assert.equal(triggered, false, 'languagechange event was not triggered'); player.dispose(); }); QUnit.test('change language multiple times should trigger languagechange event', function(assert) { const player = TestHelpers.makePlayer({}); assert.expect(3); player.on('languagechange', function() { assert.ok(true, 'languagechange event triggered'); }); player.language('es-MX'); player.language('en-EU'); // set same language should not trigger the event so we expect 3 asserts not 4. player.language('en-EU'); player.language('es-ES'); player.dispose(); }); QUnit.test('should return correct values for canPlayType', function(assert) { const player = TestHelpers.makePlayer(); assert.equal(player.canPlayType('video/mp4'), 'maybe', 'player can play mp4 files'); assert.equal(player.canPlayType('video/unsupported-format'), '', 'player can not play unsupported files'); player.dispose(); }); QUnit.test('createModal()', function(assert) { const player = TestHelpers.makePlayer(); const modal = player.createModal('foo'); const spy = sinon.spy(); modal.on('dispose', spy); assert.expect(5); assert.strictEqual(modal.el().parentNode, player.el(), 'the modal is injected into the player'); assert.strictEqual(modal.content(), 'foo', 'content is set properly'); assert.ok(modal.opened(), 'modal is opened by default'); modal.close(); assert.ok(spy.called, 'modal was disposed when closed'); assert.strictEqual(player.children().indexOf(modal), -1, 'modal was removed from player\'s children'); player.dispose(); }); QUnit.test('createModal() options object', function(assert) { const player = TestHelpers.makePlayer(); const modal = player.createModal('foo', {content: 'bar', label: 'boo'}); assert.expect(2); assert.strictEqual(modal.content(), 'foo', 'content argument takes precedence'); assert.strictEqual(modal.options_.label, 'boo', 'modal options are set properly'); modal.close(); player.dispose(); }); QUnit.test('you can clear error in the error event', function(assert) { const player = TestHelpers.makePlayer(); sinon.stub(log, 'error'); player.error({code: 4}); assert.ok(player.error(), 'we have an error'); player.error(null); player.one('error', function() { player.error(null); }); player.error({code: 4}); assert.ok(!player.error(), 'we no longer have an error'); log.error.restore(); player.dispose(); }); QUnit.test('Player#tech will return tech given the appropriate input', function(assert) { const oldLogWarn = log.warn; let warning; log.warn = function(_warning) { warning = _warning; }; const tech_ = {}; const returnedTech = Player.prototype.tech.call({tech_}, true); assert.equal(returnedTech, tech_, 'We got back the tech we wanted'); assert.notOk(warning, 'no warning was logged'); log.warn = oldLogWarn; }); QUnit.test('Player#tech logs a warning when called without a safety argument', function(assert) { const oldLogWarn = log.warn; const warningRegex = new RegExp('https://github.com/videojs/video.js/issues/2617'); let warning; log.warn = function(_warning) { warning = _warning; }; const tech_ = {}; Player.prototype.tech.call({tech_}); assert.ok(warningRegex.test(warning), 'we logged a warning'); log.warn = oldLogWarn; }); QUnit.test('player#reset loads the Html5 tech and then techCalls reset', function(assert) { let loadedTech; let loadedSource; let techCallMethod; const testPlayer = { options_: { techOrder: ['html5', 'flash'] }, resetCache_() {}, loadTech_(tech, source) { loadedTech = tech; loadedSource = source; }, techCall_(method) { techCallMethod = method; }, resetControlBarUI_() {}, poster() {}, paused() { return true; }, doReset_: Player.prototype.doReset_ }; Player.prototype.reset.call(testPlayer); assert.equal(loadedTech, 'html5', 'we loaded the html5 tech'); assert.equal(loadedSource, null, 'with a null source'); assert.equal(techCallMethod, 'reset', 'we then reset the tech'); }); QUnit.test('player#reset loads the first item in the techOrder and then techCalls reset', function(assert) { let loadedTech; let loadedSource; let techCallMethod; const testPlayer = { options_: { techOrder: ['flash', 'html5'] }, resetCache_() {}, loadTech_(tech, source) { loadedTech = tech; loadedSource = source; }, techCall_(method) { techCallMethod = method; }, resetControlBarUI_() {}, poster() {}, paused() { return true; }, doReset_: Player.prototype.doReset_ }; Player.prototype.reset.call(testPlayer); assert.equal(loadedTech, 'flash', 'we loaded the Flash tech'); assert.equal(loadedSource, null, 'with a null source'); assert.equal(techCallMethod, 'reset', 'we then reset the tech'); }); QUnit.test('player#reset clears the player cache', function(assert) { const player = TestHelpers.makePlayer(); const sources = [{ src: '//vjs.zencdn.net/v/oceans.mp4', type: 'video/mp4' }, { src: '//vjs.zencdn.net/v/oceans.webm', type: 'video/webm' }]; this.clock.tick(1); player.src(sources); player.duration(10); player.playbackRate(0.5); player.volume(0.2); assert.strictEqual(player.currentSrc(), sources[0].src, 'currentSrc is correct'); assert.deepEqual(player.currentSource(), sources[0], 'currentSource is correct'); assert.deepEqual(player.currentSources(), sources, 'currentSources is correct'); assert.strictEqual(player.duration(), 10, 'duration is correct'); assert.strictEqual(player.playbackRate(), 0.5, 'playbackRate is correct'); assert.strictEqual(player.volume(), 0.2, 'volume is correct'); assert.strictEqual(player.lastVolume_(), 0.2, 'lastVolume_ is correct'); player.reset(); assert.strictEqual(player.currentSrc(), '', 'currentSrc is correct'); assert.deepEqual(player.currentSource(), {}, 'currentSource is correct'); assert.deepEqual(player.currentSources(), [], 'currentSources is correct'); // Right now, the currentTime is not _really_ cached because it is always // retrieved from the tech. However, for completeness, we set it to zero in // the `resetCache_` method to ensure that if we do start actually caching it, // we reset it along with everything else. assert.strictEqual(player.getCache().currentTime, 0, 'currentTime is correct'); assert.ok(isNaN(player.duration()), 'duration is correct'); assert.strictEqual(player.playbackRate(), 1, 'playbackRate is correct'); assert.strictEqual(player.volume(), 1, 'volume is correct'); assert.strictEqual(player.lastVolume_(), 1, 'lastVolume_ is correct'); }); QUnit.test('player#reset removes the poster', function(assert) { const player = TestHelpers.makePlayer(); this.clock.tick(1); player.poster('foo.jpg'); assert.strictEqual(player.poster(), 'foo.jpg', 'the poster was set'); player.reset(); assert.strictEqual(player.poster(), '', 'the poster was reset'); }); QUnit.test('player#reset removes remote text tracks', function(assert) { sinon.stub(log, 'warn'); const player = TestHelpers.makePlayer(); this.clock.tick(1); player.addRemoteTextTrack({ kind: 'captions', src: 'foo.vtt', language: 'en', label: 'English' }); assert.strictEqual(player.remoteTextTracks().length, 1, 'there is one RTT'); player.reset(); assert.strictEqual(player.remoteTextTracks().length, 0, 'there are zero RTTs'); assert.strictEqual(log.warn.callCount, 1, 'one warning about for manualCleanup'); log.warn.restore(); }); QUnit.test('Remove waiting class after tech waiting when timeupdate shows a time change', function(assert) { const player = TestHelpers.makePlayer(); player.currentTime = () => 1; player.tech_.trigger('waiting'); assert.ok( /vjs-waiting/.test(player.el().className), 'vjs-waiting is added to the player el on tech waiting' ); player.trigger('timeupdate'); assert.ok( /vjs-waiting/.test(player.el().className), 'vjs-waiting still exists on the player el when time hasn\'t changed on timeupdate' ); player.currentTime = () => 2; player.trigger('timeupdate'); assert.notOk( (/vjs-waiting/).test(player.el().className), 'vjs-waiting removed from the player el when time has changed on timeupdate' ); player.dispose(); }); QUnit.test('Queues playing events when playback rate is zero while seeking', function(assert) { const player = TestHelpers.makePlayer({techOrder: ['html5']}); let canPlayCount = 0; let canPlayThroughCount = 0; let playingCount = 0; let seekedCount = 0; let seeking = false; player.on('canplay', () => canPlayCount++); player.on('canplaythrough', () => canPlayThroughCount++); player.on('playing', () => playingCount++); player.on('seeked', () => seekedCount++); player.tech_.seeking = () => { return seeking; }; player.tech_.setPlaybackRate(0); player.tech_.trigger('ratechange'); player.tech_.trigger('canplay'); player.tech_.trigger('canplaythrough'); player.tech_.trigger('playing'); player.tech_.trigger('seeked'); assert.equal(canPlayCount, 1, 'canplay event dispatched when not seeking'); assert.equal(canPlayThroughCount, 1, 'canplaythrough event dispatched when not seeking'); assert.equal(playingCount, 1, 'playing event dispatched when not seeking'); assert.equal(seekedCount, 1, 'seeked event dispatched when not seeking'); seeking = true; player.tech_.trigger('canplay'); player.tech_.trigger('canplaythrough'); player.tech_.trigger('playing'); player.tech_.trigger('seeked'); assert.equal(canPlayCount, 1, 'canplay event not dispatched'); assert.equal(canPlayThroughCount, 1, 'canplaythrough event not dispatched'); assert.equal(playingCount, 1, 'playing event not dispatched'); assert.equal(seekedCount, 1, 'seeked event not dispatched'); seeking = false; player.tech_.setPlaybackRate(1); player.tech_.trigger('ratechange'); assert.equal(canPlayCount, 2, 'canplay event dispatched after playback rate restore'); assert.equal(canPlayThroughCount, 2, 'canplaythrough event dispatched after playback rate restore'); assert.equal(playingCount, 2, 'playing event dispatched after playback rate restore'); assert.equal(seekedCount, 2, 'seeked event dispatched after playback rate restore'); }); QUnit.test('Make sure that player\'s style el respects VIDEOJS_NO_DYNAMIC_STYLE option', function(assert) { // clear the HEAD before running this test let styles = document.querySelectorAll('style'); let i = styles.length; while (i--) { const style = styles[i]; style.parentNode.removeChild(style); } let tag = TestHelpers.makeTag(); tag.id = 'vjs-no-base-theme-tag'; tag.width = 600; tag.height = 300; window.VIDEOJS_NO_DYNAMIC_STYLE = true; TestHelpers.makePlayer({}, tag); styles = document.querySelectorAll('style'); assert.equal(styles.length, 0, 'we should not get any style elements included in the DOM'); window.VIDEOJS_NO_DYNAMIC_STYLE = false; tag = TestHelpers.makeTag(); TestHelpers.makePlayer({}, tag); styles = document.querySelectorAll('style'); assert.equal(styles.length, 1, 'we should have one style element in the DOM'); assert.equal(styles[0].className, 'vjs-styles-dimensions', 'the class name is the one we expected'); }); QUnit.test('When VIDEOJS_NO_DYNAMIC_STYLE is set, apply sizing directly to the tech el', function(assert) { // clear the HEAD before running this test const styles = document.querySelectorAll('style'); let i = styles.length; while (i--) { const style = styles[i]; style.parentNode.removeChild(style); } const tag = TestHelpers.makeTag(); tag.id = 'vjs-no-base-theme-tag'; tag.width = 600; tag.height = 300; window.VIDEOJS_NO_DYNAMIC_STYLE = true; const player = TestHelpers.makePlayer({}, tag); player.width(300); player.height(600); assert.equal(player.tech_.el().width, 300, 'the width is equal to 300'); assert.equal(player.tech_.el().height, 600, 'the height is equal 600'); player.width(600); player.height(300); assert.equal(player.tech_.el().width, 600, 'the width is equal to 600'); assert.equal(player.tech_.el().height, 300, 'the height is equal 300'); player.dispose(); }); QUnit.test('should allow to register custom player when any player has not been created', function(assert) { class CustomPlayer extends Player {} videojs.registerComponent('Player', CustomPlayer); const tag = TestHelpers.makeTag(); const fixture = document.getElementById('qunit-fixture'); fixture.appendChild(tag); const player = videojs(tag); assert.equal(player instanceof CustomPlayer, true, 'player is custom'); player.dispose(); // reset the Player to the original value; videojs.registerComponent('Player', Player); }); QUnit.test('should not allow to register custom player when any player has been created', function(assert) { const tag = TestHelpers.makeTag(); const fixture = document.getElementById('qunit-fixture'); fixture.appendChild(tag); const player = videojs(tag); class CustomPlayer extends Player {} assert.throws(function() { videojs.registerComponent('Player', CustomPlayer); }, 'Can not register Player component after player has been created'); player.dispose(); // reset the Player to the original value; videojs.registerComponent('Player', Player); }); QUnit.test('techGet runs through middleware if allowedGetter', function(assert) { let cts = 0; let muts = 0; let vols = 0; let durs = 0; let lps = 0; videojs.use('video/foo', () => ({ currentTime() { cts++; }, duration() { durs++; }, loop() { lps++; }, muted() { muts++; }, volume() { vols++; } })); const tag = TestHelpers.makeTag(); const fixture = document.getElementById('qunit-fixture'); fixture.appendChild(tag); const player = videojs(tag, { techOrder: ['techFaker'] }); player.middleware_ = [middleware.getMiddleware('video/foo')[0](player)]; player.techGet_('currentTime'); player.techGet_('volume'); player.techGet_('duration'); player.techGet_('loop'); player.techGet_('muted'); assert.equal(cts, 1, 'currentTime is allowed'); assert.equal(vols, 1, 'volume is allowed'); assert.equal(durs, 1, 'duration is allowed'); assert.equal(muts, 1, 'muted is allowed'); assert.equal(lps, 0, 'loop is not allowed'); middleware.getMiddleware('video/foo').pop(); player.dispose(); }); QUnit.test('techCall runs through middleware if allowedSetter', function(assert) { let cts = 0; let muts = false; let vols = 0; let prs = 0; videojs.use('video/foo', () => ({ setCurrentTime(ct) { cts++; return ct; }, setVolume() { vols++; return vols; }, setMuted() { muts = true; return muts; }, setPlaybackRate() { prs++; return prs; } })); const tag = TestHelpers.makeTag(); const fixture = document.getElementById('qunit-fixture'); fixture.appendChild(tag); const player = videojs(tag, { techOrder: ['techFaker'] }); player.middleware_ = [middleware.getMiddleware('video/foo')[0](player)]; this.clock.tick(1); player.techCall_('setCurrentTime', 10); player.techCall_('setVolume', 0.5); player.techCall_('setMuted', true); player.techCall_('setPlaybackRate', 0.75); this.clock.tick(1); assert.equal(cts, 1, 'setCurrentTime is allowed'); assert.equal(vols, 1, 'setVolume is allowed'); assert.equal(muts, true, 'setMuted is allowed'); assert.equal(prs, 0, 'setPlaybackRate is not allowed'); middleware.getMiddleware('video/foo').pop(); player.dispose(); }); QUnit.test('src selects tech based on middleware', function(assert) { const oldTechs = Tech.techs_; const oldDefaultTechOrder = Tech.defaultTechOrder_; class FooTech extends Html5 {} class BarTech extends Html5 {} FooTech.isSupported = () => true; FooTech.canPlayType = (type) => type === 'video/mp4'; FooTech.canPlaySource = (src) => FooTech.canPlayType(src.type); BarTech.isSupported = () => true; BarTech.canPlayType = (type) => type === 'video/flv'; BarTech.canPlaySource = (src) => BarTech.canPlayType(src.type); videojs.registerTech('FooTech', FooTech); videojs.registerTech('BarTech', BarTech); videojs.use('video/foo', () => ({ setSource(src, next) { next(null, { src: 'http://example.com/video.mp4', type: 'video/mp4' }); } })); videojs.use('video/bar', () => ({ setSource(src, next) { next(null, { src: 'http://example.com/video.flv', type: 'video/flv' }); } })); const fixture = document.getElementById('qunit-fixture'); const tag = TestHelpers.makeTag(); fixture.appendChild(tag); const player = videojs(tag, { techOrder: ['fooTech', 'barTech'] }); player.src({ src: 'foo', type: 'video/foo' }); this.clock.tick(1); assert.equal(player.techName_, 'FooTech', 'the FooTech (html5) tech is chosen'); player.src({ src: 'bar', type: 'video/bar' }); this.clock.tick(1); assert.equal(player.techName_, 'BarTech', 'the BarTech (Flash) tech is chosen'); middleware.getMiddleware('video/foo').pop(); middleware.getMiddleware('video/bar').pop(); player.dispose(); delete Tech.techs_.FooTech; delete Tech.techs_.BarTech; Tech.defaultTechOrder_ = oldDefaultTechOrder; Tech.techs_ = oldTechs; }); QUnit.test('src_ does not call loadTech is name is titleCaseEquals', function(assert) { let loadTechCalled = 0; const playerProxy = { selectSource() { return { tech: 'html5' }; }, techName_: 'Html5', ready() {}, loadTech_() { loadTechCalled++; } }; Player.prototype.src_.call(playerProxy); assert.equal(loadTechCalled, 0, 'loadTech was not called'); }); QUnit.test('options: plugins', function(assert) { const optionsSpy = sinon.spy(); Plugin.registerPlugin('foo', (options) => { optionsSpy(options); }); const player = TestHelpers.makePlayer({ plugins: { foo: { bar: 1 } } }); assert.strictEqual(optionsSpy.callCount, 1, 'the plugin was set up'); assert.deepEqual(optionsSpy.getCall(0).args[0], {bar: 1}, 'the plugin got the expected options'); assert.throws( () => { TestHelpers.makePlayer({ plugins: { nope: {} } }); }, new Error('plugin "nope" does not exist'), 'plugins that do not exist cause the player to throw' ); player.dispose(); Plugin.deregisterPlugin('foo'); }); QUnit.test('should add a class with major version', function(assert) { const majorVersion = pkg.version.split('.')[0]; const player = TestHelpers.makePlayer(); assert.ok(player.hasClass('vjs-v' + majorVersion), 'the version class should be added to the player'); player.dispose(); }); QUnit.test('player.duration() returns NaN if player.cache_.duration is undefined', function(assert) { const player = TestHelpers.makePlayer(); player.cache_.duration = undefined; assert.ok(Number.isNaN(player.duration()), 'returned NaN for unkown duration'); }); QUnit.test('player.duration() returns player.cache_.duration if it is defined', function(assert) { const player = TestHelpers.makePlayer(); player.cache_.duration = 200; assert.equal(player.duration(), 200, 'returned correct integer duration'); player.cache_.duration = 942; assert.equal(player.duration(), 942, 'returned correct integer duration'); }); QUnit.test('player.duration() sets the value of player.cache_.duration', function(assert) { const player = TestHelpers.makePlayer(); // set an arbitrary initial cached duration value for testing the setter functionality player.cache_.duration = 1; player.duration(NaN); assert.ok(Number.isNaN(player.duration()), 'duration() set and get NaN duration value'); player.duration(200); assert.equal(player.duration(), 200, 'duration() set and get integer duration value'); }); QUnit.test('setPoster in tech with `techCanOverridePoster` in player should override poster', function(assert) { const player = TestHelpers.makePlayer({ techCanOverridePoster: true }); const posterchangeSpy = sinon.spy(); const firstPosterUrl = 'https://wherever.test/test.jpg'; const techPosterUrl = 'https://somewhere.text/my/image.png'; assert.equal(player.options_.techCanOverridePoster, true, 'make sure player option was passed correctly'); assert.equal(player.tech_.options_.canOverridePoster, true, 'make sure tech option was passed correctly'); player.on('posterchange', posterchangeSpy); player.poster(''); assert.ok(posterchangeSpy.notCalled, 'posterchangeSpy not called when no change of poster'); assert.equal(player.isPosterFromTech_, false, "ensure tech didn't change poster after empty call from player"); player.poster(firstPosterUrl); assert.ok(posterchangeSpy.calledOnce, 'posterchangeSpy only called once on update'); assert.equal(player.poster(), firstPosterUrl, "ensure tech didn't change poster after setting from player"); assert.equal(player.isPosterFromTech_, false, "ensure player didn't mark poster as changed by the tech"); posterchangeSpy.resetHistory(); player.tech_.setPoster(techPosterUrl); assert.ok(posterchangeSpy.calledOnce, "posterchangeSpy should've been called"); assert.equal(player.isPosterFromTech_, true, 'ensure player marked poster as set by tech after the fact'); player.dispose(); }); QUnit.test('setPoster in tech WITHOUT `techCanOverridePoster` in player should NOT override poster', function(assert) { const player = TestHelpers.makePlayer(); const posterchangeSpy = sinon.spy(); const firstPosterUrl = 'https://wherever.test/test.jpg'; const techPosterUrl = 'https://somewhere.test/my/image.png'; assert.equal(player.options_.techCanOverridePoster, undefined, "ensure player option wasn't unwittingly set"); assert.equal(player.tech_.options_.canOverridePoster, false, "ensure tech option wasn't unwittinyly set"); player.on('posterchange', posterchangeSpy); player.poster(firstPosterUrl); assert.ok(posterchangeSpy.calledOnce, 'posterchangeSpy only called once on update'); assert.equal(player.poster(), firstPosterUrl, "ensure tech didn't change poster after setting from player"); assert.equal(player.isPosterFromTech_, false, "ensure player didn't mark poster as changed by the tech"); posterchangeSpy.resetHistory(); player.tech_.setPoster(techPosterUrl); assert.ok(posterchangeSpy.notCalled, "posterchangeSpy shouldn't have been called"); assert.equal(player.isPosterFromTech_, false, "ensure tech didn't change poster because player option was false"); player.dispose(); }); QUnit.test('disposing a tech that set a poster, should unset the poster', function(assert) { const player = TestHelpers.makePlayer({ techCanOverridePoster: true }); const techPosterUrl = 'https://somewhere.test/my/image.png'; assert.equal(player.options_.techCanOverridePoster, true, 'make sure player option was passed correctly'); assert.equal(player.tech_.options_.canOverridePoster, true, 'make sure tech option was passed correctly'); player.tech_.setPoster(techPosterUrl); assert.equal(player.poster(), techPosterUrl, 'player poster should equal tech poster'); assert.equal(player.isPosterFromTech_, true, 'setting the poster with the tech should be remembered in the player'); player.unloadTech_(); assert.equal(player.poster(), '', 'ensure poster set by poster is unset after tech disposal'); player.dispose(); }); QUnit.test('disposing a tech that dit NOT set a poster, should keep the poster', function(assert) { const player = TestHelpers.makePlayer({ techCanOverridePoster: true }); const posterUrl = 'https://myposter.test/lol.jpg'; assert.equal(player.options_.techCanOverridePoster, true, 'make sure player option was passed correctly'); assert.equal(player.tech_.options_.canOverridePoster, true, 'make sure tech option was passed correctly'); player.poster(posterUrl); assert.equal(player.poster(), posterUrl, 'player poster should NOT have changed'); assert.equal(player.isPosterFromTech_, false, 'player should mark poster as set by itself'); player.unloadTech_(); assert.equal(player.poster(), posterUrl, 'player poster should stay the same after unloading / dispoing tech'); player.dispose(); }); QUnit.test('source options are retained', function(assert) { const player = TestHelpers.makePlayer(); const source = { src: 'https://some.url', type: 'someType', sourceOption: 'someOption' }; player.src(source); assert.equal(player.currentSource().sourceOption, 'someOption', 'source option retained'); }); QUnit.test('setting children to false individually, does not cause an assertion', function(assert) { const defaultChildren = Player.prototype.options_.children; defaultChildren.forEach((childName) => { const options = {}; options[childName] = false; const player = TestHelpers.makePlayer(options); this.clock.tick(1000); player.triggerReady(); player.dispose(); assert.ok(true, `${childName}: false. did not cause an assertion`); }); }); QUnit.test('setting all children to false, does not cause an assertion', function(assert) { const defaultChildren = Player.prototype.options_.children; const options = {}; defaultChildren.forEach((childName) => { options[childName] = false; }); const player = TestHelpers.makePlayer(options); this.clock.tick(1000); player.triggerReady(); player.dispose(); assert.ok(true, 'did not cause an assertion'); }); QUnit.test('controlBar behaviour with mouseenter and mouseleave events', function(assert) { const player = TestHelpers.makePlayer(); player.listenForUserActivity_(); assert.equal(player.options_.inactivityTimeout, 2000, 'inactivityTimeout default value is 2000'); const el = player.getChild('controlBar').el(); // move mouse to controlBar Events.trigger(el, 'mouseenter'); assert.equal(player.options_.inactivityTimeout, 0, 'mouseenter on control-bar, inactivityTimeout is set to 0'); // move mouse out of controlBar bounds Events.trigger(el, 'mouseleave'); assert.equal(player.options_.inactivityTimeout, player.cache_.inactivityTimeout, 'mouse leaves control-bar, inactivityTimeout is set to default value (2000)'); player.dispose(); }); QUnit.test('Should be able to set a currentTime after player initialization as soon the canplay event is fired', function(assert) { const player = TestHelpers.makePlayer({}); player.src('xyz.mp4'); player.currentTime(500); assert.strictEqual(player.currentTime(), 0, 'currentTime value was not changed'); this.clock.tick(100); player.trigger('canplay'); assert.strictEqual(player.currentTime(), 500, 'currentTime value is the one passed after initialization'); }); QUnit.test('Should accept multiple calls to currentTime after player initialization and apply the last value as soon the canplay event is fired', function(assert) { const player = TestHelpers.makePlayer({}); const spyInitTime = sinon.spy(player, 'applyInitTime_'); const spyCurrentTime = sinon.spy(player, 'currentTime'); player.src('xyz.mp4'); player.currentTime(500); player.currentTime(600); player.currentTime(700); player.currentTime(800); this.clock.tick(100); player.trigger('canplay'); assert.equal(spyInitTime.callCount, 1, 'After multiple calls to currentTime just apply the last one'); assert.ok(spyCurrentTime.calledAfter(spyInitTime), 'currentTime was called on canplay event listener'); assert.equal(player.currentTime(), 800, 'The last value passed is stored as the currentTime value'); }); QUnit.test('Should fire debugon event when debug mode is enabled', function(assert) { const player = TestHelpers.makePlayer({}); const debugOnSpy = sinon.spy(); player.on('debugon', debugOnSpy); player.debug(true); assert.ok(debugOnSpy.calledOnce, 'debugon event was fired'); player.dispose(); }); QUnit.test('Should fire debugoff event when debug mode is disabled', function(assert) { const player = TestHelpers.makePlayer({}); const debugOffSpy = sinon.spy(); player.on('debugoff', debugOffSpy); player.debug(false); assert.ok(debugOffSpy.calledOnce, 'debugoff event was fired'); player.dispose(); }); QUnit.test('Should enable debug mode and store log level when calling options', function(assert) { const player = TestHelpers.makePlayer({debug: true}); assert.ok(player.previousLogLevel_, 'debug', 'previous log level is stored when enabling debug'); player.dispose(); }); QUnit.test('Should restore previous log level when disabling debug mode', function(assert) { const player = TestHelpers.makePlayer(); player.log.level('error'); player.debug(true); assert.ok(player.log.level(), 'debug', 'log level is debug when debug is enabled'); player.debug(false); assert.ok(player.log.level(), 'error', 'previous log level was restored'); player.dispose(); }); QUnit.test('Should return if debug is enabled or disabled', function(assert) { const player = TestHelpers.makePlayer(); player.debug(true); const enabled = player.debug(); assert.ok(enabled); player.debug(false); const disabled = player.debug(); assert.notOk(disabled); player.dispose(); }); const testOrSkip = 'pictureInPictureEnabled' in document ? 'test' : 'skip'; QUnit[testOrSkip]('Should only allow requestPictureInPicture if the tech supports it', function(assert) { const player = TestHelpers.makePlayer({}); let count = 0; player.tech_.el_ = { disablePictureInPicture: false, requestPictureInPicture() { count++; } }; player.tech_.requestPictureInPicture = function() { return player.tech_.el_.requestPictureInPicture(); }; player.tech_.disablePictureInPicture = function() { return this.el_.disablePictureInPicture; }; player.requestPictureInPicture(); assert.equal(count, 1, 'requestPictureInPicture passed through to supporting tech'); player.tech_.el_.disablePictureInPicture = true; player.requestPictureInPicture(); assert.equal(count, 1, 'requestPictureInPicture not passed through when disabled on tech'); delete player.tech_.el_.disablePictureInPicture; player.requestPictureInPicture(); assert.equal(count, 1, 'requestPictureInPicture not passed through when tech does not support'); });