From a7bb6ee086f463dbf2740219273d0991a5e92e57 Mon Sep 17 00:00:00 2001 From: Wassim Gharbi Date: Sat, 15 Jul 2017 09:33:37 -0700 Subject: [PATCH 1/7] Implemented metaData and media session for amp-video --- extensions/amp-vimeo/0.1/amp-vimeo.js | 7 ++ extensions/amp-youtube/0.1/amp-youtube.js | 7 ++ src/service/video-manager-impl.js | 80 +++++++++++++++++++++++ src/video-interface.js | 27 ++++++++ 4 files changed, 121 insertions(+) diff --git a/extensions/amp-vimeo/0.1/amp-vimeo.js b/extensions/amp-vimeo/0.1/amp-vimeo.js index 59e983d15289..3d9209291635 100644 --- a/extensions/amp-vimeo/0.1/amp-vimeo.js +++ b/extensions/amp-vimeo/0.1/amp-vimeo.js @@ -74,6 +74,13 @@ class AmpVimeo extends AMP.BaseElement { })), '*'); } } + + /** @override */ + optOutOfAutomaticMediaSessionAPI() { + // Vimeo already updates the Media Session so no need for the video + // manager to update it too + return true; + } }; AMP.registerElement('amp-vimeo', AmpVimeo); diff --git a/extensions/amp-youtube/0.1/amp-youtube.js b/extensions/amp-youtube/0.1/amp-youtube.js index 124906596d9e..3f02309081ae 100644 --- a/extensions/amp-youtube/0.1/amp-youtube.js +++ b/extensions/amp-youtube/0.1/amp-youtube.js @@ -246,6 +246,13 @@ class AmpYoutube extends AMP.BaseElement { } } + /** @override */ + optOutOfAutomaticMediaSessionAPI() { + // Youtube already updates the Media Session so no need for the video + // manager to update it too + return true; + } + /** @override */ mutatedAttributesCallback(mutations) { if (mutations['data-videoid'] !== undefined) { diff --git a/src/service/video-manager-impl.js b/src/service/video-manager-impl.js index 988f5410a9f0..28369b846cf6 100644 --- a/src/service/video-manager-impl.js +++ b/src/service/video-manager-impl.js @@ -571,6 +571,9 @@ class VideoEntry { */ videoPlayed_() { this.isPlaying_ = true; + if (!this.preimplementsMediaSession()) { + this.mediaSessionUpdate_(); + } this.actionSessionManager_.beginSession(); if (this.isVisible_) { this.visibilitySessionManager_.beginSession(); @@ -617,6 +620,11 @@ class VideoEntry { this.inlineVidRect_ = this.video.element./*OK*/getBoundingClientRect(); }); + if (!this.preimplementsMediaSession()) { + this.metaData_ = this.getMetaData_(); + this.mediaSessionUpdate_(); + } + this.updateVisibility(); if (this.isVisible_) { // Handles the case when the video becomes visible before loading @@ -624,6 +632,78 @@ class VideoEntry { } } + /** + * Gets the provided metadata and fills in missing fields + * @return {!../video-interface.VideoMetaDef} + * @private + */ + getMetaData_() { + let metaData = this.video.metaData; + if (!metaData) { + metaData = {}; + } + if (!metaData.artist) { + metaData.artist = 'No artist'; + } + if (!metaData.posterUrl) { + metaData.posterUrl = this.video.element.getAttribute('poster') + || this.internalElement_.getAttribute('poster') + || this.getDefaultPoster_(); + } + if (!metaData.title) { + metaData.title = this.video.element.getAttribute('title') + || this.video.element.getAttribute('aria-label') + || this.internalElement_.getAttribute('title') + || this.internalElement_.getAttribute('aria-label') + || this.ampdoc_.win.document.title; + } + if (!metaData.album) { + metaData.album = 'No album'; + } + return metaData; + } + + /** + * Gets the provided metadata and fills in missing fields + * @private + */ + mediaSessionUpdate_() { + const win = this.ampdoc_.win; + const navigator = win.navigator; + if ('mediaSession' in navigator && win.MediaMetadata) { + + navigator.mediaSession.metadata = new win.MediaMetadata({ + title: this.metaData_.title, + artist: this.metaData_.artist, + album: this.metaData_.album, + artwork: [ + { + src: this.metaData_.posterUrl, + sizes: '512x512', + type: 'image/png', + }, + ], + }); + + navigator.mediaSession.setActionHandler('play', function() { + this.video.play(); + }); + navigator.mediaSession.setActionHandler('pause', function() { + this.video.pause(); + }); + + // TODO(@wassgha) Implement seek & next/previous + } + + } + + getDefaultPoster_() { + /*eslint-disable */ + const ampLogo = ''; + return ampLogo; + /*eslint-enable */ + } + /** * Called when visibility of a video changes. * @private diff --git a/src/video-interface.js b/src/video-interface.js index b5952805bafa..35e37fc61dc7 100644 --- a/src/video-interface.js +++ b/src/video-interface.js @@ -104,6 +104,23 @@ export class VideoInterface { */ hideControls() {} + /** + * Returns video's meta data (poster, artist, album, etc.) + * @return {!VideoMetaDef} metadata + */ + get metaData() {} + + /** + * If this returns true then it will be assumed that the player implements + * the MediaSession API internally so that the video manager does not override + * it. If not, the video manager will use the metaData variable as well as + * inferred meta-data to update the video's Media Session notification. + * + * @return {boolean} + */ + optOutOfAutomaticMediaSessionAPI() {} + + /** * Automatically comes from {@link ./base-element.BaseElement} * @@ -414,3 +431,13 @@ export const VideoAnalyticsEvents = { * }} */ export let VideoAnalyticsDetailsDef; + +/** + * @typedef {{ + * posterUrl: string, + * title: string, + * album: string, + * artist: string, + * }} + */ +export let VideoMetaDef; From 429dee4d60c83c0054fbd0878704ed4bc365361e Mon Sep 17 00:00:00 2001 From: Wassim Gharbi Date: Tue, 18 Jul 2017 15:02:15 -0700 Subject: [PATCH 2/7] Started working on requested changes --- extensions/amp-vimeo/0.1/amp-vimeo.js | 2 +- extensions/amp-youtube/0.1/amp-youtube.js | 2 +- src/service/video-manager-impl.js | 103 ++++++++++++---------- src/video-interface.js | 31 ++++--- 4 files changed, 76 insertions(+), 62 deletions(-) diff --git a/extensions/amp-vimeo/0.1/amp-vimeo.js b/extensions/amp-vimeo/0.1/amp-vimeo.js index 3d9209291635..20e33f5367fe 100644 --- a/extensions/amp-vimeo/0.1/amp-vimeo.js +++ b/extensions/amp-vimeo/0.1/amp-vimeo.js @@ -76,7 +76,7 @@ class AmpVimeo extends AMP.BaseElement { } /** @override */ - optOutOfAutomaticMediaSessionAPI() { + preimplementsMediaSessionAPI() { // Vimeo already updates the Media Session so no need for the video // manager to update it too return true; diff --git a/extensions/amp-youtube/0.1/amp-youtube.js b/extensions/amp-youtube/0.1/amp-youtube.js index 3f02309081ae..3d8719d2beed 100644 --- a/extensions/amp-youtube/0.1/amp-youtube.js +++ b/extensions/amp-youtube/0.1/amp-youtube.js @@ -247,7 +247,7 @@ class AmpYoutube extends AMP.BaseElement { } /** @override */ - optOutOfAutomaticMediaSessionAPI() { + preimplementsMediaSessionAPI() { // Youtube already updates the Media Session so no need for the video // manager to update it too return true; diff --git a/src/service/video-manager-impl.js b/src/service/video-manager-impl.js index 28369b846cf6..80b4fd2048c1 100644 --- a/src/service/video-manager-impl.js +++ b/src/service/video-manager-impl.js @@ -38,6 +38,7 @@ import { PositionObserverFidelity, PositionInViewportEntryDef, } from './position-observer-impl'; +import {map} from '../utils/object'; import {layoutRectLtwh, RelativePositions} from '../layout-rect'; import {Animation} from '../animation'; import * as st from '../style'; @@ -456,6 +457,8 @@ class VideoEntry { const element = dev().assert(video.element); + // Autoplay Variables + /** @private {boolean} */ this.userInteractedWithAutoPlay_ = false; @@ -541,7 +544,18 @@ class VideoEntry { listenOncePromise(element, VideoEvents.LOAD) .then(() => this.videoLoaded()); + // Media Session API Variables + + /** @private {!../video-interface.VideoMetaDef} */ + this.metaData_ = { + 'artist': '', + 'album': '', + 'artwork': '', + 'title': '', + }; + listenOncePromise(element, VideoEvents.LOAD) + .then(() => this.videoLoaded()); listen(element, VideoEvents.PAUSE, () => this.videoPaused_()); listen(element, VideoEvents.PLAYING, () => this.videoPlayed_()); listen(element, VideoEvents.MUTED, () => this.muted_ = true); @@ -571,9 +585,8 @@ class VideoEntry { */ videoPlayed_() { this.isPlaying_ = true; - if (!this.preimplementsMediaSession()) { - this.mediaSessionUpdate_(); - } + this.updateMediaSession_(); + this.actionSessionManager_.beginSession(); if (this.isVisible_) { this.visibilitySessionManager_.beginSession(); @@ -620,10 +633,7 @@ class VideoEntry { this.inlineVidRect_ = this.video.element./*OK*/getBoundingClientRect(); }); - if (!this.preimplementsMediaSession()) { - this.metaData_ = this.getMetaData_(); - this.mediaSessionUpdate_(); - } + this.getMetaData_(); this.updateVisibility(); if (this.isVisible_) { @@ -634,59 +644,65 @@ class VideoEntry { /** * Gets the provided metadata and fills in missing fields - * @return {!../video-interface.VideoMetaDef} * @private */ getMetaData_() { - let metaData = this.video.metaData; - if (!metaData) { - metaData = {}; + if (this.video.preimplementsMediaSessionAPI()) { + return; } - if (!metaData.artist) { - metaData.artist = 'No artist'; + + if (this.video.getMetaData()) { + this.metaData_ = map(this.video.getMetaData()); + } + + if (!this.metaData_.artist) { + const artist = 'No artist'; + if (artist) { + this.metaData_.artist = artist; + } } - if (!metaData.posterUrl) { - metaData.posterUrl = this.video.element.getAttribute('poster') - || this.internalElement_.getAttribute('poster') - || this.getDefaultPoster_(); + + if (!this.metaData_.artwork) { + const posterUrl = this.video.element.getAttribute('poster') + || this.internalElement_.getAttribute('poster'); + if (posterUrl) { + this.metaData_.artwork = posterUrl; + } } - if (!metaData.title) { - metaData.title = this.video.element.getAttribute('title') - || this.video.element.getAttribute('aria-label') - || this.internalElement_.getAttribute('title') - || this.internalElement_.getAttribute('aria-label') - || this.ampdoc_.win.document.title; + + if (!this.metaData_.title) { + const title = this.video.element.getAttribute('title') + || this.video.element.getAttribute('aria-label') + || this.internalElement_.getAttribute('title') + || this.internalElement_.getAttribute('aria-label') + || this.ampdoc_.win.document.title; + if (title) { + this.metaData_.title = title; + } } - if (!metaData.album) { - metaData.album = 'No album'; + + if (!this.metaData_.album) { + this.metaData_.album = 'No album'; } - return metaData; } /** * Gets the provided metadata and fills in missing fields * @private */ - mediaSessionUpdate_() { + updateMediaSession_() { + if (this.video.preimplementsMediaSessionAPI()) { + return; + } + const win = this.ampdoc_.win; const navigator = win.navigator; if ('mediaSession' in navigator && win.MediaMetadata) { - navigator.mediaSession.metadata = new win.MediaMetadata({ - title: this.metaData_.title, - artist: this.metaData_.artist, - album: this.metaData_.album, - artwork: [ - { - src: this.metaData_.posterUrl, - sizes: '512x512', - type: 'image/png', - }, - ], - }); + navigator.mediaSession.metadata = new win.MediaMetadata(this.metaData_); navigator.mediaSession.setActionHandler('play', function() { - this.video.play(); + this.video.play(/*isAutoplay*/ false); }); navigator.mediaSession.setActionHandler('pause', function() { this.video.pause(); @@ -697,13 +713,6 @@ class VideoEntry { } - getDefaultPoster_() { - /*eslint-disable */ - const ampLogo = ''; - return ampLogo; - /*eslint-enable */ - } - /** * Called when visibility of a video changes. * @private diff --git a/src/video-interface.js b/src/video-interface.js index 35e37fc61dc7..1f5062a3c2a1 100644 --- a/src/video-interface.js +++ b/src/video-interface.js @@ -16,6 +16,16 @@ import {ActionTrust} from './action-trust'; /* eslint no-unused-vars: 0 */ +/** + * @typedef {{ + * artwork: string, + * title: string, + * album: string, + * artist: string, + * }} + */ +export let VideoMetaDef; + /** * VideoInterface defines a common video API which any AMP component that plays * videos is expected to implement. @@ -105,10 +115,15 @@ export class VideoInterface { hideControls() {} /** - * Returns video's meta data (poster, artist, album, etc.) + * Returns video's meta data (artwork, title, artist, album, etc.) for use + * with the Media Session API + * artwork (string): URL to the poster image (preferably a 512x512 PNG) + * title (string): Name of the video + * artist (string): Name of the video's author/artist + * album (string): Name of the video's album if it exists * @return {!VideoMetaDef} metadata */ - get metaData() {} + getMetaData() {} /** * If this returns true then it will be assumed that the player implements @@ -118,7 +133,7 @@ export class VideoInterface { * * @return {boolean} */ - optOutOfAutomaticMediaSessionAPI() {} + preimplementsMediaSessionAPI() {} /** @@ -431,13 +446,3 @@ export const VideoAnalyticsEvents = { * }} */ export let VideoAnalyticsDetailsDef; - -/** - * @typedef {{ - * posterUrl: string, - * title: string, - * album: string, - * artist: string, - * }} - */ -export let VideoMetaDef; From d25e0854c9147c890f4c51343098cc1e8815287c Mon Sep 17 00:00:00 2001 From: Wassim Gharbi Date: Mon, 24 Jul 2017 12:46:08 -0700 Subject: [PATCH 3/7] Implemented metadata and mediasession api for all players & added better default posters --- extensions/amp-3q-player/0.1/amp-3q-player.js | 15 ++++ .../amp-brid-player/0.1/amp-brid-player.js | 15 ++++ .../amp-dailymotion/0.1/amp-dailymotion.js | 15 ++++ extensions/amp-ima-video/0.1/amp-ima-video.js | 15 ++++ .../0.1/amp-nexxtv-player.js | 15 ++++ .../0.1/amp-ooyala-player.js | 15 ++++ extensions/amp-video/0.1/amp-video.js | 69 ++++++++++------- extensions/amp-vimeo/0.1/amp-vimeo.js | 7 -- extensions/amp-youtube/0.1/amp-youtube.js | 24 ++++-- src/service/video-manager-impl.js | 77 +++++++++++++++---- src/video-interface.js | 13 +++- 11 files changed, 223 insertions(+), 57 deletions(-) diff --git a/extensions/amp-3q-player/0.1/amp-3q-player.js b/extensions/amp-3q-player/0.1/amp-3q-player.js index c25cf71c8056..15ce8eed43c7 100644 --- a/extensions/amp-3q-player/0.1/amp-3q-player.js +++ b/extensions/amp-3q-player/0.1/amp-3q-player.js @@ -248,6 +248,21 @@ class Amp3QPlayer extends AMP.BaseElement { return isFullscreenElement(dev().assertElement(this.iframe_)); } + /** @override */ + getMetaData() { + return { + 'artwork': [], + 'title': '', + 'artist': '', + 'album': '', + }; + } + + /** @override */ + preimplementsMediaSessionAPI() { + return false; + } + /** @override */ getCurrentTime() { // Not supported. diff --git a/extensions/amp-brid-player/0.1/amp-brid-player.js b/extensions/amp-brid-player/0.1/amp-brid-player.js index e72d24faf76f..e946e37daaeb 100644 --- a/extensions/amp-brid-player/0.1/amp-brid-player.js +++ b/extensions/amp-brid-player/0.1/amp-brid-player.js @@ -319,6 +319,21 @@ class AmpBridPlayer extends AMP.BaseElement { return isFullscreenElement(dev().assertElement(this.iframe_)); } + /** @override */ + getMetaData() { + return { + 'artwork': [], + 'title': '', + 'artist': '', + 'album': '', + }; + } + + /** @override */ + preimplementsMediaSessionAPI() { + return false; + } + /** @override */ getCurrentTime() { // Not supported. diff --git a/extensions/amp-dailymotion/0.1/amp-dailymotion.js b/extensions/amp-dailymotion/0.1/amp-dailymotion.js index 8d95e77d8784..69cfd677b7d9 100644 --- a/extensions/amp-dailymotion/0.1/amp-dailymotion.js +++ b/extensions/amp-dailymotion/0.1/amp-dailymotion.js @@ -392,6 +392,21 @@ class AmpDailymotion extends AMP.BaseElement { } } + /** @override */ + getMetaData() { + return { + 'artwork': [], + 'title': '', + 'artist': '', + 'album': '', + }; + } + + /** @override */ + preimplementsMediaSessionAPI() { + return false; + } + /** @override */ getCurrentTime() { // Not supported. diff --git a/extensions/amp-ima-video/0.1/amp-ima-video.js b/extensions/amp-ima-video/0.1/amp-ima-video.js index ebb9efc8ac1e..f1c2bbb115e1 100644 --- a/extensions/amp-ima-video/0.1/amp-ima-video.js +++ b/extensions/amp-ima-video/0.1/amp-ima-video.js @@ -310,6 +310,21 @@ class AmpImaVideo extends AMP.BaseElement { return isFullscreenElement(dev().assertElement(this.iframe_)); } + /** @override */ + getMetaData() { + return { + 'artwork': [], + 'title': '', + 'artist': '', + 'album': '', + }; + } + + /** @override */ + preimplementsMediaSessionAPI() { + return false; + } + /** @override */ getCurrentTime() { // Not supported. diff --git a/extensions/amp-nexxtv-player/0.1/amp-nexxtv-player.js b/extensions/amp-nexxtv-player/0.1/amp-nexxtv-player.js index dfd6291eb8f2..edfbbf4a9824 100644 --- a/extensions/amp-nexxtv-player/0.1/amp-nexxtv-player.js +++ b/extensions/amp-nexxtv-player/0.1/amp-nexxtv-player.js @@ -274,6 +274,21 @@ class AmpNexxtvPlayer extends AMP.BaseElement { return isFullscreenElement(dev().assertElement(this.iframe_)); } + /** @override */ + getMetaData() { + return { + 'artwork': [], + 'title': '', + 'artist': '', + 'album': '', + }; + } + + /** @override */ + preimplementsMediaSessionAPI() { + return false; + } + /** @override */ getCurrentTime() { // Not supported. diff --git a/extensions/amp-ooyala-player/0.1/amp-ooyala-player.js b/extensions/amp-ooyala-player/0.1/amp-ooyala-player.js index 068793262806..a06810bb9c7c 100644 --- a/extensions/amp-ooyala-player/0.1/amp-ooyala-player.js +++ b/extensions/amp-ooyala-player/0.1/amp-ooyala-player.js @@ -259,6 +259,21 @@ class AmpOoyalaPlayer extends AMP.BaseElement { return isFullscreenElement(dev().assertElement(this.iframe_)); } + /** @override */ + getMetaData() { + return { + 'artwork': [], + 'title': '', + 'artist': '', + 'album': '', + }; + } + + /** @override */ + preimplementsMediaSessionAPI() { + return false; + } + /** @override */ getCurrentTime() { // Not supported. diff --git a/extensions/amp-video/0.1/amp-video.js b/extensions/amp-video/0.1/amp-video.js index 901d13177ce4..b8c1d3190189 100644 --- a/extensions/amp-video/0.1/amp-video.js +++ b/extensions/amp-video/0.1/amp-video.js @@ -59,23 +59,23 @@ const ATTRS_TO_PROPAGATE = */ class AmpVideo extends AMP.BaseElement { - /** - * @param {!AmpElement} element - */ + /** + * @param {!AmpElement} element + */ constructor(element) { super(element); - /** @private {?Element} */ + /** @private {?Element} */ this.video_ = null; - /** @private {?boolean} */ + /** @private {?boolean} */ this.muted_ = false; } - /** - * @param {boolean=} opt_onLayout - * @override - */ + /** + * @param {boolean=} opt_onLayout + * @override + */ preconnectCallback(opt_onLayout) { const videoSrc = this.getVideoSource_(); if (videoSrc) { @@ -84,10 +84,10 @@ class AmpVideo extends AMP.BaseElement { } } - /** - * @private - * @return {string} - */ + /** + * @private + * @return {string} + */ getVideoSource_() { let videoSrc = this.element.getAttribute('src'); if (!videoSrc) { @@ -99,12 +99,12 @@ class AmpVideo extends AMP.BaseElement { return videoSrc; } - /** @override */ + /** @override */ isLayoutSupported(layout) { return isLayoutSizeDefined(layout); } - /** @override */ + /** @override */ buildCallback() { this.video_ = this.element.ownerDocument.createElement('video'); @@ -114,10 +114,10 @@ class AmpVideo extends AMP.BaseElement { 'No "poster" attribute has been provided for amp-video.'); } - // Enable inline play for iOS. + // Enable inline play for iOS. this.video_.setAttribute('playsinline', ''); this.video_.setAttribute('webkit-playsinline', ''); - // Disable video preload in prerender mode. + // Disable video preload in prerender mode. this.video_.setAttribute('preload', 'none'); this.propagateAttributes(ATTRS_TO_PROPAGATE_ON_BUILD, this.video_, /* opt_removeMissingAttrs */ true); @@ -129,7 +129,7 @@ class AmpVideo extends AMP.BaseElement { Services.videoManagerForDoc(this.element).register(this); } - /** @override */ + /** @override */ mutatedAttributesCallback(mutations) { if (!this.video_) { return; @@ -148,12 +148,12 @@ class AmpVideo extends AMP.BaseElement { } } - /** @override */ + /** @override */ viewportCallback(visible) { this.element.dispatchCustomEvent(VideoEvents.VISIBILITY, {visible}); } - /** @override */ + /** @override */ layoutCallback() { this.video_ = dev().assertElement(this.video_); @@ -170,7 +170,7 @@ class AmpVideo extends AMP.BaseElement { /* opt_removeMissingAttrs */ true); this.getRealChildNodes().forEach(child => { - // Skip the video we already added to the element. + // Skip the video we already added to the element. if (this.video_ === child) { return; } @@ -181,7 +181,7 @@ class AmpVideo extends AMP.BaseElement { this.video_.appendChild(child); }); - // loadPromise for media elements listens to `loadstart` + // loadPromise for media elements listens to `loadstart` return this.loadPromise(this.video_).then(() => { this.element.dispatchCustomEvent(VideoEvents.LOAD); }); @@ -241,11 +241,11 @@ class AmpVideo extends AMP.BaseElement { if (ret && ret.catch) { ret.catch(() => { - // Empty catch to prevent useless unhandled promise rejection logging. - // Play can fail for many reasons such as video getting paused before - // play() is finished. - // We use events to know the state of the video and do not care about - // the success or failure of the play()'s returned promise. + // Empty catch to prevent useless unhandled promise rejection logging. + // Play can fail for many reasons such as video getting paused before + // play() is finished. + // We use events to know the state of the video and do not care about + // the success or failure of the play()'s returned promise. }); } } @@ -304,6 +304,21 @@ class AmpVideo extends AMP.BaseElement { return isFullscreenElement(dev().assertElement(this.video_)); } + /** @override */ + getMetaData() { + return { + 'artwork': [], + 'title': '', + 'artist': '', + 'album': '', + }; + } + + /** @override */ + preimplementsMediaSessionAPI() { + return false; + } + /** @override */ getCurrentTime() { return this.video_.currentTime; diff --git a/extensions/amp-vimeo/0.1/amp-vimeo.js b/extensions/amp-vimeo/0.1/amp-vimeo.js index 20e33f5367fe..59e983d15289 100644 --- a/extensions/amp-vimeo/0.1/amp-vimeo.js +++ b/extensions/amp-vimeo/0.1/amp-vimeo.js @@ -74,13 +74,6 @@ class AmpVimeo extends AMP.BaseElement { })), '*'); } } - - /** @override */ - preimplementsMediaSessionAPI() { - // Vimeo already updates the Media Session so no need for the video - // manager to update it too - return true; - } }; AMP.registerElement('amp-vimeo', AmpVimeo); diff --git a/extensions/amp-youtube/0.1/amp-youtube.js b/extensions/amp-youtube/0.1/amp-youtube.js index 3d8719d2beed..fe14f2859ef3 100644 --- a/extensions/amp-youtube/0.1/amp-youtube.js +++ b/extensions/amp-youtube/0.1/amp-youtube.js @@ -246,13 +246,6 @@ class AmpYoutube extends AMP.BaseElement { } } - /** @override */ - preimplementsMediaSessionAPI() { - // Youtube already updates the Media Session so no need for the video - // manager to update it too - return true; - } - /** @override */ mutatedAttributesCallback(mutations) { if (mutations['data-videoid'] !== undefined) { @@ -473,6 +466,23 @@ class AmpYoutube extends AMP.BaseElement { return isFullscreenElement(dev().assertElement(this.iframe_)); } + /** @override */ + getMetaData() { + return { + 'artwork': [], + 'title': '', + 'artist': '', + 'album': '', + }; + } + + /** @override */ + preimplementsMediaSessionAPI() { + // Youtube already updates the Media Session so no need for the video + // manager to update it too + return true; + } + /** @override */ getCurrentTime() { // Not supported. diff --git a/src/service/video-manager-impl.js b/src/service/video-manager-impl.js index 80b4fd2048c1..2f126e3c5c4a 100644 --- a/src/service/video-manager-impl.js +++ b/src/service/video-manager-impl.js @@ -42,7 +42,11 @@ import {map} from '../utils/object'; import {layoutRectLtwh, RelativePositions} from '../layout-rect'; import {Animation} from '../animation'; import * as st from '../style'; +<<<<<<< HEAD import * as tr from '../transition'; +======= +import {tryParseJson} from '../json'; +>>>>>>> Implemented metadata and mediasession api for all players & added better default posters /** * @const {number} Percentage of the video that should be in viewport before it @@ -550,7 +554,7 @@ class VideoEntry { this.metaData_ = { 'artist': '', 'album': '', - 'artwork': '', + 'artwork': [], 'title': '', }; @@ -655,18 +659,68 @@ class VideoEntry { this.metaData_ = map(this.video.getMetaData()); } - if (!this.metaData_.artist) { - const artist = 'No artist'; - if (artist) { - this.metaData_.artist = artist; + if (!this.metaData_.artwork) { + const doc = this.ampdoc_.win.document; + + // Parses the schema.org json-ld formatted meta-data + const parseSchemaImage = () => { + const schema = doc.querySelector('script[type="application/ld+json"]'); + if (!schema) { + // No schema element found + return undefined; + } + const schemaJson = tryParseJson(schema.textContent); + if (!schemaJson || !schemaJson.image) { + // No image found in the schema + return undefined; + } + + if (schemaJson.image['@list'] + && schemaJson.image['@list'][0] + && typeof schemaJson.image['@list'][0] === 'string') { + return schemaJson.image['@list'][0]; + } else if (schemaJson.image[0] + && typeof schemaJson.image[0] === 'string') { + // Return the first image + return schemaJson.image[0]; + } else if (typeof schemaJson.image === 'string') { + return schemaJson.image; + } else { + return undefined; + } + }; + + // Parses the og:image if it exists + const parseOgImage = () => { + const metaTag = doc.querySelector('meta[property="og:image"]'); + if (metaTag) { + return metaTage.getAttribute('content'); + } else { + return undefined; + } + }; + + // Parses the website's Favicon + const parseFavicon = () => { + const linkTag = doc.querySelector('link[rel="shortcut icon"]') + || doc.querySelector('link[rel="icon"]'); + if (linkTag) { + return linkTag.getAttribute('href'); + } else { + return undefined; + } } - } - if (!this.metaData_.artwork) { const posterUrl = this.video.element.getAttribute('poster') - || this.internalElement_.getAttribute('poster'); + || this.internalElement_.getAttribute('poster') + || parseSchemaImage() + || parseOgImage() + || parseFavicon(); + if (posterUrl) { - this.metaData_.artwork = posterUrl; + this.metaData_.artwork = [{ + 'src':posterUrl, + }]; } } @@ -680,10 +734,6 @@ class VideoEntry { this.metaData_.title = title; } } - - if (!this.metaData_.album) { - this.metaData_.album = 'No album'; - } } /** @@ -698,7 +748,6 @@ class VideoEntry { const win = this.ampdoc_.win; const navigator = win.navigator; if ('mediaSession' in navigator && win.MediaMetadata) { - navigator.mediaSession.metadata = new win.MediaMetadata(this.metaData_); navigator.mediaSession.setActionHandler('play', function() { diff --git a/src/video-interface.js b/src/video-interface.js index 1f5062a3c2a1..577b3ea210ec 100644 --- a/src/video-interface.js +++ b/src/video-interface.js @@ -123,7 +123,14 @@ export class VideoInterface { * album (string): Name of the video's album if it exists * @return {!VideoMetaDef} metadata */ - getMetaData() {} + getMetaData() { + return { + 'artwork': '', + 'title': '', + 'artist': '', + 'album': '', + }; + } /** * If this returns true then it will be assumed that the player implements @@ -133,7 +140,9 @@ export class VideoInterface { * * @return {boolean} */ - preimplementsMediaSessionAPI() {} + preimplementsMediaSessionAPI() { + return false; + } /** From 7e142ecb222f39977992182e01d6e1e5cbaf5fd6 Mon Sep 17 00:00:00 2001 From: Wassim Gharbi Date: Fri, 28 Jul 2017 11:01:38 -0700 Subject: [PATCH 4/7] More requested changes --- src/service/video-manager-impl.js | 4 ++-- test/functional/test-video-manager.js | 31 +++++++++++++++++++++++++++ 2 files changed, 33 insertions(+), 2 deletions(-) diff --git a/src/service/video-manager-impl.js b/src/service/video-manager-impl.js index 2f126e3c5c4a..fd288d17bb00 100644 --- a/src/service/video-manager-impl.js +++ b/src/service/video-manager-impl.js @@ -694,7 +694,7 @@ class VideoEntry { const parseOgImage = () => { const metaTag = doc.querySelector('meta[property="og:image"]'); if (metaTag) { - return metaTage.getAttribute('content'); + return metaTag.getAttribute('content'); } else { return undefined; } @@ -719,7 +719,7 @@ class VideoEntry { if (posterUrl) { this.metaData_.artwork = [{ - 'src':posterUrl, + 'src': posterUrl, }]; } } diff --git a/test/functional/test-video-manager.js b/test/functional/test-video-manager.js index f76caab15e37..4ec02d02114c 100644 --- a/test/functional/test-video-manager.js +++ b/test/functional/test-video-manager.js @@ -228,6 +228,22 @@ describes.fakeWin('VideoManager', { }); + it('should change media session when video starts playing', () => { + + videoManager.register(impl); + + const mediaSessionSpy = sandbox.spy( + videoManager.getEntryForVideo_(impl), + 'updateMediaSession_' + ); + + impl.play(); + + return listenOncePromise(video, VideoEvents.PLAYING).then(() => { + expect(mediaSessionSpy.called).to.be.true; + }); + }); + beforeEach(() => { sandbox = sinon.sandbox.create(); klass = createFakeVideoPlayerClass(env.win); @@ -526,6 +542,21 @@ function createFakeVideoPlayerClass(win) { return this.currentTime_; } + /** @override */ + getMetaData() { + return { + 'artwork': '', + 'title': '', + 'artist': '', + 'album': '', + }; + } + + /** @override */ + preimplementsMediaSessionAPI() { + return false; + } + /** @override */ getDuration() { return this.duration_; From acc0054b5e2ec9b0d791ad84bc1e1515a00faa0b Mon Sep 17 00:00:00 2001 From: Wassim Gharbi Date: Sat, 29 Jul 2017 15:48:43 -0700 Subject: [PATCH 5/7] Created media session helper and implemented tests --- src/mediasession-helper.js | 124 +++++++++++++++++++ src/service/video-manager-impl.js | 127 +++++--------------- test/functional/test-mediasession-helper.js | 126 +++++++++++++++++++ test/functional/test-video-manager.js | 17 --- 4 files changed, 283 insertions(+), 111 deletions(-) create mode 100644 src/mediasession-helper.js create mode 100644 test/functional/test-mediasession-helper.js diff --git a/src/mediasession-helper.js b/src/mediasession-helper.js new file mode 100644 index 000000000000..b030269610a8 --- /dev/null +++ b/src/mediasession-helper.js @@ -0,0 +1,124 @@ +/** + * Copyright 2017 The AMP HTML Authors. All Rights Reserved. + * + * Licensed under the Apache License, Version 2.0 (the "License"); + * you may not use this file except in compliance with the License. + * You may obtain a copy of the License at + * + * http://www.apache.org/licenses/LICENSE-2.0 + * + * Unless required by applicable law or agreed to in writing, software + * distributed under the License is distributed on an "AS-IS" BASIS, + * WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied. + * See the License for the specific language governing permissions and + * limitations under the License. + */ + +import {tryParseJson} from './json'; + +/** + * Updates the Media Session API's metadata + * @param {!./ampdoc-impl.AmpDoc} ampdoc + * @param {!./video-interface.VideoMetaDef} metaData + * @param {function} playHandler + * @param {function} pauseHandler + */ +export function setMediaSession(ampdoc, + metaData, + playHandler = null, + pauseHandler = null) { + const win = ampdoc.win; + const navigator = win.navigator; + if ('mediaSession' in navigator && win.MediaMetadata) { + // Clear mediaSession (required to fix a bug when switching between two + // videos) + navigator.mediaSession.metadata = new win.MediaMetadata({ + title: '', + artist: '', + album: '', + artwork: [ + { src: ''}, + ] + }); + // Add metaData + navigator.mediaSession.metadata = new win.MediaMetadata(metaData); + + navigator.mediaSession.setActionHandler('play', playHandler); + navigator.mediaSession.setActionHandler('pause', pauseHandler); + + // TODO(@wassgha) Implement seek & next/previous + } +} + + +/** + * Parses the schema.org json-ld formatted meta-data, looks for the page's + * featured image and returns it + * @param {!./ampdoc-impl.AmpDoc} ampdoc + * @return {string|undefined} + */ +export function parseSchemaImage(ampdoc) { + const doc = ampdoc.win.document; + const schema = doc.querySelector('script[type="application/ld+json"]'); + if (!schema) { + // No schema element found + return undefined; + } + const schemaJson = tryParseJson(schema.textContent); + if (!schemaJson || !schemaJson.image) { + // No image found in the schema + return undefined; + } + + // Image definition in schema could be one of : + if (schemaJson.image['@list'] + && schemaJson.image['@list'][0] + && typeof schemaJson.image['@list'][0] === 'string') { + // 1. "image": {.., "@list": ["http://.."], ..} + return schemaJson.image['@list'][0]; + } else if (schemaJson.image['url'] + && typeof schemaJson.image['url'] === 'string') { + // 2. "image": {.., "url": "http://..", ..} + return schemaJson.image['url']; + } else if (schemaJson.image[0] + && typeof schemaJson.image[0] === 'string') { + // 3. "image": ["http://.. "] + return schemaJson.image[0]; + } else if (typeof schemaJson.image === 'string') { + // 4. "image": "http://..", + return schemaJson.image; + } else { + return undefined; + } +}; + +/** + * Parses the og:image if it exists and returns it + * @param {!./ampdoc-impl.AmpDoc} ampdoc + * @return {string|undefined} + */ +export function parseOgImage(ampdoc) { + const doc = ampdoc.win.document; + const metaTag = doc.querySelector('meta[property="og:image"]'); + if (metaTag) { + return metaTag.getAttribute('content'); + } else { + return undefined; + } +}; + +/** + * Parses the website's Favicon and returns it + * @param {!./ampdoc-impl.AmpDoc} ampdoc + * @return {string|undefined} + */ +export function parseFavicon(ampdoc) { + const doc = ampdoc.win.document; + const linkTag = doc.querySelector('link[rel="shortcut icon"]') + || doc.querySelector('link[rel="icon"]'); + if (linkTag) { + return linkTag.getAttribute('href'); + } else { + return undefined; + } +} diff --git a/src/service/video-manager-impl.js b/src/service/video-manager-impl.js index fd288d17bb00..bbf319f2826a 100644 --- a/src/service/video-manager-impl.js +++ b/src/service/video-manager-impl.js @@ -40,14 +40,15 @@ import { } from './position-observer-impl'; import {map} from '../utils/object'; import {layoutRectLtwh, RelativePositions} from '../layout-rect'; +import { + parseSchemaImage, + parseOgImage, + parseFavicon, + setMediaSession, +} from '../mediasession-helper'; import {Animation} from '../animation'; import * as st from '../style'; -<<<<<<< HEAD import * as tr from '../transition'; -======= -import {tryParseJson} from '../json'; ->>>>>>> Implemented metadata and mediasession api for all players & added better default posters - /** * @const {number} Percentage of the video that should be in viewport before it * is considered visible. @@ -589,7 +590,22 @@ class VideoEntry { */ videoPlayed_() { this.isPlaying_ = true; - this.updateMediaSession_(); + + if (!this.video.preimplementsMediaSessionAPI()) { + const playHandler = () => { + this.video.play(/*isAutoplay*/ false); + }; + const pauseHandler = () => { + this.video.pause(/*isAutoplay*/ false); + } + // Update the media session + setMediaSession( + this.ampdoc_, + this.metaData_, + playHandler, + pauseHandler + ); + } this.actionSessionManager_.beginSession(); if (this.isVisible_) { @@ -637,7 +653,7 @@ class VideoEntry { this.inlineVidRect_ = this.video.element./*OK*/getBoundingClientRect(); }); - this.getMetaData_(); + this.fillMetaData_(); this.updateVisibility(); if (this.isVisible_) { @@ -650,7 +666,7 @@ class VideoEntry { * Gets the provided metadata and fills in missing fields * @private */ - getMetaData_() { + fillMetaData_() { if (this.video.preimplementsMediaSessionAPI()) { return; } @@ -659,63 +675,12 @@ class VideoEntry { this.metaData_ = map(this.video.getMetaData()); } - if (!this.metaData_.artwork) { - const doc = this.ampdoc_.win.document; - - // Parses the schema.org json-ld formatted meta-data - const parseSchemaImage = () => { - const schema = doc.querySelector('script[type="application/ld+json"]'); - if (!schema) { - // No schema element found - return undefined; - } - const schemaJson = tryParseJson(schema.textContent); - if (!schemaJson || !schemaJson.image) { - // No image found in the schema - return undefined; - } - - if (schemaJson.image['@list'] - && schemaJson.image['@list'][0] - && typeof schemaJson.image['@list'][0] === 'string') { - return schemaJson.image['@list'][0]; - } else if (schemaJson.image[0] - && typeof schemaJson.image[0] === 'string') { - // Return the first image - return schemaJson.image[0]; - } else if (typeof schemaJson.image === 'string') { - return schemaJson.image; - } else { - return undefined; - } - }; - - // Parses the og:image if it exists - const parseOgImage = () => { - const metaTag = doc.querySelector('meta[property="og:image"]'); - if (metaTag) { - return metaTag.getAttribute('content'); - } else { - return undefined; - } - }; - - // Parses the website's Favicon - const parseFavicon = () => { - const linkTag = doc.querySelector('link[rel="shortcut icon"]') - || doc.querySelector('link[rel="icon"]'); - if (linkTag) { - return linkTag.getAttribute('href'); - } else { - return undefined; - } - } - + if (!this.metaData_.artwork || this.metaData_.artwork.length == 0) { const posterUrl = this.video.element.getAttribute('poster') - || this.internalElement_.getAttribute('poster') - || parseSchemaImage() - || parseOgImage() - || parseFavicon(); + || this.internalElement_.getAttribute('poster') + || parseSchemaImage(this.ampdoc_) + || parseOgImage(this.ampdoc_) + || parseFavicon(this.ampdoc_); if (posterUrl) { this.metaData_.artwork = [{ @@ -726,42 +691,16 @@ class VideoEntry { if (!this.metaData_.title) { const title = this.video.element.getAttribute('title') - || this.video.element.getAttribute('aria-label') - || this.internalElement_.getAttribute('title') - || this.internalElement_.getAttribute('aria-label') - || this.ampdoc_.win.document.title; + || this.video.element.getAttribute('aria-label') + || this.internalElement_.getAttribute('title') + || this.internalElement_.getAttribute('aria-label') + || this.ampdoc_.win.document.title; if (title) { this.metaData_.title = title; } } } - /** - * Gets the provided metadata and fills in missing fields - * @private - */ - updateMediaSession_() { - if (this.video.preimplementsMediaSessionAPI()) { - return; - } - - const win = this.ampdoc_.win; - const navigator = win.navigator; - if ('mediaSession' in navigator && win.MediaMetadata) { - navigator.mediaSession.metadata = new win.MediaMetadata(this.metaData_); - - navigator.mediaSession.setActionHandler('play', function() { - this.video.play(/*isAutoplay*/ false); - }); - navigator.mediaSession.setActionHandler('pause', function() { - this.video.pause(); - }); - - // TODO(@wassgha) Implement seek & next/previous - } - - } - /** * Called when visibility of a video changes. * @private diff --git a/test/functional/test-mediasession-helper.js b/test/functional/test-mediasession-helper.js new file mode 100644 index 000000000000..10dce56cecc7 --- /dev/null +++ b/test/functional/test-mediasession-helper.js @@ -0,0 +1,126 @@ +/** + * Copyright 2017 The AMP HTML Authors. All Rights Reserved. + * + * Licensed under the Apache License, Version 2.0 (the "License"); + * you may not use this file except in compliance with the License. + * You may obtain a copy of the License at + * + * http://www.apache.org/licenses/LICENSE-2.0 + * + * Unless required by applicable law or agreed to in writing, software + * distributed under the License is distributed on an "AS-IS" BASIS, + * WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied. + * See the License for the specific language governing permissions and + * limitations under the License. + */ + +import { + setMediaSession, + parseSchemaImage, + parseOgImage, + parseFavicon, +} from '../../src/mediasession-helper'; + +const template = ` + + + + + + +

Yesterday Something Happened

+

Lorem ipsum dolor set amet..

+ +`; + + +describes.sandboxed('MediaSessionAPI Helper Functions', {}, () => { + let ampdoc; + + beforeEach(() => { + document.documentElement.innerHTML = template; + ampdoc = { + win: { + document: document, + navigator: { + mediaSession: { + metadata: { + 'artist': '', + 'album': '', + 'artwork': [], + 'title': '', + }, + setActionHandler: () => { + }, + } + }, + MediaMetadata: Object, + }, + }; + }); + + afterEach(() => { + document.documentElement.innerHTML = ''; + }); + + it('should parse the schema and find the image', () => { + expect(parseSchemaImage(ampdoc)).to.equal('http://example.com/image.png'); + }); + + it('should parse the og-image', () => { + expect(parseOgImage(ampdoc)).to.equal('http://example.com/og-image.png'); + }); + + it('should parse the favicon', () => { + expect(parseFavicon(ampdoc)).to.equal('http://example.com/favicon.ico'); + }); + + it('should set the media session', () => { + expect(ampdoc.win.navigator.mediaSession.metadata).to.deep.equal({ + 'artist': '', + 'album': '', + 'artwork': [], + 'title': '', + }); + const fakeMetaData = { + 'artist': 'Some artist', + 'album': 'Some album', + 'artwork': [ + 'http://example.com/image.png' + ], + 'title': 'Some title', + }; + setMediaSession(ampdoc, fakeMetaData); + expect(ampdoc.win.navigator.mediaSession.metadata).to.deep.equal(fakeMetaData); + }); +}); diff --git a/test/functional/test-video-manager.js b/test/functional/test-video-manager.js index 4ec02d02114c..340c7b42fb59 100644 --- a/test/functional/test-video-manager.js +++ b/test/functional/test-video-manager.js @@ -227,23 +227,6 @@ describes.fakeWin('VideoManager', { }); }); - - it('should change media session when video starts playing', () => { - - videoManager.register(impl); - - const mediaSessionSpy = sandbox.spy( - videoManager.getEntryForVideo_(impl), - 'updateMediaSession_' - ); - - impl.play(); - - return listenOncePromise(video, VideoEvents.PLAYING).then(() => { - expect(mediaSessionSpy.called).to.be.true; - }); - }); - beforeEach(() => { sandbox = sinon.sandbox.create(); klass = createFakeVideoPlayerClass(env.win); From ca3ff51a8985f7728346b67d18b9e1302c61480c Mon Sep 17 00:00:00 2001 From: Wassim Gharbi Date: Mon, 31 Jul 2017 13:59:29 -0700 Subject: [PATCH 6/7] Requested changes for MediaSessionAPI implementation --- extensions/amp-3q-player/0.1/amp-3q-player.js | 9 +- .../amp-brid-player/0.1/amp-brid-player.js | 9 +- .../amp-dailymotion/0.1/amp-dailymotion.js | 9 +- extensions/amp-ima-video/0.1/amp-ima-video.js | 9 +- .../0.1/amp-nexxtv-player.js | 9 +- .../0.1/amp-ooyala-player.js | 9 +- extensions/amp-video/0.1/amp-video.js | 19 ++-- extensions/amp-youtube/0.1/amp-youtube.js | 9 +- src/mediasession-helper.js | 87 ++++++++++--------- src/service/video-manager-impl.js | 38 ++++---- src/video-interface.js | 19 ++-- test/functional/test-mediasession-helper.js | 19 ++-- test/functional/test-video-manager.js | 9 +- 13 files changed, 103 insertions(+), 151 deletions(-) diff --git a/extensions/amp-3q-player/0.1/amp-3q-player.js b/extensions/amp-3q-player/0.1/amp-3q-player.js index 15ce8eed43c7..b795ac9a512c 100644 --- a/extensions/amp-3q-player/0.1/amp-3q-player.js +++ b/extensions/amp-3q-player/0.1/amp-3q-player.js @@ -249,13 +249,8 @@ class Amp3QPlayer extends AMP.BaseElement { } /** @override */ - getMetaData() { - return { - 'artwork': [], - 'title': '', - 'artist': '', - 'album': '', - }; + getMetadata() { + // Not implemented } /** @override */ diff --git a/extensions/amp-brid-player/0.1/amp-brid-player.js b/extensions/amp-brid-player/0.1/amp-brid-player.js index e946e37daaeb..5b48eeced6f6 100644 --- a/extensions/amp-brid-player/0.1/amp-brid-player.js +++ b/extensions/amp-brid-player/0.1/amp-brid-player.js @@ -320,13 +320,8 @@ class AmpBridPlayer extends AMP.BaseElement { } /** @override */ - getMetaData() { - return { - 'artwork': [], - 'title': '', - 'artist': '', - 'album': '', - }; + getMetadata() { + // Not implemented } /** @override */ diff --git a/extensions/amp-dailymotion/0.1/amp-dailymotion.js b/extensions/amp-dailymotion/0.1/amp-dailymotion.js index 69cfd677b7d9..5f7469b4f00d 100644 --- a/extensions/amp-dailymotion/0.1/amp-dailymotion.js +++ b/extensions/amp-dailymotion/0.1/amp-dailymotion.js @@ -393,13 +393,8 @@ class AmpDailymotion extends AMP.BaseElement { } /** @override */ - getMetaData() { - return { - 'artwork': [], - 'title': '', - 'artist': '', - 'album': '', - }; + getMetadata() { + // Not implemented } /** @override */ diff --git a/extensions/amp-ima-video/0.1/amp-ima-video.js b/extensions/amp-ima-video/0.1/amp-ima-video.js index f1c2bbb115e1..0012bab67f00 100644 --- a/extensions/amp-ima-video/0.1/amp-ima-video.js +++ b/extensions/amp-ima-video/0.1/amp-ima-video.js @@ -311,13 +311,8 @@ class AmpImaVideo extends AMP.BaseElement { } /** @override */ - getMetaData() { - return { - 'artwork': [], - 'title': '', - 'artist': '', - 'album': '', - }; + getMetadata() { + // Not implemented } /** @override */ diff --git a/extensions/amp-nexxtv-player/0.1/amp-nexxtv-player.js b/extensions/amp-nexxtv-player/0.1/amp-nexxtv-player.js index edfbbf4a9824..046e935f4ab5 100644 --- a/extensions/amp-nexxtv-player/0.1/amp-nexxtv-player.js +++ b/extensions/amp-nexxtv-player/0.1/amp-nexxtv-player.js @@ -275,13 +275,8 @@ class AmpNexxtvPlayer extends AMP.BaseElement { } /** @override */ - getMetaData() { - return { - 'artwork': [], - 'title': '', - 'artist': '', - 'album': '', - }; + getMetadata() { + // Not implemented } /** @override */ diff --git a/extensions/amp-ooyala-player/0.1/amp-ooyala-player.js b/extensions/amp-ooyala-player/0.1/amp-ooyala-player.js index a06810bb9c7c..9cc160dedb47 100644 --- a/extensions/amp-ooyala-player/0.1/amp-ooyala-player.js +++ b/extensions/amp-ooyala-player/0.1/amp-ooyala-player.js @@ -260,13 +260,8 @@ class AmpOoyalaPlayer extends AMP.BaseElement { } /** @override */ - getMetaData() { - return { - 'artwork': [], - 'title': '', - 'artist': '', - 'album': '', - }; + getMetadata() { + // Not implemented } /** @override */ diff --git a/extensions/amp-video/0.1/amp-video.js b/extensions/amp-video/0.1/amp-video.js index b8c1d3190189..09401a31ecf4 100644 --- a/extensions/amp-video/0.1/amp-video.js +++ b/extensions/amp-video/0.1/amp-video.js @@ -305,13 +305,18 @@ class AmpVideo extends AMP.BaseElement { } /** @override */ - getMetaData() { - return { - 'artwork': [], - 'title': '', - 'artist': '', - 'album': '', - }; + getMetadata() { + const poster = this.element.getAttribute('poster'); + if (poster) { + return { + 'title': '', + 'artist': '', + 'album': '', + 'artwork': [ + {'src': poster}, + ], + }; + } } /** @override */ diff --git a/extensions/amp-youtube/0.1/amp-youtube.js b/extensions/amp-youtube/0.1/amp-youtube.js index fe14f2859ef3..16a67d0947dd 100644 --- a/extensions/amp-youtube/0.1/amp-youtube.js +++ b/extensions/amp-youtube/0.1/amp-youtube.js @@ -467,13 +467,8 @@ class AmpYoutube extends AMP.BaseElement { } /** @override */ - getMetaData() { - return { - 'artwork': [], - 'title': '', - 'artist': '', - 'album': '', - }; + getMetadata() { + // Not implemented } /** @override */ diff --git a/src/mediasession-helper.js b/src/mediasession-helper.js index b030269610a8..76bb161aabfe 100644 --- a/src/mediasession-helper.js +++ b/src/mediasession-helper.js @@ -16,32 +16,35 @@ import {tryParseJson} from './json'; +/** @const {./video-interface.VideoMetaDef} Dummy metadata used to fix a bug */ +export const EMPTY_METADATA = { + 'title': '', + 'artist': '', + 'album': '', + 'artwork': [ + {'src': ''}, + ], +}; + /** * Updates the Media Session API's metadata - * @param {!./ampdoc-impl.AmpDoc} ampdoc - * @param {!./video-interface.VideoMetaDef} metaData - * @param {function} playHandler - * @param {function} pauseHandler + * @param {!./service/ampdoc-impl.AmpDoc} ampdoc + * @param {!./video-interface.VideoMetaDef} metadata + * @param {function()=} playHandler + * @param {function()=} pauseHandler */ export function setMediaSession(ampdoc, - metaData, - playHandler = null, - pauseHandler = null) { + metadata, + playHandler, + pauseHandler) { const win = ampdoc.win; const navigator = win.navigator; if ('mediaSession' in navigator && win.MediaMetadata) { // Clear mediaSession (required to fix a bug when switching between two // videos) - navigator.mediaSession.metadata = new win.MediaMetadata({ - title: '', - artist: '', - album: '', - artwork: [ - { src: ''}, - ] - }); - // Add metaData - navigator.mediaSession.metadata = new win.MediaMetadata(metaData); + navigator.mediaSession.metadata = new win.MediaMetadata(EMPTY_METADATA); + // Add metadata + navigator.mediaSession.metadata = new win.MediaMetadata(metadata); navigator.mediaSession.setActionHandler('play', playHandler); navigator.mediaSession.setActionHandler('pause', pauseHandler); @@ -54,7 +57,7 @@ export function setMediaSession(ampdoc, /** * Parses the schema.org json-ld formatted meta-data, looks for the page's * featured image and returns it - * @param {!./ampdoc-impl.AmpDoc} ampdoc + * @param {!./service/ampdoc-impl.AmpDoc} ampdoc * @return {string|undefined} */ export function parseSchemaImage(ampdoc) { @@ -62,39 +65,39 @@ export function parseSchemaImage(ampdoc) { const schema = doc.querySelector('script[type="application/ld+json"]'); if (!schema) { // No schema element found - return undefined; + return; } const schemaJson = tryParseJson(schema.textContent); - if (!schemaJson || !schemaJson.image) { + if (!schemaJson || !schemaJson['image']) { // No image found in the schema - return undefined; + return; } // Image definition in schema could be one of : - if (schemaJson.image['@list'] - && schemaJson.image['@list'][0] - && typeof schemaJson.image['@list'][0] === 'string') { - // 1. "image": {.., "@list": ["http://.."], ..} - return schemaJson.image['@list'][0]; - } else if (schemaJson.image['url'] - && typeof schemaJson.image['url'] === 'string') { - // 2. "image": {.., "url": "http://..", ..} - return schemaJson.image['url']; - } else if (schemaJson.image[0] - && typeof schemaJson.image[0] === 'string') { - // 3. "image": ["http://.. "] - return schemaJson.image[0]; - } else if (typeof schemaJson.image === 'string') { - // 4. "image": "http://..", - return schemaJson.image; + if (typeof schemaJson['image'] === 'string') { + // 1. "image": "http://..", + return schemaJson['image']; + } else if (schemaJson['image']['@list'] + && schemaJson['image']['@list'][0] + && typeof schemaJson['image']['@list'][0] === 'string') { + // 2. "image": {.., "@list": ["http://.."], ..} + return schemaJson['image']['@list'][0]; + } else if (schemaJson['image']['url'] + && typeof schemaJson['image']['url'] === 'string') { + // 3. "image": {.., "url": "http://..", ..} + return schemaJson['image']['url']; + } else if (schemaJson['image'][0] + && typeof schemaJson['image'][0] === 'string') { + // 4. "image": ["http://.. "] + return schemaJson['image'][0]; } else { - return undefined; + return; } }; /** * Parses the og:image if it exists and returns it - * @param {!./ampdoc-impl.AmpDoc} ampdoc + * @param {!./service/ampdoc-impl.AmpDoc} ampdoc * @return {string|undefined} */ export function parseOgImage(ampdoc) { @@ -103,13 +106,13 @@ export function parseOgImage(ampdoc) { if (metaTag) { return metaTag.getAttribute('content'); } else { - return undefined; + return; } }; /** * Parses the website's Favicon and returns it - * @param {!./ampdoc-impl.AmpDoc} ampdoc + * @param {!./service/ampdoc-impl.AmpDoc} ampdoc * @return {string|undefined} */ export function parseFavicon(ampdoc) { @@ -119,6 +122,6 @@ export function parseFavicon(ampdoc) { if (linkTag) { return linkTag.getAttribute('href'); } else { - return undefined; + return; } } diff --git a/src/service/video-manager-impl.js b/src/service/video-manager-impl.js index bbf319f2826a..869c08128a05 100644 --- a/src/service/video-manager-impl.js +++ b/src/service/video-manager-impl.js @@ -41,6 +41,7 @@ import { import {map} from '../utils/object'; import {layoutRectLtwh, RelativePositions} from '../layout-rect'; import { + EMPTY_METADATA, parseSchemaImage, parseOgImage, parseFavicon, @@ -546,18 +547,10 @@ class VideoEntry { this.hasFullscreenOnLandscape = fsOnLandscapeAttr == '' || fsOnLandscapeAttr == 'always'; - listenOncePromise(element, VideoEvents.LOAD) - .then(() => this.videoLoaded()); - // Media Session API Variables /** @private {!../video-interface.VideoMetaDef} */ - this.metaData_ = { - 'artist': '', - 'album': '', - 'artwork': [], - 'title': '', - }; + this.metadata_ = EMPTY_METADATA; listenOncePromise(element, VideoEvents.LOAD) .then(() => this.videoLoaded()); @@ -596,12 +589,12 @@ class VideoEntry { this.video.play(/*isAutoplay*/ false); }; const pauseHandler = () => { - this.video.pause(/*isAutoplay*/ false); - } + this.video.pause(); + }; // Update the media session setMediaSession( this.ampdoc_, - this.metaData_, + this.metadata_, playHandler, pauseHandler ); @@ -653,7 +646,7 @@ class VideoEntry { this.inlineVidRect_ = this.video.element./*OK*/getBoundingClientRect(); }); - this.fillMetaData_(); + this.fillMediaSessionMetadata_(); this.updateVisibility(); if (this.isVisible_) { @@ -666,37 +659,36 @@ class VideoEntry { * Gets the provided metadata and fills in missing fields * @private */ - fillMetaData_() { + fillMediaSessionMetadata_() { if (this.video.preimplementsMediaSessionAPI()) { return; } - if (this.video.getMetaData()) { - this.metaData_ = map(this.video.getMetaData()); + if (this.video.getMetadata()) { + const mapped = map(this.video.getMetadata()); + this.metadata_ = /** @type {!../video-interface.VideoMetaDef} */ (mapped); } - if (!this.metaData_.artwork || this.metaData_.artwork.length == 0) { - const posterUrl = this.video.element.getAttribute('poster') - || this.internalElement_.getAttribute('poster') - || parseSchemaImage(this.ampdoc_) + if (!this.metadata_.artwork || this.metadata_.artwork.length == 0) { + const posterUrl = parseSchemaImage(this.ampdoc_) || parseOgImage(this.ampdoc_) || parseFavicon(this.ampdoc_); if (posterUrl) { - this.metaData_.artwork = [{ + this.metadata_.artwork = [{ 'src': posterUrl, }]; } } - if (!this.metaData_.title) { + if (!this.metadata_.title) { const title = this.video.element.getAttribute('title') || this.video.element.getAttribute('aria-label') || this.internalElement_.getAttribute('title') || this.internalElement_.getAttribute('aria-label') || this.ampdoc_.win.document.title; if (title) { - this.metaData_.title = title; + this.metadata_.title = title; } } } diff --git a/src/video-interface.js b/src/video-interface.js index 577b3ea210ec..5ee4f067b384 100644 --- a/src/video-interface.js +++ b/src/video-interface.js @@ -18,7 +18,7 @@ import {ActionTrust} from './action-trust'; /* eslint no-unused-vars: 0 */ /** * @typedef {{ - * artwork: string, + * artwork: Array, * title: string, * album: string, * artist: string, @@ -121,28 +121,19 @@ export class VideoInterface { * title (string): Name of the video * artist (string): Name of the video's author/artist * album (string): Name of the video's album if it exists - * @return {!VideoMetaDef} metadata + * @return {!VideoMetaDef|undefined} metadata */ - getMetaData() { - return { - 'artwork': '', - 'title': '', - 'artist': '', - 'album': '', - }; - } + getMetadata() {} /** * If this returns true then it will be assumed that the player implements * the MediaSession API internally so that the video manager does not override - * it. If not, the video manager will use the metaData variable as well as + * it. If not, the video manager will use the metadata variable as well as * inferred meta-data to update the video's Media Session notification. * * @return {boolean} */ - preimplementsMediaSessionAPI() { - return false; - } + preimplementsMediaSessionAPI() {} /** diff --git a/test/functional/test-mediasession-helper.js b/test/functional/test-mediasession-helper.js index 10dce56cecc7..747a9cc8dff3 100644 --- a/test/functional/test-mediasession-helper.js +++ b/test/functional/test-mediasession-helper.js @@ -71,20 +71,20 @@ describes.sandboxed('MediaSessionAPI Helper Functions', {}, () => { document.documentElement.innerHTML = template; ampdoc = { win: { - document: document, - navigator: { - mediaSession: { - metadata: { + 'document': document, + 'navigator': { + 'mediaSession': { + 'metadata': { 'artist': '', 'album': '', 'artwork': [], 'title': '', }, - setActionHandler: () => { + 'setActionHandler': () => { }, - } + }, }, - MediaMetadata: Object, + 'MediaMetadata': Object, }, }; }); @@ -116,11 +116,12 @@ describes.sandboxed('MediaSessionAPI Helper Functions', {}, () => { 'artist': 'Some artist', 'album': 'Some album', 'artwork': [ - 'http://example.com/image.png' + 'http://example.com/image.png', ], 'title': 'Some title', }; setMediaSession(ampdoc, fakeMetaData); - expect(ampdoc.win.navigator.mediaSession.metadata).to.deep.equal(fakeMetaData); + const newMetaData = ampdoc.win.navigator.mediaSession.metadata; + expect(newMetaData).to.deep.equal(fakeMetaData); }); }); diff --git a/test/functional/test-video-manager.js b/test/functional/test-video-manager.js index 340c7b42fb59..bdb9de859b78 100644 --- a/test/functional/test-video-manager.js +++ b/test/functional/test-video-manager.js @@ -526,13 +526,8 @@ function createFakeVideoPlayerClass(win) { } /** @override */ - getMetaData() { - return { - 'artwork': '', - 'title': '', - 'artist': '', - 'album': '', - }; + getMetadata() { + // Not supported } /** @override */ From 588aeb1695de2146b3e5ff1de6c88a20410bcdd8 Mon Sep 17 00:00:00 2001 From: Wassim Gharbi Date: Tue, 1 Aug 2017 14:20:45 -0700 Subject: [PATCH 7/7] Fixed test --- src/mediasession-helper.js | 7 +- src/video-interface.js | 2 +- test/functional/test-mediasession-helper.js | 95 ++++++++++++--------- 3 files changed, 56 insertions(+), 48 deletions(-) diff --git a/src/mediasession-helper.js b/src/mediasession-helper.js index 76bb161aabfe..d13f52f176d4 100644 --- a/src/mediasession-helper.js +++ b/src/mediasession-helper.js @@ -78,16 +78,13 @@ export function parseSchemaImage(ampdoc) { // 1. "image": "http://..", return schemaJson['image']; } else if (schemaJson['image']['@list'] - && schemaJson['image']['@list'][0] && typeof schemaJson['image']['@list'][0] === 'string') { // 2. "image": {.., "@list": ["http://.."], ..} return schemaJson['image']['@list'][0]; - } else if (schemaJson['image']['url'] - && typeof schemaJson['image']['url'] === 'string') { + } else if (typeof schemaJson['image']['url'] === 'string') { // 3. "image": {.., "url": "http://..", ..} return schemaJson['image']['url']; - } else if (schemaJson['image'][0] - && typeof schemaJson['image'][0] === 'string') { + } else if (typeof schemaJson['image'][0] === 'string') { // 4. "image": ["http://.. "] return schemaJson['image'][0]; } else { diff --git a/src/video-interface.js b/src/video-interface.js index 5ee4f067b384..ee0b0e97ee44 100644 --- a/src/video-interface.js +++ b/src/video-interface.js @@ -117,7 +117,7 @@ export class VideoInterface { /** * Returns video's meta data (artwork, title, artist, album, etc.) for use * with the Media Session API - * artwork (string): URL to the poster image (preferably a 512x512 PNG) + * artwork (Array): URL to the poster image (preferably a 512x512 PNG) * title (string): Name of the video * artist (string): Name of the video's author/artist * album (string): Name of the video's album if it exists diff --git a/test/functional/test-mediasession-helper.js b/test/functional/test-mediasession-helper.js index 747a9cc8dff3..f1ce57b832d0 100644 --- a/test/functional/test-mediasession-helper.js +++ b/test/functional/test-mediasession-helper.js @@ -21,54 +21,63 @@ import { parseFavicon, } from '../../src/mediasession-helper'; -const template = ` - - - - - - -

Yesterday Something Happened

-

Lorem ipsum dolor set amet..

- +const schemaTemplate = ` +{ + "@context": "http://schema.org", + "@type": "NewsArticle", + "mainEntityOfPage": "something", + "headline": "Something Happened", + "datePublished": "Fri Jul 28 12:45:00 EDT 2017", + "dateModified": "Fri Jul 28 12:45:00 EDT 2017", + "description": "Appearantly, yesterday something happened", + "author": { + "@type": "Person", + "name": "Awesome Author" + }, + "publisher": { + "@type": "Organization", + "name": "Aperture Science", + "logo": { + "@type": "ImageObject", + "url": "logo-url", + "width": 133, + "height": 60 + } + }, + "image": { + "@type": "ImageObject", + "url": "http://example.com/image.png", + "height": 392, + "width": 696 + } +} `; describes.sandboxed('MediaSessionAPI Helper Functions', {}, () => { let ampdoc; + let favicon; + let schema; + let ogImage; + let head; beforeEach(() => { - document.documentElement.innerHTML = template; + head = document.querySelector('head'); + // Favicon + favicon = document.createElement('link'); + favicon.setAttribute('rel', 'icon'); + favicon.setAttribute('href', 'http://example.com/favicon.ico'); + head.appendChild(favicon); + // Schema + schema = document.createElement('script'); + schema.setAttribute('type', 'application/ld+json'); + schema.innerHTML = schemaTemplate; + head.appendChild(schema); + // og-image + ogImage = document.createElement('meta'); + ogImage.setAttribute('property', 'og:image'); + ogImage.setAttribute('content', 'http://example.com/og-image.png'); + head.appendChild(ogImage); ampdoc = { win: { 'document': document, @@ -90,7 +99,9 @@ describes.sandboxed('MediaSessionAPI Helper Functions', {}, () => { }); afterEach(() => { - document.documentElement.innerHTML = ''; + head.removeChild(favicon); + head.removeChild(schema); + head.removeChild(ogImage); }); it('should parse the schema and find the image', () => {