mirror of
https://github.com/WordPress/WordPress.git
synced 2025-01-22 08:11:52 +01:00
887998f7ed
* Update deprecated `isHostedVideo` method to always return `true` since all oEmbeds are now supported. * Disable `scanImage` for non-image media widgets. * Ensure embed URL field element is matched from current media frame instance due to bug where media frames are not destroyed upon closing. * Update error message for invalid video URLs to ask user to check the URL for validity. Props gk.loveweb, octalmage, westonruter. See #42039, #40935. Fixes #41052. Built from https://develop.svn.wordpress.org/trunk@41933 git-svn-id: http://core.svn.wordpress.org/trunk@41767 1a063a9b-81f0-0310-95a4-ce76da25c4cd
251 lines
6.4 KiB
JavaScript
251 lines
6.4 KiB
JavaScript
/* eslint consistent-this: [ "error", "control" ] */
|
|
(function( component ) {
|
|
'use strict';
|
|
|
|
var VideoWidgetModel, VideoWidgetControl, VideoDetailsMediaFrame;
|
|
|
|
/**
|
|
* Custom video details frame that removes the replace-video state.
|
|
*
|
|
* @class VideoDetailsMediaFrame
|
|
* @constructor
|
|
*/
|
|
VideoDetailsMediaFrame = wp.media.view.MediaFrame.VideoDetails.extend({
|
|
|
|
/**
|
|
* Create the default states.
|
|
*
|
|
* @returns {void}
|
|
*/
|
|
createStates: function createStates() {
|
|
this.states.add([
|
|
new wp.media.controller.VideoDetails({
|
|
media: this.media
|
|
}),
|
|
|
|
new wp.media.controller.MediaLibrary({
|
|
type: 'video',
|
|
id: 'add-video-source',
|
|
title: wp.media.view.l10n.videoAddSourceTitle,
|
|
toolbar: 'add-video-source',
|
|
media: this.media,
|
|
menu: false
|
|
}),
|
|
|
|
new wp.media.controller.MediaLibrary({
|
|
type: 'text',
|
|
id: 'add-track',
|
|
title: wp.media.view.l10n.videoAddTrackTitle,
|
|
toolbar: 'add-track',
|
|
media: this.media,
|
|
menu: 'video-details'
|
|
})
|
|
]);
|
|
}
|
|
});
|
|
|
|
/**
|
|
* Video widget model.
|
|
*
|
|
* See WP_Widget_Video::enqueue_admin_scripts() for amending prototype from PHP exports.
|
|
*
|
|
* @class VideoWidgetModel
|
|
* @constructor
|
|
*/
|
|
VideoWidgetModel = component.MediaWidgetModel.extend({});
|
|
|
|
/**
|
|
* Video widget control.
|
|
*
|
|
* See WP_Widget_Video::enqueue_admin_scripts() for amending prototype from PHP exports.
|
|
*
|
|
* @class VideoWidgetControl
|
|
* @constructor
|
|
*/
|
|
VideoWidgetControl = component.MediaWidgetControl.extend({
|
|
|
|
/**
|
|
* Show display settings.
|
|
*
|
|
* @type {boolean}
|
|
*/
|
|
showDisplaySettings: false,
|
|
|
|
/**
|
|
* Cache of oembed responses.
|
|
*
|
|
* @type {Object}
|
|
*/
|
|
oembedResponses: {},
|
|
|
|
/**
|
|
* Map model props to media frame props.
|
|
*
|
|
* @param {Object} modelProps - Model props.
|
|
* @returns {Object} Media frame props.
|
|
*/
|
|
mapModelToMediaFrameProps: function mapModelToMediaFrameProps( modelProps ) {
|
|
var control = this, mediaFrameProps;
|
|
mediaFrameProps = component.MediaWidgetControl.prototype.mapModelToMediaFrameProps.call( control, modelProps );
|
|
mediaFrameProps.link = 'embed';
|
|
return mediaFrameProps;
|
|
},
|
|
|
|
/**
|
|
* Fetches embed data for external videos.
|
|
*
|
|
* @returns {void}
|
|
*/
|
|
fetchEmbed: function fetchEmbed() {
|
|
var control = this, url;
|
|
url = control.model.get( 'url' );
|
|
|
|
// If we already have a local cache of the embed response, return.
|
|
if ( control.oembedResponses[ url ] ) {
|
|
return;
|
|
}
|
|
|
|
// If there is an in-flight embed request, abort it.
|
|
if ( control.fetchEmbedDfd && 'pending' === control.fetchEmbedDfd.state() ) {
|
|
control.fetchEmbedDfd.abort();
|
|
}
|
|
|
|
control.fetchEmbedDfd = wp.apiRequest({
|
|
url: wp.media.view.settings.oEmbedProxyUrl,
|
|
data: {
|
|
url: control.model.get( 'url' ),
|
|
maxwidth: control.model.get( 'width' ),
|
|
maxheight: control.model.get( 'height' ),
|
|
discover: false
|
|
},
|
|
type: 'GET',
|
|
dataType: 'json',
|
|
context: control
|
|
});
|
|
|
|
control.fetchEmbedDfd.done( function( response ) {
|
|
control.oembedResponses[ url ] = response;
|
|
control.renderPreview();
|
|
});
|
|
|
|
control.fetchEmbedDfd.fail( function() {
|
|
control.oembedResponses[ url ] = null;
|
|
});
|
|
},
|
|
|
|
/**
|
|
* Whether a url is a supported external host.
|
|
*
|
|
* @deprecated since 4.9.
|
|
*
|
|
* @returns {boolean} Whether url is a supported video host.
|
|
*/
|
|
isHostedVideo: function isHostedVideo() {
|
|
return true;
|
|
},
|
|
|
|
/**
|
|
* Render preview.
|
|
*
|
|
* @returns {void}
|
|
*/
|
|
renderPreview: function renderPreview() {
|
|
var control = this, previewContainer, previewTemplate, attachmentId, attachmentUrl, poster, html = '', isOEmbed = false, mime, error, urlParser, matches;
|
|
attachmentId = control.model.get( 'attachment_id' );
|
|
attachmentUrl = control.model.get( 'url' );
|
|
error = control.model.get( 'error' );
|
|
|
|
if ( ! attachmentId && ! attachmentUrl ) {
|
|
return;
|
|
}
|
|
|
|
// Verify the selected attachment mime is supported.
|
|
mime = control.selectedAttachment.get( 'mime' );
|
|
if ( mime && attachmentId ) {
|
|
if ( ! _.contains( _.values( wp.media.view.settings.embedMimes ), mime ) ) {
|
|
error = 'unsupported_file_type';
|
|
}
|
|
} else if ( ! attachmentId ) {
|
|
urlParser = document.createElement( 'a' );
|
|
urlParser.href = attachmentUrl;
|
|
matches = urlParser.pathname.toLowerCase().match( /\.(\w+)$/ );
|
|
if ( matches ) {
|
|
if ( ! _.contains( _.keys( wp.media.view.settings.embedMimes ), matches[1] ) ) {
|
|
error = 'unsupported_file_type';
|
|
}
|
|
} else {
|
|
isOEmbed = true;
|
|
}
|
|
}
|
|
|
|
if ( isOEmbed ) {
|
|
control.fetchEmbed();
|
|
if ( control.oembedResponses[ attachmentUrl ] ) {
|
|
poster = control.oembedResponses[ attachmentUrl ].thumbnail_url;
|
|
html = control.oembedResponses[ attachmentUrl ].html.replace( /\swidth="\d+"/, ' width="100%"' ).replace( /\sheight="\d+"/, '' );
|
|
}
|
|
}
|
|
|
|
previewContainer = control.$el.find( '.media-widget-preview' );
|
|
previewTemplate = wp.template( 'wp-media-widget-video-preview' );
|
|
|
|
previewContainer.html( previewTemplate({
|
|
model: {
|
|
attachment_id: attachmentId,
|
|
html: html,
|
|
src: attachmentUrl,
|
|
poster: poster
|
|
},
|
|
is_oembed: isOEmbed,
|
|
error: error
|
|
}));
|
|
wp.mediaelement.initialize();
|
|
},
|
|
|
|
/**
|
|
* Open the media image-edit frame to modify the selected item.
|
|
*
|
|
* @returns {void}
|
|
*/
|
|
editMedia: function editMedia() {
|
|
var control = this, mediaFrame, metadata, updateCallback;
|
|
|
|
metadata = control.mapModelToMediaFrameProps( control.model.toJSON() );
|
|
|
|
// Set up the media frame.
|
|
mediaFrame = new VideoDetailsMediaFrame({
|
|
frame: 'video',
|
|
state: 'video-details',
|
|
metadata: metadata
|
|
});
|
|
wp.media.frame = mediaFrame;
|
|
mediaFrame.$el.addClass( 'media-widget' );
|
|
|
|
updateCallback = function( mediaFrameProps ) {
|
|
|
|
// Update cached attachment object to avoid having to re-fetch. This also triggers re-rendering of preview.
|
|
control.selectedAttachment.set( mediaFrameProps );
|
|
|
|
control.model.set( _.extend(
|
|
_.omit( control.model.defaults(), 'title' ),
|
|
control.mapMediaToModelProps( mediaFrameProps ),
|
|
{ error: false }
|
|
) );
|
|
};
|
|
|
|
mediaFrame.state( 'video-details' ).on( 'update', updateCallback );
|
|
mediaFrame.state( 'replace-video' ).on( 'replace', updateCallback );
|
|
mediaFrame.on( 'close', function() {
|
|
mediaFrame.detach();
|
|
});
|
|
|
|
mediaFrame.open();
|
|
}
|
|
});
|
|
|
|
// Exports.
|
|
component.controlConstructors.media_video = VideoWidgetControl;
|
|
component.modelConstructors.media_video = VideoWidgetModel;
|
|
|
|
})( wp.mediaWidgets );
|