From a3be137f739d78f771f54aa8e2a6ecab7f3dd529 Mon Sep 17 00:00:00 2001 From: ferdiemmen Date: Mon, 19 Jun 2017 17:05:47 +0200 Subject: [PATCH 1/2] Fix mobile ads on iPhone --- demo/css/plyr-ads.css | 8 ++++---- demo/js/plyr-ads.js | 8 +++----- src/js/plyr-ads.js | 8 +++----- src/scss/plyr-ads.scss | 7 ++++--- 4 files changed, 14 insertions(+), 17 deletions(-) diff --git a/demo/css/plyr-ads.css b/demo/css/plyr-ads.css index de1231e..96e1326 100644 --- a/demo/css/plyr-ads.css +++ b/demo/css/plyr-ads.css @@ -1,13 +1,13 @@ +.plyr--ready { + position: relative; } + .plyr-ads { position: absolute; top: 0; left: 0; right: 0; bottom: 0; - width: 100%; - height: 100%; - z-index: 10; - overflow: hidden; } + z-index: 10; } .plyr-ads__skip-button { position: absolute; z-index: 11; diff --git a/demo/js/plyr-ads.js b/demo/js/plyr-ads.js index 17efca4..9692dc6 100644 --- a/demo/js/plyr-ads.js +++ b/demo/js/plyr-ads.js @@ -108,13 +108,13 @@ this.plyrAdContainer = _insertElement('div', this.plyr.getContainer(), { class: 'plyr-ads' }); - this.plyrAdContainer.addEventListener('click', () => { + this.plyrAdContainer.parentNode.addEventListener('touchend', () => { this.playAds(); }, false); } function _createPlyrAdsSkipButton() { - this.plyrAdSkipButton = _insertElement('button', this.plyr.getContainer(), { + this.plyrAdSkipButton = _insertElement('touchend', this.plyr.getContainer(), { class: 'plyr-ads__skip-button' }); this.plyrAdSkipButton.textContent = this.options.skipButton.text; @@ -371,9 +371,7 @@ plyr.forEach(instance => { // Only add ads to video instances. if (instance.getType() !== 'audio') { - instance.on('ready', () => { - instance.plyrAds = new PlyrAds(instance, options); - }); + instance.plyrAds = new PlyrAds(instance, options); } }); } diff --git a/src/js/plyr-ads.js b/src/js/plyr-ads.js index 17efca4..9692dc6 100644 --- a/src/js/plyr-ads.js +++ b/src/js/plyr-ads.js @@ -108,13 +108,13 @@ this.plyrAdContainer = _insertElement('div', this.plyr.getContainer(), { class: 'plyr-ads' }); - this.plyrAdContainer.addEventListener('click', () => { + this.plyrAdContainer.parentNode.addEventListener('touchend', () => { this.playAds(); }, false); } function _createPlyrAdsSkipButton() { - this.plyrAdSkipButton = _insertElement('button', this.plyr.getContainer(), { + this.plyrAdSkipButton = _insertElement('touchend', this.plyr.getContainer(), { class: 'plyr-ads__skip-button' }); this.plyrAdSkipButton.textContent = this.options.skipButton.text; @@ -371,9 +371,7 @@ plyr.forEach(instance => { // Only add ads to video instances. if (instance.getType() !== 'audio') { - instance.on('ready', () => { - instance.plyrAds = new PlyrAds(instance, options); - }); + instance.plyrAds = new PlyrAds(instance, options); } }); } diff --git a/src/scss/plyr-ads.scss b/src/scss/plyr-ads.scss index 8e233f1..f7bdcb0 100644 --- a/src/scss/plyr-ads.scss +++ b/src/scss/plyr-ads.scss @@ -1,14 +1,15 @@ +.plyr--ready { + position: relative; +} + .plyr-ads { position: absolute; top: 0; left: 0; right: 0; bottom: 0; - width: 100%; - height: 100%; z-index: 10; - overflow: hidden; &__skip-button { position: absolute; From 58751c11f2b6c54b9f24399549b32d1396d3ff96 Mon Sep 17 00:00:00 2001 From: ferdiemmen Date: Tue, 20 Jun 2017 12:27:59 +0200 Subject: [PATCH 2/2] Fix ads on iPhone and iPad --- demo/css/plyr-ads.css | 13 ++----- demo/js/plyr-ads.js | 79 +++++++++++++++++------------------------- src/js/plyr-ads.js | 79 +++++++++++++++++------------------------- src/scss/plyr-ads.scss | 13 ++----- 4 files changed, 66 insertions(+), 118 deletions(-) diff --git a/demo/css/plyr-ads.css b/demo/css/plyr-ads.css index 96e1326..6c72ced 100644 --- a/demo/css/plyr-ads.css +++ b/demo/css/plyr-ads.css @@ -8,14 +8,5 @@ right: 0; bottom: 0; z-index: 10; } - .plyr-ads__skip-button { - position: absolute; - z-index: 11; - top: 1em; - right: 0; - border: none; - font-size: 1.2em; - padding: 10px 20px; - background: rgba(40, 40, 40, 0.8); - color: #fff; - display: none; } + .plyr-ads video { + left: 0; } diff --git a/demo/js/plyr-ads.js b/demo/js/plyr-ads.js index 9692dc6..80f11aa 100644 --- a/demo/js/plyr-ads.js +++ b/demo/js/plyr-ads.js @@ -28,7 +28,7 @@ 'use strict'; // Default config - const defaults = { + var defaults = { container: 'plyr-ads', classes: [], skipButton: { @@ -38,7 +38,7 @@ }; // Check variable types. - const _is = { + var _is = { object: function (input) { return input !== null && typeof (input) === 'object'; }, @@ -83,10 +83,7 @@ } // Add ad container to DOM. - this.createPlyrAdsContainer(); - - // Add ad skip button to DOM. - this.createPlyrAdsSkipButton(); + this.createAdDisplayContainer(); // Setup IMA. this.setUpIMA(); @@ -96,31 +93,30 @@ PlyrAds.prototype.playVideo = _playVideo; PlyrAds.prototype.setUpIMA = _setUpIMA; PlyrAds.prototype.createAdDisplayContainer = _createAdDisplayContainer; - PlyrAds.prototype.createPlyrAdsSkipButton = _createPlyrAdsSkipButton; - PlyrAds.prototype.createPlyrAdsContainer = _createPlyrAdsContainer; PlyrAds.prototype.onAdEvent = _onAdEvent; PlyrAds.prototype.onAdError = _onAdError; PlyrAds.prototype.onAdsManagerLoaded = _onAdsManagerLoaded; PlyrAds.prototype.onContentResumeRequested = _onContentResumeRequested; PlyrAds.prototype.onContentSkippable = _onContentSkippable; - function _createPlyrAdsContainer() { - this.plyrAdContainer = _insertElement('div', this.plyr.getContainer(), { - class: 'plyr-ads' - }); - this.plyrAdContainer.parentNode.addEventListener('touchend', () => { + function _createAdDisplayContainer() { + this.adDisplayContainer = new google.ima.AdDisplayContainer( + this.plyr.getContainer()); + this.adDisplayContainer.I.setAttribute('class', 'plyr-ads'); + this.adDisplayContainer.I.addEventListener(_getStartEvent(), function() { this.playAds(); - }, false); + }.bind(this), false); } - function _createPlyrAdsSkipButton() { - this.plyrAdSkipButton = _insertElement('touchend', this.plyr.getContainer(), { - class: 'plyr-ads__skip-button' - }); - this.plyrAdSkipButton.textContent = this.options.skipButton.text; - this.plyrAdSkipButton.addEventListener('click', () => { - this.playVideo(); - }, false); + function _getStartEvent() { + // Set the correct event based on userAgent. + var startEvent = 'click'; + if (navigator.userAgent.match(/iPhone/i) || + navigator.userAgent.match(/iPad/i) || + navigator.userAgent.match(/Android/i)) { + startEvent = 'touchend'; + } + return startEvent; } // Prepend child @@ -130,7 +126,7 @@ // Set attributes function _setAttributes(element, attributes) { - for (let key in attributes) { + for (var key in attributes) { element.setAttribute(key, (_is.boolean(attributes[key]) && attributes[key]) ? '' : attributes[key]); } } @@ -138,7 +134,7 @@ // Insert a HTML element function _insertElement(type, parent, attributes) { // Create a new - let element = document.createElement(type); + var element = document.createElement(type); // Set all passed attributes _setAttributes(element, attributes); @@ -148,14 +144,11 @@ } function _playVideo() { - this.plyrAdContainer.remove(); - this.plyrAdSkipButton.remove(); + this.adDisplayContainer.I.remove(); this.plyr.play(); } function _setUpIMA() { - // Create the ad display container. - this.createAdDisplayContainer(); // Create ads loader. this.adsLoader = new google.ima.AdsLoader(this.adDisplayContainer); // Listen and respond to ads loaded and error events. @@ -168,23 +161,16 @@ this.adsLoader.addEventListener( google.ima.AdErrorEvent.Type.AD_ERROR, function (adError) { - this.plyrAdContainer.remove(); + this.adDisplayContainer.I.remove(); }.bind(this), false); // Request video ads. - let adsRequest = new google.ima.AdsRequest(); + var adsRequest = new google.ima.AdsRequest(); adsRequest.adTagUrl = this.options.adTagUrl; this.adsLoader.requestAds(adsRequest); } - function _createAdDisplayContainer() { - // We assume the adContainer is the DOM id of the element that will house - // the ads. - this.adDisplayContainer = new google.ima.AdDisplayContainer( - this.plyrAdContainer); - } - function _playAds() { // Initialize the container. Must be done via a user action on mobile devices. @@ -192,19 +178,16 @@ // Initialize the ads manager. Ad rules playlist will start at this time. this.adsManager.init(this.plyrContainer.offsetWidth, this.plyrContainer.offsetHeight, google.ima.ViewMode.NORMAL); + // Call play to start showing the ad. Single video and overlay ads will // start at this time; the call will be ignored for ad rules. - try { - this.adsManager.start(); - } catch (adError) { - // An error may be thrown if there was a problem with the VAST response. - this.playVideo(); - } + console.log(this.adsManager); + this.adsManager.start(); } function _onAdsManagerLoaded(adsManagerLoadedEvent) { // Get the ads manager. - let adsRenderingSettings = new google.ima.AdsRenderingSettings(); + var adsRenderingSettings = new google.ima.AdsRenderingSettings(); adsRenderingSettings.restoreCustomPlaybackStateOnAdBreakComplete = true; this.adsManager = adsManagerLoadedEvent.getAdsManager(adsRenderingSettings); @@ -262,7 +245,7 @@ function _onAdEvent(adEvent) { // Retrieve the ad from the event. Some events (e.g. ALL_ADS_COMPLETED) // don't have ad object associated. - let ad = adEvent.getAd(); + var ad = adEvent.getAd(); switch (adEvent.type) { case google.ima.AdEvent.Type.LOADED: // This is the first event sent for an ad - it is possible to @@ -282,7 +265,7 @@ // the remaining time. this.intervalTimer = setInterval( function () { - let remainingTime = Math.round(this.adsManager.getRemainingTime()); + var remainingTime = Math.round(this.adsManager.getRemainingTime()); }.bind(this), 300); // every 300ms } @@ -365,10 +348,10 @@ if (!window.plyr) return false; // Set options from defaults and config. - let options = _extend({}, defaults, config); + var options = _extend({}, defaults, config); // Loop through plyr instances and add ads. - plyr.forEach(instance => { + plyr.forEach(function(instance) { // Only add ads to video instances. if (instance.getType() !== 'audio') { instance.plyrAds = new PlyrAds(instance, options); diff --git a/src/js/plyr-ads.js b/src/js/plyr-ads.js index 9692dc6..80f11aa 100644 --- a/src/js/plyr-ads.js +++ b/src/js/plyr-ads.js @@ -28,7 +28,7 @@ 'use strict'; // Default config - const defaults = { + var defaults = { container: 'plyr-ads', classes: [], skipButton: { @@ -38,7 +38,7 @@ }; // Check variable types. - const _is = { + var _is = { object: function (input) { return input !== null && typeof (input) === 'object'; }, @@ -83,10 +83,7 @@ } // Add ad container to DOM. - this.createPlyrAdsContainer(); - - // Add ad skip button to DOM. - this.createPlyrAdsSkipButton(); + this.createAdDisplayContainer(); // Setup IMA. this.setUpIMA(); @@ -96,31 +93,30 @@ PlyrAds.prototype.playVideo = _playVideo; PlyrAds.prototype.setUpIMA = _setUpIMA; PlyrAds.prototype.createAdDisplayContainer = _createAdDisplayContainer; - PlyrAds.prototype.createPlyrAdsSkipButton = _createPlyrAdsSkipButton; - PlyrAds.prototype.createPlyrAdsContainer = _createPlyrAdsContainer; PlyrAds.prototype.onAdEvent = _onAdEvent; PlyrAds.prototype.onAdError = _onAdError; PlyrAds.prototype.onAdsManagerLoaded = _onAdsManagerLoaded; PlyrAds.prototype.onContentResumeRequested = _onContentResumeRequested; PlyrAds.prototype.onContentSkippable = _onContentSkippable; - function _createPlyrAdsContainer() { - this.plyrAdContainer = _insertElement('div', this.plyr.getContainer(), { - class: 'plyr-ads' - }); - this.plyrAdContainer.parentNode.addEventListener('touchend', () => { + function _createAdDisplayContainer() { + this.adDisplayContainer = new google.ima.AdDisplayContainer( + this.plyr.getContainer()); + this.adDisplayContainer.I.setAttribute('class', 'plyr-ads'); + this.adDisplayContainer.I.addEventListener(_getStartEvent(), function() { this.playAds(); - }, false); + }.bind(this), false); } - function _createPlyrAdsSkipButton() { - this.plyrAdSkipButton = _insertElement('touchend', this.plyr.getContainer(), { - class: 'plyr-ads__skip-button' - }); - this.plyrAdSkipButton.textContent = this.options.skipButton.text; - this.plyrAdSkipButton.addEventListener('click', () => { - this.playVideo(); - }, false); + function _getStartEvent() { + // Set the correct event based on userAgent. + var startEvent = 'click'; + if (navigator.userAgent.match(/iPhone/i) || + navigator.userAgent.match(/iPad/i) || + navigator.userAgent.match(/Android/i)) { + startEvent = 'touchend'; + } + return startEvent; } // Prepend child @@ -130,7 +126,7 @@ // Set attributes function _setAttributes(element, attributes) { - for (let key in attributes) { + for (var key in attributes) { element.setAttribute(key, (_is.boolean(attributes[key]) && attributes[key]) ? '' : attributes[key]); } } @@ -138,7 +134,7 @@ // Insert a HTML element function _insertElement(type, parent, attributes) { // Create a new - let element = document.createElement(type); + var element = document.createElement(type); // Set all passed attributes _setAttributes(element, attributes); @@ -148,14 +144,11 @@ } function _playVideo() { - this.plyrAdContainer.remove(); - this.plyrAdSkipButton.remove(); + this.adDisplayContainer.I.remove(); this.plyr.play(); } function _setUpIMA() { - // Create the ad display container. - this.createAdDisplayContainer(); // Create ads loader. this.adsLoader = new google.ima.AdsLoader(this.adDisplayContainer); // Listen and respond to ads loaded and error events. @@ -168,23 +161,16 @@ this.adsLoader.addEventListener( google.ima.AdErrorEvent.Type.AD_ERROR, function (adError) { - this.plyrAdContainer.remove(); + this.adDisplayContainer.I.remove(); }.bind(this), false); // Request video ads. - let adsRequest = new google.ima.AdsRequest(); + var adsRequest = new google.ima.AdsRequest(); adsRequest.adTagUrl = this.options.adTagUrl; this.adsLoader.requestAds(adsRequest); } - function _createAdDisplayContainer() { - // We assume the adContainer is the DOM id of the element that will house - // the ads. - this.adDisplayContainer = new google.ima.AdDisplayContainer( - this.plyrAdContainer); - } - function _playAds() { // Initialize the container. Must be done via a user action on mobile devices. @@ -192,19 +178,16 @@ // Initialize the ads manager. Ad rules playlist will start at this time. this.adsManager.init(this.plyrContainer.offsetWidth, this.plyrContainer.offsetHeight, google.ima.ViewMode.NORMAL); + // Call play to start showing the ad. Single video and overlay ads will // start at this time; the call will be ignored for ad rules. - try { - this.adsManager.start(); - } catch (adError) { - // An error may be thrown if there was a problem with the VAST response. - this.playVideo(); - } + console.log(this.adsManager); + this.adsManager.start(); } function _onAdsManagerLoaded(adsManagerLoadedEvent) { // Get the ads manager. - let adsRenderingSettings = new google.ima.AdsRenderingSettings(); + var adsRenderingSettings = new google.ima.AdsRenderingSettings(); adsRenderingSettings.restoreCustomPlaybackStateOnAdBreakComplete = true; this.adsManager = adsManagerLoadedEvent.getAdsManager(adsRenderingSettings); @@ -262,7 +245,7 @@ function _onAdEvent(adEvent) { // Retrieve the ad from the event. Some events (e.g. ALL_ADS_COMPLETED) // don't have ad object associated. - let ad = adEvent.getAd(); + var ad = adEvent.getAd(); switch (adEvent.type) { case google.ima.AdEvent.Type.LOADED: // This is the first event sent for an ad - it is possible to @@ -282,7 +265,7 @@ // the remaining time. this.intervalTimer = setInterval( function () { - let remainingTime = Math.round(this.adsManager.getRemainingTime()); + var remainingTime = Math.round(this.adsManager.getRemainingTime()); }.bind(this), 300); // every 300ms } @@ -365,10 +348,10 @@ if (!window.plyr) return false; // Set options from defaults and config. - let options = _extend({}, defaults, config); + var options = _extend({}, defaults, config); // Loop through plyr instances and add ads. - plyr.forEach(instance => { + plyr.forEach(function(instance) { // Only add ads to video instances. if (instance.getType() !== 'audio') { instance.plyrAds = new PlyrAds(instance, options); diff --git a/src/scss/plyr-ads.scss b/src/scss/plyr-ads.scss index f7bdcb0..9b4ce48 100644 --- a/src/scss/plyr-ads.scss +++ b/src/scss/plyr-ads.scss @@ -11,16 +11,7 @@ bottom: 0; z-index: 10; - &__skip-button { - position: absolute; - z-index: 11; - top: 1em; - right: 0; - border: none; - font-size: 1.2em; - padding: 10px 20px; - background: rgba(40,40,40,.8); - color: #fff; - display: none; + video { + left: 0; } }