X-Git-Url: https://jasonwoof.com/gitweb/?a=blobdiff_plain;f=_source%2Fplugins%2Fwysiwygarea%2Fplugin.js;h=d3b9e629d459699a04f79e90c68a9112df0d4ce3;hb=7cd80714081a8ffdf4a1a8d2c72f120ed5ef3d6d;hp=225f2f36464db7ad499e477de0f75268af07cee3;hpb=ea7e3453c7b0f023b050aca6d9f83ab372860d91;p=ckeditor.git
diff --git a/_source/plugins/wysiwygarea/plugin.js b/_source/plugins/wysiwygarea/plugin.js
index 225f2f3..d3b9e62 100644
--- a/_source/plugins/wysiwygarea/plugin.js
+++ b/_source/plugins/wysiwygarea/plugin.js
@@ -15,7 +15,7 @@ For licensing, see LICENSE.html or http://ckeditor.com/license
*/
var nonExitableElementNames = { table:1,pre:1 };
// Matching an empty paragraph at the end of document.
- var emptyParagraphRegexp = /\s*<(p|div|address|h\d|center)[^>]*>\s*(?:
]*>| | )\s*(:?<\/\1>)?\s*$/gi;
+ var emptyParagraphRegexp = /\s*<(p|div|address|h\d|center)[^>]*>\s*(?:
]*>| |\u00A0| )?\s*(:?<\/\1>)?\s*$/gi;
function onInsertHtml( evt )
{
@@ -81,14 +81,27 @@ For licensing, see LICENSE.html or http://ckeditor.com/license
// If we're inserting a block at dtd-violated position, split
// the parent blocks until we reach blockLimit.
- var parent, dtd;
- if ( this.config.enterMode != CKEDITOR.ENTER_BR && isBlock )
+ var current, dtd;
+ if ( isBlock )
{
- while( ( parent = range.getCommonAncestor( false, true ) )
- && ( dtd = CKEDITOR.dtd[ parent.getName() ] )
+ while( ( current = range.getCommonAncestor( false, true ) )
+ && ( dtd = CKEDITOR.dtd[ current.getName() ] )
&& !( dtd && dtd [ elementName ] ) )
{
- range.splitBlock();
+ // Split up inline elements.
+ if ( current.getName() in CKEDITOR.dtd.span )
+ range.splitElement( current );
+ // If we're in an empty block which indicate a new paragraph,
+ // simply replace it with the inserting block.(#3664)
+ else if ( range.checkStartOfBlock()
+ && range.checkEndOfBlock() )
+ {
+ range.setStartBefore( current );
+ range.collapse( true );
+ current.remove();
+ }
+ else
+ range.splitBlock();
}
}
@@ -122,6 +135,27 @@ For licensing, see LICENSE.html or http://ckeditor.com/license
}
}
+ // DOM modification here should not bother dirty flag.(#4385)
+ function restoreDirty( editor )
+ {
+ if( !editor.checkDirty() )
+ setTimeout( function(){ editor.resetDirty(); } );
+ }
+
+ var isNotWhitespace = CKEDITOR.dom.walker.whitespaces( true ),
+ isNotBookmark = CKEDITOR.dom.walker.bookmark( false, true );
+
+ function isNotEmpty( node )
+ {
+ return isNotWhitespace( node ) && isNotBookmark( node );
+ }
+
+ function isNbsp( node )
+ {
+ return node.type == CKEDITOR.NODE_TEXT
+ && CKEDITOR.tools.trim( node.getText() ).match( /^(?: |\xa0)$/ );
+ }
+
/**
* Auto-fixing block-less content by wrapping paragraph (#3190), prevent
* non-exitable-block by padding extra br.(#3189)
@@ -143,60 +177,53 @@ For licensing, see LICENSE.html or http://ckeditor.com/license
&& blockLimit.getName() == 'body'
&& !path.block )
{
- var bms = selection.createBookmarks(),
- fixedBlock = range.fixBlock( true,
+ restoreDirty( editor );
+ var fixedBlock = range.fixBlock( true,
editor.config.enterMode == CKEDITOR.ENTER_DIV ? 'div' : 'p' );
- // For IE, we'll be removing any bogus br ( introduce by fixing body )
- // right now to prevent it introducing visual line break.
+ // For IE, we should remove any filler node which was introduced before.
if ( CKEDITOR.env.ie )
{
- var brNodeList = fixedBlock.getElementsByTag( 'br' ), brNode;
- for ( var i = 0 ; i < brNodeList.count() ; i++ )
- {
- if( ( brNode = brNodeList.getItem( i ) ) && brNode.hasAttribute( '_cke_bogus' ) )
- brNode.remove();
- }
+ var first = fixedBlock.getFirst( isNotEmpty );
+ first && isNbsp( first ) && first.remove();
}
- selection.selectBookmarks( bms );
-
- // If the fixed block is blank and is already followed by a exitable
- // block, we should drop it and move to the exist block(#3684).
- var children = fixedBlock.getChildren(),
- count = children.count(),
- firstChild,
- whitespaceGuard = CKEDITOR.dom.walker.whitespaces( true ),
- previousElement = fixedBlock.getPrevious( whitespaceGuard ),
- nextElement = fixedBlock.getNext( whitespaceGuard ),
- enterBlock;
- if ( previousElement && previousElement.getName
- && !( previousElement.getName() in nonExitableElementNames ) )
- enterBlock = previousElement;
- else if ( nextElement && nextElement.getName
- && !( nextElement.getName() in nonExitableElementNames ) )
- enterBlock = nextElement;
-
- // Not all blocks are editable, e.g.
, further checking it.(#3994)
- if( ( !count
- || ( firstChild = children.getItem( 0 ) ) && firstChild.is && firstChild.is( 'br' ) )
- && enterBlock
- && range.moveToElementEditStart( enterBlock ) )
+ // If the fixed block is blank and already followed by a exitable
+ // block, we should revert the fix. (#3684)
+ if( fixedBlock.getOuterHtml().match( emptyParagraphRegexp ) )
{
- fixedBlock.remove();
- range.select();
+ var previousElement = fixedBlock.getPrevious( isNotWhitespace ),
+ nextElement = fixedBlock.getNext( isNotWhitespace );
+
+
+ if ( previousElement && previousElement.getName
+ && !( previousElement.getName() in nonExitableElementNames )
+ && range.moveToElementEditStart( previousElement )
+ || nextElement && nextElement.getName
+ && !( nextElement.getName() in nonExitableElementNames )
+ && range.moveToElementEditStart( nextElement ) )
+ {
+ fixedBlock.remove();
+ }
}
+
+ range.select();
+ // Notify non-IE that selection has changed.
+ if( !CKEDITOR.env.ie )
+ editor.selectionChange();
}
- // Inserting the padding-br before body if it's preceded by an
- // unexitable block.
+ // All browsers are incapable to moving cursor out of certain non-exitable
+ // blocks (e.g. table, list, pre) at the end of document, make this happen by
+ // place a bogus node there, which would be later removed by dataprocessor.
var lastNode = body.getLast( CKEDITOR.dom.walker.whitespaces( true ) );
if ( lastNode && lastNode.getName && ( lastNode.getName() in nonExitableElementNames ) )
{
- var paddingBlock = editor.document.createElement(
- ( CKEDITOR.env.ie && enterMode != CKEDITOR.ENTER_BR ) ?
- '
' : 'br' );
- body.append( paddingBlock );
+ restoreDirty( editor );
+ if( !CKEDITOR.env.ie )
+ body.appendBogus();
+ else
+ body.append( editor.document.createText( '\xa0' ) );
}
}
@@ -310,16 +337,11 @@ For licensing, see LICENSE.html or http://ckeditor.com/license
mainElement.append( iframe );
};
- // The script that is appended to the data being loaded. It
- // enables editing, and makes some
+ // The script that launches the bootstrap logic on 'domReady', so the document
+ // is fully editable even before the editing iframe is fully loaded (#4455).
var activationScript =
'';
// Editing area bootstrap code.
@@ -363,26 +385,6 @@ For licensing, see LICENSE.html or http://ckeditor.com/license
domWindow = editor.window = new CKEDITOR.dom.window( domWindow );
domDocument = editor.document = new CKEDITOR.dom.document( domDocument );
- // Gecko need a key event to 'wake up' the editing
- // ability when document is empty.(#3864)
- var firstNode = domDocument.getBody().getFirst();
- if ( CKEDITOR.env.gecko
- && firstNode && firstNode.is
- && firstNode.is( 'br' ) && firstNode.hasAttribute( '_moz_editor_bogus_node' ) )
- {
- var keyEventSimulate = domDocument.$.createEvent( "KeyEvents" );
- keyEventSimulate.initKeyEvent( 'keypress', true, true, domWindow.$, false,
- false, false, false, 0, 32 );
- domDocument.$.dispatchEvent( keyEventSimulate );
- var bogusText = domDocument.getBody().getFirst() ;
- // Compensate the line maintaining
if enterMode is not block.
- if ( editor.config.enterMode == CKEDITOR.ENTER_BR )
- domDocument.createElement( 'br', { attributes: { '_moz_dirty' : "" } } )
- .replace( bogusText );
- else
- bogusText.remove();
- }
-
// Gecko/Webkit need some help when selecting control type elements. (#3448)
if ( !( CKEDITOR.env.ie || CKEDITOR.env.opera) )
{
@@ -412,7 +414,23 @@ For licensing, see LICENSE.html or http://ckeditor.com/license
} );
}
- var focusTarget = ( CKEDITOR.env.ie || CKEDITOR.env.safari ) ?
+ // IE standard compliant in editing frame doesn't focus the editor when
+ // clicking outside actual content, manually apply the focus. (#1659)
+ if( CKEDITOR.env.ie
+ && domDocument.$.compatMode == 'CSS1Compat' )
+ {
+ var htmlElement = domDocument.getDocumentElement();
+ htmlElement.on( 'mousedown', function( evt )
+ {
+ // Setting focus directly on editor doesn't work, we
+ // have to use here a temporary element to 'redirect'
+ // the focus.
+ if ( evt.data.getTarget().equals( htmlElement ) )
+ ieFocusGrabber.focus();
+ } );
+ }
+
+ var focusTarget = ( CKEDITOR.env.ie || CKEDITOR.env.webkit ) ?
domWindow : domDocument;
focusTarget.on( 'blur', function()
@@ -422,6 +440,32 @@ For licensing, see LICENSE.html or http://ckeditor.com/license
focusTarget.on( 'focus', function()
{
+ // Gecko need a key event to 'wake up' the editing
+ // ability when document is empty.(#3864)
+ if ( CKEDITOR.env.gecko )
+ {
+ var first = body;
+ while( first.firstChild )
+ first = first.firstChild;
+
+ if( !first.nextSibling
+ && ( 'BR' == first.tagName )
+ && first.hasAttribute( '_moz_editor_bogus_node' ) )
+ {
+ var keyEventSimulate = domDocument.$.createEvent( "KeyEvents" );
+ keyEventSimulate.initKeyEvent( 'keypress', true, true, domWindow.$, false,
+ false, false, false, 0, 32 );
+ domDocument.$.dispatchEvent( keyEventSimulate );
+ var bogusText = domDocument.getBody().getFirst() ;
+ // Compensate the line maintaining
if enterMode is not block.
+ if ( editor.config.enterMode == CKEDITOR.ENTER_BR )
+ domDocument.createElement( 'br', { attributes: { '_moz_dirty' : "" } } )
+ .replace( bogusText );
+ else
+ bogusText.remove();
+ }
+ }
+
editor.focusManager.focus();
});
@@ -429,9 +473,46 @@ For licensing, see LICENSE.html or http://ckeditor.com/license
if ( keystrokeHandler )
keystrokeHandler.attach( domDocument );
+ if ( CKEDITOR.env.ie )
+ {
+ // Cancel default action for backspace in IE on control types. (#4047)
+ domDocument.on( 'keydown', function( evt )
+ {
+ // Backspace.
+ var control = evt.data.getKeystroke() == 8
+ && editor.getSelection().getSelectedElement();
+ if ( control )
+ {
+ // Make undo snapshot.
+ editor.fire( 'saveSnapshot' );
+ // Remove manually.
+ control.remove();
+ editor.fire( 'saveSnapshot' );
+ evt.cancel();
+ }
+ } );
+
+ // PageUp/PageDown scrolling is broken in document
+ // with standard doctype, manually fix it. (#4736)
+ if( domDocument.$.compatMode == 'CSS1Compat' )
+ {
+ var pageUpDownKeys = { 33 : 1, 34 : 1 };
+ domDocument.on( 'keydown', function( evt )
+ {
+ if( evt.data.getKeystroke() in pageUpDownKeys )
+ {
+ setTimeout( function ()
+ {
+ editor.getSelection().scrollIntoView();
+ }, 0 );
+ }
+ } );
+ }
+ }
+
// Adds the document body as a context menu target.
if ( editor.contextMenu )
- editor.contextMenu.addTarget( domDocument );
+ editor.contextMenu.addTarget( domDocument, editor.config.browserContextMenuOnCtrl !== false );
setTimeout( function()
{
@@ -451,6 +532,10 @@ For licensing, see LICENSE.html or http://ckeditor.com/license
editor.focus();
isPendingFocus = false;
}
+ setTimeout( function()
+ {
+ editor.fire( 'dataReady' );
+ }, 0 );
/*
* IE BUG: IE might have rendered the iframe with invisible contents.
@@ -511,7 +596,9 @@ For licensing, see LICENSE.html or http://ckeditor.com/license
editor.config.docType +
'' +
'' +
- '' +
+ '' +
''+
@@ -574,6 +661,25 @@ For licensing, see LICENSE.html or http://ckeditor.com/license
else if ( editor.window )
{
editor.window.focus();
+
+ // Force the selection to happen, in this way
+ // we guarantee the focus will be there. (#4848)
+ if ( CKEDITOR.env.ie )
+ {
+ try
+ {
+ var sel = editor.getSelection();
+ sel = sel && sel.getNative();
+ var range = sel && sel.type && sel.createRange();
+ if ( range )
+ {
+ sel.empty();
+ range.select();
+ }
+ }
+ catch (e) {}
+ }
+
editor.selectionChange();
}
}
@@ -584,8 +690,60 @@ For licensing, see LICENSE.html or http://ckeditor.com/license
// Auto fixing on some document structure weakness to enhance usabilities. (#3190 and #3189)
editor.on( 'selectionChange', onSelectionChangeFixBody, null, null, 1 );
});
+
+ // Create an invisible element to grab focus.
+ if( CKEDITOR.env.ie )
+ {
+ var ieFocusGrabber;
+ editor.on( 'uiReady', function()
+ {
+ ieFocusGrabber = editor.container.append( CKEDITOR.dom.element.createFromHtml(
+ '' ) );
+
+ ieFocusGrabber.on( 'focus', function()
+ {
+ editor.focus();
+ } );
+ } );
+ }
}
});
+
+ // Fixing Firefox 'Back-Forward Cache' break design mode. (#4514)
+ if( CKEDITOR.env.gecko )
+ {
+ var topWin = window.top;
+
+ ( function ()
+ {
+ var topBody = topWin.document.body;
+
+ if( !topBody )
+ topWin.addEventListener( 'load', arguments.callee, false );
+ else
+ {
+ topBody.setAttribute( 'onpageshow', topBody.getAttribute( 'onpageshow' )
+ + ';event.persisted && CKEDITOR.tools.callFunction(' +
+ CKEDITOR.tools.addFunction( function()
+ {
+ var allInstances = CKEDITOR.instances,
+ editor,
+ doc;
+ for( var i in allInstances )
+ {
+ editor = allInstances[ i ];
+ doc = editor.document;
+ if( doc )
+ {
+ doc.$.designMode = 'off';
+ doc.$.designMode = 'on';
+ }
+ }
+ } ) + ')' );
+ }
+ } )();
+
+ }
})();
/**