* @output wp-includes/js/mce-view.js
* Note: this API is "experimental" meaning that it will probably change
* in the next few releases based on feedback from 3.9.0.
* If you decide to use it, please follow the development closely.
* |- registered view constructor (type)
* | |- view instance (unique text)
( function( window, wp, shortcode, $ ) {
* A set of utilities that simplifies adding custom UI within a TinyMCE editor.
* At its core, it serves as a series of converters, transforming text to a
* custom UI, and back again.
* Registers a new view type.
* @param {string} type The view type.
* @param {Object} extend An object to extend wp.mce.View.prototype with.
register: function( type, extend ) {
views[ type ] = wp.mce.View.extend( _.extend( extend, { type: type } ) );
* Unregisters a view type.
* @param {string} type The view type.
unregister: function( type ) {
* Returns the settings of a view type.
* @param {string} type The view type.
* @return {Function} The view constructor.
* Unbinds all view nodes.
* Runs before removing all view nodes from the DOM.
_.each( instances, function( instance ) {
* Scans a given string for each view's pattern,
* replacing any matches with markers,
* and creates a new instance for every match.
* @param {string} content The string to scan.
* @param {tinymce.Editor} editor The editor.
* @return {string} The string with markers.
setMarkers: function( content, editor ) {
var pieces = [ { content: content } ],
_.each( views, function( view, type ) {
current = pieces.slice();
_.each( current, function( piece ) {
var remaining = piece.content,
// Ignore processed pieces, but retain their location.
// Iterate through the string progressively matching views
// and slicing the string as we go.
while ( remaining && ( result = view.prototype.match( remaining ) ) ) {
// Any text before the match becomes an unprocessed piece.
pieces.push( { content: remaining.substring( 0, result.index ) } );
result.options.editor = editor;
instance = self.createInstance( type, result.content, result.options );
text = instance.loader ? '.' : instance.text;
// Add the processed piece for the match.
content: instance.ignore ? text : '<p data-wpview-marker="' + instance.encodedText + '">' + text + '</p>',
// Update the remaining content.
remaining = remaining.slice( result.index + result.content.length );
// There are no additional matches.
// If any content remains, add it as an unprocessed piece.
pieces.push( { content: remaining } );
content = _.pluck( pieces, 'content' ).join( '' );
return content.replace( /<p>\s*<p data-wpview-marker=/g, '<p data-wpview-marker=' ).replace( /<\/p>\s*<\/p>/g, '</p>' );
* Create a view instance.
* @param {string} type The view type.
* @param {string} text The textual representation of the view.
* @param {Object} options Options.
* @param {boolean} force Recreate the instance. Optional.
* @return {wp.mce.View} The view instance.
createInstance: function( type, text, options, force ) {
var View = this.get( type ),
if ( text.indexOf( '[' ) !== -1 && text.indexOf( ']' ) !== -1 ) {
// Looks like a shortcode? Remove any line breaks from inside of shortcodes
// or autop will replace them with <p> and <br> later and the string won't match.
text = text.replace( /\[[^\]]+\]/g, function( match ) {
return match.replace( /[\r\n]/g, '' );
instance = this.getInstance( text );
encodedText = encodeURIComponent( text );
options = _.extend( options || {}, {
return instances[ encodedText ] = new View( options );
* @param {(string|HTMLElement)} object The textual representation of the view or the view node.
* @return {wp.mce.View} The view instance or undefined.
getInstance: function( object ) {
if ( typeof object === 'string' ) {
return instances[ encodeURIComponent( object ) ];
return instances[ $( object ).attr( 'data-wpview-text' ) ];
* Given a view node, get the view's text.
* @param {HTMLElement} node The view node.
* @return {string} The textual representation of the view.
getText: function( node ) {
return decodeURIComponent( $( node ).attr( 'data-wpview-text' ) || '' );
* Renders all view nodes that are not yet rendered.
* @param {boolean} force Rerender all view nodes.
render: function( force ) {
_.each( instances, function( instance ) {
instance.render( null, force );
* Update the text of a given view node.
* @param {string} text The new text.
* @param {tinymce.Editor} editor The TinyMCE editor instance the view node is in.
* @param {HTMLElement} node The view node to update.
* @param {boolean} force Recreate the instance. Optional.
update: function( text, editor, node, force ) {
var instance = this.getInstance( node );
instance.update( text, editor, node, force );
* Renders any editing interface based on the view type.
* @param {tinymce.Editor} editor The TinyMCE editor instance the view node is in.
* @param {HTMLElement} node The view node to edit.
edit: function( editor, node ) {
var instance = this.getInstance( node );
if ( instance && instance.edit ) {
instance.edit( instance.text, function( text, force ) {
instance.update( text, editor, node, force );
* Remove a given view node from the DOM.
* @param {tinymce.Editor} editor The TinyMCE editor instance the view node is in.
* @param {HTMLElement} node The view node to remove.
remove: function( editor, node ) {
var instance = this.getInstance( node );
instance.remove( editor, node );
* A Backbone-like View constructor intended for use when rendering a TinyMCE View.
* The main difference is that the TinyMCE View is not tied to a particular DOM node.
* @param {Object} options Options.
wp.mce.View = function( options ) {
_.extend( this, options );
wp.mce.View.extend = Backbone.View.extend;
_.extend( wp.mce.View.prototype, /** @lends wp.mce.View.prototype */{
* Whether or not to display a loader.
* Runs after the view instance is created.
initialize: function() {},
* Returns the content to render in the view node.
* Renders all view nodes tied to this view instance that are not yet rendered.
* @param {string} content The content to render. Optional.
* @param {boolean} force Rerender all view nodes tied to this view instance. Optional.
render: function( content, force ) {
content = this.getContent();
// If there's nothing to render an no loader needs to be shown, stop.
if ( ! this.loader && ! content ) {
// We're about to rerender all views of this instance, so unbind rendered views.
// Replace any left over markers.
this.setContent( content, function( editor, node ) {
$( node ).data( 'rendered', true );
this.bindNode.call( this, editor, node );
}, force ? null : false );
* Binds a given node after its content is added to the DOM.
* Unbinds a given node before its content is removed from the DOM.
unbindNode: function() {},
* Unbinds all view nodes tied to this view instance.
* Runs before their content is removed from the DOM.
this.getNodes( function( editor, node ) {
this.unbindNode.call( this, editor, node );
* Gets all the TinyMCE editor instances that support views.
* @param {Function} callback A callback.
getEditors: function( callback ) {
_.each( tinymce.editors, function( editor ) {
if ( editor.plugins.wpview ) {
callback.call( this, editor );
* Gets all view nodes tied to this view instance.
* @param {Function} callback A callback.
* @param {boolean} rendered Get (un)rendered view nodes. Optional.
getNodes: function( callback, rendered ) {
this.getEditors( function( editor ) {
.find( '[data-wpview-text="' + self.encodedText + '"]' )
if ( rendered == null ) {
data = $( this ).data( 'rendered' ) === true;
return rendered ? data : ! data;
callback.call( self, editor, this, this /* back compat */ );
* Gets all marker nodes tied to this view instance.
* @param {Function} callback A callback.
getMarkers: function( callback ) {
this.getEditors( function( editor ) {
.find( '[data-wpview-marker="' + this.encodedText + '"]' )
callback.call( self, editor, this );
* Replaces all marker nodes tied to this view instance.
replaceMarkers: function() {
this.getMarkers( function( editor, node ) {
var selected = node === editor.selection.getNode();
if ( ! this.loader && $( node ).text() !== tinymce.DOM.decode( this.text ) ) {
editor.dom.setAttrib( node, 'data-wpview-marker', null );
'<div class="wpview wpview-wrap" data-wpview-text="' + this.encodedText + '" data-wpview-type="' + this.type + '" contenteditable="false"></div>'
editor.undoManager.ignore( function() {
editor.$( node ).replaceWith( $viewNode );
editor.undoManager.ignore( function() {
editor.selection.select( $viewNode[0] );
editor.selection.collapse();
* Removes all marker nodes tied to this view instance.
removeMarkers: function() {
this.getMarkers( function( editor, node ) {
editor.dom.setAttrib( node, 'data-wpview-marker', null );
* Sets the content for all view nodes tied to this view instance.
* @param {*} content The content to set.
* @param {Function} callback A callback. Optional.
* @param {boolean} rendered Only set for (un)rendered nodes. Optional.
setContent: function( content, callback, rendered ) {
if ( _.isObject( content ) && ( content.sandbox || content.head || content.body.indexOf( '<script' ) !== -1 ) ) {
this.setIframes( content.head || '', content.body, callback, rendered );
} else if ( _.isString( content ) && content.indexOf( '<script' ) !== -1 ) {
this.setIframes( '', content, callback, rendered );
this.getNodes( function( editor, node ) {
content = content.body || content;
if ( content.indexOf( '<iframe' ) !== -1 ) {
content += '<span class="mce-shim"></span>';
editor.undoManager.transact( function() {
node.appendChild( _.isString( content ) ? editor.dom.createFragment( content ) : content );
editor.dom.add( node, 'span', { 'class': 'wpview-end' } );
callback && callback.call( this, editor, node );