Editor:
- Fix keeping text selection and scroll position when there are embeds from URL. - Add editor setting to disable keeping selection and scroll position. - Remove dependency on Underscore.js. - Fix error in the Text widget editor. Props biskobe. Fixes #42059, see #40854. git-svn-id: https://develop.svn.wordpress.org/trunk@41783 602fd350-edb4-49c9-b593-d223f7449a82
This commit is contained in:
parent
59bd66aeec
commit
ef701b01e9
@ -628,6 +628,7 @@ if ( post_type_supports($post_type, 'editor') ) {
|
|||||||
'resize' => false,
|
'resize' => false,
|
||||||
'wp_autoresize_on' => $_wp_editor_expand,
|
'wp_autoresize_on' => $_wp_editor_expand,
|
||||||
'add_unload_trigger' => false,
|
'add_unload_trigger' => false,
|
||||||
|
'wp_keep_scroll_position' => true,
|
||||||
),
|
),
|
||||||
) ); ?>
|
) ); ?>
|
||||||
<table id="post-status-info"><tbody><tr>
|
<table id="post-status-info"><tbody><tr>
|
||||||
|
@ -99,8 +99,18 @@ window.wp = window.wp || {};
|
|||||||
|
|
||||||
editorHeight = parseInt( textarea.style.height, 10 ) || 0;
|
editorHeight = parseInt( textarea.style.height, 10 ) || 0;
|
||||||
|
|
||||||
// Save the selection
|
var keepSelection = false;
|
||||||
addHTMLBookmarkInTextAreaContent( $textarea, $ );
|
if ( editor ) {
|
||||||
|
keepSelection = editor.getParam( 'wp_keep_scroll_position' )
|
||||||
|
} else {
|
||||||
|
keepSelection = window.tinyMCEPreInit.mceInit[ id ] &&
|
||||||
|
window.tinyMCEPreInit.mceInit[ id ]['wp_keep_scroll_position']
|
||||||
|
}
|
||||||
|
|
||||||
|
if ( keepSelection ) {
|
||||||
|
// Save the selection
|
||||||
|
addHTMLBookmarkInTextAreaContent( $textarea );
|
||||||
|
}
|
||||||
|
|
||||||
if ( editor ) {
|
if ( editor ) {
|
||||||
editor.show();
|
editor.show();
|
||||||
@ -116,8 +126,10 @@ window.wp = window.wp || {};
|
|||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
// Restore the selection
|
if ( editor.getParam( 'wp_keep_scroll_position' ) ) {
|
||||||
focusHTMLBookmarkInVisualEditor( editor );
|
// Restore the selection
|
||||||
|
focusHTMLBookmarkInVisualEditor( editor );
|
||||||
|
}
|
||||||
} else {
|
} else {
|
||||||
tinymce.init( window.tinyMCEPreInit.mceInit[ id ] );
|
tinymce.init( window.tinyMCEPreInit.mceInit[ id ] );
|
||||||
}
|
}
|
||||||
@ -132,7 +144,6 @@ window.wp = window.wp || {};
|
|||||||
return false;
|
return false;
|
||||||
}
|
}
|
||||||
|
|
||||||
var selectionRange = null;
|
|
||||||
if ( editor ) {
|
if ( editor ) {
|
||||||
// Don't resize the textarea in iOS. The iframe is forced to 100% height there, we shouldn't match it.
|
// Don't resize the textarea in iOS. The iframe is forced to 100% height there, we shouldn't match it.
|
||||||
if ( ! tinymce.Env.iOS ) {
|
if ( ! tinymce.Env.iOS ) {
|
||||||
@ -150,7 +161,11 @@ window.wp = window.wp || {};
|
|||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
selectionRange = findBookmarkedPosition( editor );
|
var selectionRange = null;
|
||||||
|
|
||||||
|
if ( editor.getParam( 'wp_keep_scroll_position' ) ) {
|
||||||
|
selectionRange = findBookmarkedPosition( editor );
|
||||||
|
}
|
||||||
|
|
||||||
editor.hide();
|
editor.hide();
|
||||||
|
|
||||||
@ -234,9 +249,13 @@ window.wp = window.wp || {};
|
|||||||
function getShortcodeWrapperInfo( content, cursorPosition ) {
|
function getShortcodeWrapperInfo( content, cursorPosition ) {
|
||||||
var contentShortcodes = getShortCodePositionsInText( content );
|
var contentShortcodes = getShortCodePositionsInText( content );
|
||||||
|
|
||||||
return _.find( contentShortcodes, function( element ) {
|
for ( var i = 0; i < contentShortcodes.length; i++ ) {
|
||||||
return cursorPosition >= element.startIndex && cursorPosition <= element.endIndex;
|
var element = contentShortcodes[ i ];
|
||||||
} );
|
|
||||||
|
if ( cursorPosition >= element.startIndex && cursorPosition <= element.endIndex ) {
|
||||||
|
return element;
|
||||||
|
}
|
||||||
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
/**
|
/**
|
||||||
@ -245,13 +264,20 @@ window.wp = window.wp || {};
|
|||||||
* @param {string} content The content we want to scan for shortcodes.
|
* @param {string} content The content we want to scan for shortcodes.
|
||||||
*/
|
*/
|
||||||
function getShortcodesInText( content ) {
|
function getShortcodesInText( content ) {
|
||||||
var shortcodes = content.match( /\[+([\w_-])+/g );
|
var shortcodes = content.match( /\[+([\w_-])+/g ),
|
||||||
|
result = [];
|
||||||
|
|
||||||
return _.uniq(
|
if ( shortcodes ) {
|
||||||
_.map( shortcodes, function( element ) {
|
for ( var i = 0; i < shortcodes.length; i++ ) {
|
||||||
return element.replace( /^\[+/g, '' );
|
var shortcode = shortcodes[ i ].replace( /^\[+/g, '' );
|
||||||
} )
|
|
||||||
);
|
if ( result.indexOf( shortcode ) === -1 ) {
|
||||||
|
result.push( shortcode );
|
||||||
|
}
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
return result;
|
||||||
}
|
}
|
||||||
|
|
||||||
/**
|
/**
|
||||||
@ -335,6 +361,34 @@ window.wp = window.wp || {};
|
|||||||
shortcodesDetails.push( shortcodeInfo );
|
shortcodesDetails.push( shortcodeInfo );
|
||||||
}
|
}
|
||||||
|
|
||||||
|
/**
|
||||||
|
* Get all URL matches, and treat them as embeds.
|
||||||
|
*
|
||||||
|
* Since there isn't a good way to detect if a URL by itself on a line is a previewable
|
||||||
|
* object, it's best to treat all of them as such.
|
||||||
|
*
|
||||||
|
* This means that the selection will capture the whole URL, in a similar way shrotcodes
|
||||||
|
* are treated.
|
||||||
|
*/
|
||||||
|
var urlRegexp = new RegExp(
|
||||||
|
'(^|[\\n\\r][\\n\\r]|<p>)(https?:\\/\\/[^\s"]+?)(<\\/p>\s*|[\\n\\r][\\n\\r]|$)', 'gi'
|
||||||
|
);
|
||||||
|
|
||||||
|
while ( shortcodeMatch = urlRegexp.exec( content ) ) {
|
||||||
|
shortcodeInfo = {
|
||||||
|
shortcodeName: 'url',
|
||||||
|
showAsPlainText: false,
|
||||||
|
startIndex: shortcodeMatch.index,
|
||||||
|
endIndex: shortcodeMatch.index + shortcodeMatch[ 0 ].length,
|
||||||
|
length: shortcodeMatch[ 0 ].length,
|
||||||
|
isPreviewable: true,
|
||||||
|
urlAtStartOfContent: shortcodeMatch[ 1 ] === '',
|
||||||
|
urlAtEndOfContent: shortcodeMatch[ 3 ] === ''
|
||||||
|
};
|
||||||
|
|
||||||
|
shortcodesDetails.push( shortcodeInfo );
|
||||||
|
}
|
||||||
|
|
||||||
return shortcodesDetails;
|
return shortcodesDetails;
|
||||||
}
|
}
|
||||||
|
|
||||||
@ -399,8 +453,7 @@ window.wp = window.wp || {};
|
|||||||
*/
|
*/
|
||||||
if ( voidElements.indexOf( isCursorStartInTag.tagType ) !== -1 ) {
|
if ( voidElements.indexOf( isCursorStartInTag.tagType ) !== -1 ) {
|
||||||
cursorStart = isCursorStartInTag.ltPos;
|
cursorStart = isCursorStartInTag.ltPos;
|
||||||
}
|
} else {
|
||||||
else {
|
|
||||||
cursorStart = isCursorStartInTag.gtPos;
|
cursorStart = isCursorStartInTag.gtPos;
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
@ -412,12 +465,28 @@ window.wp = window.wp || {};
|
|||||||
|
|
||||||
var isCursorStartInShortcode = getShortcodeWrapperInfo( content, cursorStart );
|
var isCursorStartInShortcode = getShortcodeWrapperInfo( content, cursorStart );
|
||||||
if ( isCursorStartInShortcode && isCursorStartInShortcode.isPreviewable ) {
|
if ( isCursorStartInShortcode && isCursorStartInShortcode.isPreviewable ) {
|
||||||
cursorStart = isCursorStartInShortcode.startIndex;
|
/**
|
||||||
|
* If a URL is at the start or the end of the content,
|
||||||
|
* the selection doesn't work, because it inserts a marker in the text,
|
||||||
|
* which breaks the embedURL detection.
|
||||||
|
*
|
||||||
|
* The best way to avoid that and not modify the user content is to
|
||||||
|
* adjust the cursor to either after or before URL.
|
||||||
|
*/
|
||||||
|
if ( isCursorStartInShortcode.urlAtStartOfContent ) {
|
||||||
|
cursorStart = isCursorStartInShortcode.endIndex;
|
||||||
|
} else {
|
||||||
|
cursorStart = isCursorStartInShortcode.startIndex;
|
||||||
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
var isCursorEndInShortcode = getShortcodeWrapperInfo( content, cursorEnd );
|
var isCursorEndInShortcode = getShortcodeWrapperInfo( content, cursorEnd );
|
||||||
if ( isCursorEndInShortcode && isCursorEndInShortcode.isPreviewable ) {
|
if ( isCursorEndInShortcode && isCursorEndInShortcode.isPreviewable ) {
|
||||||
cursorEnd = isCursorEndInShortcode.endIndex;
|
if ( isCursorEndInShortcode.urlAtEndOfContent ) {
|
||||||
|
cursorEnd = isCursorEndInShortcode.startIndex;
|
||||||
|
} else {
|
||||||
|
cursorEnd = isCursorEndInShortcode.endIndex;
|
||||||
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
return {
|
return {
|
||||||
@ -455,7 +524,7 @@ window.wp = window.wp || {};
|
|||||||
mode = htmlModeCursorStartPosition !== htmlModeCursorEndPosition ? 'range' : 'single',
|
mode = htmlModeCursorStartPosition !== htmlModeCursorEndPosition ? 'range' : 'single',
|
||||||
|
|
||||||
selectedText = null,
|
selectedText = null,
|
||||||
cursorMarkerSkeleton = getCursorMarkerSpan( $$, '' );
|
cursorMarkerSkeleton = getCursorMarkerSpan( $$, '' ).attr( 'data-mce-type','bookmark' );
|
||||||
|
|
||||||
if ( mode === 'range' ) {
|
if ( mode === 'range' ) {
|
||||||
var markedText = textArea.value.slice( htmlModeCursorStartPosition, htmlModeCursorEndPosition ),
|
var markedText = textArea.value.slice( htmlModeCursorStartPosition, htmlModeCursorEndPosition ),
|
||||||
@ -470,7 +539,7 @@ window.wp = window.wp || {};
|
|||||||
textArea.value = [
|
textArea.value = [
|
||||||
textArea.value.slice( 0, htmlModeCursorStartPosition ), // text until the cursor/selection position
|
textArea.value.slice( 0, htmlModeCursorStartPosition ), // text until the cursor/selection position
|
||||||
cursorMarkerSkeleton.clone() // cursor/selection start marker
|
cursorMarkerSkeleton.clone() // cursor/selection start marker
|
||||||
.addClass( 'mce_SELRES_start')[0].outerHTML,
|
.addClass( 'mce_SELRES_start' )[0].outerHTML,
|
||||||
selectedText, // selected text with end cursor/position marker
|
selectedText, // selected text with end cursor/position marker
|
||||||
textArea.value.slice( htmlModeCursorEndPosition ) // text from last cursor/selection position to end
|
textArea.value.slice( htmlModeCursorEndPosition ) // text from last cursor/selection position to end
|
||||||
].join( '' );
|
].join( '' );
|
||||||
@ -487,8 +556,8 @@ window.wp = window.wp || {};
|
|||||||
* @param {Object} editor TinyMCE editor instance.
|
* @param {Object} editor TinyMCE editor instance.
|
||||||
*/
|
*/
|
||||||
function focusHTMLBookmarkInVisualEditor( editor ) {
|
function focusHTMLBookmarkInVisualEditor( editor ) {
|
||||||
var startNode = editor.$( '.mce_SELRES_start' ),
|
var startNode = editor.$( '.mce_SELRES_start' ).attr( 'data-mce-bogus', 1 ),
|
||||||
endNode = editor.$( '.mce_SELRES_end' );
|
endNode = editor.$( '.mce_SELRES_end' ).attr( 'data-mce-bogus', 1 );
|
||||||
|
|
||||||
if ( startNode.length ) {
|
if ( startNode.length ) {
|
||||||
editor.focus();
|
editor.focus();
|
||||||
@ -505,8 +574,9 @@ window.wp = window.wp || {};
|
|||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
scrollVisualModeToStartElement( editor, startNode );
|
if ( editor.getParam( 'wp_keep_scroll_position' ) ) {
|
||||||
|
scrollVisualModeToStartElement( editor, startNode );
|
||||||
|
}
|
||||||
|
|
||||||
removeSelectionMarker( startNode );
|
removeSelectionMarker( startNode );
|
||||||
removeSelectionMarker( endNode );
|
removeSelectionMarker( endNode );
|
||||||
@ -548,13 +618,18 @@ window.wp = window.wp || {};
|
|||||||
var elementTop = editor.$( element ).offset().top,
|
var elementTop = editor.$( element ).offset().top,
|
||||||
TinyMCEContentAreaTop = editor.$( editor.getContentAreaContainer() ).offset().top,
|
TinyMCEContentAreaTop = editor.$( editor.getContentAreaContainer() ).offset().top,
|
||||||
|
|
||||||
edTools = $( '#wp-content-editor-tools' ),
|
|
||||||
edToolsHeight = edTools.height(),
|
|
||||||
edToolsOffsetTop = edTools.offset().top,
|
|
||||||
|
|
||||||
toolbarHeight = getToolbarHeight( editor ),
|
toolbarHeight = getToolbarHeight( editor ),
|
||||||
|
|
||||||
windowHeight = window.innerHeight || document.documentElement.clientHeight || document.body.clientHeight,
|
edTools = $( '#wp-content-editor-tools' ),
|
||||||
|
edToolsHeight = 0,
|
||||||
|
edToolsOffsetTop = 0;
|
||||||
|
|
||||||
|
if ( edTools.length ) {
|
||||||
|
edToolsHeight = edTools.height();
|
||||||
|
edToolsOffsetTop = edTools.offset().top;
|
||||||
|
}
|
||||||
|
|
||||||
|
var windowHeight = window.innerHeight || document.documentElement.clientHeight || document.body.clientHeight,
|
||||||
|
|
||||||
selectionPosition = TinyMCEContentAreaTop + elementTop,
|
selectionPosition = TinyMCEContentAreaTop + elementTop,
|
||||||
visibleAreaHeight = windowHeight - ( edToolsHeight + toolbarHeight );
|
visibleAreaHeight = windowHeight - ( edToolsHeight + toolbarHeight );
|
||||||
@ -675,8 +750,8 @@ window.wp = window.wp || {};
|
|||||||
* This way we can adjust the selection to properly select only the content, ignoring
|
* This way we can adjust the selection to properly select only the content, ignoring
|
||||||
* whitespace inserted around the selected object by the Editor.
|
* whitespace inserted around the selected object by the Editor.
|
||||||
*/
|
*/
|
||||||
startElement.attr('data-mce-object-selection', 'true');
|
startElement.attr( 'data-mce-object-selection', 'true' );
|
||||||
endElement.attr('data-mce-object-selection', 'true');
|
endElement.attr( 'data-mce-object-selection', 'true' );
|
||||||
|
|
||||||
editor.$( startNode ).before( startElement[0] );
|
editor.$( startNode ).before( startElement[0] );
|
||||||
editor.$( startNode ).after( endElement[0] );
|
editor.$( startNode ).after( endElement[0] );
|
||||||
|
@ -981,6 +981,7 @@ final class _WP_Editors {
|
|||||||
'end_container_on_empty_block' => true,
|
'end_container_on_empty_block' => true,
|
||||||
'wpeditimage_html5_captions' => true,
|
'wpeditimage_html5_captions' => true,
|
||||||
'wp_lang_attr' => get_bloginfo( 'language' ),
|
'wp_lang_attr' => get_bloginfo( 'language' ),
|
||||||
|
'wp_keep_scroll_position' => false,
|
||||||
'wp_shortcut_labels' => wp_json_encode( $shortcut_labels ),
|
'wp_shortcut_labels' => wp_json_encode( $shortcut_labels ),
|
||||||
);
|
);
|
||||||
|
|
||||||
|
Loading…
Reference in New Issue
Block a user