wpView:
- Don't wrap single-line URLs in [embed]. Use them directly in generating a view. - If the embedding HTML contains a script, "sandbox" it in an iframe to prevent it from changing the editor DOM. - Automatically add toolbar and overlay when needed. - Try to embed single-line URLs only if they are pasted in an empty paragraph. Props avryl, see #28195 git-svn-id: https://develop.svn.wordpress.org/trunk@28748 602fd350-edb4-49c9-b593-d223f7449a82
This commit is contained in:
parent
e0a089d9d8
commit
09d4b541fb
@ -8,7 +8,9 @@
|
||||
// Ensure the global `wp` object exists.
|
||||
window.wp = window.wp || {};
|
||||
|
||||
(function($){
|
||||
( function( $ ) {
|
||||
'use strict';
|
||||
|
||||
var views = {},
|
||||
instances = {},
|
||||
media = wp.media,
|
||||
@ -27,6 +29,7 @@ window.wp = window.wp || {};
|
||||
*/
|
||||
wp.mce.View = function( options ) {
|
||||
options = options || {};
|
||||
this.type = options.type;
|
||||
_.extend( this, _.pick( options, viewOptions ) );
|
||||
this.initialize.apply( this, arguments );
|
||||
};
|
||||
@ -35,22 +38,67 @@ window.wp = window.wp || {};
|
||||
initialize: function() {},
|
||||
getHtml: function() {},
|
||||
render: function() {
|
||||
var html = this.getHtml();
|
||||
// Search all tinymce editor instances and update the placeholders
|
||||
this.setContent(
|
||||
'<div class="toolbar">' +
|
||||
( _.isFunction( views[ this.type ].edit ) ? '<div class="dashicons dashicons-edit edit"></div>' : '' ) +
|
||||
'<div class="dashicons dashicons-no-alt remove"></div>' +
|
||||
'</div>' +
|
||||
'<div class="wpview-content">' +
|
||||
this.getHtml() +
|
||||
'</div>' +
|
||||
( this.overlay ? '<div class="wpview-overlay"></div>' : '' ) +
|
||||
// The <ins> is used to mark the end of the wrapper div (has to be the last child node).
|
||||
// Needed when comparing the content as string for preventing extra undo levels.
|
||||
'<ins data-wpview-end="1"></ins>',
|
||||
function( self, editor, node ) {
|
||||
$( self ).trigger( 'ready', [ editor, node ] );
|
||||
}
|
||||
);
|
||||
},
|
||||
unbind: function() {},
|
||||
setContent: function( html, callback, replace ) {
|
||||
_.each( tinymce.editors, function( editor ) {
|
||||
var self = this;
|
||||
if ( editor.plugins.wpview ) {
|
||||
$( editor.getDoc() ).find( '[data-wpview-text="' + this.encodedText + '"]' ).each( function ( i, element ) {
|
||||
var node = $( element );
|
||||
// The <ins> is used to mark the end of the wrapper div. Needed when comparing
|
||||
// the content as string for preventing extra undo levels.
|
||||
node.html( html ).append( '<ins data-wpview-end="1"></ins>' );
|
||||
$( self ).trigger( 'ready', element );
|
||||
});
|
||||
$( editor.getBody() )
|
||||
.find( '[data-wpview-text="' + this.encodedText + '"]' )
|
||||
.each( function ( i, element ) {
|
||||
var contentWrap = $( element ).children( '.wpview-content' ),
|
||||
wrap = element;
|
||||
|
||||
if ( contentWrap.length ) {
|
||||
element = contentWrap = contentWrap[0];
|
||||
}
|
||||
|
||||
if ( _.isString( html ) ) {
|
||||
if ( replace ) {
|
||||
element = editor.dom.replace( editor.dom.createFragment( html ), wrap );
|
||||
} else {
|
||||
editor.dom.setHTML( element, html );
|
||||
}
|
||||
} else {
|
||||
if ( replace ) {
|
||||
element = editor.dom.replace( html, wrap );
|
||||
} else {
|
||||
element.appendChild( html );
|
||||
}
|
||||
}
|
||||
|
||||
if ( _.isFunction( callback ) ) {
|
||||
callback( self, editor, $( element ).children( '.wpview-content' )[0] );
|
||||
}
|
||||
} );
|
||||
}
|
||||
}, this );
|
||||
},
|
||||
unbind: function() {}
|
||||
setError: function( message, dashicon ) {
|
||||
this.setContent(
|
||||
'<div class="wpview-error">' +
|
||||
'<div class="dashicons dashicons-' + ( dashicon ? dashicon : 'no' ) + '"></div>' +
|
||||
'<p>' + message + '</p>' +
|
||||
'</div>'
|
||||
);
|
||||
}
|
||||
} );
|
||||
|
||||
// take advantage of the Backbone extend method
|
||||
@ -209,6 +257,7 @@ window.wp = window.wp || {};
|
||||
|
||||
if ( ! wp.mce.views.getInstance( encodedText ) ) {
|
||||
viewOptions = options;
|
||||
viewOptions.type = viewType;
|
||||
viewOptions.encodedText = encodedText;
|
||||
instance = new view.View( viewOptions );
|
||||
instances[ encodedText ] = instance;
|
||||
@ -244,6 +293,7 @@ window.wp = window.wp || {};
|
||||
if ( ! instance ) {
|
||||
result = view.toView( text );
|
||||
viewOptions = result.options;
|
||||
viewOptions.type = view.type;
|
||||
viewOptions.encodedText = encodedText;
|
||||
instance = new view.View( viewOptions );
|
||||
instances[ encodedText ] = instance;
|
||||
@ -331,7 +381,6 @@ window.wp = window.wp || {};
|
||||
};
|
||||
|
||||
return this.template( options );
|
||||
|
||||
}
|
||||
},
|
||||
|
||||
@ -361,6 +410,8 @@ window.wp = window.wp || {};
|
||||
loaded: false,
|
||||
|
||||
View: _.extend( {}, wp.media.mixin, {
|
||||
overlay: true,
|
||||
|
||||
initialize: function( options ) {
|
||||
this.players = [];
|
||||
this.shortcode = options.shortcode;
|
||||
@ -375,32 +426,23 @@ window.wp = window.wp || {};
|
||||
*
|
||||
* @global MediaElementPlayer
|
||||
*
|
||||
* @param {Event} e
|
||||
* @param {Event} event
|
||||
* @param {Object} editor
|
||||
* @param {HTMLElement} node
|
||||
*/
|
||||
setPlayer: function(e, node) {
|
||||
// if the ready event fires on an empty node
|
||||
if ( ! node ) {
|
||||
return;
|
||||
}
|
||||
|
||||
setPlayer: function( event, editor, node ) {
|
||||
var self = this,
|
||||
media,
|
||||
firefox = this.ua.is( 'ff' ),
|
||||
className = '.wp-' + this.shortcode.tag + '-shortcode';
|
||||
media;
|
||||
|
||||
media = $( node ).find( className );
|
||||
media = $( node ).find( '.wp-' + this.shortcode.tag + '-shortcode' );
|
||||
|
||||
if ( ! this.isCompatible( media ) ) {
|
||||
media.closest( '.wpview-wrap' ).addClass( 'wont-play' );
|
||||
if ( ! media.parent().hasClass( 'wpview-wrap' ) ) {
|
||||
media.parent().replaceWith( media );
|
||||
}
|
||||
media.replaceWith( '<p>' + media.find( 'source' ).eq(0).prop( 'src' ) + '</p>' );
|
||||
return;
|
||||
} else {
|
||||
media.closest( '.wpview-wrap' ).removeClass( 'wont-play' );
|
||||
if ( firefox ) {
|
||||
if ( this.ua.is( 'ff' ) ) {
|
||||
media.prop( 'preload', 'metadata' );
|
||||
} else {
|
||||
media.prop( 'preload', 'none' );
|
||||
@ -508,6 +550,7 @@ window.wp = window.wp || {};
|
||||
state: ['playlist-edit', 'video-playlist-edit'],
|
||||
View: _.extend( {}, wp.media.mixin, {
|
||||
template: media.template( 'editor-playlist' ),
|
||||
overlay: true,
|
||||
|
||||
initialize: function( options ) {
|
||||
this.players = [];
|
||||
@ -531,7 +574,7 @@ window.wp = window.wp || {};
|
||||
this.dfd = this.attachments.more().done( _.bind( this.render, this ) );
|
||||
},
|
||||
|
||||
setPlaylist: function( event, element ) {
|
||||
setPlaylist: function( event, editor, element ) {
|
||||
if ( ! this.data.tracks ) {
|
||||
return;
|
||||
}
|
||||
@ -634,60 +677,161 @@ window.wp = window.wp || {};
|
||||
/**
|
||||
* TinyMCE handler for the embed shortcode
|
||||
*/
|
||||
wp.mce.views.register( 'embed', {
|
||||
View: _.extend( {}, wp.media.mixin, {
|
||||
template: media.template( 'editor-embed' ),
|
||||
initialize: function( options ) {
|
||||
this.players = [];
|
||||
this.content = options.content;
|
||||
this.parsed = false;
|
||||
this.shortcode = options.shortcode;
|
||||
_.bindAll( this, 'setHtml', 'setNode', 'fetch' );
|
||||
$( this ).on( 'ready', this.setNode );
|
||||
},
|
||||
unbind: function() {
|
||||
var self = this;
|
||||
_.each( this.players, function ( player ) {
|
||||
player.pause();
|
||||
self.removePlayer( player );
|
||||
} );
|
||||
this.players = [];
|
||||
},
|
||||
setNode: function ( e, node ) {
|
||||
this.node = node;
|
||||
if ( this.parsed ) {
|
||||
this.parseMediaShortcodes();
|
||||
} else {
|
||||
this.fetch();
|
||||
}
|
||||
},
|
||||
fetch: function () {
|
||||
wp.ajax.send( 'parse-embed', {
|
||||
data: {
|
||||
post_ID: $( '#post_ID' ).val(),
|
||||
content: this.shortcode.string()
|
||||
}
|
||||
} ).done( this.setHtml );
|
||||
},
|
||||
setHtml: function ( content ) {
|
||||
this.parsed = content;
|
||||
$( this.node ).html( this.getHtml() );
|
||||
this.parseMediaShortcodes();
|
||||
},
|
||||
parseMediaShortcodes: function () {
|
||||
var self = this;
|
||||
$( '.wp-audio-shortcode, .wp-video-shortcode', this.node ).each( function ( i, element ) {
|
||||
self.players.push( new MediaElementPlayer( element, self.mejsSettings ) );
|
||||
} );
|
||||
},
|
||||
getHtml: function() {
|
||||
if ( ! this.parsed ) {
|
||||
return '';
|
||||
}
|
||||
return this.template({ content: this.parsed });
|
||||
wp.mce.embedView = _.extend( {}, wp.media.mixin, {
|
||||
overlay: true,
|
||||
initialize: function( options ) {
|
||||
this.players = [];
|
||||
this.content = options.content;
|
||||
this.fetching = false;
|
||||
this.parsed = false;
|
||||
this.original = options.url || options.shortcode.string();
|
||||
|
||||
if ( options.url ) {
|
||||
this.shortcode = '[embed]' + options.url + '[/embed]';
|
||||
} else {
|
||||
this.shortcode = options.shortcode.string();
|
||||
}
|
||||
} ),
|
||||
edit: function() {}
|
||||
|
||||
_.bindAll( this, 'setHtml', 'setNode', 'fetch' );
|
||||
$( this ).on( 'ready', this.setNode );
|
||||
},
|
||||
unbind: function() {
|
||||
var self = this;
|
||||
_.each( this.players, function ( player ) {
|
||||
player.pause();
|
||||
self.removePlayer( player );
|
||||
} );
|
||||
this.players = [];
|
||||
},
|
||||
setNode: function () {
|
||||
if ( this.parsed ) {
|
||||
this.setHtml( this.parsed );
|
||||
this.parseMediaShortcodes();
|
||||
} else if ( ! this.fetching ) {
|
||||
this.fetch();
|
||||
}
|
||||
},
|
||||
fetch: function () {
|
||||
var self = this;
|
||||
|
||||
this.fetching = true;
|
||||
|
||||
wp.ajax.send( 'parse-embed', {
|
||||
data: {
|
||||
post_ID: $( '#post_ID' ).val(),
|
||||
content: this.shortcode
|
||||
}
|
||||
} )
|
||||
.done( function( content ) {
|
||||
self.fetching = false;
|
||||
|
||||
if ( content.substring( 0, ( '<a href' ).length ) === '<a href' ) {
|
||||
if ( self.type === 'embed' ) {
|
||||
self.setError( self.original + ' failed to embed.', 'admin-media' );
|
||||
} else {
|
||||
self.setContent( self.original, null, true );
|
||||
}
|
||||
} else {
|
||||
self.parsed = content;
|
||||
self.setHtml( content );
|
||||
}
|
||||
} )
|
||||
.fail( function() {
|
||||
self.fetching = false;
|
||||
self.setError( self.original + ' failed to embed due to a server error.', 'admin-media' );
|
||||
} );
|
||||
},
|
||||
/* jshint scripturl: true */
|
||||
setHtml: function ( content ) {
|
||||
var MutationObserver = window.MutationObserver || window.WebKitMutationObserver || window.MozMutationObserver,
|
||||
iframe, iframeDoc, i, resize,
|
||||
dom = tinymce.DOM;
|
||||
|
||||
if ( content.indexOf( '<script' ) !== -1 ) {
|
||||
iframe = dom.create( 'iframe', {
|
||||
src: tinymce.Env.ie ? 'javascript:""' : '',
|
||||
frameBorder: '0',
|
||||
allowTransparency: 'true',
|
||||
style: {
|
||||
width: '100%',
|
||||
display: 'block'
|
||||
}
|
||||
} );
|
||||
|
||||
this.setContent( iframe );
|
||||
iframeDoc = iframe.contentWindow.document;
|
||||
|
||||
iframeDoc.open();
|
||||
iframeDoc.write(
|
||||
'<!DOCTYPE html>' +
|
||||
'<html>' +
|
||||
'<head>' +
|
||||
'<meta http-equiv="Content-Type" content="text/html; charset=UTF-8" />' +
|
||||
'</head>' +
|
||||
'<body>' +
|
||||
content +
|
||||
'</body>' +
|
||||
'</html>'
|
||||
);
|
||||
iframeDoc.close();
|
||||
|
||||
resize = function() {
|
||||
$( iframe ).height( $( iframeDoc ).height() );
|
||||
};
|
||||
|
||||
if ( MutationObserver ) {
|
||||
new MutationObserver( _.debounce( function() {
|
||||
resize();
|
||||
}, 100 ) )
|
||||
.observe( iframeDoc.body, {
|
||||
attributes: true,
|
||||
childList: true,
|
||||
subtree: true
|
||||
} );
|
||||
} else {
|
||||
for ( i = 1; i < 6; i++ ) {
|
||||
setTimeout( resize, i * 700 );
|
||||
}
|
||||
}
|
||||
} else {
|
||||
this.setContent( content );
|
||||
}
|
||||
|
||||
this.parseMediaShortcodes();
|
||||
},
|
||||
parseMediaShortcodes: function () {
|
||||
var self = this;
|
||||
$( '.wp-audio-shortcode, .wp-video-shortcode', this.node ).each( function ( i, element ) {
|
||||
self.players.push( new MediaElementPlayer( element, self.mejsSettings ) );
|
||||
} );
|
||||
},
|
||||
getHtml: function() {
|
||||
return '';
|
||||
}
|
||||
} );
|
||||
|
||||
wp.mce.views.register( 'embed', {
|
||||
View: wp.mce.embedView
|
||||
} );
|
||||
|
||||
wp.mce.views.register( 'embedURL', {
|
||||
toView: function( content ) {
|
||||
var re = /(?:^|<p>)(https?:\/\/[^\s"]+?)(?:<\/p>\s*|$)/gi,
|
||||
match = re.exec( tinymce.trim( content ) );
|
||||
|
||||
if ( ! match ) {
|
||||
return;
|
||||
}
|
||||
|
||||
return {
|
||||
index: match.index,
|
||||
content: match[0],
|
||||
options: {
|
||||
url: match[1]
|
||||
}
|
||||
};
|
||||
},
|
||||
View: wp.mce.embedView
|
||||
} );
|
||||
|
||||
}(jQuery));
|
||||
|
@ -158,6 +158,8 @@ tinymce.PluginManager.add( 'wpview', function( editor ) {
|
||||
// matching view patterns, and transform the matches into
|
||||
// view wrappers.
|
||||
editor.on( 'BeforeSetContent', function( event ) {
|
||||
var node;
|
||||
|
||||
if ( ! event.content ) {
|
||||
return;
|
||||
}
|
||||
@ -166,15 +168,17 @@ tinymce.PluginManager.add( 'wpview', function( editor ) {
|
||||
wp.mce.views.unbind( editor );
|
||||
}
|
||||
|
||||
node = editor.selection.getNode();
|
||||
|
||||
// When a url is pasted, only try to embed it when pasted in an empty paragrapgh.
|
||||
if ( event.content.match( /^\s*(https?:\/\/[^\s"]+)\s*$/i ) &&
|
||||
( node.nodeName !== 'P' || node.parentNode !== editor.getBody() || ! editor.dom.isEmpty( node ) ) ) {
|
||||
return;
|
||||
}
|
||||
|
||||
event.content = wp.mce.views.toViews( event.content );
|
||||
});
|
||||
|
||||
editor.on( 'PastePreProcess', function( event ) {
|
||||
if ( event.content.match( /^\s*(https?:\/\/[^\s"]+)\s*$/im ) ) {
|
||||
event.content = '[embed]' + event.content + '[/embed]';
|
||||
}
|
||||
} );
|
||||
|
||||
// When the editor's content has been updated and the DOM has been
|
||||
// processed, render the views in the document.
|
||||
editor.on( 'SetContent', function( event ) {
|
||||
|
@ -994,9 +994,6 @@ function wp_print_media_templates() {
|
||||
</script>
|
||||
|
||||
<script type="text/html" id="tmpl-editor-gallery">
|
||||
<div class="toolbar">
|
||||
<div class="dashicons dashicons-edit edit"></div><div class="dashicons dashicons-no-alt remove"></div>
|
||||
</div>
|
||||
<# if ( data.attachments ) { #>
|
||||
<div class="gallery gallery-columns-{{ data.columns }}">
|
||||
<# _.each( data.attachments, function( attachment, index ) { #>
|
||||
@ -1027,30 +1024,16 @@ function wp_print_media_templates() {
|
||||
</script>
|
||||
|
||||
<script type="text/html" id="tmpl-editor-audio">
|
||||
<div class="toolbar">
|
||||
<div class="dashicons dashicons-edit edit"></div>
|
||||
<div class="dashicons dashicons-no-alt remove"></div>
|
||||
</div>
|
||||
<?php wp_underscore_audio_template() ?>
|
||||
<div class="wpview-overlay"></div>
|
||||
</script>
|
||||
|
||||
<script type="text/html" id="tmpl-editor-video">
|
||||
<div class="toolbar">
|
||||
<div class="dashicons dashicons-edit edit"></div>
|
||||
<div class="dashicons dashicons-no-alt remove"></div>
|
||||
</div>
|
||||
<?php wp_underscore_video_template() ?>
|
||||
<div class="wpview-overlay"></div>
|
||||
</script>
|
||||
|
||||
<?php wp_underscore_playlist_templates() ?>
|
||||
|
||||
<script type="text/html" id="tmpl-editor-playlist">
|
||||
<div class="toolbar">
|
||||
<div class="dashicons dashicons-edit edit"></div>
|
||||
<div class="dashicons dashicons-no-alt remove"></div>
|
||||
</div>
|
||||
<# if ( data.tracks ) { #>
|
||||
<div class="wp-playlist wp-{{ data.type }}-playlist wp-playlist-{{ data.style }}">
|
||||
<# if ( 'audio' === data.type ){ #>
|
||||
@ -1062,7 +1045,6 @@ function wp_print_media_templates() {
|
||||
<div class="wp-playlist-next"></div>
|
||||
<div class="wp-playlist-prev"></div>
|
||||
</div>
|
||||
<div class="wpview-overlay"></div>
|
||||
<# } else { #>
|
||||
<div class="wpview-error">
|
||||
<div class="dashicons dashicons-video-alt3"></div><p><?php _e( 'No items found.' ); ?></p>
|
||||
@ -1075,14 +1057,6 @@ function wp_print_media_templates() {
|
||||
<div class="upload-errors"></div>
|
||||
</script>
|
||||
|
||||
<script type="text/html" id="tmpl-editor-embed">
|
||||
<div class="toolbar">
|
||||
<div class="dashicons dashicons-no-alt remove"></div>
|
||||
</div>
|
||||
{{{ data.content }}}
|
||||
<div class="wpview-overlay"></div>
|
||||
</script>
|
||||
|
||||
<?php
|
||||
|
||||
/**
|
||||
|
Loading…
Reference in New Issue
Block a user