angelovcom.net

Unnamed repository; edit this file 'description' to name the repository.
Log | Files | Refs

editor.js (45316B)


      1 /**
      2  * @output wp-admin/js/editor.js
      3  */
      4 
      5 window.wp = window.wp || {};
      6 
      7 ( function( $, wp ) {
      8 	wp.editor = wp.editor || {};
      9 
     10 	/**
     11 	 * Utility functions for the editor.
     12 	 *
     13 	 * @since 2.5.0
     14 	 */
     15 	function SwitchEditors() {
     16 		var tinymce, $$,
     17 			exports = {};
     18 
     19 		function init() {
     20 			if ( ! tinymce && window.tinymce ) {
     21 				tinymce = window.tinymce;
     22 				$$ = tinymce.$;
     23 
     24 				/**
     25 				 * Handles onclick events for the Visual/Text tabs.
     26 				 *
     27 				 * @since 4.3.0
     28 				 *
     29 				 * @return {void}
     30 				 */
     31 				$$( document ).on( 'click', function( event ) {
     32 					var id, mode,
     33 						target = $$( event.target );
     34 
     35 					if ( target.hasClass( 'wp-switch-editor' ) ) {
     36 						id = target.attr( 'data-wp-editor-id' );
     37 						mode = target.hasClass( 'switch-tmce' ) ? 'tmce' : 'html';
     38 						switchEditor( id, mode );
     39 					}
     40 				});
     41 			}
     42 		}
     43 
     44 		/**
     45 		 * Returns the height of the editor toolbar(s) in px.
     46 		 *
     47 		 * @since 3.9.0
     48 		 *
     49 		 * @param {Object} editor The TinyMCE editor.
     50 		 * @return {number} If the height is between 10 and 200 return the height,
     51 		 * else return 30.
     52 		 */
     53 		function getToolbarHeight( editor ) {
     54 			var node = $$( '.mce-toolbar-grp', editor.getContainer() )[0],
     55 				height = node && node.clientHeight;
     56 
     57 			if ( height && height > 10 && height < 200 ) {
     58 				return parseInt( height, 10 );
     59 			}
     60 
     61 			return 30;
     62 		}
     63 
     64 		/**
     65 		 * Switches the editor between Visual and Text mode.
     66 		 *
     67 		 * @since 2.5.0
     68 		 *
     69 		 * @memberof switchEditors
     70 		 *
     71 		 * @param {string} id The id of the editor you want to change the editor mode for. Default: `content`.
     72 		 * @param {string} mode The mode you want to switch to. Default: `toggle`.
     73 		 * @return {void}
     74 		 */
     75 		function switchEditor( id, mode ) {
     76 			id = id || 'content';
     77 			mode = mode || 'toggle';
     78 
     79 			var editorHeight, toolbarHeight, iframe,
     80 				editor = tinymce.get( id ),
     81 				wrap = $$( '#wp-' + id + '-wrap' ),
     82 				$textarea = $$( '#' + id ),
     83 				textarea = $textarea[0];
     84 
     85 			if ( 'toggle' === mode ) {
     86 				if ( editor && ! editor.isHidden() ) {
     87 					mode = 'html';
     88 				} else {
     89 					mode = 'tmce';
     90 				}
     91 			}
     92 
     93 			if ( 'tmce' === mode || 'tinymce' === mode ) {
     94 				// If the editor is visible we are already in `tinymce` mode.
     95 				if ( editor && ! editor.isHidden() ) {
     96 					return false;
     97 				}
     98 
     99 				// Insert closing tags for any open tags in QuickTags.
    100 				if ( typeof( window.QTags ) !== 'undefined' ) {
    101 					window.QTags.closeAllTags( id );
    102 				}
    103 
    104 				editorHeight = parseInt( textarea.style.height, 10 ) || 0;
    105 
    106 				var keepSelection = false;
    107 				if ( editor ) {
    108 					keepSelection = editor.getParam( 'wp_keep_scroll_position' );
    109 				} else {
    110 					keepSelection = window.tinyMCEPreInit.mceInit[ id ] &&
    111 									window.tinyMCEPreInit.mceInit[ id ].wp_keep_scroll_position;
    112 				}
    113 
    114 				if ( keepSelection ) {
    115 					// Save the selection.
    116 					addHTMLBookmarkInTextAreaContent( $textarea );
    117 				}
    118 
    119 				if ( editor ) {
    120 					editor.show();
    121 
    122 					// No point to resize the iframe in iOS.
    123 					if ( ! tinymce.Env.iOS && editorHeight ) {
    124 						toolbarHeight = getToolbarHeight( editor );
    125 						editorHeight = editorHeight - toolbarHeight + 14;
    126 
    127 						// Sane limit for the editor height.
    128 						if ( editorHeight > 50 && editorHeight < 5000 ) {
    129 							editor.theme.resizeTo( null, editorHeight );
    130 						}
    131 					}
    132 
    133 					if ( editor.getParam( 'wp_keep_scroll_position' ) ) {
    134 						// Restore the selection.
    135 						focusHTMLBookmarkInVisualEditor( editor );
    136 					}
    137 				} else {
    138 					tinymce.init( window.tinyMCEPreInit.mceInit[ id ] );
    139 				}
    140 
    141 				wrap.removeClass( 'html-active' ).addClass( 'tmce-active' );
    142 				$textarea.attr( 'aria-hidden', true );
    143 				window.setUserSetting( 'editor', 'tinymce' );
    144 
    145 			} else if ( 'html' === mode ) {
    146 				// If the editor is hidden (Quicktags is shown) we don't need to switch.
    147 				if ( editor && editor.isHidden() ) {
    148 					return false;
    149 				}
    150 
    151 				if ( editor ) {
    152 					// Don't resize the textarea in iOS.
    153 					// The iframe is forced to 100% height there, we shouldn't match it.
    154 					if ( ! tinymce.Env.iOS ) {
    155 						iframe = editor.iframeElement;
    156 						editorHeight = iframe ? parseInt( iframe.style.height, 10 ) : 0;
    157 
    158 						if ( editorHeight ) {
    159 							toolbarHeight = getToolbarHeight( editor );
    160 							editorHeight = editorHeight + toolbarHeight - 14;
    161 
    162 							// Sane limit for the textarea height.
    163 							if ( editorHeight > 50 && editorHeight < 5000 ) {
    164 								textarea.style.height = editorHeight + 'px';
    165 							}
    166 						}
    167 					}
    168 
    169 					var selectionRange = null;
    170 
    171 					if ( editor.getParam( 'wp_keep_scroll_position' ) ) {
    172 						selectionRange = findBookmarkedPosition( editor );
    173 					}
    174 
    175 					editor.hide();
    176 
    177 					if ( selectionRange ) {
    178 						selectTextInTextArea( editor, selectionRange );
    179 					}
    180 				} else {
    181 					// There is probably a JS error on the page.
    182 					// The TinyMCE editor instance doesn't exist. Show the textarea.
    183 					$textarea.css({ 'display': '', 'visibility': '' });
    184 				}
    185 
    186 				wrap.removeClass( 'tmce-active' ).addClass( 'html-active' );
    187 				$textarea.attr( 'aria-hidden', false );
    188 				window.setUserSetting( 'editor', 'html' );
    189 			}
    190 		}
    191 
    192 		/**
    193 		 * Checks if a cursor is inside an HTML tag or comment.
    194 		 *
    195 		 * In order to prevent breaking HTML tags when selecting text, the cursor
    196 		 * must be moved to either the start or end of the tag.
    197 		 *
    198 		 * This will prevent the selection marker to be inserted in the middle of an HTML tag.
    199 		 *
    200 		 * This function gives information whether the cursor is inside a tag or not, as well as
    201 		 * the tag type, if it is a closing tag and check if the HTML tag is inside a shortcode tag,
    202 		 * e.g. `[caption]<img.../>..`.
    203 		 *
    204 		 * @param {string} content The test content where the cursor is.
    205 		 * @param {number} cursorPosition The cursor position inside the content.
    206 		 *
    207 		 * @return {(null|Object)} Null if cursor is not in a tag, Object if the cursor is inside a tag.
    208 		 */
    209 		function getContainingTagInfo( content, cursorPosition ) {
    210 			var lastLtPos = content.lastIndexOf( '<', cursorPosition - 1 ),
    211 				lastGtPos = content.lastIndexOf( '>', cursorPosition );
    212 
    213 			if ( lastLtPos > lastGtPos || content.substr( cursorPosition, 1 ) === '>' ) {
    214 				// Find what the tag is.
    215 				var tagContent = content.substr( lastLtPos ),
    216 					tagMatch = tagContent.match( /<\s*(\/)?(\w+|\!-{2}.*-{2})/ );
    217 
    218 				if ( ! tagMatch ) {
    219 					return null;
    220 				}
    221 
    222 				var tagType = tagMatch[2],
    223 					closingGt = tagContent.indexOf( '>' );
    224 
    225 				return {
    226 					ltPos: lastLtPos,
    227 					gtPos: lastLtPos + closingGt + 1, // Offset by one to get the position _after_ the character.
    228 					tagType: tagType,
    229 					isClosingTag: !! tagMatch[1]
    230 				};
    231 			}
    232 			return null;
    233 		}
    234 
    235 		/**
    236 		 * Checks if the cursor is inside a shortcode
    237 		 *
    238 		 * If the cursor is inside a shortcode wrapping tag, e.g. `[caption]` it's better to
    239 		 * move the selection marker to before or after the shortcode.
    240 		 *
    241 		 * For example `[caption]` rewrites/removes anything that's between the `[caption]` tag and the
    242 		 * `<img/>` tag inside.
    243 		 *
    244 		 * `[caption]<span>ThisIsGone</span><img .../>[caption]`
    245 		 *
    246 		 * Moving the selection to before or after the short code is better, since it allows to select
    247 		 * something, instead of just losing focus and going to the start of the content.
    248 		 *
    249 		 * @param {string} content The text content to check against.
    250 		 * @param {number} cursorPosition    The cursor position to check.
    251 		 *
    252 		 * @return {(undefined|Object)} Undefined if the cursor is not wrapped in a shortcode tag.
    253 		 *                              Information about the wrapping shortcode tag if it's wrapped in one.
    254 		 */
    255 		function getShortcodeWrapperInfo( content, cursorPosition ) {
    256 			var contentShortcodes = getShortCodePositionsInText( content );
    257 
    258 			for ( var i = 0; i < contentShortcodes.length; i++ ) {
    259 				var element = contentShortcodes[ i ];
    260 
    261 				if ( cursorPosition >= element.startIndex && cursorPosition <= element.endIndex ) {
    262 					return element;
    263 				}
    264 			}
    265 		}
    266 
    267 		/**
    268 		 * Gets a list of unique shortcodes or shortcode-look-alikes in the content.
    269 		 *
    270 		 * @param {string} content The content we want to scan for shortcodes.
    271 		 */
    272 		function getShortcodesInText( content ) {
    273 			var shortcodes = content.match( /\[+([\w_-])+/g ),
    274 				result = [];
    275 
    276 			if ( shortcodes ) {
    277 				for ( var i = 0; i < shortcodes.length; i++ ) {
    278 					var shortcode = shortcodes[ i ].replace( /^\[+/g, '' );
    279 
    280 					if ( result.indexOf( shortcode ) === -1 ) {
    281 						result.push( shortcode );
    282 					}
    283 				}
    284 			}
    285 
    286 			return result;
    287 		}
    288 
    289 		/**
    290 		 * Gets all shortcodes and their positions in the content
    291 		 *
    292 		 * This function returns all the shortcodes that could be found in the textarea content
    293 		 * along with their character positions and boundaries.
    294 		 *
    295 		 * This is used to check if the selection cursor is inside the boundaries of a shortcode
    296 		 * and move it accordingly, to avoid breakage.
    297 		 *
    298 		 * @link adjustTextAreaSelectionCursors
    299 		 *
    300 		 * The information can also be used in other cases when we need to lookup shortcode data,
    301 		 * as it's already structured!
    302 		 *
    303 		 * @param {string} content The content we want to scan for shortcodes
    304 		 */
    305 		function getShortCodePositionsInText( content ) {
    306 			var allShortcodes = getShortcodesInText( content ), shortcodeInfo;
    307 
    308 			if ( allShortcodes.length === 0 ) {
    309 				return [];
    310 			}
    311 
    312 			var shortcodeDetailsRegexp = wp.shortcode.regexp( allShortcodes.join( '|' ) ),
    313 				shortcodeMatch, // Define local scope for the variable to be used in the loop below.
    314 				shortcodesDetails = [];
    315 
    316 			while ( shortcodeMatch = shortcodeDetailsRegexp.exec( content ) ) {
    317 				/**
    318 				 * Check if the shortcode should be shown as plain text.
    319 				 *
    320 				 * This corresponds to the [[shortcode]] syntax, which doesn't parse the shortcode
    321 				 * and just shows it as text.
    322 				 */
    323 				var showAsPlainText = shortcodeMatch[1] === '[';
    324 
    325 				shortcodeInfo = {
    326 					shortcodeName: shortcodeMatch[2],
    327 					showAsPlainText: showAsPlainText,
    328 					startIndex: shortcodeMatch.index,
    329 					endIndex: shortcodeMatch.index + shortcodeMatch[0].length,
    330 					length: shortcodeMatch[0].length
    331 				};
    332 
    333 				shortcodesDetails.push( shortcodeInfo );
    334 			}
    335 
    336 			/**
    337 			 * Get all URL matches, and treat them as embeds.
    338 			 *
    339 			 * Since there isn't a good way to detect if a URL by itself on a line is a previewable
    340 			 * object, it's best to treat all of them as such.
    341 			 *
    342 			 * This means that the selection will capture the whole URL, in a similar way shrotcodes
    343 			 * are treated.
    344 			 */
    345 			var urlRegexp = new RegExp(
    346 				'(^|[\\n\\r][\\n\\r]|<p>)(https?:\\/\\/[^\s"]+?)(<\\/p>\s*|[\\n\\r][\\n\\r]|$)', 'gi'
    347 			);
    348 
    349 			while ( shortcodeMatch = urlRegexp.exec( content ) ) {
    350 				shortcodeInfo = {
    351 					shortcodeName: 'url',
    352 					showAsPlainText: false,
    353 					startIndex: shortcodeMatch.index,
    354 					endIndex: shortcodeMatch.index + shortcodeMatch[ 0 ].length,
    355 					length: shortcodeMatch[ 0 ].length,
    356 					urlAtStartOfContent: shortcodeMatch[ 1 ] === '',
    357 					urlAtEndOfContent: shortcodeMatch[ 3 ] === ''
    358 				};
    359 
    360 				shortcodesDetails.push( shortcodeInfo );
    361 			}
    362 
    363 			return shortcodesDetails;
    364 		}
    365 
    366 		/**
    367 		 * Generate a cursor marker element to be inserted in the content.
    368 		 *
    369 		 * `span` seems to be the least destructive element that can be used.
    370 		 *
    371 		 * Using DomQuery syntax to create it, since it's used as both text and as a DOM element.
    372 		 *
    373 		 * @param {Object} domLib DOM library instance.
    374 		 * @param {string} content The content to insert into the cursor marker element.
    375 		 */
    376 		function getCursorMarkerSpan( domLib, content ) {
    377 			return domLib( '<span>' ).css( {
    378 						display: 'inline-block',
    379 						width: 0,
    380 						overflow: 'hidden',
    381 						'line-height': 0
    382 					} )
    383 					.html( content ? content : '' );
    384 		}
    385 
    386 		/**
    387 		 * Gets adjusted selection cursor positions according to HTML tags, comments, and shortcodes.
    388 		 *
    389 		 * Shortcodes and HTML codes are a bit of a special case when selecting, since they may render
    390 		 * content in Visual mode. If we insert selection markers somewhere inside them, it's really possible
    391 		 * to break the syntax and render the HTML tag or shortcode broken.
    392 		 *
    393 		 * @link getShortcodeWrapperInfo
    394 		 *
    395 		 * @param {string} content Textarea content that the cursors are in
    396 		 * @param {{cursorStart: number, cursorEnd: number}} cursorPositions Cursor start and end positions
    397 		 *
    398 		 * @return {{cursorStart: number, cursorEnd: number}}
    399 		 */
    400 		function adjustTextAreaSelectionCursors( content, cursorPositions ) {
    401 			var voidElements = [
    402 				'area', 'base', 'br', 'col', 'embed', 'hr', 'img', 'input',
    403 				'keygen', 'link', 'meta', 'param', 'source', 'track', 'wbr'
    404 			];
    405 
    406 			var cursorStart = cursorPositions.cursorStart,
    407 				cursorEnd = cursorPositions.cursorEnd,
    408 				// Check if the cursor is in a tag and if so, adjust it.
    409 				isCursorStartInTag = getContainingTagInfo( content, cursorStart );
    410 
    411 			if ( isCursorStartInTag ) {
    412 				/**
    413 				 * Only move to the start of the HTML tag (to select the whole element) if the tag
    414 				 * is part of the voidElements list above.
    415 				 *
    416 				 * This list includes tags that are self-contained and don't need a closing tag, according to the
    417 				 * HTML5 specification.
    418 				 *
    419 				 * This is done in order to make selection of text a bit more consistent when selecting text in
    420 				 * `<p>` tags or such.
    421 				 *
    422 				 * In cases where the tag is not a void element, the cursor is put to the end of the tag,
    423 				 * so it's either between the opening and closing tag elements or after the closing tag.
    424 				 */
    425 				if ( voidElements.indexOf( isCursorStartInTag.tagType ) !== -1 ) {
    426 					cursorStart = isCursorStartInTag.ltPos;
    427 				} else {
    428 					cursorStart = isCursorStartInTag.gtPos;
    429 				}
    430 			}
    431 
    432 			var isCursorEndInTag = getContainingTagInfo( content, cursorEnd );
    433 			if ( isCursorEndInTag ) {
    434 				cursorEnd = isCursorEndInTag.gtPos;
    435 			}
    436 
    437 			var isCursorStartInShortcode = getShortcodeWrapperInfo( content, cursorStart );
    438 			if ( isCursorStartInShortcode && ! isCursorStartInShortcode.showAsPlainText ) {
    439 				/**
    440 				 * If a URL is at the start or the end of the content,
    441 				 * the selection doesn't work, because it inserts a marker in the text,
    442 				 * which breaks the embedURL detection.
    443 				 *
    444 				 * The best way to avoid that and not modify the user content is to
    445 				 * adjust the cursor to either after or before URL.
    446 				 */
    447 				if ( isCursorStartInShortcode.urlAtStartOfContent ) {
    448 					cursorStart = isCursorStartInShortcode.endIndex;
    449 				} else {
    450 					cursorStart = isCursorStartInShortcode.startIndex;
    451 				}
    452 			}
    453 
    454 			var isCursorEndInShortcode = getShortcodeWrapperInfo( content, cursorEnd );
    455 			if ( isCursorEndInShortcode && ! isCursorEndInShortcode.showAsPlainText ) {
    456 				if ( isCursorEndInShortcode.urlAtEndOfContent ) {
    457 					cursorEnd = isCursorEndInShortcode.startIndex;
    458 				} else {
    459 					cursorEnd = isCursorEndInShortcode.endIndex;
    460 				}
    461 			}
    462 
    463 			return {
    464 				cursorStart: cursorStart,
    465 				cursorEnd: cursorEnd
    466 			};
    467 		}
    468 
    469 		/**
    470 		 * Adds text selection markers in the editor textarea.
    471 		 *
    472 		 * Adds selection markers in the content of the editor `textarea`.
    473 		 * The method directly manipulates the `textarea` content, to allow TinyMCE plugins
    474 		 * to run after the markers are added.
    475 		 *
    476 		 * @param {Object} $textarea TinyMCE's textarea wrapped as a DomQuery object
    477 		 */
    478 		function addHTMLBookmarkInTextAreaContent( $textarea ) {
    479 			if ( ! $textarea || ! $textarea.length ) {
    480 				// If no valid $textarea object is provided, there's nothing we can do.
    481 				return;
    482 			}
    483 
    484 			var textArea = $textarea[0],
    485 				textAreaContent = textArea.value,
    486 
    487 				adjustedCursorPositions = adjustTextAreaSelectionCursors( textAreaContent, {
    488 					cursorStart: textArea.selectionStart,
    489 					cursorEnd: textArea.selectionEnd
    490 				} ),
    491 
    492 				htmlModeCursorStartPosition = adjustedCursorPositions.cursorStart,
    493 				htmlModeCursorEndPosition = adjustedCursorPositions.cursorEnd,
    494 
    495 				mode = htmlModeCursorStartPosition !== htmlModeCursorEndPosition ? 'range' : 'single',
    496 
    497 				selectedText = null,
    498 				cursorMarkerSkeleton = getCursorMarkerSpan( $$, '&#65279;' ).attr( 'data-mce-type','bookmark' );
    499 
    500 			if ( mode === 'range' ) {
    501 				var markedText = textArea.value.slice( htmlModeCursorStartPosition, htmlModeCursorEndPosition ),
    502 					bookMarkEnd = cursorMarkerSkeleton.clone().addClass( 'mce_SELRES_end' );
    503 
    504 				selectedText = [
    505 					markedText,
    506 					bookMarkEnd[0].outerHTML
    507 				].join( '' );
    508 			}
    509 
    510 			textArea.value = [
    511 				textArea.value.slice( 0, htmlModeCursorStartPosition ), // Text until the cursor/selection position.
    512 				cursorMarkerSkeleton.clone()							// Cursor/selection start marker.
    513 					.addClass( 'mce_SELRES_start' )[0].outerHTML,
    514 				selectedText, 											// Selected text with end cursor/position marker.
    515 				textArea.value.slice( htmlModeCursorEndPosition )		// Text from last cursor/selection position to end.
    516 			].join( '' );
    517 		}
    518 
    519 		/**
    520 		 * Focuses the selection markers in Visual mode.
    521 		 *
    522 		 * The method checks for existing selection markers inside the editor DOM (Visual mode)
    523 		 * and create a selection between the two nodes using the DOM `createRange` selection API
    524 		 *
    525 		 * If there is only a single node, select only the single node through TinyMCE's selection API
    526 		 *
    527 		 * @param {Object} editor TinyMCE editor instance.
    528 		 */
    529 		function focusHTMLBookmarkInVisualEditor( editor ) {
    530 			var startNode = editor.$( '.mce_SELRES_start' ).attr( 'data-mce-bogus', 1 ),
    531 				endNode = editor.$( '.mce_SELRES_end' ).attr( 'data-mce-bogus', 1 );
    532 
    533 			if ( startNode.length ) {
    534 				editor.focus();
    535 
    536 				if ( ! endNode.length ) {
    537 					editor.selection.select( startNode[0] );
    538 				} else {
    539 					var selection = editor.getDoc().createRange();
    540 
    541 					selection.setStartAfter( startNode[0] );
    542 					selection.setEndBefore( endNode[0] );
    543 
    544 					editor.selection.setRng( selection );
    545 				}
    546 			}
    547 
    548 			if ( editor.getParam( 'wp_keep_scroll_position' ) ) {
    549 				scrollVisualModeToStartElement( editor, startNode );
    550 			}
    551 
    552 			removeSelectionMarker( startNode );
    553 			removeSelectionMarker( endNode );
    554 
    555 			editor.save();
    556 		}
    557 
    558 		/**
    559 		 * Removes selection marker and the parent node if it is an empty paragraph.
    560 		 *
    561 		 * By default TinyMCE wraps loose inline tags in a `<p>`.
    562 		 * When removing selection markers an empty `<p>` may be left behind, remove it.
    563 		 *
    564 		 * @param {Object} $marker The marker to be removed from the editor DOM, wrapped in an instnce of `editor.$`
    565 		 */
    566 		function removeSelectionMarker( $marker ) {
    567 			var $markerParent = $marker.parent();
    568 
    569 			$marker.remove();
    570 
    571 			//Remove empty paragraph left over after removing the marker.
    572 			if ( $markerParent.is( 'p' ) && ! $markerParent.children().length && ! $markerParent.text() ) {
    573 				$markerParent.remove();
    574 			}
    575 		}
    576 
    577 		/**
    578 		 * Scrolls the content to place the selected element in the center of the screen.
    579 		 *
    580 		 * Takes an element, that is usually the selection start element, selected in
    581 		 * `focusHTMLBookmarkInVisualEditor()` and scrolls the screen so the element appears roughly
    582 		 * in the middle of the screen.
    583 		 *
    584 		 * I order to achieve the proper positioning, the editor media bar and toolbar are subtracted
    585 		 * from the window height, to get the proper viewport window, that the user sees.
    586 		 *
    587 		 * @param {Object} editor TinyMCE editor instance.
    588 		 * @param {Object} element HTMLElement that should be scrolled into view.
    589 		 */
    590 		function scrollVisualModeToStartElement( editor, element ) {
    591 			var elementTop = editor.$( element ).offset().top,
    592 				TinyMCEContentAreaTop = editor.$( editor.getContentAreaContainer() ).offset().top,
    593 
    594 				toolbarHeight = getToolbarHeight( editor ),
    595 
    596 				edTools = $( '#wp-content-editor-tools' ),
    597 				edToolsHeight = 0,
    598 				edToolsOffsetTop = 0,
    599 
    600 				$scrollArea;
    601 
    602 			if ( edTools.length ) {
    603 				edToolsHeight = edTools.height();
    604 				edToolsOffsetTop = edTools.offset().top;
    605 			}
    606 
    607 			var windowHeight = window.innerHeight || document.documentElement.clientHeight || document.body.clientHeight,
    608 
    609 				selectionPosition = TinyMCEContentAreaTop + elementTop,
    610 				visibleAreaHeight = windowHeight - ( edToolsHeight + toolbarHeight );
    611 
    612 			// There's no need to scroll if the selection is inside the visible area.
    613 			if ( selectionPosition < visibleAreaHeight ) {
    614 				return;
    615 			}
    616 
    617 			/**
    618 			 * The minimum scroll height should be to the top of the editor, to offer a consistent
    619 			 * experience.
    620 			 *
    621 			 * In order to find the top of the editor, we calculate the offset of `#wp-content-editor-tools` and
    622 			 * subtracting the height. This gives the scroll position where the top of the editor tools aligns with
    623 			 * the top of the viewport (under the Master Bar)
    624 			 */
    625 			var adjustedScroll;
    626 			if ( editor.settings.wp_autoresize_on ) {
    627 				$scrollArea = $( 'html,body' );
    628 				adjustedScroll = Math.max( selectionPosition - visibleAreaHeight / 2, edToolsOffsetTop - edToolsHeight );
    629 			} else {
    630 				$scrollArea = $( editor.contentDocument ).find( 'html,body' );
    631 				adjustedScroll = elementTop;
    632 			}
    633 
    634 			$scrollArea.animate( {
    635 				scrollTop: parseInt( adjustedScroll, 10 )
    636 			}, 100 );
    637 		}
    638 
    639 		/**
    640 		 * This method was extracted from the `SaveContent` hook in
    641 		 * `wp-includes/js/tinymce/plugins/wordpress/plugin.js`.
    642 		 *
    643 		 * It's needed here, since the method changes the content a bit, which confuses the cursor position.
    644 		 *
    645 		 * @param {Object} event TinyMCE event object.
    646 		 */
    647 		function fixTextAreaContent( event ) {
    648 			// Keep empty paragraphs :(
    649 			event.content = event.content.replace( /<p>(?:<br ?\/?>|\u00a0|\uFEFF| )*<\/p>/g, '<p>&nbsp;</p>' );
    650 		}
    651 
    652 		/**
    653 		 * Finds the current selection position in the Visual editor.
    654 		 *
    655 		 * Find the current selection in the Visual editor by inserting marker elements at the start
    656 		 * and end of the selection.
    657 		 *
    658 		 * Uses the standard DOM selection API to achieve that goal.
    659 		 *
    660 		 * Check the notes in the comments in the code below for more information on some gotchas
    661 		 * and why this solution was chosen.
    662 		 *
    663 		 * @param {Object} editor The editor where we must find the selection.
    664 		 * @return {(null|Object)} The selection range position in the editor.
    665 		 */
    666 		function findBookmarkedPosition( editor ) {
    667 			// Get the TinyMCE `window` reference, since we need to access the raw selection.
    668 			var TinyMCEWindow = editor.getWin(),
    669 				selection = TinyMCEWindow.getSelection();
    670 
    671 			if ( ! selection || selection.rangeCount < 1 ) {
    672 				// no selection, no need to continue.
    673 				return;
    674 			}
    675 
    676 			/**
    677 			 * The ID is used to avoid replacing user generated content, that may coincide with the
    678 			 * format specified below.
    679 			 * @type {string}
    680 			 */
    681 			var selectionID = 'SELRES_' + Math.random();
    682 
    683 			/**
    684 			 * Create two marker elements that will be used to mark the start and the end of the range.
    685 			 *
    686 			 * The elements have hardcoded style that makes them invisible. This is done to avoid seeing
    687 			 * random content flickering in the editor when switching between modes.
    688 			 */
    689 			var spanSkeleton = getCursorMarkerSpan( editor.$, selectionID ),
    690 				startElement = spanSkeleton.clone().addClass( 'mce_SELRES_start' ),
    691 				endElement = spanSkeleton.clone().addClass( 'mce_SELRES_end' );
    692 
    693 			/**
    694 			 * Inspired by:
    695 			 * @link https://stackoverflow.com/a/17497803/153310
    696 			 *
    697 			 * Why do it this way and not with TinyMCE's bookmarks?
    698 			 *
    699 			 * TinyMCE's bookmarks are very nice when working with selections and positions, BUT
    700 			 * there is no way to determine the precise position of the bookmark when switching modes, since
    701 			 * TinyMCE does some serialization of the content, to fix things like shortcodes, run plugins, prettify
    702 			 * HTML code and so on. In this process, the bookmark markup gets lost.
    703 			 *
    704 			 * If we decide to hook right after the bookmark is added, we can see where the bookmark is in the raw HTML
    705 			 * in TinyMCE. Unfortunately this state is before the serialization, so any visual markup in the content will
    706 			 * throw off the positioning.
    707 			 *
    708 			 * To avoid this, we insert two custom `span`s that will serve as the markers at the beginning and end of the
    709 			 * selection.
    710 			 *
    711 			 * Why not use TinyMCE's selection API or the DOM API to wrap the contents? Because if we do that, this creates
    712 			 * a new node, which is inserted in the dom. Now this will be fine, if we worked with fixed selections to
    713 			 * full nodes. Unfortunately in our case, the user can select whatever they like, which means that the
    714 			 * selection may start in the middle of one node and end in the middle of a completely different one. If we
    715 			 * wrap the selection in another node, this will create artifacts in the content.
    716 			 *
    717 			 * Using the method below, we insert the custom `span` nodes at the start and at the end of the selection.
    718 			 * This helps us not break the content and also gives us the option to work with multi-node selections without
    719 			 * breaking the markup.
    720 			 */
    721 			var range = selection.getRangeAt( 0 ),
    722 				startNode = range.startContainer,
    723 				startOffset = range.startOffset,
    724 				boundaryRange = range.cloneRange();
    725 
    726 			/**
    727 			 * If the selection is on a shortcode with Live View, TinyMCE creates a bogus markup,
    728 			 * which we have to account for.
    729 			 */
    730 			if ( editor.$( startNode ).parents( '.mce-offscreen-selection' ).length > 0 ) {
    731 				startNode = editor.$( '[data-mce-selected]' )[0];
    732 
    733 				/**
    734 				 * Marking the start and end element with `data-mce-object-selection` helps
    735 				 * discern when the selected object is a Live Preview selection.
    736 				 *
    737 				 * This way we can adjust the selection to properly select only the content, ignoring
    738 				 * whitespace inserted around the selected object by the Editor.
    739 				 */
    740 				startElement.attr( 'data-mce-object-selection', 'true' );
    741 				endElement.attr( 'data-mce-object-selection', 'true' );
    742 
    743 				editor.$( startNode ).before( startElement[0] );
    744 				editor.$( startNode ).after( endElement[0] );
    745 			} else {
    746 				boundaryRange.collapse( false );
    747 				boundaryRange.insertNode( endElement[0] );
    748 
    749 				boundaryRange.setStart( startNode, startOffset );
    750 				boundaryRange.collapse( true );
    751 				boundaryRange.insertNode( startElement[0] );
    752 
    753 				range.setStartAfter( startElement[0] );
    754 				range.setEndBefore( endElement[0] );
    755 				selection.removeAllRanges();
    756 				selection.addRange( range );
    757 			}
    758 
    759 			/**
    760 			 * Now the editor's content has the start/end nodes.
    761 			 *
    762 			 * Unfortunately the content goes through some more changes after this step, before it gets inserted
    763 			 * in the `textarea`. This means that we have to do some minor cleanup on our own here.
    764 			 */
    765 			editor.on( 'GetContent', fixTextAreaContent );
    766 
    767 			var content = removep( editor.getContent() );
    768 
    769 			editor.off( 'GetContent', fixTextAreaContent );
    770 
    771 			startElement.remove();
    772 			endElement.remove();
    773 
    774 			var startRegex = new RegExp(
    775 				'<span[^>]*\\s*class="mce_SELRES_start"[^>]+>\\s*' + selectionID + '[^<]*<\\/span>(\\s*)'
    776 			);
    777 
    778 			var endRegex = new RegExp(
    779 				'(\\s*)<span[^>]*\\s*class="mce_SELRES_end"[^>]+>\\s*' + selectionID + '[^<]*<\\/span>'
    780 			);
    781 
    782 			var startMatch = content.match( startRegex ),
    783 				endMatch = content.match( endRegex );
    784 
    785 			if ( ! startMatch ) {
    786 				return null;
    787 			}
    788 
    789 			var startIndex = startMatch.index,
    790 				startMatchLength = startMatch[0].length,
    791 				endIndex = null;
    792 
    793 			if (endMatch) {
    794 				/**
    795 				 * Adjust the selection index, if the selection contains a Live Preview object or not.
    796 				 *
    797 				 * Check where the `data-mce-object-selection` attribute is set above for more context.
    798 				 */
    799 				if ( startMatch[0].indexOf( 'data-mce-object-selection' ) !== -1 ) {
    800 					startMatchLength -= startMatch[1].length;
    801 				}
    802 
    803 				var endMatchIndex = endMatch.index;
    804 
    805 				if ( endMatch[0].indexOf( 'data-mce-object-selection' ) !== -1 ) {
    806 					endMatchIndex -= endMatch[1].length;
    807 				}
    808 
    809 				// We need to adjust the end position to discard the length of the range start marker.
    810 				endIndex = endMatchIndex - startMatchLength;
    811 			}
    812 
    813 			return {
    814 				start: startIndex,
    815 				end: endIndex
    816 			};
    817 		}
    818 
    819 		/**
    820 		 * Selects text in the TinyMCE `textarea`.
    821 		 *
    822 		 * Selects the text in TinyMCE's textarea that's between `selection.start` and `selection.end`.
    823 		 *
    824 		 * For `selection` parameter:
    825 		 * @link findBookmarkedPosition
    826 		 *
    827 		 * @param {Object} editor TinyMCE's editor instance.
    828 		 * @param {Object} selection Selection data.
    829 		 */
    830 		function selectTextInTextArea( editor, selection ) {
    831 			// Only valid in the text area mode and if we have selection.
    832 			if ( ! selection ) {
    833 				return;
    834 			}
    835 
    836 			var textArea = editor.getElement(),
    837 				start = selection.start,
    838 				end = selection.end || selection.start;
    839 
    840 			if ( textArea.focus ) {
    841 				// Wait for the Visual editor to be hidden, then focus and scroll to the position.
    842 				setTimeout( function() {
    843 					textArea.setSelectionRange( start, end );
    844 					if ( textArea.blur ) {
    845 						// Defocus before focusing.
    846 						textArea.blur();
    847 					}
    848 					textArea.focus();
    849 				}, 100 );
    850 			}
    851 		}
    852 
    853 		// Restore the selection when the editor is initialized. Needed when the Text editor is the default.
    854 		$( document ).on( 'tinymce-editor-init.keep-scroll-position', function( event, editor ) {
    855 			if ( editor.$( '.mce_SELRES_start' ).length ) {
    856 				focusHTMLBookmarkInVisualEditor( editor );
    857 			}
    858 		} );
    859 
    860 		/**
    861 		 * Replaces <p> tags with two line breaks. "Opposite" of wpautop().
    862 		 *
    863 		 * Replaces <p> tags with two line breaks except where the <p> has attributes.
    864 		 * Unifies whitespace.
    865 		 * Indents <li>, <dt> and <dd> for better readability.
    866 		 *
    867 		 * @since 2.5.0
    868 		 *
    869 		 * @memberof switchEditors
    870 		 *
    871 		 * @param {string} html The content from the editor.
    872 		 * @return {string} The content with stripped paragraph tags.
    873 		 */
    874 		function removep( html ) {
    875 			var blocklist = 'blockquote|ul|ol|li|dl|dt|dd|table|thead|tbody|tfoot|tr|th|td|h[1-6]|fieldset|figure',
    876 				blocklist1 = blocklist + '|div|p',
    877 				blocklist2 = blocklist + '|pre',
    878 				preserve_linebreaks = false,
    879 				preserve_br = false,
    880 				preserve = [];
    881 
    882 			if ( ! html ) {
    883 				return '';
    884 			}
    885 
    886 			// Protect script and style tags.
    887 			if ( html.indexOf( '<script' ) !== -1 || html.indexOf( '<style' ) !== -1 ) {
    888 				html = html.replace( /<(script|style)[^>]*>[\s\S]*?<\/\1>/g, function( match ) {
    889 					preserve.push( match );
    890 					return '<wp-preserve>';
    891 				} );
    892 			}
    893 
    894 			// Protect pre tags.
    895 			if ( html.indexOf( '<pre' ) !== -1 ) {
    896 				preserve_linebreaks = true;
    897 				html = html.replace( /<pre[^>]*>[\s\S]+?<\/pre>/g, function( a ) {
    898 					a = a.replace( /<br ?\/?>(\r\n|\n)?/g, '<wp-line-break>' );
    899 					a = a.replace( /<\/?p( [^>]*)?>(\r\n|\n)?/g, '<wp-line-break>' );
    900 					return a.replace( /\r?\n/g, '<wp-line-break>' );
    901 				});
    902 			}
    903 
    904 			// Remove line breaks but keep <br> tags inside image captions.
    905 			if ( html.indexOf( '[caption' ) !== -1 ) {
    906 				preserve_br = true;
    907 				html = html.replace( /\[caption[\s\S]+?\[\/caption\]/g, function( a ) {
    908 					return a.replace( /<br([^>]*)>/g, '<wp-temp-br$1>' ).replace( /[\r\n\t]+/, '' );
    909 				});
    910 			}
    911 
    912 			// Normalize white space characters before and after block tags.
    913 			html = html.replace( new RegExp( '\\s*</(' + blocklist1 + ')>\\s*', 'g' ), '</$1>\n' );
    914 			html = html.replace( new RegExp( '\\s*<((?:' + blocklist1 + ')(?: [^>]*)?)>', 'g' ), '\n<$1>' );
    915 
    916 			// Mark </p> if it has any attributes.
    917 			html = html.replace( /(<p [^>]+>.*?)<\/p>/g, '$1</p#>' );
    918 
    919 			// Preserve the first <p> inside a <div>.
    920 			html = html.replace( /<div( [^>]*)?>\s*<p>/gi, '<div$1>\n\n' );
    921 
    922 			// Remove paragraph tags.
    923 			html = html.replace( /\s*<p>/gi, '' );
    924 			html = html.replace( /\s*<\/p>\s*/gi, '\n\n' );
    925 
    926 			// Normalize white space chars and remove multiple line breaks.
    927 			html = html.replace( /\n[\s\u00a0]+\n/g, '\n\n' );
    928 
    929 			// Replace <br> tags with line breaks.
    930 			html = html.replace( /(\s*)<br ?\/?>\s*/gi, function( match, space ) {
    931 				if ( space && space.indexOf( '\n' ) !== -1 ) {
    932 					return '\n\n';
    933 				}
    934 
    935 				return '\n';
    936 			});
    937 
    938 			// Fix line breaks around <div>.
    939 			html = html.replace( /\s*<div/g, '\n<div' );
    940 			html = html.replace( /<\/div>\s*/g, '</div>\n' );
    941 
    942 			// Fix line breaks around caption shortcodes.
    943 			html = html.replace( /\s*\[caption([^\[]+)\[\/caption\]\s*/gi, '\n\n[caption$1[/caption]\n\n' );
    944 			html = html.replace( /caption\]\n\n+\[caption/g, 'caption]\n\n[caption' );
    945 
    946 			// Pad block elements tags with a line break.
    947 			html = html.replace( new RegExp('\\s*<((?:' + blocklist2 + ')(?: [^>]*)?)\\s*>', 'g' ), '\n<$1>' );
    948 			html = html.replace( new RegExp('\\s*</(' + blocklist2 + ')>\\s*', 'g' ), '</$1>\n' );
    949 
    950 			// Indent <li>, <dt> and <dd> tags.
    951 			html = html.replace( /<((li|dt|dd)[^>]*)>/g, ' \t<$1>' );
    952 
    953 			// Fix line breaks around <select> and <option>.
    954 			if ( html.indexOf( '<option' ) !== -1 ) {
    955 				html = html.replace( /\s*<option/g, '\n<option' );
    956 				html = html.replace( /\s*<\/select>/g, '\n</select>' );
    957 			}
    958 
    959 			// Pad <hr> with two line breaks.
    960 			if ( html.indexOf( '<hr' ) !== -1 ) {
    961 				html = html.replace( /\s*<hr( [^>]*)?>\s*/g, '\n\n<hr$1>\n\n' );
    962 			}
    963 
    964 			// Remove line breaks in <object> tags.
    965 			if ( html.indexOf( '<object' ) !== -1 ) {
    966 				html = html.replace( /<object[\s\S]+?<\/object>/g, function( a ) {
    967 					return a.replace( /[\r\n]+/g, '' );
    968 				});
    969 			}
    970 
    971 			// Unmark special paragraph closing tags.
    972 			html = html.replace( /<\/p#>/g, '</p>\n' );
    973 
    974 			// Pad remaining <p> tags whit a line break.
    975 			html = html.replace( /\s*(<p [^>]+>[\s\S]*?<\/p>)/g, '\n$1' );
    976 
    977 			// Trim.
    978 			html = html.replace( /^\s+/, '' );
    979 			html = html.replace( /[\s\u00a0]+$/, '' );
    980 
    981 			if ( preserve_linebreaks ) {
    982 				html = html.replace( /<wp-line-break>/g, '\n' );
    983 			}
    984 
    985 			if ( preserve_br ) {
    986 				html = html.replace( /<wp-temp-br([^>]*)>/g, '<br$1>' );
    987 			}
    988 
    989 			// Restore preserved tags.
    990 			if ( preserve.length ) {
    991 				html = html.replace( /<wp-preserve>/g, function() {
    992 					return preserve.shift();
    993 				} );
    994 			}
    995 
    996 			return html;
    997 		}
    998 
    999 		/**
   1000 		 * Replaces two line breaks with a paragraph tag and one line break with a <br>.
   1001 		 *
   1002 		 * Similar to `wpautop()` in formatting.php.
   1003 		 *
   1004 		 * @since 2.5.0
   1005 		 *
   1006 		 * @memberof switchEditors
   1007 		 *
   1008 		 * @param {string} text The text input.
   1009 		 * @return {string} The formatted text.
   1010 		 */
   1011 		function autop( text ) {
   1012 			var preserve_linebreaks = false,
   1013 				preserve_br = false,
   1014 				blocklist = 'table|thead|tfoot|caption|col|colgroup|tbody|tr|td|th|div|dl|dd|dt|ul|ol|li|pre' +
   1015 					'|form|map|area|blockquote|address|math|style|p|h[1-6]|hr|fieldset|legend|section' +
   1016 					'|article|aside|hgroup|header|footer|nav|figure|figcaption|details|menu|summary';
   1017 
   1018 			// Normalize line breaks.
   1019 			text = text.replace( /\r\n|\r/g, '\n' );
   1020 
   1021 			// Remove line breaks from <object>.
   1022 			if ( text.indexOf( '<object' ) !== -1 ) {
   1023 				text = text.replace( /<object[\s\S]+?<\/object>/g, function( a ) {
   1024 					return a.replace( /\n+/g, '' );
   1025 				});
   1026 			}
   1027 
   1028 			// Remove line breaks from tags.
   1029 			text = text.replace( /<[^<>]+>/g, function( a ) {
   1030 				return a.replace( /[\n\t ]+/g, ' ' );
   1031 			});
   1032 
   1033 			// Preserve line breaks in <pre> and <script> tags.
   1034 			if ( text.indexOf( '<pre' ) !== -1 || text.indexOf( '<script' ) !== -1 ) {
   1035 				preserve_linebreaks = true;
   1036 				text = text.replace( /<(pre|script)[^>]*>[\s\S]*?<\/\1>/g, function( a ) {
   1037 					return a.replace( /\n/g, '<wp-line-break>' );
   1038 				});
   1039 			}
   1040 
   1041 			if ( text.indexOf( '<figcaption' ) !== -1 ) {
   1042 				text = text.replace( /\s*(<figcaption[^>]*>)/g, '$1' );
   1043 				text = text.replace( /<\/figcaption>\s*/g, '</figcaption>' );
   1044 			}
   1045 
   1046 			// Keep <br> tags inside captions.
   1047 			if ( text.indexOf( '[caption' ) !== -1 ) {
   1048 				preserve_br = true;
   1049 
   1050 				text = text.replace( /\[caption[\s\S]+?\[\/caption\]/g, function( a ) {
   1051 					a = a.replace( /<br([^>]*)>/g, '<wp-temp-br$1>' );
   1052 
   1053 					a = a.replace( /<[^<>]+>/g, function( b ) {
   1054 						return b.replace( /[\n\t ]+/, ' ' );
   1055 					});
   1056 
   1057 					return a.replace( /\s*\n\s*/g, '<wp-temp-br />' );
   1058 				});
   1059 			}
   1060 
   1061 			text = text + '\n\n';
   1062 			text = text.replace( /<br \/>\s*<br \/>/gi, '\n\n' );
   1063 
   1064 			// Pad block tags with two line breaks.
   1065 			text = text.replace( new RegExp( '(<(?:' + blocklist + ')(?: [^>]*)?>)', 'gi' ), '\n\n$1' );
   1066 			text = text.replace( new RegExp( '(</(?:' + blocklist + ')>)', 'gi' ), '$1\n\n' );
   1067 			text = text.replace( /<hr( [^>]*)?>/gi, '<hr$1>\n\n' );
   1068 
   1069 			// Remove white space chars around <option>.
   1070 			text = text.replace( /\s*<option/gi, '<option' );
   1071 			text = text.replace( /<\/option>\s*/gi, '</option>' );
   1072 
   1073 			// Normalize multiple line breaks and white space chars.
   1074 			text = text.replace( /\n\s*\n+/g, '\n\n' );
   1075 
   1076 			// Convert two line breaks to a paragraph.
   1077 			text = text.replace( /([\s\S]+?)\n\n/g, '<p>$1</p>\n' );
   1078 
   1079 			// Remove empty paragraphs.
   1080 			text = text.replace( /<p>\s*?<\/p>/gi, '');
   1081 
   1082 			// Remove <p> tags that are around block tags.
   1083 			text = text.replace( new RegExp( '<p>\\s*(</?(?:' + blocklist + ')(?: [^>]*)?>)\\s*</p>', 'gi' ), '$1' );
   1084 			text = text.replace( /<p>(<li.+?)<\/p>/gi, '$1');
   1085 
   1086 			// Fix <p> in blockquotes.
   1087 			text = text.replace( /<p>\s*<blockquote([^>]*)>/gi, '<blockquote$1><p>');
   1088 			text = text.replace( /<\/blockquote>\s*<\/p>/gi, '</p></blockquote>');
   1089 
   1090 			// Remove <p> tags that are wrapped around block tags.
   1091 			text = text.replace( new RegExp( '<p>\\s*(</?(?:' + blocklist + ')(?: [^>]*)?>)', 'gi' ), '$1' );
   1092 			text = text.replace( new RegExp( '(</?(?:' + blocklist + ')(?: [^>]*)?>)\\s*</p>', 'gi' ), '$1' );
   1093 
   1094 			text = text.replace( /(<br[^>]*>)\s*\n/gi, '$1' );
   1095 
   1096 			// Add <br> tags.
   1097 			text = text.replace( /\s*\n/g, '<br />\n');
   1098 
   1099 			// Remove <br> tags that are around block tags.
   1100 			text = text.replace( new RegExp( '(</?(?:' + blocklist + ')[^>]*>)\\s*<br />', 'gi' ), '$1' );
   1101 			text = text.replace( /<br \/>(\s*<\/?(?:p|li|div|dl|dd|dt|th|pre|td|ul|ol)>)/gi, '$1' );
   1102 
   1103 			// Remove <p> and <br> around captions.
   1104 			text = text.replace( /(?:<p>|<br ?\/?>)*\s*\[caption([^\[]+)\[\/caption\]\s*(?:<\/p>|<br ?\/?>)*/gi, '[caption$1[/caption]' );
   1105 
   1106 			// Make sure there is <p> when there is </p> inside block tags that can contain other blocks.
   1107 			text = text.replace( /(<(?:div|th|td|form|fieldset|dd)[^>]*>)(.*?)<\/p>/g, function( a, b, c ) {
   1108 				if ( c.match( /<p( [^>]*)?>/ ) ) {
   1109 					return a;
   1110 				}
   1111 
   1112 				return b + '<p>' + c + '</p>';
   1113 			});
   1114 
   1115 			// Restore the line breaks in <pre> and <script> tags.
   1116 			if ( preserve_linebreaks ) {
   1117 				text = text.replace( /<wp-line-break>/g, '\n' );
   1118 			}
   1119 
   1120 			// Restore the <br> tags in captions.
   1121 			if ( preserve_br ) {
   1122 				text = text.replace( /<wp-temp-br([^>]*)>/g, '<br$1>' );
   1123 			}
   1124 
   1125 			return text;
   1126 		}
   1127 
   1128 		/**
   1129 		 * Fires custom jQuery events `beforePreWpautop` and `afterPreWpautop` when jQuery is available.
   1130 		 *
   1131 		 * @since 2.9.0
   1132 		 *
   1133 		 * @memberof switchEditors
   1134 		 *
   1135 		 * @param {string} html The content from the visual editor.
   1136 		 * @return {string} the filtered content.
   1137 		 */
   1138 		function pre_wpautop( html ) {
   1139 			var obj = { o: exports, data: html, unfiltered: html };
   1140 
   1141 			if ( $ ) {
   1142 				$( 'body' ).trigger( 'beforePreWpautop', [ obj ] );
   1143 			}
   1144 
   1145 			obj.data = removep( obj.data );
   1146 
   1147 			if ( $ ) {
   1148 				$( 'body' ).trigger( 'afterPreWpautop', [ obj ] );
   1149 			}
   1150 
   1151 			return obj.data;
   1152 		}
   1153 
   1154 		/**
   1155 		 * Fires custom jQuery events `beforeWpautop` and `afterWpautop` when jQuery is available.
   1156 		 *
   1157 		 * @since 2.9.0
   1158 		 *
   1159 		 * @memberof switchEditors
   1160 		 *
   1161 		 * @param {string} text The content from the text editor.
   1162 		 * @return {string} filtered content.
   1163 		 */
   1164 		function wpautop( text ) {
   1165 			var obj = { o: exports, data: text, unfiltered: text };
   1166 
   1167 			if ( $ ) {
   1168 				$( 'body' ).trigger( 'beforeWpautop', [ obj ] );
   1169 			}
   1170 
   1171 			obj.data = autop( obj.data );
   1172 
   1173 			if ( $ ) {
   1174 				$( 'body' ).trigger( 'afterWpautop', [ obj ] );
   1175 			}
   1176 
   1177 			return obj.data;
   1178 		}
   1179 
   1180 		if ( $ ) {
   1181 			$( init );
   1182 		} else if ( document.addEventListener ) {
   1183 			document.addEventListener( 'DOMContentLoaded', init, false );
   1184 			window.addEventListener( 'load', init, false );
   1185 		} else if ( window.attachEvent ) {
   1186 			window.attachEvent( 'onload', init );
   1187 			document.attachEvent( 'onreadystatechange', function() {
   1188 				if ( 'complete' === document.readyState ) {
   1189 					init();
   1190 				}
   1191 			} );
   1192 		}
   1193 
   1194 		wp.editor.autop = wpautop;
   1195 		wp.editor.removep = pre_wpautop;
   1196 
   1197 		exports = {
   1198 			go: switchEditor,
   1199 			wpautop: wpautop,
   1200 			pre_wpautop: pre_wpautop,
   1201 			_wp_Autop: autop,
   1202 			_wp_Nop: removep
   1203 		};
   1204 
   1205 		return exports;
   1206 	}
   1207 
   1208 	/**
   1209 	 * Expose the switch editors to be used globally.
   1210 	 *
   1211 	 * @namespace switchEditors
   1212 	 */
   1213 	window.switchEditors = new SwitchEditors();
   1214 
   1215 	/**
   1216 	 * Initialize TinyMCE and/or Quicktags. For use with wp_enqueue_editor() (PHP).
   1217 	 *
   1218 	 * Intended for use with an existing textarea that will become the Text editor tab.
   1219 	 * The editor width will be the width of the textarea container, height will be adjustable.
   1220 	 *
   1221 	 * Settings for both TinyMCE and Quicktags can be passed on initialization, and are "filtered"
   1222 	 * with custom jQuery events on the document element, wp-before-tinymce-init and wp-before-quicktags-init.
   1223 	 *
   1224 	 * @since 4.8.0
   1225 	 *
   1226 	 * @param {string} id The HTML id of the textarea that is used for the editor.
   1227 	 *                    Has to be jQuery compliant. No brackets, special chars, etc.
   1228 	 * @param {Object} settings Example:
   1229 	 * settings = {
   1230 	 *    // See https://www.tinymce.com/docs/configure/integration-and-setup/.
   1231 	 *    // Alternatively set to `true` to use the defaults.
   1232 	 *    tinymce: {
   1233 	 *        setup: function( editor ) {
   1234 	 *            console.log( 'Editor initialized', editor );
   1235 	 *        }
   1236 	 *    }
   1237 	 *
   1238 	 *    // Alternatively set to `true` to use the defaults.
   1239 	 *	  quicktags: {
   1240 	 *        buttons: 'strong,em,link'
   1241 	 *    }
   1242 	 * }
   1243 	 */
   1244 	wp.editor.initialize = function( id, settings ) {
   1245 		var init;
   1246 		var defaults;
   1247 
   1248 		if ( ! $ || ! id || ! wp.editor.getDefaultSettings ) {
   1249 			return;
   1250 		}
   1251 
   1252 		defaults = wp.editor.getDefaultSettings();
   1253 
   1254 		// Initialize TinyMCE by default.
   1255 		if ( ! settings ) {
   1256 			settings = {
   1257 				tinymce: true
   1258 			};
   1259 		}
   1260 
   1261 		// Add wrap and the Visual|Text tabs.
   1262 		if ( settings.tinymce && settings.quicktags ) {
   1263 			var $textarea = $( '#' + id );
   1264 
   1265 			var $wrap = $( '<div>' ).attr( {
   1266 					'class': 'wp-core-ui wp-editor-wrap tmce-active',
   1267 					id: 'wp-' + id + '-wrap'
   1268 				} );
   1269 
   1270 			var $editorContainer = $( '<div class="wp-editor-container">' );
   1271 
   1272 			var $button = $( '<button>' ).attr( {
   1273 					type: 'button',
   1274 					'data-wp-editor-id': id
   1275 				} );
   1276 
   1277 			var $editorTools = $( '<div class="wp-editor-tools">' );
   1278 
   1279 			if ( settings.mediaButtons ) {
   1280 				var buttonText = 'Add Media';
   1281 
   1282 				if ( window._wpMediaViewsL10n && window._wpMediaViewsL10n.addMedia ) {
   1283 					buttonText = window._wpMediaViewsL10n.addMedia;
   1284 				}
   1285 
   1286 				var $addMediaButton = $( '<button type="button" class="button insert-media add_media">' );
   1287 
   1288 				$addMediaButton.append( '<span class="wp-media-buttons-icon"></span>' );
   1289 				$addMediaButton.append( document.createTextNode( ' ' + buttonText ) );
   1290 				$addMediaButton.data( 'editor', id );
   1291 
   1292 				$editorTools.append(
   1293 					$( '<div class="wp-media-buttons">' )
   1294 						.append( $addMediaButton )
   1295 				);
   1296 			}
   1297 
   1298 			$wrap.append(
   1299 				$editorTools
   1300 					.append( $( '<div class="wp-editor-tabs">' )
   1301 						.append( $button.clone().attr({
   1302 							id: id + '-tmce',
   1303 							'class': 'wp-switch-editor switch-tmce'
   1304 						}).text( window.tinymce.translate( 'Visual' ) ) )
   1305 						.append( $button.attr({
   1306 							id: id + '-html',
   1307 							'class': 'wp-switch-editor switch-html'
   1308 						}).text( window.tinymce.translate( 'Text' ) ) )
   1309 					).append( $editorContainer )
   1310 			);
   1311 
   1312 			$textarea.after( $wrap );
   1313 			$editorContainer.append( $textarea );
   1314 		}
   1315 
   1316 		if ( window.tinymce && settings.tinymce ) {
   1317 			if ( typeof settings.tinymce !== 'object' ) {
   1318 				settings.tinymce = {};
   1319 			}
   1320 
   1321 			init = $.extend( {}, defaults.tinymce, settings.tinymce );
   1322 			init.selector = '#' + id;
   1323 
   1324 			$( document ).trigger( 'wp-before-tinymce-init', init );
   1325 			window.tinymce.init( init );
   1326 
   1327 			if ( ! window.wpActiveEditor ) {
   1328 				window.wpActiveEditor = id;
   1329 			}
   1330 		}
   1331 
   1332 		if ( window.quicktags && settings.quicktags ) {
   1333 			if ( typeof settings.quicktags !== 'object' ) {
   1334 				settings.quicktags = {};
   1335 			}
   1336 
   1337 			init = $.extend( {}, defaults.quicktags, settings.quicktags );
   1338 			init.id = id;
   1339 
   1340 			$( document ).trigger( 'wp-before-quicktags-init', init );
   1341 			window.quicktags( init );
   1342 
   1343 			if ( ! window.wpActiveEditor ) {
   1344 				window.wpActiveEditor = init.id;
   1345 			}
   1346 		}
   1347 	};
   1348 
   1349 	/**
   1350 	 * Remove one editor instance.
   1351 	 *
   1352 	 * Intended for use with editors that were initialized with wp.editor.initialize().
   1353 	 *
   1354 	 * @since 4.8.0
   1355 	 *
   1356 	 * @param {string} id The HTML id of the editor textarea.
   1357 	 */
   1358 	wp.editor.remove = function( id ) {
   1359 		var mceInstance, qtInstance,
   1360 			$wrap = $( '#wp-' + id + '-wrap' );
   1361 
   1362 		if ( window.tinymce ) {
   1363 			mceInstance = window.tinymce.get( id );
   1364 
   1365 			if ( mceInstance ) {
   1366 				if ( ! mceInstance.isHidden() ) {
   1367 					mceInstance.save();
   1368 				}
   1369 
   1370 				mceInstance.remove();
   1371 			}
   1372 		}
   1373 
   1374 		if ( window.quicktags ) {
   1375 			qtInstance = window.QTags.getInstance( id );
   1376 
   1377 			if ( qtInstance ) {
   1378 				qtInstance.remove();
   1379 			}
   1380 		}
   1381 
   1382 		if ( $wrap.length ) {
   1383 			$wrap.after( $( '#' + id ) );
   1384 			$wrap.remove();
   1385 		}
   1386 	};
   1387 
   1388 	/**
   1389 	 * Get the editor content.
   1390 	 *
   1391 	 * Intended for use with editors that were initialized with wp.editor.initialize().
   1392 	 *
   1393 	 * @since 4.8.0
   1394 	 *
   1395 	 * @param {string} id The HTML id of the editor textarea.
   1396 	 * @return The editor content.
   1397 	 */
   1398 	wp.editor.getContent = function( id ) {
   1399 		var editor;
   1400 
   1401 		if ( ! $ || ! id ) {
   1402 			return;
   1403 		}
   1404 
   1405 		if ( window.tinymce ) {
   1406 			editor = window.tinymce.get( id );
   1407 
   1408 			if ( editor && ! editor.isHidden() ) {
   1409 				editor.save();
   1410 			}
   1411 		}
   1412 
   1413 		return $( '#' + id ).val();
   1414 	};
   1415 
   1416 }( window.jQuery, window.wp ));