module('Component'); var getFakePlayer = function(){ return { // Fake player requries an ID id: function(){ return 'player_1'; }, reportUserActivity: function(){} }; }; test('should create an element', function(){ var comp = new vjs.Component(getFakePlayer(), {}); ok(comp.el().nodeName); }); test('should add a child component', function(){ var comp = new vjs.Component(getFakePlayer()); var child = comp.addChild('component'); ok(comp.children().length === 1); ok(comp.children()[0] === child); ok(comp.el().childNodes[0] === child.el()); ok(comp.getChild('component') === child); ok(comp.getChildById(child.id()) === child); }); test('should init child components from options', function(){ var comp = new vjs.Component(getFakePlayer(), { children: { 'component': true } }); ok(comp.children().length === 1); ok(comp.el().childNodes.length === 1); }); test('should init child components from simple children array', function(){ var comp = new vjs.Component(getFakePlayer(), { children: [ 'component', 'component', 'component' ] }); ok(comp.children().length === 3); ok(comp.el().childNodes.length === 3); }); test('should init child components from children array of objects', function(){ var comp = new vjs.Component(getFakePlayer(), { children: [ { 'name': 'component' }, { 'name': 'component' }, { 'name': 'component' } ] }); ok(comp.children().length === 3); ok(comp.el().childNodes.length === 3); }); test('should do a deep merge of child options', function(){ // Create a default option for component vjs.Component.prototype.options_ = { 'example': { 'childOne': { 'foo': 'bar', 'asdf': 'fdsa' }, 'childTwo': {}, 'childThree': {} } }; var comp = new vjs.Component(getFakePlayer(), { 'example': { 'childOne': { 'foo': 'baz', 'abc': '123' }, 'childThree': null, 'childFour': {} } }); var mergedOptions = comp.options(); var children = mergedOptions['example']; ok(children['childOne']['foo'] === 'baz', 'value three levels deep overridden'); ok(children['childOne']['asdf'] === 'fdsa', 'value three levels deep maintained'); ok(children['childOne']['abc'] === '123', 'value three levels deep added'); ok(children['childTwo'], 'object two levels deep maintained'); ok(children['childThree'] === null, 'object two levels deep removed'); ok(children['childFour'], 'object two levels deep added'); ok(vjs.Component.prototype.options_['example']['childOne']['foo'] === 'bar', 'prototype options were not overridden'); // Reset default component options to none vjs.Component.prototype.options_ = null; }); test('should dispose of component and children', function(){ var comp = new vjs.Component(getFakePlayer()); // Add a child var child = comp.addChild('Component'); ok(comp.children().length === 1); // Add a listener comp.on('click', function(){ return true; }); var data = vjs.getData(comp.el()); var id = comp.el()[vjs.expando]; var hasDisposed = false; var bubbles = null; comp.on('dispose', function(event){ hasDisposed = true; bubbles = event.bubbles; }); comp.dispose(); ok(hasDisposed, 'component fired dispose event'); ok(bubbles === false, 'dispose event does not bubble'); ok(!comp.children(), 'component children were deleted'); ok(!comp.el(), 'component element was deleted'); ok(!child.children(), 'child children were deleted'); ok(!child.el(), 'child element was deleted'); ok(!vjs.cache[id], 'listener cache nulled'); ok(vjs.isEmpty(data), 'original listener cache object was emptied'); }); test('should add and remove event listeners to element', function(){ var comp = new vjs.Component(getFakePlayer(), {}); // No need to make this async because we're triggering events inline. // We're going to trigger the event after removing the listener, // So if we get extra asserts that's a problem. expect(2); var testListener = function(){ ok(true, 'fired event once'); ok(this === comp, 'listener has the component as context'); }; comp.on('test-event', testListener); comp.trigger('test-event'); comp.off('test-event', testListener); comp.trigger('test-event'); }); test('should trigger a listener once using one()', function(){ var comp = new vjs.Component(getFakePlayer(), {}); expect(1); var testListener = function(){ ok(true, 'fired event once'); }; comp.one('test-event', testListener); comp.trigger('test-event'); comp.trigger('test-event'); }); test('should trigger a listener when ready', function(){ expect(2); var optionsReadyListener = function(){ ok(true, 'options listener fired'); }; var methodReadyListener = function(){ ok(true, 'ready method listener fired'); }; var comp = new vjs.Component(getFakePlayer(), {}, optionsReadyListener); comp.triggerReady(); comp.ready(methodReadyListener); // First two listeners should only be fired once and then removed comp.triggerReady(); }); test('should add and remove a CSS class', function(){ var comp = new vjs.Component(getFakePlayer(), {}); comp.addClass('test-class'); ok(comp.el().className.indexOf('test-class') !== -1); comp.removeClass('test-class'); ok(comp.el().className.indexOf('test-class') === -1); }); test('should show and hide an element', function(){ var comp = new vjs.Component(getFakePlayer(), {}); comp.hide(); ok(comp.el().style.display === 'none'); comp.show(); ok(comp.el().style.display === 'block'); }); test('should change the width and height of a component', function(){ var container = document.createElement('div'); var comp = new vjs.Component(getFakePlayer(), {}); var el = comp.el(); var fixture = document.getElementById('qunit-fixture'); fixture.appendChild(container); container.appendChild(el); // Container of el needs dimensions or the component won't have dimensions container.style.width = '1000px'; container.style.height = '1000px'; comp.width('50%'); comp.height('123px'); ok(comp.width() === 500, 'percent values working'); var compStyle = vjs.getComputedDimension(el, 'width'); ok(compStyle === comp.width() + 'px', 'matches computed style'); ok(comp.height() === 123, 'px values working'); comp.width(321); ok(comp.width() === 321, 'integer values working'); comp.width('auto'); comp.height('auto'); ok(comp.width() === 1000, 'forced width was removed'); ok(comp.height() === 0, 'forced height was removed'); }); test('should use a defined content el for appending children', function(){ var CompWithContent = vjs.Component.extend(); CompWithContent.prototype.createEl = function(){ // Create the main componenent element var el = vjs.createEl('div'); // Create the element where children will be appended this.contentEl_ = vjs.createEl('div', { 'id': 'contentEl' }); el.appendChild(this.contentEl_); return el; }; var comp = new CompWithContent(getFakePlayer()); var child = comp.addChild('component'); ok(comp.children().length === 1); ok(comp.el().childNodes[0]['id'] === 'contentEl'); ok(comp.el().childNodes[0].childNodes[0] === child.el()); comp.removeChild(child); ok(comp.children().length === 0, 'Length should now be zero'); ok(comp.el().childNodes[0]['id'] === 'contentEl', 'Content El should still exist'); ok(comp.el().childNodes[0].childNodes[0] !== child.el(), 'Child el should be removed.'); }); test('should emit a tap event', function(){ expect(2); // Fake touch support. Real touch support isn't needed for this test. var origTouch = vjs.TOUCH_ENABLED; vjs.TOUCH_ENABLED = true; var comp = new vjs.Component(getFakePlayer()); comp.emitTapEvents(); comp.on('tap', function(){ ok(true, 'Tap event emitted'); }); // A touchstart followed by touchend should trigger a tap vjs.trigger(comp.el(), {type: 'touchstart', touches: [{}]}); comp.trigger('touchend'); // A touchmove with a lot of movement should not trigger a tap vjs.trigger(comp.el(), {type: 'touchstart', touches: [ { pageX: 0, pageY: 0 } ]}); vjs.trigger(comp.el(), {type: 'touchmove', touches: [ { pageX: 100, pageY: 100 } ]}); comp.trigger('touchend'); // A touchmove with not much movement should still allow a tap vjs.trigger(comp.el(), {type: 'touchstart', touches: [ { pageX: 0, pageY: 0 } ]}); vjs.trigger(comp.el(), {type: 'touchmove', touches: [ { pageX: 10, pageY: 10 } ]}); comp.trigger('touchend'); // Reset to orignial value vjs.TOUCH_ENABLED = origTouch; });