2015-05-01 14:07:28 -04:00
|
|
|
|
/* Copyright © 2011-2015 by Neil Jenkins. MIT Licensed. */
|
2013-04-08 13:27:06 +10:00
|
|
|
|
|
2014-04-07 13:05:44 +10:00
|
|
|
|
( function ( doc, undefined ) {
|
2013-04-08 13:27:06 +10:00
|
|
|
|
|
|
|
|
|
"use strict";
|
|
|
|
|
|
|
|
|
|
var DOCUMENT_POSITION_PRECEDING = 2; // Node.DOCUMENT_POSITION_PRECEDING
|
|
|
|
|
var ELEMENT_NODE = 1; // Node.ELEMENT_NODE;
|
|
|
|
|
var TEXT_NODE = 3; // Node.TEXT_NODE;
|
2016-03-22 17:57:00 +11:00
|
|
|
|
var DOCUMENT_NODE = 9; // Node.DOCUMENT_NODE;
|
2015-06-22 09:59:34 +07:00
|
|
|
|
var DOCUMENT_FRAGMENT_NODE = 11; // Node.DOCUMENT_FRAGMENT_NODE;
|
2013-04-08 13:27:06 +10:00
|
|
|
|
var SHOW_ELEMENT = 1; // NodeFilter.SHOW_ELEMENT;
|
|
|
|
|
var SHOW_TEXT = 4; // NodeFilter.SHOW_TEXT;
|
|
|
|
|
|
|
|
|
|
var START_TO_START = 0; // Range.START_TO_START
|
|
|
|
|
var START_TO_END = 1; // Range.START_TO_END
|
|
|
|
|
var END_TO_END = 2; // Range.END_TO_END
|
|
|
|
|
var END_TO_START = 3; // Range.END_TO_START
|
|
|
|
|
|
2014-12-25 15:09:46 +07:00
|
|
|
|
var ZWS = '\u200B';
|
|
|
|
|
|
2013-04-08 13:27:06 +10:00
|
|
|
|
var win = doc.defaultView;
|
|
|
|
|
|
|
|
|
|
var ua = navigator.userAgent;
|
2013-05-20 16:14:28 +10:00
|
|
|
|
|
2016-12-11 12:30:09 +11:00
|
|
|
|
var isAndroid = /Android/.test( ua );
|
2013-05-20 16:14:28 +10:00
|
|
|
|
var isMac = /Mac OS X/.test( ua );
|
2016-12-11 12:30:09 +11:00
|
|
|
|
var isWin = /Windows NT/.test( ua );
|
2020-03-11 14:35:50 +11:00
|
|
|
|
var isIOS = /iP(?:ad|hone|od)/.test( ua ) ||
|
|
|
|
|
( isMac && !!navigator.maxTouchPoints );
|
2016-07-14 11:32:38 +10:00
|
|
|
|
|
2013-04-08 13:27:06 +10:00
|
|
|
|
var isGecko = /Gecko\//.test( ua );
|
2016-03-11 15:22:49 +11:00
|
|
|
|
var isEdge = /Edge\//.test( ua );
|
|
|
|
|
var isWebKit = !isEdge && /WebKit\//.test( ua );
|
2016-08-05 20:47:02 +10:00
|
|
|
|
var isIE = /Trident\/[4567]\./.test( ua );
|
2013-04-08 13:27:06 +10:00
|
|
|
|
|
2013-05-20 16:14:28 +10:00
|
|
|
|
var ctrlKey = isMac ? 'meta-' : 'ctrl-';
|
|
|
|
|
|
2020-03-11 14:48:30 +11:00
|
|
|
|
var cantFocusEmptyTextNodes = isWebKit;
|
2013-04-08 13:27:06 +10:00
|
|
|
|
|
2014-12-27 13:48:15 +07:00
|
|
|
|
var canObserveMutations = typeof MutationObserver !== 'undefined';
|
2016-11-15 11:07:15 +11:00
|
|
|
|
var canWeakMap = typeof WeakMap !== 'undefined';
|
2014-12-27 13:48:15 +07:00
|
|
|
|
|
2013-11-04 18:21:57 +11:00
|
|
|
|
// Use [^ \t\r\n] instead of \S so that nbsp does not count as white-space
|
|
|
|
|
var notWS = /[^ \t\r\n]/;
|
2013-04-08 13:27:06 +10:00
|
|
|
|
|
|
|
|
|
var indexOf = Array.prototype.indexOf;
|
|
|
|
|
|
|
|
|
|
/*
|
|
|
|
|
Native TreeWalker is buggy in IE and Opera:
|
|
|
|
|
* IE9/10 sometimes throw errors when calling TreeWalker#nextNode or
|
|
|
|
|
TreeWalker#previousNode. No way to feature detect this.
|
|
|
|
|
* Some versions of Opera have a bug in TreeWalker#previousNode which makes
|
|
|
|
|
it skip to the wrong node.
|
|
|
|
|
|
|
|
|
|
Rather than risk further bugs, it's easiest just to implement our own
|
|
|
|
|
(subset) of the spec in all browsers.
|
|
|
|
|
*/
|
|
|
|
|
|
|
|
|
|
var typeToBitArray = {
|
|
|
|
|
// ELEMENT_NODE
|
|
|
|
|
1: 1,
|
|
|
|
|
// ATTRIBUTE_NODE
|
|
|
|
|
2: 2,
|
|
|
|
|
// TEXT_NODE
|
|
|
|
|
3: 4,
|
|
|
|
|
// COMMENT_NODE
|
|
|
|
|
8: 128,
|
|
|
|
|
// DOCUMENT_NODE
|
|
|
|
|
9: 256,
|
|
|
|
|
// DOCUMENT_FRAGMENT_NODE
|
|
|
|
|
11: 1024
|
|
|
|
|
};
|
|
|
|
|
|
2018-07-27 09:21:50 +10:00
|
|
|
|
var always = function () {
|
|
|
|
|
return true;
|
|
|
|
|
};
|
|
|
|
|
|
2013-04-08 13:27:06 +10:00
|
|
|
|
function TreeWalker ( root, nodeType, filter ) {
|
|
|
|
|
this.root = this.currentNode = root;
|
|
|
|
|
this.nodeType = nodeType;
|
2018-07-27 09:21:50 +10:00
|
|
|
|
this.filter = filter || always;
|
2013-04-08 13:27:06 +10:00
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
TreeWalker.prototype.nextNode = function () {
|
|
|
|
|
var current = this.currentNode,
|
|
|
|
|
root = this.root,
|
|
|
|
|
nodeType = this.nodeType,
|
|
|
|
|
filter = this.filter,
|
|
|
|
|
node;
|
|
|
|
|
while ( true ) {
|
|
|
|
|
node = current.firstChild;
|
|
|
|
|
while ( !node && current ) {
|
|
|
|
|
if ( current === root ) {
|
|
|
|
|
break;
|
|
|
|
|
}
|
|
|
|
|
node = current.nextSibling;
|
|
|
|
|
if ( !node ) { current = current.parentNode; }
|
|
|
|
|
}
|
|
|
|
|
if ( !node ) {
|
|
|
|
|
return null;
|
|
|
|
|
}
|
|
|
|
|
if ( ( typeToBitArray[ node.nodeType ] & nodeType ) &&
|
2014-09-04 17:19:02 +07:00
|
|
|
|
filter( node ) ) {
|
2013-04-08 13:27:06 +10:00
|
|
|
|
this.currentNode = node;
|
|
|
|
|
return node;
|
|
|
|
|
}
|
|
|
|
|
current = node;
|
|
|
|
|
}
|
|
|
|
|
};
|
|
|
|
|
|
|
|
|
|
TreeWalker.prototype.previousNode = function () {
|
|
|
|
|
var current = this.currentNode,
|
|
|
|
|
root = this.root,
|
|
|
|
|
nodeType = this.nodeType,
|
|
|
|
|
filter = this.filter,
|
|
|
|
|
node;
|
|
|
|
|
while ( true ) {
|
|
|
|
|
if ( current === root ) {
|
|
|
|
|
return null;
|
|
|
|
|
}
|
|
|
|
|
node = current.previousSibling;
|
|
|
|
|
if ( node ) {
|
|
|
|
|
while ( current = node.lastChild ) {
|
|
|
|
|
node = current;
|
|
|
|
|
}
|
|
|
|
|
} else {
|
|
|
|
|
node = current.parentNode;
|
|
|
|
|
}
|
|
|
|
|
if ( !node ) {
|
|
|
|
|
return null;
|
|
|
|
|
}
|
|
|
|
|
if ( ( typeToBitArray[ node.nodeType ] & nodeType ) &&
|
2014-09-04 17:19:02 +07:00
|
|
|
|
filter( node ) ) {
|
2013-04-08 13:27:06 +10:00
|
|
|
|
this.currentNode = node;
|
|
|
|
|
return node;
|
|
|
|
|
}
|
|
|
|
|
current = node;
|
|
|
|
|
}
|
|
|
|
|
};
|
|
|
|
|
|
2015-06-22 09:59:34 +07:00
|
|
|
|
// Previous node in post-order.
|
|
|
|
|
TreeWalker.prototype.previousPONode = function () {
|
|
|
|
|
var current = this.currentNode,
|
|
|
|
|
root = this.root,
|
|
|
|
|
nodeType = this.nodeType,
|
|
|
|
|
filter = this.filter,
|
|
|
|
|
node;
|
|
|
|
|
while ( true ) {
|
|
|
|
|
node = current.lastChild;
|
|
|
|
|
while ( !node && current ) {
|
|
|
|
|
if ( current === root ) {
|
|
|
|
|
break;
|
|
|
|
|
}
|
|
|
|
|
node = current.previousSibling;
|
|
|
|
|
if ( !node ) { current = current.parentNode; }
|
|
|
|
|
}
|
|
|
|
|
if ( !node ) {
|
|
|
|
|
return null;
|
|
|
|
|
}
|
|
|
|
|
if ( ( typeToBitArray[ node.nodeType ] & nodeType ) &&
|
|
|
|
|
filter( node ) ) {
|
|
|
|
|
this.currentNode = node;
|
|
|
|
|
return node;
|
|
|
|
|
}
|
|
|
|
|
current = node;
|
|
|
|
|
}
|
|
|
|
|
};
|
|
|
|
|
|
2016-08-01 13:43:07 -07:00
|
|
|
|
var inlineNodeNames = /^(?:#text|A(?:BBR|CRONYM)?|B(?:R|D[IO])?|C(?:ITE|ODE)|D(?:ATA|EL|FN)|EM|FONT|HR|I(?:FRAME|MG|NPUT|NS)?|KBD|Q|R(?:P|T|UBY)|S(?:AMP|MALL|PAN|TR(?:IKE|ONG)|U[BP])?|TIME|U|VAR|WBR)$/;
|
2013-04-08 13:27:06 +10:00
|
|
|
|
|
|
|
|
|
var leafNodeNames = {
|
|
|
|
|
BR: 1,
|
2016-05-01 16:33:47 +10:00
|
|
|
|
HR: 1,
|
2016-05-01 16:36:43 +10:00
|
|
|
|
IFRAME: 1,
|
2013-04-08 13:27:06 +10:00
|
|
|
|
IMG: 1,
|
|
|
|
|
INPUT: 1
|
|
|
|
|
};
|
|
|
|
|
|
|
|
|
|
function every ( nodeList, fn ) {
|
|
|
|
|
var l = nodeList.length;
|
|
|
|
|
while ( l-- ) {
|
|
|
|
|
if ( !fn( nodeList[l] ) ) {
|
|
|
|
|
return false;
|
|
|
|
|
}
|
|
|
|
|
}
|
|
|
|
|
return true;
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
// ---
|
|
|
|
|
|
2016-11-15 11:07:15 +11:00
|
|
|
|
var UNKNOWN = 0;
|
|
|
|
|
var INLINE = 1;
|
|
|
|
|
var BLOCK = 2;
|
|
|
|
|
var CONTAINER = 3;
|
|
|
|
|
|
|
|
|
|
var nodeCategoryCache = canWeakMap ? new WeakMap() : null;
|
|
|
|
|
|
2013-04-08 13:27:06 +10:00
|
|
|
|
function isLeaf ( node ) {
|
2016-11-15 11:07:15 +11:00
|
|
|
|
return node.nodeType === ELEMENT_NODE && !!leafNodeNames[ node.nodeName ];
|
2013-04-08 13:27:06 +10:00
|
|
|
|
}
|
2016-11-15 11:07:15 +11:00
|
|
|
|
function getNodeCategory ( node ) {
|
2016-11-21 10:59:11 +11:00
|
|
|
|
switch ( node.nodeType ) {
|
|
|
|
|
case TEXT_NODE:
|
|
|
|
|
return INLINE;
|
|
|
|
|
case ELEMENT_NODE:
|
|
|
|
|
case DOCUMENT_FRAGMENT_NODE:
|
|
|
|
|
if ( canWeakMap && nodeCategoryCache.has( node ) ) {
|
|
|
|
|
return nodeCategoryCache.get( node );
|
|
|
|
|
}
|
|
|
|
|
break;
|
|
|
|
|
default:
|
|
|
|
|
return UNKNOWN;
|
2016-11-15 11:07:15 +11:00
|
|
|
|
}
|
2016-11-21 10:59:11 +11:00
|
|
|
|
|
2016-11-15 11:07:15 +11:00
|
|
|
|
var nodeCategory;
|
2016-11-21 10:59:11 +11:00
|
|
|
|
if ( !every( node.childNodes, isInline ) ) {
|
2016-09-26 11:37:38 +01:00
|
|
|
|
// Malformed HTML can have block tags inside inline tags. Need to treat
|
|
|
|
|
// these as containers rather than inline. See #239.
|
2016-11-15 11:07:15 +11:00
|
|
|
|
nodeCategory = CONTAINER;
|
|
|
|
|
} else if ( inlineNodeNames.test( node.nodeName ) ) {
|
|
|
|
|
nodeCategory = INLINE;
|
|
|
|
|
} else {
|
|
|
|
|
nodeCategory = BLOCK;
|
|
|
|
|
}
|
|
|
|
|
if ( canWeakMap ) {
|
|
|
|
|
nodeCategoryCache.set( node, nodeCategory );
|
|
|
|
|
}
|
|
|
|
|
return nodeCategory;
|
|
|
|
|
}
|
|
|
|
|
function isInline ( node ) {
|
|
|
|
|
return getNodeCategory( node ) === INLINE;
|
2013-04-08 13:27:06 +10:00
|
|
|
|
}
|
|
|
|
|
function isBlock ( node ) {
|
2016-11-15 11:07:15 +11:00
|
|
|
|
return getNodeCategory( node ) === BLOCK;
|
2013-04-08 13:27:06 +10:00
|
|
|
|
}
|
|
|
|
|
function isContainer ( node ) {
|
2016-11-15 11:07:15 +11:00
|
|
|
|
return getNodeCategory( node ) === CONTAINER;
|
2013-04-08 13:27:06 +10:00
|
|
|
|
}
|
|
|
|
|
|
2016-03-22 17:57:00 +11:00
|
|
|
|
function getBlockWalker ( node, root ) {
|
2016-04-08 17:42:47 +10:00
|
|
|
|
var walker = new TreeWalker( root, SHOW_ELEMENT, isBlock );
|
2013-04-08 13:27:06 +10:00
|
|
|
|
walker.currentNode = node;
|
|
|
|
|
return walker;
|
|
|
|
|
}
|
2016-03-22 17:57:00 +11:00
|
|
|
|
function getPreviousBlock ( node, root ) {
|
|
|
|
|
node = getBlockWalker( node, root ).previousNode();
|
|
|
|
|
return node !== root ? node : null;
|
|
|
|
|
}
|
|
|
|
|
function getNextBlock ( node, root ) {
|
|
|
|
|
node = getBlockWalker( node, root ).nextNode();
|
|
|
|
|
return node !== root ? node : null;
|
|
|
|
|
}
|
2013-04-08 13:27:06 +10:00
|
|
|
|
|
2017-07-15 20:33:32 +10:00
|
|
|
|
function isEmptyBlock ( block ) {
|
|
|
|
|
return !block.textContent && !block.querySelector( 'IMG' );
|
|
|
|
|
}
|
|
|
|
|
|
2016-03-22 17:57:00 +11:00
|
|
|
|
function areAlike ( node, node2 ) {
|
|
|
|
|
return !isLeaf( node ) && (
|
|
|
|
|
node.nodeType === node2.nodeType &&
|
|
|
|
|
node.nodeName === node2.nodeName &&
|
2016-05-26 10:32:45 +10:00
|
|
|
|
node.nodeName !== 'A' &&
|
2016-03-22 17:57:00 +11:00
|
|
|
|
node.className === node2.className &&
|
|
|
|
|
( ( !node.style && !node2.style ) ||
|
|
|
|
|
node.style.cssText === node2.style.cssText )
|
|
|
|
|
);
|
2013-04-08 13:27:06 +10:00
|
|
|
|
}
|
2016-03-22 17:57:00 +11:00
|
|
|
|
function hasTagAttributes ( node, tag, attributes ) {
|
|
|
|
|
if ( node.nodeName !== tag ) {
|
|
|
|
|
return false;
|
|
|
|
|
}
|
|
|
|
|
for ( var attr in attributes ) {
|
|
|
|
|
if ( node.getAttribute( attr ) !== attributes[ attr ] ) {
|
|
|
|
|
return false;
|
|
|
|
|
}
|
|
|
|
|
}
|
|
|
|
|
return true;
|
2013-04-08 13:27:06 +10:00
|
|
|
|
}
|
2016-03-22 17:57:00 +11:00
|
|
|
|
function getNearest ( node, root, tag, attributes ) {
|
|
|
|
|
while ( node && node !== root ) {
|
2013-04-08 13:27:06 +10:00
|
|
|
|
if ( hasTagAttributes( node, tag, attributes ) ) {
|
|
|
|
|
return node;
|
|
|
|
|
}
|
2016-03-22 17:57:00 +11:00
|
|
|
|
node = node.parentNode;
|
|
|
|
|
}
|
2013-04-08 13:27:06 +10:00
|
|
|
|
return null;
|
|
|
|
|
}
|
2016-03-22 17:57:00 +11:00
|
|
|
|
function isOrContains ( parent, node ) {
|
|
|
|
|
while ( node ) {
|
|
|
|
|
if ( node === parent ) {
|
|
|
|
|
return true;
|
|
|
|
|
}
|
|
|
|
|
node = node.parentNode;
|
|
|
|
|
}
|
|
|
|
|
return false;
|
|
|
|
|
}
|
2013-04-08 13:27:06 +10:00
|
|
|
|
|
2018-07-12 13:35:20 +10:00
|
|
|
|
function getPath ( node, root, config ) {
|
2016-03-26 11:32:59 +11:00
|
|
|
|
var path = '';
|
2018-07-12 13:35:20 +10:00
|
|
|
|
var id, className, classNames, dir, styleNames;
|
2016-03-26 11:32:59 +11:00
|
|
|
|
if ( node && node !== root ) {
|
2018-07-12 13:35:20 +10:00
|
|
|
|
path = getPath( node.parentNode, root, config );
|
2016-03-22 17:57:00 +11:00
|
|
|
|
if ( node.nodeType === ELEMENT_NODE ) {
|
|
|
|
|
path += ( path ? '>' : '' ) + node.nodeName;
|
|
|
|
|
if ( id = node.id ) {
|
|
|
|
|
path += '#' + id;
|
|
|
|
|
}
|
|
|
|
|
if ( className = node.className.trim() ) {
|
|
|
|
|
classNames = className.split( /\s\s*/ );
|
|
|
|
|
classNames.sort();
|
|
|
|
|
path += '.';
|
|
|
|
|
path += classNames.join( '.' );
|
|
|
|
|
}
|
|
|
|
|
if ( dir = node.dir ) {
|
|
|
|
|
path += '[dir=' + dir + ']';
|
|
|
|
|
}
|
2016-07-07 16:09:22 +10:00
|
|
|
|
if ( classNames ) {
|
2018-07-12 13:35:20 +10:00
|
|
|
|
styleNames = config.classNames;
|
|
|
|
|
if ( indexOf.call( classNames, styleNames.highlight ) > -1 ) {
|
2016-07-07 16:09:22 +10:00
|
|
|
|
path += '[backgroundColor=' +
|
|
|
|
|
node.style.backgroundColor.replace( / /g,'' ) + ']';
|
2016-07-06 13:27:22 -04:00
|
|
|
|
}
|
2018-07-12 13:35:20 +10:00
|
|
|
|
if ( indexOf.call( classNames, styleNames.colour ) > -1 ) {
|
2016-07-07 16:09:22 +10:00
|
|
|
|
path += '[color=' +
|
|
|
|
|
node.style.color.replace( / /g,'' ) + ']';
|
2016-07-06 13:27:22 -04:00
|
|
|
|
}
|
2018-07-12 13:35:20 +10:00
|
|
|
|
if ( indexOf.call( classNames, styleNames.fontFamily ) > -1 ) {
|
2016-07-07 16:09:22 +10:00
|
|
|
|
path += '[fontFamily=' +
|
|
|
|
|
node.style.fontFamily.replace( / /g,'' ) + ']';
|
2016-07-06 13:27:22 -04:00
|
|
|
|
}
|
2018-07-12 13:35:20 +10:00
|
|
|
|
if ( indexOf.call( classNames, styleNames.fontSize ) > -1 ) {
|
2016-07-06 13:27:22 -04:00
|
|
|
|
path += '[fontSize=' + node.style.fontSize + ']';
|
|
|
|
|
}
|
|
|
|
|
}
|
2015-05-09 17:14:51 +07:00
|
|
|
|
}
|
2013-04-08 13:27:06 +10:00
|
|
|
|
}
|
|
|
|
|
return path;
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
function getLength ( node ) {
|
|
|
|
|
var nodeType = node.nodeType;
|
2017-07-15 20:33:32 +10:00
|
|
|
|
return nodeType === ELEMENT_NODE || nodeType === DOCUMENT_FRAGMENT_NODE ?
|
2013-04-08 13:27:06 +10:00
|
|
|
|
node.childNodes.length : node.length || 0;
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
function detach ( node ) {
|
|
|
|
|
var parent = node.parentNode;
|
|
|
|
|
if ( parent ) {
|
|
|
|
|
parent.removeChild( node );
|
|
|
|
|
}
|
|
|
|
|
return node;
|
|
|
|
|
}
|
|
|
|
|
function replaceWith ( node, node2 ) {
|
|
|
|
|
var parent = node.parentNode;
|
|
|
|
|
if ( parent ) {
|
|
|
|
|
parent.replaceChild( node2, node );
|
|
|
|
|
}
|
|
|
|
|
}
|
|
|
|
|
function empty ( node ) {
|
|
|
|
|
var frag = node.ownerDocument.createDocumentFragment(),
|
|
|
|
|
childNodes = node.childNodes,
|
|
|
|
|
l = childNodes ? childNodes.length : 0;
|
|
|
|
|
while ( l-- ) {
|
|
|
|
|
frag.appendChild( node.firstChild );
|
|
|
|
|
}
|
|
|
|
|
return frag;
|
|
|
|
|
}
|
|
|
|
|
|
2014-10-02 16:36:39 +07:00
|
|
|
|
function createElement ( doc, tag, props, children ) {
|
|
|
|
|
var el = doc.createElement( tag ),
|
|
|
|
|
attr, value, i, l;
|
|
|
|
|
if ( props instanceof Array ) {
|
|
|
|
|
children = props;
|
|
|
|
|
props = null;
|
|
|
|
|
}
|
|
|
|
|
if ( props ) {
|
|
|
|
|
for ( attr in props ) {
|
|
|
|
|
value = props[ attr ];
|
|
|
|
|
if ( value !== undefined ) {
|
2020-01-23 10:42:44 +11:00
|
|
|
|
el.setAttribute( attr, value );
|
2014-10-02 16:36:39 +07:00
|
|
|
|
}
|
|
|
|
|
}
|
|
|
|
|
}
|
|
|
|
|
if ( children ) {
|
|
|
|
|
for ( i = 0, l = children.length; i < l; i += 1 ) {
|
|
|
|
|
el.appendChild( children[i] );
|
|
|
|
|
}
|
|
|
|
|
}
|
|
|
|
|
return el;
|
|
|
|
|
}
|
|
|
|
|
|
2016-03-22 17:57:00 +11:00
|
|
|
|
function fixCursor ( node, root ) {
|
2013-04-08 13:27:06 +10:00
|
|
|
|
// In Webkit and Gecko, block level elements are collapsed and
|
2020-04-14 06:28:48 +10:00
|
|
|
|
// unfocusable if they have no content. To remedy this, a <BR> must be
|
2013-04-08 13:27:06 +10:00
|
|
|
|
// inserted. In Opera and IE, we just need a textnode in order for the
|
|
|
|
|
// cursor to appear.
|
2016-12-07 17:54:13 +11:00
|
|
|
|
var self = root.__squire__;
|
|
|
|
|
var doc = node.ownerDocument;
|
|
|
|
|
var originalNode = node;
|
|
|
|
|
var fixer, child;
|
2013-04-08 13:27:06 +10:00
|
|
|
|
|
2016-03-22 17:57:00 +11:00
|
|
|
|
if ( node === root ) {
|
2013-04-08 13:27:06 +10:00
|
|
|
|
if ( !( child = node.firstChild ) || child.nodeName === 'BR' ) {
|
2016-12-07 17:54:13 +11:00
|
|
|
|
fixer = self.createDefaultBlock();
|
2013-04-08 13:27:06 +10:00
|
|
|
|
if ( child ) {
|
|
|
|
|
node.replaceChild( fixer, child );
|
|
|
|
|
}
|
|
|
|
|
else {
|
|
|
|
|
node.appendChild( fixer );
|
|
|
|
|
}
|
|
|
|
|
node = fixer;
|
|
|
|
|
fixer = null;
|
|
|
|
|
}
|
|
|
|
|
}
|
|
|
|
|
|
2016-04-07 13:08:45 +10:00
|
|
|
|
if ( node.nodeType === TEXT_NODE ) {
|
|
|
|
|
return originalNode;
|
|
|
|
|
}
|
|
|
|
|
|
2013-04-08 13:27:06 +10:00
|
|
|
|
if ( isInline( node ) ) {
|
2014-08-28 08:10:31 +07:00
|
|
|
|
child = node.firstChild;
|
|
|
|
|
while ( cantFocusEmptyTextNodes && child &&
|
|
|
|
|
child.nodeType === TEXT_NODE && !child.data ) {
|
|
|
|
|
node.removeChild( child );
|
|
|
|
|
child = node.firstChild;
|
|
|
|
|
}
|
|
|
|
|
if ( !child ) {
|
2013-04-08 13:27:06 +10:00
|
|
|
|
if ( cantFocusEmptyTextNodes ) {
|
2014-12-25 15:09:46 +07:00
|
|
|
|
fixer = doc.createTextNode( ZWS );
|
2016-12-07 17:54:13 +11:00
|
|
|
|
self._didAddZWS();
|
2013-04-08 13:27:06 +10:00
|
|
|
|
} else {
|
|
|
|
|
fixer = doc.createTextNode( '' );
|
|
|
|
|
}
|
|
|
|
|
}
|
2020-03-11 14:48:30 +11:00
|
|
|
|
} else if ( !node.querySelector( 'BR' ) ) {
|
|
|
|
|
fixer = createElement( doc, 'BR' );
|
|
|
|
|
while ( ( child = node.lastElementChild ) && !isInline( child ) ) {
|
|
|
|
|
node = child;
|
2013-04-08 13:27:06 +10:00
|
|
|
|
}
|
|
|
|
|
}
|
|
|
|
|
if ( fixer ) {
|
2016-03-24 22:04:42 +11:00
|
|
|
|
try {
|
|
|
|
|
node.appendChild( fixer );
|
|
|
|
|
} catch ( error ) {
|
2016-12-07 17:54:13 +11:00
|
|
|
|
self.didError({
|
2016-03-24 22:04:42 +11:00
|
|
|
|
name: 'Squire: fixCursor – ' + error,
|
|
|
|
|
message: 'Parent: ' + node.nodeName + '/' + node.innerHTML +
|
|
|
|
|
' appendChild: ' + fixer.nodeName
|
|
|
|
|
});
|
|
|
|
|
}
|
2013-04-08 13:27:06 +10:00
|
|
|
|
}
|
|
|
|
|
|
2016-03-22 17:57:00 +11:00
|
|
|
|
return originalNode;
|
2013-04-08 13:27:06 +10:00
|
|
|
|
}
|
|
|
|
|
|
2014-10-02 16:36:39 +07:00
|
|
|
|
// Recursively examine container nodes and wrap any inline children.
|
2016-03-22 17:57:00 +11:00
|
|
|
|
function fixContainer ( container, root ) {
|
2016-12-07 17:54:13 +11:00
|
|
|
|
var children = container.childNodes;
|
|
|
|
|
var doc = container.ownerDocument;
|
|
|
|
|
var wrapper = null;
|
|
|
|
|
var i, l, child, isBR;
|
2015-05-09 17:14:51 +07:00
|
|
|
|
|
2014-10-02 16:36:39 +07:00
|
|
|
|
for ( i = 0, l = children.length; i < l; i += 1 ) {
|
|
|
|
|
child = children[i];
|
|
|
|
|
isBR = child.nodeName === 'BR';
|
|
|
|
|
if ( !isBR && isInline( child ) ) {
|
2015-05-09 17:14:51 +07:00
|
|
|
|
if ( !wrapper ) {
|
2020-01-23 10:42:44 +11:00
|
|
|
|
wrapper = createElement( doc, 'div' );
|
2015-05-09 17:14:51 +07:00
|
|
|
|
}
|
2014-10-02 16:36:39 +07:00
|
|
|
|
wrapper.appendChild( child );
|
|
|
|
|
i -= 1;
|
|
|
|
|
l -= 1;
|
|
|
|
|
} else if ( isBR || wrapper ) {
|
2015-05-09 17:14:51 +07:00
|
|
|
|
if ( !wrapper ) {
|
2020-01-23 10:42:44 +11:00
|
|
|
|
wrapper = createElement( doc, 'div' );
|
2015-05-09 17:14:51 +07:00
|
|
|
|
}
|
2016-03-22 17:57:00 +11:00
|
|
|
|
fixCursor( wrapper, root );
|
2014-10-02 16:36:39 +07:00
|
|
|
|
if ( isBR ) {
|
|
|
|
|
container.replaceChild( wrapper, child );
|
|
|
|
|
} else {
|
|
|
|
|
container.insertBefore( wrapper, child );
|
|
|
|
|
i += 1;
|
|
|
|
|
l += 1;
|
|
|
|
|
}
|
|
|
|
|
wrapper = null;
|
|
|
|
|
}
|
|
|
|
|
if ( isContainer( child ) ) {
|
2016-03-22 17:57:00 +11:00
|
|
|
|
fixContainer( child, root );
|
2014-10-02 16:36:39 +07:00
|
|
|
|
}
|
|
|
|
|
}
|
|
|
|
|
if ( wrapper ) {
|
2016-03-22 17:57:00 +11:00
|
|
|
|
container.appendChild( fixCursor( wrapper, root ) );
|
2014-10-02 16:36:39 +07:00
|
|
|
|
}
|
|
|
|
|
return container;
|
|
|
|
|
}
|
|
|
|
|
|
2016-03-22 17:57:00 +11:00
|
|
|
|
function split ( node, offset, stopNode, root ) {
|
2013-04-08 13:27:06 +10:00
|
|
|
|
var nodeType = node.nodeType,
|
|
|
|
|
parent, clone, next;
|
2013-04-10 13:43:31 +10:00
|
|
|
|
if ( nodeType === TEXT_NODE && node !== stopNode ) {
|
2016-03-22 17:57:00 +11:00
|
|
|
|
return split(
|
|
|
|
|
node.parentNode, node.splitText( offset ), stopNode, root );
|
2013-04-08 13:27:06 +10:00
|
|
|
|
}
|
|
|
|
|
if ( nodeType === ELEMENT_NODE ) {
|
|
|
|
|
if ( typeof( offset ) === 'number' ) {
|
|
|
|
|
offset = offset < node.childNodes.length ?
|
|
|
|
|
node.childNodes[ offset ] : null;
|
|
|
|
|
}
|
|
|
|
|
if ( node === stopNode ) {
|
|
|
|
|
return offset;
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
// Clone node without children
|
2013-06-20 23:15:18 +10:00
|
|
|
|
parent = node.parentNode;
|
2013-04-08 13:27:06 +10:00
|
|
|
|
clone = node.cloneNode( false );
|
|
|
|
|
|
|
|
|
|
// Add right-hand siblings to the clone
|
|
|
|
|
while ( offset ) {
|
|
|
|
|
next = offset.nextSibling;
|
|
|
|
|
clone.appendChild( offset );
|
|
|
|
|
offset = next;
|
|
|
|
|
}
|
|
|
|
|
|
2015-03-29 18:38:40 +11:00
|
|
|
|
// Maintain li numbering if inside a quote.
|
2016-03-22 17:57:00 +11:00
|
|
|
|
if ( node.nodeName === 'OL' &&
|
|
|
|
|
getNearest( node, root, 'BLOCKQUOTE' ) ) {
|
2015-02-06 14:19:06 +07:00
|
|
|
|
clone.start = ( +node.start || 1 ) + node.childNodes.length - 1;
|
|
|
|
|
}
|
|
|
|
|
|
2013-04-08 13:27:06 +10:00
|
|
|
|
// DO NOT NORMALISE. This may undo the fixCursor() call
|
|
|
|
|
// of a node lower down the tree!
|
|
|
|
|
|
|
|
|
|
// We need something in the element in order for the cursor to appear.
|
2016-03-22 17:57:00 +11:00
|
|
|
|
fixCursor( node, root );
|
|
|
|
|
fixCursor( clone, root );
|
2013-04-08 13:27:06 +10:00
|
|
|
|
|
|
|
|
|
// Inject clone after original node
|
|
|
|
|
if ( next = node.nextSibling ) {
|
|
|
|
|
parent.insertBefore( clone, next );
|
|
|
|
|
} else {
|
|
|
|
|
parent.appendChild( clone );
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
// Keep on splitting up the tree
|
2016-03-22 17:57:00 +11:00
|
|
|
|
return split( parent, clone, stopNode, root );
|
2013-04-08 13:27:06 +10:00
|
|
|
|
}
|
2013-04-10 13:43:31 +10:00
|
|
|
|
return offset;
|
2013-04-08 13:27:06 +10:00
|
|
|
|
}
|
|
|
|
|
|
2016-07-14 12:15:06 +10:00
|
|
|
|
function _mergeInlines ( node, fakeRange ) {
|
2013-04-08 13:27:06 +10:00
|
|
|
|
var children = node.childNodes,
|
|
|
|
|
l = children.length,
|
|
|
|
|
frags = [],
|
|
|
|
|
child, prev, len;
|
|
|
|
|
while ( l-- ) {
|
|
|
|
|
child = children[l];
|
|
|
|
|
prev = l && children[ l - 1 ];
|
|
|
|
|
if ( l && isInline( child ) && areAlike( child, prev ) &&
|
|
|
|
|
!leafNodeNames[ child.nodeName ] ) {
|
2016-07-14 12:15:06 +10:00
|
|
|
|
if ( fakeRange.startContainer === child ) {
|
|
|
|
|
fakeRange.startContainer = prev;
|
|
|
|
|
fakeRange.startOffset += getLength( prev );
|
2013-04-08 13:27:06 +10:00
|
|
|
|
}
|
2016-07-14 12:15:06 +10:00
|
|
|
|
if ( fakeRange.endContainer === child ) {
|
|
|
|
|
fakeRange.endContainer = prev;
|
|
|
|
|
fakeRange.endOffset += getLength( prev );
|
2013-04-08 13:27:06 +10:00
|
|
|
|
}
|
2016-07-14 12:15:06 +10:00
|
|
|
|
if ( fakeRange.startContainer === node ) {
|
|
|
|
|
if ( fakeRange.startOffset > l ) {
|
|
|
|
|
fakeRange.startOffset -= 1;
|
2013-04-08 13:27:06 +10:00
|
|
|
|
}
|
2016-07-14 12:15:06 +10:00
|
|
|
|
else if ( fakeRange.startOffset === l ) {
|
|
|
|
|
fakeRange.startContainer = prev;
|
|
|
|
|
fakeRange.startOffset = getLength( prev );
|
2013-04-08 13:27:06 +10:00
|
|
|
|
}
|
|
|
|
|
}
|
2016-07-14 12:15:06 +10:00
|
|
|
|
if ( fakeRange.endContainer === node ) {
|
|
|
|
|
if ( fakeRange.endOffset > l ) {
|
|
|
|
|
fakeRange.endOffset -= 1;
|
2013-04-08 13:27:06 +10:00
|
|
|
|
}
|
2016-07-14 12:15:06 +10:00
|
|
|
|
else if ( fakeRange.endOffset === l ) {
|
|
|
|
|
fakeRange.endContainer = prev;
|
|
|
|
|
fakeRange.endOffset = getLength( prev );
|
2013-04-08 13:27:06 +10:00
|
|
|
|
}
|
|
|
|
|
}
|
|
|
|
|
detach( child );
|
|
|
|
|
if ( child.nodeType === TEXT_NODE ) {
|
2014-04-16 18:06:10 +10:00
|
|
|
|
prev.appendData( child.data );
|
2013-04-08 13:27:06 +10:00
|
|
|
|
}
|
|
|
|
|
else {
|
|
|
|
|
frags.push( empty( child ) );
|
|
|
|
|
}
|
|
|
|
|
}
|
|
|
|
|
else if ( child.nodeType === ELEMENT_NODE ) {
|
|
|
|
|
len = frags.length;
|
|
|
|
|
while ( len-- ) {
|
|
|
|
|
child.appendChild( frags.pop() );
|
|
|
|
|
}
|
2016-07-14 12:15:06 +10:00
|
|
|
|
_mergeInlines( child, fakeRange );
|
2013-04-08 13:27:06 +10:00
|
|
|
|
}
|
|
|
|
|
}
|
|
|
|
|
}
|
|
|
|
|
|
2016-07-14 12:15:06 +10:00
|
|
|
|
function mergeInlines ( node, range ) {
|
|
|
|
|
if ( node.nodeType === TEXT_NODE ) {
|
|
|
|
|
node = node.parentNode;
|
|
|
|
|
}
|
|
|
|
|
if ( node.nodeType === ELEMENT_NODE ) {
|
|
|
|
|
var fakeRange = {
|
|
|
|
|
startContainer: range.startContainer,
|
|
|
|
|
startOffset: range.startOffset,
|
|
|
|
|
endContainer: range.endContainer,
|
|
|
|
|
endOffset: range.endOffset
|
|
|
|
|
};
|
|
|
|
|
_mergeInlines( node, fakeRange );
|
|
|
|
|
range.setStart( fakeRange.startContainer, fakeRange.startOffset );
|
|
|
|
|
range.setEnd( fakeRange.endContainer, fakeRange.endOffset );
|
|
|
|
|
}
|
|
|
|
|
}
|
|
|
|
|
|
2017-07-15 20:33:32 +10:00
|
|
|
|
function mergeWithBlock ( block, next, range, root ) {
|
|
|
|
|
var container = next;
|
|
|
|
|
var parent, last, offset;
|
|
|
|
|
while ( ( parent = container.parentNode ) &&
|
|
|
|
|
parent !== root &&
|
|
|
|
|
parent.nodeType === ELEMENT_NODE &&
|
|
|
|
|
parent.childNodes.length === 1 ) {
|
|
|
|
|
container = parent;
|
2013-04-08 13:27:06 +10:00
|
|
|
|
}
|
|
|
|
|
detach( container );
|
|
|
|
|
|
|
|
|
|
offset = block.childNodes.length;
|
|
|
|
|
|
|
|
|
|
// Remove extra <BR> fixer if present.
|
|
|
|
|
last = block.lastChild;
|
|
|
|
|
if ( last && last.nodeName === 'BR' ) {
|
|
|
|
|
block.removeChild( last );
|
|
|
|
|
offset -= 1;
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
block.appendChild( empty( next ) );
|
|
|
|
|
|
2016-07-14 12:15:06 +10:00
|
|
|
|
range.setStart( block, offset );
|
2013-04-08 13:27:06 +10:00
|
|
|
|
range.collapse( true );
|
2016-07-14 12:15:06 +10:00
|
|
|
|
mergeInlines( block, range );
|
2013-04-08 13:27:06 +10:00
|
|
|
|
}
|
|
|
|
|
|
2016-03-22 17:57:00 +11:00
|
|
|
|
function mergeContainers ( node, root ) {
|
2013-04-08 13:27:06 +10:00
|
|
|
|
var prev = node.previousSibling,
|
2014-04-07 13:05:44 +10:00
|
|
|
|
first = node.firstChild,
|
2014-06-02 10:17:00 +10:00
|
|
|
|
doc = node.ownerDocument,
|
2014-05-23 15:26:47 +10:00
|
|
|
|
isListItem = ( node.nodeName === 'LI' ),
|
2014-05-27 11:31:06 +10:00
|
|
|
|
needsFix, block;
|
2014-04-07 13:05:44 +10:00
|
|
|
|
|
|
|
|
|
// Do not merge LIs, unless it only contains a UL
|
|
|
|
|
if ( isListItem && ( !first || !/^[OU]L$/.test( first.nodeName ) ) ) {
|
|
|
|
|
return;
|
|
|
|
|
}
|
|
|
|
|
|
2014-05-23 15:26:47 +10:00
|
|
|
|
if ( prev && areAlike( prev, node ) ) {
|
|
|
|
|
if ( !isContainer( prev ) ) {
|
|
|
|
|
if ( isListItem ) {
|
2015-03-29 18:35:03 +11:00
|
|
|
|
block = createElement( doc, 'DIV' );
|
2014-05-23 15:26:47 +10:00
|
|
|
|
block.appendChild( empty( prev ) );
|
|
|
|
|
prev.appendChild( block );
|
|
|
|
|
} else {
|
|
|
|
|
return;
|
|
|
|
|
}
|
|
|
|
|
}
|
2013-04-08 13:27:06 +10:00
|
|
|
|
detach( node );
|
2014-05-27 11:31:06 +10:00
|
|
|
|
needsFix = !isContainer( node );
|
2013-04-08 13:27:06 +10:00
|
|
|
|
prev.appendChild( empty( node ) );
|
2014-05-27 11:31:06 +10:00
|
|
|
|
if ( needsFix ) {
|
2016-03-22 17:57:00 +11:00
|
|
|
|
fixContainer( prev, root );
|
2014-05-27 11:31:06 +10:00
|
|
|
|
}
|
2013-04-08 13:27:06 +10:00
|
|
|
|
if ( first ) {
|
2016-03-22 17:57:00 +11:00
|
|
|
|
mergeContainers( first, root );
|
2013-04-08 13:27:06 +10:00
|
|
|
|
}
|
2014-04-07 13:05:44 +10:00
|
|
|
|
} else if ( isListItem ) {
|
2015-03-29 18:35:03 +11:00
|
|
|
|
prev = createElement( doc, 'DIV' );
|
2014-04-07 13:05:44 +10:00
|
|
|
|
node.insertBefore( prev, first );
|
2016-03-22 17:57:00 +11:00
|
|
|
|
fixCursor( prev, root );
|
2013-04-08 13:27:06 +10:00
|
|
|
|
}
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
var getNodeBefore = function ( node, offset ) {
|
|
|
|
|
var children = node.childNodes;
|
|
|
|
|
while ( offset && node.nodeType === ELEMENT_NODE ) {
|
|
|
|
|
node = children[ offset - 1 ];
|
|
|
|
|
children = node.childNodes;
|
|
|
|
|
offset = children.length;
|
|
|
|
|
}
|
|
|
|
|
return node;
|
|
|
|
|
};
|
|
|
|
|
|
|
|
|
|
var getNodeAfter = function ( node, offset ) {
|
|
|
|
|
if ( node.nodeType === ELEMENT_NODE ) {
|
|
|
|
|
var children = node.childNodes;
|
|
|
|
|
if ( offset < children.length ) {
|
|
|
|
|
node = children[ offset ];
|
|
|
|
|
} else {
|
|
|
|
|
while ( node && !node.nextSibling ) {
|
|
|
|
|
node = node.parentNode;
|
|
|
|
|
}
|
|
|
|
|
if ( node ) { node = node.nextSibling; }
|
|
|
|
|
}
|
|
|
|
|
}
|
|
|
|
|
return node;
|
|
|
|
|
};
|
|
|
|
|
|
2013-06-20 21:03:01 +10:00
|
|
|
|
// ---
|
2013-04-08 13:27:06 +10:00
|
|
|
|
|
2013-06-20 21:03:01 +10:00
|
|
|
|
var insertNodeInRange = function ( range, node ) {
|
2013-04-08 13:27:06 +10:00
|
|
|
|
// Insert at start.
|
2013-06-20 21:03:01 +10:00
|
|
|
|
var startContainer = range.startContainer,
|
|
|
|
|
startOffset = range.startOffset,
|
|
|
|
|
endContainer = range.endContainer,
|
|
|
|
|
endOffset = range.endOffset,
|
2013-04-08 13:27:06 +10:00
|
|
|
|
parent, children, childCount, afterSplit;
|
|
|
|
|
|
|
|
|
|
// If part way through a text node, split it.
|
|
|
|
|
if ( startContainer.nodeType === TEXT_NODE ) {
|
|
|
|
|
parent = startContainer.parentNode;
|
|
|
|
|
children = parent.childNodes;
|
|
|
|
|
if ( startOffset === startContainer.length ) {
|
|
|
|
|
startOffset = indexOf.call( children, startContainer ) + 1;
|
2013-06-20 21:03:01 +10:00
|
|
|
|
if ( range.collapsed ) {
|
2013-04-08 13:27:06 +10:00
|
|
|
|
endContainer = parent;
|
|
|
|
|
endOffset = startOffset;
|
|
|
|
|
}
|
|
|
|
|
} else {
|
|
|
|
|
if ( startOffset ) {
|
|
|
|
|
afterSplit = startContainer.splitText( startOffset );
|
|
|
|
|
if ( endContainer === startContainer ) {
|
|
|
|
|
endOffset -= startOffset;
|
|
|
|
|
endContainer = afterSplit;
|
|
|
|
|
}
|
|
|
|
|
else if ( endContainer === parent ) {
|
|
|
|
|
endOffset += 1;
|
|
|
|
|
}
|
|
|
|
|
startContainer = afterSplit;
|
|
|
|
|
}
|
|
|
|
|
startOffset = indexOf.call( children, startContainer );
|
|
|
|
|
}
|
|
|
|
|
startContainer = parent;
|
|
|
|
|
} else {
|
|
|
|
|
children = startContainer.childNodes;
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
childCount = children.length;
|
|
|
|
|
|
2015-09-28 15:48:23 +02:00
|
|
|
|
if ( startOffset === childCount ) {
|
2013-04-08 13:27:06 +10:00
|
|
|
|
startContainer.appendChild( node );
|
|
|
|
|
} else {
|
|
|
|
|
startContainer.insertBefore( node, children[ startOffset ] );
|
|
|
|
|
}
|
2015-02-06 14:09:37 +07:00
|
|
|
|
|
2013-04-08 13:27:06 +10:00
|
|
|
|
if ( startContainer === endContainer ) {
|
|
|
|
|
endOffset += children.length - childCount;
|
|
|
|
|
}
|
|
|
|
|
|
2013-06-20 21:03:01 +10:00
|
|
|
|
range.setStart( startContainer, startOffset );
|
|
|
|
|
range.setEnd( endContainer, endOffset );
|
2013-04-08 13:27:06 +10:00
|
|
|
|
};
|
|
|
|
|
|
2016-03-22 17:57:00 +11:00
|
|
|
|
var extractContentsOfRange = function ( range, common, root ) {
|
2013-06-20 21:03:01 +10:00
|
|
|
|
var startContainer = range.startContainer,
|
|
|
|
|
startOffset = range.startOffset,
|
|
|
|
|
endContainer = range.endContainer,
|
|
|
|
|
endOffset = range.endOffset;
|
2013-04-08 13:27:06 +10:00
|
|
|
|
|
|
|
|
|
if ( !common ) {
|
2013-06-20 21:03:01 +10:00
|
|
|
|
common = range.commonAncestorContainer;
|
2013-04-08 13:27:06 +10:00
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
if ( common.nodeType === TEXT_NODE ) {
|
|
|
|
|
common = common.parentNode;
|
|
|
|
|
}
|
|
|
|
|
|
2016-03-22 17:57:00 +11:00
|
|
|
|
var endNode = split( endContainer, endOffset, common, root ),
|
|
|
|
|
startNode = split( startContainer, startOffset, common, root ),
|
2013-04-08 13:27:06 +10:00
|
|
|
|
frag = common.ownerDocument.createDocumentFragment(),
|
2018-10-05 16:25:28 +10:00
|
|
|
|
next, before, after, beforeText, afterText;
|
2013-04-08 13:27:06 +10:00
|
|
|
|
|
|
|
|
|
// End node will be null if at end of child nodes list.
|
|
|
|
|
while ( startNode !== endNode ) {
|
|
|
|
|
next = startNode.nextSibling;
|
|
|
|
|
frag.appendChild( startNode );
|
|
|
|
|
startNode = next;
|
|
|
|
|
}
|
|
|
|
|
|
2015-03-04 13:15:21 +08:00
|
|
|
|
startContainer = common;
|
|
|
|
|
startOffset = endNode ?
|
2013-04-08 13:27:06 +10:00
|
|
|
|
indexOf.call( common.childNodes, endNode ) :
|
2015-03-04 13:15:21 +08:00
|
|
|
|
common.childNodes.length;
|
|
|
|
|
|
|
|
|
|
// Merge text nodes if adjacent. IE10 in particular will not focus
|
|
|
|
|
// between two text nodes
|
|
|
|
|
after = common.childNodes[ startOffset ];
|
|
|
|
|
before = after && after.previousSibling;
|
|
|
|
|
if ( before &&
|
|
|
|
|
before.nodeType === TEXT_NODE &&
|
|
|
|
|
after.nodeType === TEXT_NODE ) {
|
|
|
|
|
startContainer = before;
|
|
|
|
|
startOffset = before.length;
|
2018-10-05 16:25:28 +10:00
|
|
|
|
beforeText = before.data;
|
|
|
|
|
afterText = after.data;
|
|
|
|
|
|
|
|
|
|
// If we now have two adjacent spaces, the second one needs to become
|
|
|
|
|
// a nbsp, otherwise the browser will swallow it due to HTML whitespace
|
|
|
|
|
// collapsing.
|
|
|
|
|
if ( beforeText.charAt( beforeText.length - 1 ) === ' ' &&
|
|
|
|
|
afterText.charAt( 0 ) === ' ' ) {
|
|
|
|
|
afterText = ' ' + afterText.slice( 1 ); // nbsp
|
|
|
|
|
}
|
|
|
|
|
before.appendData( afterText );
|
2015-03-04 13:15:21 +08:00
|
|
|
|
detach( after );
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
range.setStart( startContainer, startOffset );
|
2013-06-20 21:03:01 +10:00
|
|
|
|
range.collapse( true );
|
2013-04-08 13:27:06 +10:00
|
|
|
|
|
2016-03-22 17:57:00 +11:00
|
|
|
|
fixCursor( common, root );
|
2013-04-08 13:27:06 +10:00
|
|
|
|
|
|
|
|
|
return frag;
|
|
|
|
|
};
|
|
|
|
|
|
2016-03-22 17:57:00 +11:00
|
|
|
|
var deleteContentsOfRange = function ( range, root ) {
|
2017-01-09 14:07:20 +11:00
|
|
|
|
var startBlock = getStartBlockOfRange( range, root );
|
|
|
|
|
var endBlock = getEndBlockOfRange( range, root );
|
|
|
|
|
var needsMerge = ( startBlock !== endBlock );
|
|
|
|
|
var frag, child;
|
2013-04-08 13:27:06 +10:00
|
|
|
|
|
2017-01-09 14:07:20 +11:00
|
|
|
|
// Move boundaries up as much as possible without exiting block,
|
|
|
|
|
// to reduce need to split.
|
2017-01-13 15:15:59 +11:00
|
|
|
|
moveRangeBoundariesDownTree( range );
|
|
|
|
|
moveRangeBoundariesUpTree( range, startBlock, endBlock, root );
|
2015-07-25 18:03:51 -07:00
|
|
|
|
|
2013-04-08 13:27:06 +10:00
|
|
|
|
// Remove selected range
|
2017-01-09 14:07:20 +11:00
|
|
|
|
frag = extractContentsOfRange( range, null, root );
|
2013-04-08 13:27:06 +10:00
|
|
|
|
|
2017-01-09 14:07:20 +11:00
|
|
|
|
// Move boundaries back down tree as far as possible.
|
2014-12-26 14:48:13 +07:00
|
|
|
|
moveRangeBoundariesDownTree( range );
|
|
|
|
|
|
2013-04-08 13:27:06 +10:00
|
|
|
|
// If we split into two different blocks, merge the blocks.
|
2015-07-25 18:03:51 -07:00
|
|
|
|
if ( needsMerge ) {
|
2017-01-09 14:07:20 +11:00
|
|
|
|
// endBlock will have been split, so need to refetch
|
2016-03-22 17:57:00 +11:00
|
|
|
|
endBlock = getEndBlockOfRange( range, root );
|
2015-07-25 18:03:51 -07:00
|
|
|
|
if ( startBlock && endBlock && startBlock !== endBlock ) {
|
2017-07-15 20:33:32 +10:00
|
|
|
|
mergeWithBlock( startBlock, endBlock, range, root );
|
2015-07-25 18:03:51 -07:00
|
|
|
|
}
|
2013-04-08 13:27:06 +10:00
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
// Ensure block has necessary children
|
|
|
|
|
if ( startBlock ) {
|
2016-03-22 17:57:00 +11:00
|
|
|
|
fixCursor( startBlock, root );
|
2013-04-08 13:27:06 +10:00
|
|
|
|
}
|
|
|
|
|
|
2016-03-22 17:57:00 +11:00
|
|
|
|
// Ensure root has a block-level element in it.
|
2017-01-09 14:07:20 +11:00
|
|
|
|
child = root.firstChild;
|
2013-04-08 13:27:06 +10:00
|
|
|
|
if ( !child || child.nodeName === 'BR' ) {
|
2016-03-22 17:57:00 +11:00
|
|
|
|
fixCursor( root, root );
|
|
|
|
|
range.selectNodeContents( root.firstChild );
|
2015-07-25 18:03:51 -07:00
|
|
|
|
} else {
|
2017-01-09 14:07:20 +11:00
|
|
|
|
range.collapse( true );
|
2013-04-08 13:27:06 +10:00
|
|
|
|
}
|
2016-03-11 15:22:49 +11:00
|
|
|
|
return frag;
|
2013-04-08 13:27:06 +10:00
|
|
|
|
};
|
|
|
|
|
|
|
|
|
|
// ---
|
|
|
|
|
|
2017-07-15 20:33:32 +10:00
|
|
|
|
// Contents of range will be deleted.
|
|
|
|
|
// After method, range will be around inserted content
|
2016-03-22 17:57:00 +11:00
|
|
|
|
var insertTreeFragmentIntoRange = function ( range, frag, root ) {
|
2020-02-24 14:43:40 +11:00
|
|
|
|
var firstInFragIsInline = frag.firstChild && isInline( frag.firstChild );
|
2017-07-15 20:33:32 +10:00
|
|
|
|
var node, block, blockContentsAfterSplit, stopPoint, container, offset;
|
2017-11-01 10:41:53 +11:00
|
|
|
|
var replaceBlock, firstBlockInFrag, nodeAfterSplit, nodeBeforeSplit;
|
|
|
|
|
var tempRange;
|
2017-07-15 20:33:32 +10:00
|
|
|
|
|
|
|
|
|
// Fixup content: ensure no top-level inline, and add cursor fix elements.
|
|
|
|
|
fixContainer( frag, root );
|
|
|
|
|
node = frag;
|
|
|
|
|
while ( ( node = getNextBlock( node, root ) ) ) {
|
|
|
|
|
fixCursor( node, root );
|
2013-04-08 13:27:06 +10:00
|
|
|
|
}
|
|
|
|
|
|
2017-07-15 20:33:32 +10:00
|
|
|
|
// Delete any selected content.
|
2013-06-20 21:03:01 +10:00
|
|
|
|
if ( !range.collapsed ) {
|
2016-03-22 17:57:00 +11:00
|
|
|
|
deleteContentsOfRange( range, root );
|
2013-04-08 13:27:06 +10:00
|
|
|
|
}
|
|
|
|
|
|
2017-07-15 20:33:32 +10:00
|
|
|
|
// Move range down into text nodes.
|
2013-06-20 21:03:01 +10:00
|
|
|
|
moveRangeBoundariesDownTree( range );
|
2017-07-15 20:33:32 +10:00
|
|
|
|
range.collapse( false ); // collapse to end
|
|
|
|
|
|
|
|
|
|
// Where will we split up to? First blockquote parent, otherwise root.
|
|
|
|
|
stopPoint = getNearest( range.endContainer, root, 'BLOCKQUOTE' ) || root;
|
|
|
|
|
|
|
|
|
|
// Merge the contents of the first block in the frag with the focused block.
|
|
|
|
|
// If there are contents in the block after the focus point, collect this
|
2020-02-24 14:43:40 +11:00
|
|
|
|
// up to insert in the last block later. This preserves the style that was
|
|
|
|
|
// present in this bit of the page.
|
|
|
|
|
//
|
|
|
|
|
// If the block being inserted into is empty though, replace it instead of
|
|
|
|
|
// merging if the fragment had block contents.
|
|
|
|
|
// e.g. <blockquote><p>Foo</p></blockquote>
|
|
|
|
|
// This seems a reasonable approximation of user intent.
|
|
|
|
|
|
2017-09-04 10:22:49 +10:00
|
|
|
|
block = getStartBlockOfRange( range, root );
|
|
|
|
|
firstBlockInFrag = getNextBlock( frag, frag );
|
2020-02-24 14:43:40 +11:00
|
|
|
|
replaceBlock = !firstInFragIsInline && !!block && isEmptyBlock( block );
|
2017-11-01 10:41:53 +11:00
|
|
|
|
if ( block && firstBlockInFrag && !replaceBlock &&
|
2017-09-05 11:42:54 +10:00
|
|
|
|
// Don't merge table cells or PRE elements into block
|
|
|
|
|
!getNearest( firstBlockInFrag, frag, 'PRE' ) &&
|
2017-09-04 10:22:49 +10:00
|
|
|
|
!getNearest( firstBlockInFrag, frag, 'TABLE' ) ) {
|
2017-07-15 20:33:32 +10:00
|
|
|
|
moveRangeBoundariesUpTree( range, block, block, root );
|
|
|
|
|
range.collapse( true ); // collapse to start
|
|
|
|
|
container = range.endContainer;
|
|
|
|
|
offset = range.endOffset;
|
|
|
|
|
// Remove trailing <br> – we don't want this considered content to be
|
|
|
|
|
// inserted again later
|
|
|
|
|
cleanupBRs( block, root, false );
|
|
|
|
|
if ( isInline( container ) ) {
|
|
|
|
|
// Split up to block parent.
|
2016-03-22 17:57:00 +11:00
|
|
|
|
nodeAfterSplit = split(
|
2017-07-15 20:33:32 +10:00
|
|
|
|
container, offset, getPreviousBlock( container, root ), root );
|
|
|
|
|
container = nodeAfterSplit.parentNode;
|
|
|
|
|
offset = indexOf.call( container.childNodes, nodeAfterSplit );
|
|
|
|
|
}
|
|
|
|
|
if ( /*isBlock( container ) && */offset !== getLength( container ) ) {
|
|
|
|
|
// Collect any inline contents of the block after the range point
|
|
|
|
|
blockContentsAfterSplit =
|
|
|
|
|
root.ownerDocument.createDocumentFragment();
|
|
|
|
|
while ( ( node = container.childNodes[ offset ] ) ) {
|
|
|
|
|
blockContentsAfterSplit.appendChild( node );
|
2015-06-19 10:00:55 +07:00
|
|
|
|
}
|
2013-04-08 13:27:06 +10:00
|
|
|
|
}
|
2017-07-15 20:33:32 +10:00
|
|
|
|
// And merge the first block in.
|
2017-09-04 10:22:49 +10:00
|
|
|
|
mergeWithBlock( container, firstBlockInFrag, range, root );
|
2013-04-08 13:27:06 +10:00
|
|
|
|
|
2017-07-15 20:33:32 +10:00
|
|
|
|
// And where we will insert
|
|
|
|
|
offset = indexOf.call( container.parentNode.childNodes, container ) + 1;
|
|
|
|
|
container = container.parentNode;
|
|
|
|
|
range.setEnd( container, offset );
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
// Is there still any content in the fragment?
|
|
|
|
|
if ( getLength( frag ) ) {
|
2017-11-01 10:41:53 +11:00
|
|
|
|
if ( replaceBlock ) {
|
|
|
|
|
range.setEndBefore( block );
|
|
|
|
|
range.collapse( false );
|
|
|
|
|
detach( block );
|
|
|
|
|
}
|
2017-07-15 20:33:32 +10:00
|
|
|
|
moveRangeBoundariesUpTree( range, stopPoint, stopPoint, root );
|
|
|
|
|
// Now split after block up to blockquote (if a parent) or root
|
|
|
|
|
nodeAfterSplit = split(
|
|
|
|
|
range.endContainer, range.endOffset, stopPoint, root );
|
|
|
|
|
nodeBeforeSplit = nodeAfterSplit ?
|
|
|
|
|
nodeAfterSplit.previousSibling :
|
|
|
|
|
stopPoint.lastChild;
|
|
|
|
|
stopPoint.insertBefore( frag, nodeAfterSplit );
|
|
|
|
|
if ( nodeAfterSplit ) {
|
|
|
|
|
range.setEndBefore( nodeAfterSplit );
|
|
|
|
|
} else {
|
|
|
|
|
range.setEnd( stopPoint, getLength( stopPoint ) );
|
2013-04-08 13:27:06 +10:00
|
|
|
|
}
|
2017-07-15 20:33:32 +10:00
|
|
|
|
block = getEndBlockOfRange( range, root );
|
2013-04-08 13:27:06 +10:00
|
|
|
|
|
2017-07-15 20:33:32 +10:00
|
|
|
|
// Get a reference that won't be invalidated if we merge containers.
|
|
|
|
|
moveRangeBoundariesDownTree( range );
|
|
|
|
|
container = range.endContainer;
|
|
|
|
|
offset = range.endOffset;
|
2015-06-12 17:48:06 +07:00
|
|
|
|
|
|
|
|
|
// Merge inserted containers with edges of split
|
2015-06-17 15:38:12 +07:00
|
|
|
|
if ( nodeAfterSplit && isContainer( nodeAfterSplit ) ) {
|
2016-03-22 17:57:00 +11:00
|
|
|
|
mergeContainers( nodeAfterSplit, root );
|
2013-04-08 13:27:06 +10:00
|
|
|
|
}
|
2017-07-15 20:33:32 +10:00
|
|
|
|
nodeAfterSplit = nodeBeforeSplit && nodeBeforeSplit.nextSibling;
|
|
|
|
|
if ( nodeAfterSplit && isContainer( nodeAfterSplit ) ) {
|
|
|
|
|
mergeContainers( nodeAfterSplit, root );
|
|
|
|
|
}
|
|
|
|
|
range.setEnd( container, offset );
|
|
|
|
|
}
|
2013-04-08 13:27:06 +10:00
|
|
|
|
|
2017-07-15 20:33:32 +10:00
|
|
|
|
// Insert inline content saved from before.
|
|
|
|
|
if ( blockContentsAfterSplit ) {
|
|
|
|
|
tempRange = range.cloneRange();
|
|
|
|
|
mergeWithBlock( block, blockContentsAfterSplit, tempRange, root );
|
|
|
|
|
range.setEnd( tempRange.endContainer, tempRange.endOffset );
|
2013-04-08 13:27:06 +10:00
|
|
|
|
}
|
2017-07-15 20:33:32 +10:00
|
|
|
|
moveRangeBoundariesDownTree( range );
|
2013-04-08 13:27:06 +10:00
|
|
|
|
};
|
|
|
|
|
|
|
|
|
|
// ---
|
|
|
|
|
|
2013-06-20 21:03:01 +10:00
|
|
|
|
var isNodeContainedInRange = function ( range, node, partial ) {
|
|
|
|
|
var nodeRange = node.ownerDocument.createRange();
|
2013-04-08 13:27:06 +10:00
|
|
|
|
|
|
|
|
|
nodeRange.selectNode( node );
|
|
|
|
|
|
|
|
|
|
if ( partial ) {
|
|
|
|
|
// Node must not finish before range starts or start after range
|
|
|
|
|
// finishes.
|
|
|
|
|
var nodeEndBeforeStart = ( range.compareBoundaryPoints(
|
|
|
|
|
END_TO_START, nodeRange ) > -1 ),
|
|
|
|
|
nodeStartAfterEnd = ( range.compareBoundaryPoints(
|
|
|
|
|
START_TO_END, nodeRange ) < 1 );
|
|
|
|
|
return ( !nodeEndBeforeStart && !nodeStartAfterEnd );
|
|
|
|
|
}
|
|
|
|
|
else {
|
|
|
|
|
// Node must start after range starts and finish before range
|
|
|
|
|
// finishes
|
|
|
|
|
var nodeStartAfterStart = ( range.compareBoundaryPoints(
|
|
|
|
|
START_TO_START, nodeRange ) < 1 ),
|
|
|
|
|
nodeEndBeforeEnd = ( range.compareBoundaryPoints(
|
|
|
|
|
END_TO_END, nodeRange ) > -1 );
|
|
|
|
|
return ( nodeStartAfterStart && nodeEndBeforeEnd );
|
|
|
|
|
}
|
|
|
|
|
};
|
|
|
|
|
|
2013-06-20 21:03:01 +10:00
|
|
|
|
var moveRangeBoundariesDownTree = function ( range ) {
|
|
|
|
|
var startContainer = range.startContainer,
|
|
|
|
|
startOffset = range.startOffset,
|
|
|
|
|
endContainer = range.endContainer,
|
|
|
|
|
endOffset = range.endOffset,
|
2016-09-25 21:43:47 +01:00
|
|
|
|
maySkipBR = true,
|
2013-04-08 13:27:06 +10:00
|
|
|
|
child;
|
|
|
|
|
|
|
|
|
|
while ( startContainer.nodeType !== TEXT_NODE ) {
|
|
|
|
|
child = startContainer.childNodes[ startOffset ];
|
|
|
|
|
if ( !child || isLeaf( child ) ) {
|
|
|
|
|
break;
|
|
|
|
|
}
|
|
|
|
|
startContainer = child;
|
|
|
|
|
startOffset = 0;
|
|
|
|
|
}
|
|
|
|
|
if ( endOffset ) {
|
|
|
|
|
while ( endContainer.nodeType !== TEXT_NODE ) {
|
|
|
|
|
child = endContainer.childNodes[ endOffset - 1 ];
|
|
|
|
|
if ( !child || isLeaf( child ) ) {
|
2016-09-25 21:43:47 +01:00
|
|
|
|
if ( maySkipBR && child && child.nodeName === 'BR' ) {
|
|
|
|
|
endOffset -= 1;
|
|
|
|
|
maySkipBR = false;
|
|
|
|
|
continue;
|
|
|
|
|
}
|
2013-04-08 13:27:06 +10:00
|
|
|
|
break;
|
|
|
|
|
}
|
|
|
|
|
endContainer = child;
|
|
|
|
|
endOffset = getLength( endContainer );
|
|
|
|
|
}
|
|
|
|
|
} else {
|
|
|
|
|
while ( endContainer.nodeType !== TEXT_NODE ) {
|
|
|
|
|
child = endContainer.firstChild;
|
|
|
|
|
if ( !child || isLeaf( child ) ) {
|
|
|
|
|
break;
|
|
|
|
|
}
|
|
|
|
|
endContainer = child;
|
|
|
|
|
}
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
// If collapsed, this algorithm finds the nearest text node positions
|
|
|
|
|
// *outside* the range rather than inside, but also it flips which is
|
|
|
|
|
// assigned to which.
|
2013-06-20 21:03:01 +10:00
|
|
|
|
if ( range.collapsed ) {
|
|
|
|
|
range.setStart( endContainer, endOffset );
|
|
|
|
|
range.setEnd( startContainer, startOffset );
|
2013-04-08 13:27:06 +10:00
|
|
|
|
} else {
|
2013-06-20 21:03:01 +10:00
|
|
|
|
range.setStart( startContainer, startOffset );
|
|
|
|
|
range.setEnd( endContainer, endOffset );
|
2013-04-08 13:27:06 +10:00
|
|
|
|
}
|
|
|
|
|
};
|
|
|
|
|
|
2017-01-13 15:15:59 +11:00
|
|
|
|
var moveRangeBoundariesUpTree = function ( range, startMax, endMax, root ) {
|
2017-01-09 14:07:20 +11:00
|
|
|
|
var startContainer = range.startContainer;
|
|
|
|
|
var startOffset = range.startOffset;
|
|
|
|
|
var endContainer = range.endContainer;
|
|
|
|
|
var endOffset = range.endOffset;
|
|
|
|
|
var maySkipBR = true;
|
|
|
|
|
var parent;
|
2013-04-08 13:27:06 +10:00
|
|
|
|
|
2017-01-09 14:07:20 +11:00
|
|
|
|
if ( !startMax ) {
|
|
|
|
|
startMax = range.commonAncestorContainer;
|
|
|
|
|
}
|
|
|
|
|
if ( !endMax ) {
|
|
|
|
|
endMax = startMax;
|
2013-04-08 13:27:06 +10:00
|
|
|
|
}
|
|
|
|
|
|
2017-01-13 15:15:59 +11:00
|
|
|
|
while ( !startOffset &&
|
|
|
|
|
startContainer !== startMax &&
|
|
|
|
|
startContainer !== root ) {
|
2013-04-08 13:27:06 +10:00
|
|
|
|
parent = startContainer.parentNode;
|
|
|
|
|
startOffset = indexOf.call( parent.childNodes, startContainer );
|
|
|
|
|
startContainer = parent;
|
|
|
|
|
}
|
|
|
|
|
|
2016-09-25 21:43:47 +01:00
|
|
|
|
while ( true ) {
|
2020-06-03 14:53:05 +10:00
|
|
|
|
if ( endContainer === endMax || endContainer === root ) {
|
|
|
|
|
break;
|
|
|
|
|
}
|
2016-09-25 21:43:47 +01:00
|
|
|
|
if ( maySkipBR &&
|
|
|
|
|
endContainer.nodeType !== TEXT_NODE &&
|
|
|
|
|
endContainer.childNodes[ endOffset ] &&
|
|
|
|
|
endContainer.childNodes[ endOffset ].nodeName === 'BR' ) {
|
|
|
|
|
endOffset += 1;
|
|
|
|
|
maySkipBR = false;
|
|
|
|
|
}
|
2020-06-03 14:53:05 +10:00
|
|
|
|
if ( endOffset !== getLength( endContainer ) ) {
|
2016-09-25 21:43:47 +01:00
|
|
|
|
break;
|
|
|
|
|
}
|
2013-04-08 13:27:06 +10:00
|
|
|
|
parent = endContainer.parentNode;
|
|
|
|
|
endOffset = indexOf.call( parent.childNodes, endContainer ) + 1;
|
|
|
|
|
endContainer = parent;
|
|
|
|
|
}
|
|
|
|
|
|
2013-06-20 21:03:01 +10:00
|
|
|
|
range.setStart( startContainer, startOffset );
|
|
|
|
|
range.setEnd( endContainer, endOffset );
|
2013-04-08 13:27:06 +10:00
|
|
|
|
};
|
|
|
|
|
|
2020-06-03 14:53:05 +10:00
|
|
|
|
var moveRangeBoundaryOutOf = function ( range, nodeName, root ) {
|
|
|
|
|
var parent = getNearest( range.endContainer, root, 'A' );
|
|
|
|
|
if ( parent ) {
|
|
|
|
|
var clone = range.cloneRange();
|
|
|
|
|
parent = parent.parentNode;
|
|
|
|
|
moveRangeBoundariesUpTree( clone, parent, parent, root );
|
|
|
|
|
if ( clone.endContainer === parent ) {
|
|
|
|
|
range.setStart( clone.endContainer, clone.endOffset );
|
|
|
|
|
range.setEnd( clone.endContainer, clone.endOffset );
|
|
|
|
|
}
|
|
|
|
|
}
|
|
|
|
|
return range;
|
|
|
|
|
};
|
|
|
|
|
|
2013-04-08 13:27:06 +10:00
|
|
|
|
// Returns the first block at least partially contained by the range,
|
|
|
|
|
// or null if no block is contained by the range.
|
2016-03-22 17:57:00 +11:00
|
|
|
|
var getStartBlockOfRange = function ( range, root ) {
|
2013-06-20 21:03:01 +10:00
|
|
|
|
var container = range.startContainer,
|
2013-04-08 13:27:06 +10:00
|
|
|
|
block;
|
|
|
|
|
|
|
|
|
|
// If inline, get the containing block.
|
|
|
|
|
if ( isInline( container ) ) {
|
2016-03-22 17:57:00 +11:00
|
|
|
|
block = getPreviousBlock( container, root );
|
2016-09-22 09:44:03 +01:00
|
|
|
|
} else if ( container !== root && isBlock( container ) ) {
|
2013-04-08 13:27:06 +10:00
|
|
|
|
block = container;
|
|
|
|
|
} else {
|
2013-06-20 21:03:01 +10:00
|
|
|
|
block = getNodeBefore( container, range.startOffset );
|
2016-03-22 17:57:00 +11:00
|
|
|
|
block = getNextBlock( block, root );
|
2013-04-08 13:27:06 +10:00
|
|
|
|
}
|
|
|
|
|
// Check the block actually intersects the range
|
2013-06-20 21:03:01 +10:00
|
|
|
|
return block && isNodeContainedInRange( range, block, true ) ? block : null;
|
2013-04-08 13:27:06 +10:00
|
|
|
|
};
|
|
|
|
|
|
|
|
|
|
// Returns the last block at least partially contained by the range,
|
|
|
|
|
// or null if no block is contained by the range.
|
2016-03-22 17:57:00 +11:00
|
|
|
|
var getEndBlockOfRange = function ( range, root ) {
|
2013-06-20 21:03:01 +10:00
|
|
|
|
var container = range.endContainer,
|
2013-04-08 13:27:06 +10:00
|
|
|
|
block, child;
|
|
|
|
|
|
|
|
|
|
// If inline, get the containing block.
|
|
|
|
|
if ( isInline( container ) ) {
|
2016-03-22 17:57:00 +11:00
|
|
|
|
block = getPreviousBlock( container, root );
|
2016-09-22 09:44:03 +01:00
|
|
|
|
} else if ( container !== root && isBlock( container ) ) {
|
2013-04-08 13:27:06 +10:00
|
|
|
|
block = container;
|
|
|
|
|
} else {
|
2013-06-20 21:03:01 +10:00
|
|
|
|
block = getNodeAfter( container, range.endOffset );
|
2016-05-26 11:39:40 +10:00
|
|
|
|
if ( !block || !isOrContains( root, block ) ) {
|
2016-03-22 17:57:00 +11:00
|
|
|
|
block = root;
|
2013-04-08 13:27:06 +10:00
|
|
|
|
while ( child = block.lastChild ) {
|
|
|
|
|
block = child;
|
|
|
|
|
}
|
|
|
|
|
}
|
2016-03-22 17:57:00 +11:00
|
|
|
|
block = getPreviousBlock( block, root );
|
2013-04-08 13:27:06 +10:00
|
|
|
|
}
|
|
|
|
|
// Check the block actually intersects the range
|
2013-06-20 21:03:01 +10:00
|
|
|
|
return block && isNodeContainedInRange( range, block, true ) ? block : null;
|
2013-04-08 13:27:06 +10:00
|
|
|
|
};
|
|
|
|
|
|
2014-09-03 10:27:38 +07:00
|
|
|
|
var contentWalker = new TreeWalker( null,
|
|
|
|
|
SHOW_TEXT|SHOW_ELEMENT,
|
|
|
|
|
function ( node ) {
|
|
|
|
|
return node.nodeType === TEXT_NODE ?
|
|
|
|
|
notWS.test( node.data ) :
|
|
|
|
|
node.nodeName === 'IMG';
|
|
|
|
|
}
|
|
|
|
|
);
|
|
|
|
|
|
2016-03-22 17:57:00 +11:00
|
|
|
|
var rangeDoesStartAtBlockBoundary = function ( range, root ) {
|
2016-05-26 11:39:40 +10:00
|
|
|
|
var startContainer = range.startContainer;
|
|
|
|
|
var startOffset = range.startOffset;
|
|
|
|
|
var nodeAfterCursor;
|
2013-04-08 13:27:06 +10:00
|
|
|
|
|
2014-09-03 10:27:38 +07:00
|
|
|
|
// If in the middle or end of a text node, we're not at the boundary.
|
2015-06-19 10:00:55 +07:00
|
|
|
|
contentWalker.root = null;
|
2014-09-03 10:27:38 +07:00
|
|
|
|
if ( startContainer.nodeType === TEXT_NODE ) {
|
2013-04-08 13:27:06 +10:00
|
|
|
|
if ( startOffset ) {
|
|
|
|
|
return false;
|
|
|
|
|
}
|
2016-05-26 11:39:40 +10:00
|
|
|
|
nodeAfterCursor = startContainer;
|
2014-09-03 10:27:38 +07:00
|
|
|
|
} else {
|
2016-05-26 11:39:40 +10:00
|
|
|
|
nodeAfterCursor = getNodeAfter( startContainer, startOffset );
|
|
|
|
|
if ( nodeAfterCursor && !isOrContains( root, nodeAfterCursor ) ) {
|
|
|
|
|
nodeAfterCursor = null;
|
|
|
|
|
}
|
|
|
|
|
// The cursor was right at the end of the document
|
|
|
|
|
if ( !nodeAfterCursor ) {
|
|
|
|
|
nodeAfterCursor = getNodeBefore( startContainer, startOffset );
|
|
|
|
|
if ( nodeAfterCursor.nodeType === TEXT_NODE &&
|
|
|
|
|
nodeAfterCursor.length ) {
|
|
|
|
|
return false;
|
|
|
|
|
}
|
2016-05-16 10:08:29 -04:00
|
|
|
|
}
|
2013-04-08 13:27:06 +10:00
|
|
|
|
}
|
2014-09-03 10:27:38 +07:00
|
|
|
|
|
|
|
|
|
// Otherwise, look for any previous content in the same block.
|
2016-05-26 11:39:40 +10:00
|
|
|
|
contentWalker.currentNode = nodeAfterCursor;
|
2016-03-22 17:57:00 +11:00
|
|
|
|
contentWalker.root = getStartBlockOfRange( range, root );
|
2014-09-03 10:27:38 +07:00
|
|
|
|
|
|
|
|
|
return !contentWalker.previousNode();
|
2013-04-08 13:27:06 +10:00
|
|
|
|
};
|
|
|
|
|
|
2016-03-22 17:57:00 +11:00
|
|
|
|
var rangeDoesEndAtBlockBoundary = function ( range, root ) {
|
2013-06-20 21:03:01 +10:00
|
|
|
|
var endContainer = range.endContainer,
|
|
|
|
|
endOffset = range.endOffset,
|
2014-09-03 10:27:38 +07:00
|
|
|
|
length;
|
2013-04-08 13:27:06 +10:00
|
|
|
|
|
2014-09-03 10:27:38 +07:00
|
|
|
|
// If in a text node with content, and not at the end, we're not
|
|
|
|
|
// at the boundary
|
2015-06-19 10:00:55 +07:00
|
|
|
|
contentWalker.root = null;
|
2014-09-03 10:27:38 +07:00
|
|
|
|
if ( endContainer.nodeType === TEXT_NODE ) {
|
|
|
|
|
length = endContainer.data.length;
|
|
|
|
|
if ( length && endOffset < length ) {
|
2013-04-08 13:27:06 +10:00
|
|
|
|
return false;
|
|
|
|
|
}
|
2014-09-03 10:27:38 +07:00
|
|
|
|
contentWalker.currentNode = endContainer;
|
|
|
|
|
} else {
|
|
|
|
|
contentWalker.currentNode = getNodeBefore( endContainer, endOffset );
|
2013-04-08 13:27:06 +10:00
|
|
|
|
}
|
2014-09-03 10:27:38 +07:00
|
|
|
|
|
|
|
|
|
// Otherwise, look for any further content in the same block.
|
2016-03-22 17:57:00 +11:00
|
|
|
|
contentWalker.root = getEndBlockOfRange( range, root );
|
2014-09-03 10:27:38 +07:00
|
|
|
|
|
|
|
|
|
return !contentWalker.nextNode();
|
2013-04-08 13:27:06 +10:00
|
|
|
|
};
|
|
|
|
|
|
2016-03-22 17:57:00 +11:00
|
|
|
|
var expandRangeToBlockBoundaries = function ( range, root ) {
|
|
|
|
|
var start = getStartBlockOfRange( range, root ),
|
|
|
|
|
end = getEndBlockOfRange( range, root ),
|
2013-04-08 13:27:06 +10:00
|
|
|
|
parent;
|
|
|
|
|
|
|
|
|
|
if ( start && end ) {
|
|
|
|
|
parent = start.parentNode;
|
2013-06-20 21:03:01 +10:00
|
|
|
|
range.setStart( parent, indexOf.call( parent.childNodes, start ) );
|
2013-04-08 13:27:06 +10:00
|
|
|
|
parent = end.parentNode;
|
2013-06-20 21:03:01 +10:00
|
|
|
|
range.setEnd( parent, indexOf.call( parent.childNodes, end ) + 1 );
|
2013-04-08 13:27:06 +10:00
|
|
|
|
}
|
|
|
|
|
};
|
2014-10-02 16:36:39 +07:00
|
|
|
|
|
2015-06-18 15:54:37 +07:00
|
|
|
|
var keys = {
|
|
|
|
|
8: 'backspace',
|
|
|
|
|
9: 'tab',
|
|
|
|
|
13: 'enter',
|
|
|
|
|
32: 'space',
|
2015-06-25 10:54:53 +07:00
|
|
|
|
33: 'pageup',
|
|
|
|
|
34: 'pagedown',
|
2015-06-18 15:54:37 +07:00
|
|
|
|
37: 'left',
|
|
|
|
|
39: 'right',
|
|
|
|
|
46: 'delete',
|
|
|
|
|
219: '[',
|
|
|
|
|
221: ']'
|
|
|
|
|
};
|
|
|
|
|
|
|
|
|
|
// Ref: http://unixpapa.com/js/key.html
|
|
|
|
|
var onKey = function ( event ) {
|
|
|
|
|
var code = event.keyCode,
|
|
|
|
|
key = keys[ code ],
|
|
|
|
|
modifiers = '',
|
|
|
|
|
range = this.getSelection();
|
|
|
|
|
|
2015-07-13 09:25:44 +07:00
|
|
|
|
if ( event.defaultPrevented ) {
|
|
|
|
|
return;
|
|
|
|
|
}
|
|
|
|
|
|
2015-06-18 15:54:37 +07:00
|
|
|
|
if ( !key ) {
|
|
|
|
|
key = String.fromCharCode( code ).toLowerCase();
|
|
|
|
|
// Only reliable for letters and numbers
|
|
|
|
|
if ( !/^[A-Za-z0-9]$/.test( key ) ) {
|
|
|
|
|
key = '';
|
|
|
|
|
}
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
// Function keys
|
|
|
|
|
if ( 111 < code && code < 124 ) {
|
|
|
|
|
key = 'f' + ( code - 111 );
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
// We need to apply the backspace/delete handlers regardless of
|
|
|
|
|
// control key modifiers.
|
|
|
|
|
if ( key !== 'backspace' && key !== 'delete' ) {
|
|
|
|
|
if ( event.altKey ) { modifiers += 'alt-'; }
|
|
|
|
|
if ( event.ctrlKey ) { modifiers += 'ctrl-'; }
|
|
|
|
|
if ( event.metaKey ) { modifiers += 'meta-'; }
|
2020-05-11 11:16:11 +10:00
|
|
|
|
if ( event.shiftKey ) { modifiers += 'shift-'; }
|
2015-06-18 15:54:37 +07:00
|
|
|
|
}
|
|
|
|
|
// However, on Windows, shift-delete is apparently "cut" (WTF right?), so
|
2020-05-11 11:16:11 +10:00
|
|
|
|
// we want to let the browser handle shift-delete in this situation.
|
|
|
|
|
if ( isWin && event.shiftKey && key === 'delete' ) {
|
|
|
|
|
modifiers += 'shift-';
|
|
|
|
|
}
|
2015-06-18 15:54:37 +07:00
|
|
|
|
|
|
|
|
|
key = modifiers + key;
|
|
|
|
|
|
|
|
|
|
if ( this._keyHandlers[ key ] ) {
|
|
|
|
|
this._keyHandlers[ key ]( this, event, range );
|
2018-10-15 08:38:24 +11:00
|
|
|
|
// !event.isComposing stops us from blatting Kana-Kanji conversion in Safari
|
|
|
|
|
} else if ( !range.collapsed && !event.isComposing &&
|
|
|
|
|
!event.ctrlKey && !event.metaKey &&
|
2018-04-04 20:14:53 +10:00
|
|
|
|
( event.key || key ).length === 1 ) {
|
2015-06-18 15:54:37 +07:00
|
|
|
|
// Record undo checkpoint.
|
2016-03-11 15:56:34 +11:00
|
|
|
|
this.saveUndoState( range );
|
2015-06-18 15:54:37 +07:00
|
|
|
|
// Delete the selection
|
2016-03-22 17:57:00 +11:00
|
|
|
|
deleteContentsOfRange( range, this._root );
|
2015-06-18 15:54:37 +07:00
|
|
|
|
this._ensureBottomLine();
|
|
|
|
|
this.setSelection( range );
|
|
|
|
|
this._updatePath( range, true );
|
|
|
|
|
}
|
|
|
|
|
};
|
|
|
|
|
|
2015-04-29 11:16:11 +07:00
|
|
|
|
var mapKeyTo = function ( method ) {
|
|
|
|
|
return function ( self, event ) {
|
|
|
|
|
event.preventDefault();
|
|
|
|
|
self[ method ]();
|
|
|
|
|
};
|
|
|
|
|
};
|
2013-04-08 13:27:06 +10:00
|
|
|
|
|
2015-04-29 11:16:11 +07:00
|
|
|
|
var mapKeyToFormat = function ( tag, remove ) {
|
|
|
|
|
remove = remove || null;
|
|
|
|
|
return function ( self, event ) {
|
|
|
|
|
event.preventDefault();
|
|
|
|
|
var range = self.getSelection();
|
|
|
|
|
if ( self.hasFormat( tag, null, range ) ) {
|
|
|
|
|
self.changeFormat( null, { tag: tag }, range );
|
|
|
|
|
} else {
|
|
|
|
|
self.changeFormat( { tag: tag }, remove, range );
|
2015-03-29 18:35:03 +11:00
|
|
|
|
}
|
2015-04-29 11:16:11 +07:00
|
|
|
|
};
|
|
|
|
|
};
|
2013-06-20 23:15:18 +10:00
|
|
|
|
|
2015-04-29 11:16:11 +07:00
|
|
|
|
// If you delete the content inside a span with a font styling, Webkit will
|
|
|
|
|
// replace it with a <font> tag (!). If you delete all the text inside a
|
|
|
|
|
// link in Opera, it won't delete the link. Let's make things consistent. If
|
|
|
|
|
// you delete all text inside an inline tag, remove the inline tag.
|
|
|
|
|
var afterDelete = function ( self, range ) {
|
|
|
|
|
try {
|
|
|
|
|
if ( !range ) { range = self.getSelection(); }
|
|
|
|
|
var node = range.startContainer,
|
|
|
|
|
parent;
|
|
|
|
|
// Climb the tree from the focus point while we are inside an empty
|
|
|
|
|
// inline element
|
|
|
|
|
if ( node.nodeType === TEXT_NODE ) {
|
|
|
|
|
node = node.parentNode;
|
|
|
|
|
}
|
|
|
|
|
parent = node;
|
|
|
|
|
while ( isInline( parent ) &&
|
|
|
|
|
( !parent.textContent || parent.textContent === ZWS ) ) {
|
|
|
|
|
node = parent;
|
|
|
|
|
parent = node.parentNode;
|
|
|
|
|
}
|
2016-06-10 11:02:24 +10:00
|
|
|
|
// If focused in empty inline element
|
2015-04-29 11:16:11 +07:00
|
|
|
|
if ( node !== parent ) {
|
|
|
|
|
// Move focus to just before empty inline(s)
|
|
|
|
|
range.setStart( parent,
|
|
|
|
|
indexOf.call( parent.childNodes, node ) );
|
|
|
|
|
range.collapse( true );
|
|
|
|
|
// Remove empty inline(s)
|
|
|
|
|
parent.removeChild( node );
|
|
|
|
|
// Fix cursor in block
|
|
|
|
|
if ( !isBlock( parent ) ) {
|
2016-03-22 17:57:00 +11:00
|
|
|
|
parent = getPreviousBlock( parent, self._root );
|
2015-04-29 11:16:11 +07:00
|
|
|
|
}
|
2016-03-22 17:57:00 +11:00
|
|
|
|
fixCursor( parent, self._root );
|
2015-04-29 11:16:11 +07:00
|
|
|
|
// Move cursor into text node
|
|
|
|
|
moveRangeBoundariesDownTree( range );
|
|
|
|
|
}
|
2015-07-13 06:39:07 -07:00
|
|
|
|
// If you delete the last character in the sole <div> in Chrome,
|
|
|
|
|
// it removes the div and replaces it with just a <br> inside the
|
2016-03-22 17:57:00 +11:00
|
|
|
|
// root. Detach the <br>; the _ensureBottomLine call will insert a new
|
2015-07-13 06:39:07 -07:00
|
|
|
|
// block.
|
2016-03-22 17:57:00 +11:00
|
|
|
|
if ( node === self._root &&
|
2015-07-13 06:39:07 -07:00
|
|
|
|
( node = node.firstChild ) && node.nodeName === 'BR' ) {
|
|
|
|
|
detach( node );
|
|
|
|
|
}
|
2015-04-29 11:16:11 +07:00
|
|
|
|
self._ensureBottomLine();
|
|
|
|
|
self.setSelection( range );
|
|
|
|
|
self._updatePath( range, true );
|
|
|
|
|
} catch ( error ) {
|
|
|
|
|
self.didError( error );
|
2013-06-20 23:15:18 +10:00
|
|
|
|
}
|
2015-04-29 11:16:11 +07:00
|
|
|
|
};
|
2013-06-20 23:15:18 +10:00
|
|
|
|
|
2020-03-29 13:42:15 +11:00
|
|
|
|
var detachUneditableNode = function ( node, root ) {
|
|
|
|
|
var parent;
|
|
|
|
|
while (( parent = node.parentNode )) {
|
|
|
|
|
if ( parent === root || parent.isContentEditable ) {
|
|
|
|
|
break;
|
|
|
|
|
}
|
|
|
|
|
node = parent;
|
|
|
|
|
}
|
|
|
|
|
detach( node );
|
|
|
|
|
};
|
|
|
|
|
|
2019-11-20 22:38:14 +08:00
|
|
|
|
var handleEnter = function ( self, shiftKey, range ) {
|
|
|
|
|
var root = self._root;
|
|
|
|
|
var block, parent, node, offset, nodeAfterSplit;
|
|
|
|
|
|
|
|
|
|
// Save undo checkpoint and add any links in the preceding section.
|
|
|
|
|
// Remove any zws so we don't think there's content in an empty
|
|
|
|
|
// block.
|
|
|
|
|
self._recordUndoState( range );
|
2019-12-23 01:04:10 +02:00
|
|
|
|
if ( self._config.addLinks ) {
|
|
|
|
|
addLinks( range.startContainer, root, self );
|
|
|
|
|
}
|
2019-11-20 22:38:14 +08:00
|
|
|
|
self._removeZWS();
|
|
|
|
|
self._getRangeAndRemoveBookmark( range );
|
|
|
|
|
|
|
|
|
|
// Selected text is overwritten, therefore delete the contents
|
|
|
|
|
// to collapse selection.
|
|
|
|
|
if ( !range.collapsed ) {
|
|
|
|
|
deleteContentsOfRange( range, root );
|
|
|
|
|
}
|
2013-06-20 23:15:18 +10:00
|
|
|
|
|
2019-11-20 22:38:14 +08:00
|
|
|
|
block = getStartBlockOfRange( range, root );
|
2014-06-02 10:22:52 +10:00
|
|
|
|
|
2019-11-20 22:38:14 +08:00
|
|
|
|
// Inside a PRE, insert literal newline, unless on blank line.
|
|
|
|
|
if ( block && ( parent = getNearest( block, root, 'PRE' ) ) ) {
|
|
|
|
|
moveRangeBoundariesDownTree( range );
|
|
|
|
|
node = range.startContainer;
|
|
|
|
|
offset = range.startOffset;
|
|
|
|
|
if ( node.nodeType !== TEXT_NODE ) {
|
|
|
|
|
node = self._doc.createTextNode( '' );
|
|
|
|
|
parent.insertBefore( node, parent.firstChild );
|
|
|
|
|
}
|
|
|
|
|
// If blank line: split and insert default block
|
|
|
|
|
if ( !shiftKey &&
|
|
|
|
|
( node.data.charAt( offset - 1 ) === '\n' ||
|
|
|
|
|
rangeDoesStartAtBlockBoundary( range, root ) ) &&
|
|
|
|
|
( node.data.charAt( offset ) === '\n' ||
|
|
|
|
|
rangeDoesEndAtBlockBoundary( range, root ) ) ) {
|
|
|
|
|
node.deleteData( offset && offset - 1, offset ? 2 : 1 );
|
|
|
|
|
nodeAfterSplit =
|
|
|
|
|
split( node, offset && offset - 1, root, root );
|
|
|
|
|
node = nodeAfterSplit.previousSibling;
|
|
|
|
|
if ( !node.textContent ) {
|
|
|
|
|
detach( node );
|
2018-07-27 10:29:49 +10:00
|
|
|
|
}
|
2019-11-20 22:38:14 +08:00
|
|
|
|
node = self.createDefaultBlock();
|
|
|
|
|
nodeAfterSplit.parentNode.insertBefore( node, nodeAfterSplit );
|
|
|
|
|
if ( !nodeAfterSplit.textContent ) {
|
|
|
|
|
detach( nodeAfterSplit );
|
2018-07-27 10:29:49 +10:00
|
|
|
|
}
|
2019-11-20 22:38:14 +08:00
|
|
|
|
range.setStart( node, 0 );
|
|
|
|
|
} else {
|
|
|
|
|
node.insertData( offset, '\n' );
|
|
|
|
|
fixCursor( parent, root );
|
|
|
|
|
// Firefox bug: if you set the selection in the text node after
|
|
|
|
|
// the new line, it draws the cursor before the line break still
|
|
|
|
|
// but if you set the selection to the equivalent position
|
|
|
|
|
// in the parent, it works.
|
|
|
|
|
if ( node.length === offset + 1 ) {
|
|
|
|
|
range.setStartAfter( node );
|
|
|
|
|
} else {
|
|
|
|
|
range.setStart( node, offset + 1 );
|
2017-03-21 17:05:41 +11:00
|
|
|
|
}
|
2015-04-29 11:16:11 +07:00
|
|
|
|
}
|
2019-11-20 22:38:14 +08:00
|
|
|
|
range.collapse( true );
|
|
|
|
|
self.setSelection( range );
|
|
|
|
|
self._updatePath( range, true );
|
|
|
|
|
self._docWasChanged();
|
|
|
|
|
return;
|
|
|
|
|
}
|
2013-06-20 23:15:18 +10:00
|
|
|
|
|
2019-11-20 22:38:14 +08:00
|
|
|
|
// If this is a malformed bit of document or in a table;
|
|
|
|
|
// just play it safe and insert a <br>.
|
|
|
|
|
if ( !block || shiftKey || /^T[HD]$/.test( block.nodeName ) ) {
|
|
|
|
|
// If inside an <a>, move focus out
|
2020-06-03 14:53:05 +10:00
|
|
|
|
moveRangeBoundaryOutOf( range, 'A', root );
|
2019-11-20 22:38:14 +08:00
|
|
|
|
insertNodeInRange( range, self.createElement( 'BR' ) );
|
|
|
|
|
range.collapse( false );
|
|
|
|
|
self.setSelection( range );
|
|
|
|
|
self._updatePath( range, true );
|
|
|
|
|
return;
|
|
|
|
|
}
|
2013-06-20 23:15:18 +10:00
|
|
|
|
|
2019-11-20 22:38:14 +08:00
|
|
|
|
// If in a list, we'll split the LI instead.
|
|
|
|
|
if ( parent = getNearest( block, root, 'LI' ) ) {
|
|
|
|
|
block = parent;
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
if ( isEmptyBlock( block ) ) {
|
|
|
|
|
// Break list
|
|
|
|
|
if ( getNearest( block, root, 'UL' ) ||
|
|
|
|
|
getNearest( block, root, 'OL' ) ) {
|
|
|
|
|
return self.decreaseListLevel( range );
|
|
|
|
|
}
|
|
|
|
|
// Break blockquote
|
|
|
|
|
else if ( getNearest( block, root, 'BLOCKQUOTE' ) ) {
|
|
|
|
|
return self.modifyBlocks( removeBlockQuote, range );
|
2015-04-29 11:16:11 +07:00
|
|
|
|
}
|
2019-11-20 22:38:14 +08:00
|
|
|
|
}
|
2013-06-20 23:15:18 +10:00
|
|
|
|
|
2019-11-20 22:38:14 +08:00
|
|
|
|
// Otherwise, split at cursor point.
|
|
|
|
|
nodeAfterSplit = splitBlock( self, block,
|
|
|
|
|
range.startContainer, range.startOffset );
|
2013-06-20 23:15:18 +10:00
|
|
|
|
|
2019-11-20 22:38:14 +08:00
|
|
|
|
// Clean up any empty inlines if we hit enter at the beginning of the
|
|
|
|
|
// block
|
|
|
|
|
removeZWS( block );
|
|
|
|
|
removeEmptyInlines( block );
|
|
|
|
|
fixCursor( block, root );
|
2013-06-20 23:15:18 +10:00
|
|
|
|
|
2019-11-20 22:38:14 +08:00
|
|
|
|
// Focus cursor
|
|
|
|
|
// If there's a <b>/<i> etc. at the beginning of the split
|
|
|
|
|
// make sure we focus inside it.
|
|
|
|
|
while ( nodeAfterSplit.nodeType === ELEMENT_NODE ) {
|
|
|
|
|
var child = nodeAfterSplit.firstChild,
|
|
|
|
|
next;
|
|
|
|
|
|
|
|
|
|
// Don't continue links over a block break; unlikely to be the
|
|
|
|
|
// desired outcome.
|
|
|
|
|
if ( nodeAfterSplit.nodeName === 'A' &&
|
|
|
|
|
( !nodeAfterSplit.textContent ||
|
|
|
|
|
nodeAfterSplit.textContent === ZWS ) ) {
|
|
|
|
|
child = self._doc.createTextNode( '' );
|
|
|
|
|
replaceWith( nodeAfterSplit, child );
|
|
|
|
|
nodeAfterSplit = child;
|
|
|
|
|
break;
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
while ( child && child.nodeType === TEXT_NODE && !child.data ) {
|
|
|
|
|
next = child.nextSibling;
|
|
|
|
|
if ( !next || next.nodeName === 'BR' ) {
|
2015-04-29 11:16:11 +07:00
|
|
|
|
break;
|
|
|
|
|
}
|
2019-11-20 22:38:14 +08:00
|
|
|
|
detach( child );
|
|
|
|
|
child = next;
|
2015-04-29 11:16:11 +07:00
|
|
|
|
}
|
2019-11-20 22:38:14 +08:00
|
|
|
|
|
|
|
|
|
// 'BR's essentially don't count; they're a browser hack.
|
|
|
|
|
// If you try to select the contents of a 'BR', FF will not let
|
|
|
|
|
// you type anything!
|
|
|
|
|
if ( !child || child.nodeName === 'BR' ||
|
2020-03-11 14:40:10 +11:00
|
|
|
|
child.nodeType === TEXT_NODE ) {
|
2019-11-20 22:38:14 +08:00
|
|
|
|
break;
|
|
|
|
|
}
|
|
|
|
|
nodeAfterSplit = child;
|
|
|
|
|
}
|
|
|
|
|
range = self.createRange( nodeAfterSplit, 0 );
|
|
|
|
|
self.setSelection( range );
|
|
|
|
|
self._updatePath( range, true );
|
|
|
|
|
};
|
|
|
|
|
|
|
|
|
|
var keyHandlers = {
|
|
|
|
|
// This song and dance is to force iOS to do enable the shift key
|
|
|
|
|
// automatically on enter. When you do the DOM split manipulation yourself,
|
|
|
|
|
// WebKit doesn't reset the IME state and so presents auto-complete options
|
|
|
|
|
// as though you were continuing to type on the previous line, and doesn't
|
|
|
|
|
// auto-enable the shift key. The old trick of blurring and focussing
|
|
|
|
|
// again no longer works in iOS 13, and I tried various execCommand options
|
|
|
|
|
// but they didn't seem to do anything. The only solution I've found is to
|
|
|
|
|
// let iOS handle the enter key, then after it's done that reset the HTML
|
|
|
|
|
// to what it was before and handle it properly in Squire; the IME state of
|
|
|
|
|
// course doesn't reset so you end up in the correct state!
|
|
|
|
|
enter: isIOS ? function ( self, event, range ) {
|
|
|
|
|
self._saveRangeToBookmark( range );
|
|
|
|
|
var html = self._getHTML();
|
|
|
|
|
var restoreAndDoEnter = function () {
|
|
|
|
|
self.removeEventListener( 'keyup', restoreAndDoEnter );
|
|
|
|
|
self._setHTML( html );
|
|
|
|
|
range = self._getRangeAndRemoveBookmark();
|
|
|
|
|
// Ignore the shift key on iOS, as this is for auto-capitalisation.
|
|
|
|
|
handleEnter( self, false, range );
|
|
|
|
|
};
|
|
|
|
|
self.addEventListener( 'keyup', restoreAndDoEnter );
|
|
|
|
|
} : function ( self, event, range ) {
|
|
|
|
|
event.preventDefault();
|
|
|
|
|
handleEnter( self, event.shiftKey, range );
|
2015-04-29 11:16:11 +07:00
|
|
|
|
},
|
2018-07-27 09:23:09 +10:00
|
|
|
|
|
|
|
|
|
'shift-enter': function ( self, event, range ) {
|
|
|
|
|
return self._keyHandlers.enter( self, event, range );
|
|
|
|
|
},
|
|
|
|
|
|
2015-04-29 11:16:11 +07:00
|
|
|
|
backspace: function ( self, event, range ) {
|
2016-03-22 17:57:00 +11:00
|
|
|
|
var root = self._root;
|
2015-04-29 11:16:11 +07:00
|
|
|
|
self._removeZWS();
|
|
|
|
|
// Record undo checkpoint.
|
2016-03-11 15:56:34 +11:00
|
|
|
|
self.saveUndoState( range );
|
2015-04-29 11:16:11 +07:00
|
|
|
|
// If not collapsed, delete contents
|
|
|
|
|
if ( !range.collapsed ) {
|
|
|
|
|
event.preventDefault();
|
2016-03-22 17:57:00 +11:00
|
|
|
|
deleteContentsOfRange( range, root );
|
2015-04-29 11:16:11 +07:00
|
|
|
|
afterDelete( self, range );
|
|
|
|
|
}
|
|
|
|
|
// If at beginning of block, merge with previous
|
2016-03-22 17:57:00 +11:00
|
|
|
|
else if ( rangeDoesStartAtBlockBoundary( range, root ) ) {
|
2015-04-29 11:16:11 +07:00
|
|
|
|
event.preventDefault();
|
2016-03-22 17:57:00 +11:00
|
|
|
|
var current = getStartBlockOfRange( range, root );
|
2016-02-26 14:28:53 +11:00
|
|
|
|
var previous;
|
|
|
|
|
if ( !current ) {
|
|
|
|
|
return;
|
|
|
|
|
}
|
|
|
|
|
// In case inline data has somehow got between blocks.
|
2016-03-22 17:57:00 +11:00
|
|
|
|
fixContainer( current.parentNode, root );
|
2016-02-26 14:28:53 +11:00
|
|
|
|
// Now get previous block
|
2016-03-22 17:57:00 +11:00
|
|
|
|
previous = getPreviousBlock( current, root );
|
2015-04-29 11:16:11 +07:00
|
|
|
|
// Must not be at the very beginning of the text area.
|
|
|
|
|
if ( previous ) {
|
|
|
|
|
// If not editable, just delete whole block.
|
|
|
|
|
if ( !previous.isContentEditable ) {
|
2020-03-29 13:42:15 +11:00
|
|
|
|
detachUneditableNode( previous, root );
|
2015-04-29 11:16:11 +07:00
|
|
|
|
return;
|
|
|
|
|
}
|
|
|
|
|
// Otherwise merge.
|
2017-07-15 20:33:32 +10:00
|
|
|
|
mergeWithBlock( previous, current, range, root );
|
2015-04-29 11:16:11 +07:00
|
|
|
|
// If deleted line between containers, merge newly adjacent
|
|
|
|
|
// containers.
|
|
|
|
|
current = previous.parentNode;
|
2016-03-26 11:20:28 +11:00
|
|
|
|
while ( current !== root && !current.nextSibling ) {
|
2015-04-29 11:16:11 +07:00
|
|
|
|
current = current.parentNode;
|
|
|
|
|
}
|
2016-03-26 11:20:28 +11:00
|
|
|
|
if ( current !== root && ( current = current.nextSibling ) ) {
|
2016-03-22 17:57:00 +11:00
|
|
|
|
mergeContainers( current, root );
|
2015-04-29 11:16:11 +07:00
|
|
|
|
}
|
|
|
|
|
self.setSelection( range );
|
|
|
|
|
}
|
|
|
|
|
// If at very beginning of text area, allow backspace
|
|
|
|
|
// to break lists/blockquote.
|
|
|
|
|
else if ( current ) {
|
|
|
|
|
// Break list
|
2016-03-22 17:57:00 +11:00
|
|
|
|
if ( getNearest( current, root, 'UL' ) ||
|
|
|
|
|
getNearest( current, root, 'OL' ) ) {
|
2017-08-15 11:11:48 +10:00
|
|
|
|
return self.decreaseListLevel( range );
|
2015-04-29 11:16:11 +07:00
|
|
|
|
}
|
|
|
|
|
// Break blockquote
|
2016-03-22 17:57:00 +11:00
|
|
|
|
else if ( getNearest( current, root, 'BLOCKQUOTE' ) ) {
|
2015-04-29 11:16:11 +07:00
|
|
|
|
return self.modifyBlocks( decreaseBlockQuoteLevel, range );
|
|
|
|
|
}
|
|
|
|
|
self.setSelection( range );
|
|
|
|
|
self._updatePath( range, true );
|
|
|
|
|
}
|
|
|
|
|
}
|
|
|
|
|
// Otherwise, leave to browser but check afterwards whether it has
|
|
|
|
|
// left behind an empty inline tag.
|
|
|
|
|
else {
|
|
|
|
|
self.setSelection( range );
|
|
|
|
|
setTimeout( function () { afterDelete( self ); }, 0 );
|
|
|
|
|
}
|
|
|
|
|
},
|
|
|
|
|
'delete': function ( self, event, range ) {
|
2016-03-22 17:57:00 +11:00
|
|
|
|
var root = self._root;
|
2016-02-26 14:28:53 +11:00
|
|
|
|
var current, next, originalRange,
|
|
|
|
|
cursorContainer, cursorOffset, nodeAfterCursor;
|
2015-04-29 11:16:11 +07:00
|
|
|
|
self._removeZWS();
|
|
|
|
|
// Record undo checkpoint.
|
2016-03-11 15:56:34 +11:00
|
|
|
|
self.saveUndoState( range );
|
2015-04-29 11:16:11 +07:00
|
|
|
|
// If not collapsed, delete contents
|
|
|
|
|
if ( !range.collapsed ) {
|
|
|
|
|
event.preventDefault();
|
2016-03-22 17:57:00 +11:00
|
|
|
|
deleteContentsOfRange( range, root );
|
2015-04-29 11:16:11 +07:00
|
|
|
|
afterDelete( self, range );
|
|
|
|
|
}
|
|
|
|
|
// If at end of block, merge next into this block
|
2016-03-22 17:57:00 +11:00
|
|
|
|
else if ( rangeDoesEndAtBlockBoundary( range, root ) ) {
|
2015-04-29 11:16:11 +07:00
|
|
|
|
event.preventDefault();
|
2016-03-22 17:57:00 +11:00
|
|
|
|
current = getStartBlockOfRange( range, root );
|
2016-02-26 14:28:53 +11:00
|
|
|
|
if ( !current ) {
|
|
|
|
|
return;
|
|
|
|
|
}
|
|
|
|
|
// In case inline data has somehow got between blocks.
|
2016-03-22 17:57:00 +11:00
|
|
|
|
fixContainer( current.parentNode, root );
|
2016-02-26 14:28:53 +11:00
|
|
|
|
// Now get next block
|
2016-03-22 17:57:00 +11:00
|
|
|
|
next = getNextBlock( current, root );
|
2015-04-29 11:16:11 +07:00
|
|
|
|
// Must not be at the very end of the text area.
|
|
|
|
|
if ( next ) {
|
|
|
|
|
// If not editable, just delete whole block.
|
|
|
|
|
if ( !next.isContentEditable ) {
|
2020-03-29 13:42:15 +11:00
|
|
|
|
detachUneditableNode( next, root );
|
2015-04-29 11:16:11 +07:00
|
|
|
|
return;
|
|
|
|
|
}
|
|
|
|
|
// Otherwise merge.
|
2017-07-15 20:33:32 +10:00
|
|
|
|
mergeWithBlock( current, next, range, root );
|
2015-04-29 11:16:11 +07:00
|
|
|
|
// If deleted line between containers, merge newly adjacent
|
|
|
|
|
// containers.
|
|
|
|
|
next = current.parentNode;
|
2016-03-26 11:20:28 +11:00
|
|
|
|
while ( next !== root && !next.nextSibling ) {
|
2015-04-29 11:16:11 +07:00
|
|
|
|
next = next.parentNode;
|
|
|
|
|
}
|
2016-03-26 11:20:28 +11:00
|
|
|
|
if ( next !== root && ( next = next.nextSibling ) ) {
|
2016-03-22 17:57:00 +11:00
|
|
|
|
mergeContainers( next, root );
|
2015-04-29 11:16:11 +07:00
|
|
|
|
}
|
|
|
|
|
self.setSelection( range );
|
|
|
|
|
self._updatePath( range, true );
|
|
|
|
|
}
|
|
|
|
|
}
|
|
|
|
|
// Otherwise, leave to browser but check afterwards whether it has
|
|
|
|
|
// left behind an empty inline tag.
|
|
|
|
|
else {
|
2015-06-17 17:18:34 +07:00
|
|
|
|
// But first check if the cursor is just before an IMG tag. If so,
|
|
|
|
|
// delete it ourselves, because the browser won't if it is not
|
|
|
|
|
// inline.
|
2016-02-26 14:28:53 +11:00
|
|
|
|
originalRange = range.cloneRange();
|
2017-01-13 15:15:59 +11:00
|
|
|
|
moveRangeBoundariesUpTree( range, root, root, root );
|
2015-06-17 17:18:34 +07:00
|
|
|
|
cursorContainer = range.endContainer;
|
|
|
|
|
cursorOffset = range.endOffset;
|
|
|
|
|
if ( cursorContainer.nodeType === ELEMENT_NODE ) {
|
|
|
|
|
nodeAfterCursor = cursorContainer.childNodes[ cursorOffset ];
|
|
|
|
|
if ( nodeAfterCursor && nodeAfterCursor.nodeName === 'IMG' ) {
|
|
|
|
|
event.preventDefault();
|
|
|
|
|
detach( nodeAfterCursor );
|
|
|
|
|
moveRangeBoundariesDownTree( range );
|
|
|
|
|
afterDelete( self, range );
|
|
|
|
|
return;
|
|
|
|
|
}
|
|
|
|
|
}
|
|
|
|
|
self.setSelection( originalRange );
|
2015-04-29 11:16:11 +07:00
|
|
|
|
setTimeout( function () { afterDelete( self ); }, 0 );
|
|
|
|
|
}
|
|
|
|
|
},
|
|
|
|
|
tab: function ( self, event, range ) {
|
2016-03-22 17:57:00 +11:00
|
|
|
|
var root = self._root;
|
2015-04-29 11:16:11 +07:00
|
|
|
|
var node, parent;
|
|
|
|
|
self._removeZWS();
|
2015-10-22 16:08:53 -07:00
|
|
|
|
// If no selection and at start of block
|
2016-03-22 17:57:00 +11:00
|
|
|
|
if ( range.collapsed && rangeDoesStartAtBlockBoundary( range, root ) ) {
|
|
|
|
|
node = getStartBlockOfRange( range, root );
|
2015-04-29 11:16:11 +07:00
|
|
|
|
// Iterate through the block's parents
|
2017-08-15 11:11:48 +10:00
|
|
|
|
while ( ( parent = node.parentNode ) ) {
|
2015-04-29 11:16:11 +07:00
|
|
|
|
// If we find a UL or OL (so are in a list, node must be an LI)
|
|
|
|
|
if ( parent.nodeName === 'UL' || parent.nodeName === 'OL' ) {
|
2016-06-28 21:31:31 -07:00
|
|
|
|
// Then increase the list level
|
|
|
|
|
event.preventDefault();
|
2017-08-15 11:11:48 +10:00
|
|
|
|
self.increaseListLevel( range );
|
2015-04-29 11:16:11 +07:00
|
|
|
|
break;
|
|
|
|
|
}
|
|
|
|
|
node = parent;
|
|
|
|
|
}
|
|
|
|
|
}
|
|
|
|
|
},
|
2015-11-17 16:50:55 +11:00
|
|
|
|
'shift-tab': function ( self, event, range ) {
|
2016-03-22 17:57:00 +11:00
|
|
|
|
var root = self._root;
|
|
|
|
|
var node;
|
2015-11-17 16:50:55 +11:00
|
|
|
|
self._removeZWS();
|
|
|
|
|
// If no selection and at start of block
|
2016-03-22 17:57:00 +11:00
|
|
|
|
if ( range.collapsed && rangeDoesStartAtBlockBoundary( range, root ) ) {
|
2015-11-17 16:50:55 +11:00
|
|
|
|
// Break list
|
2016-03-22 17:57:00 +11:00
|
|
|
|
node = range.startContainer;
|
|
|
|
|
if ( getNearest( node, root, 'UL' ) ||
|
|
|
|
|
getNearest( node, root, 'OL' ) ) {
|
2015-11-17 16:50:55 +11:00
|
|
|
|
event.preventDefault();
|
2017-08-15 11:11:48 +10:00
|
|
|
|
self.decreaseListLevel( range );
|
2015-11-17 16:50:55 +11:00
|
|
|
|
}
|
|
|
|
|
}
|
|
|
|
|
},
|
2015-04-29 11:16:11 +07:00
|
|
|
|
space: function ( self, _, range ) {
|
2020-07-08 16:12:32 +10:00
|
|
|
|
var node;
|
2018-10-05 16:25:28 +10:00
|
|
|
|
var root = self._root;
|
2015-04-29 11:16:11 +07:00
|
|
|
|
self._recordUndoState( range );
|
2019-12-23 01:04:10 +02:00
|
|
|
|
if ( self._config.addLinks ) {
|
|
|
|
|
addLinks( range.startContainer, root, self );
|
|
|
|
|
}
|
2015-04-29 11:16:11 +07:00
|
|
|
|
self._getRangeAndRemoveBookmark( range );
|
|
|
|
|
|
|
|
|
|
// If the cursor is at the end of a link (<a>foo|</a>) then move it
|
|
|
|
|
// outside of the link (<a>foo</a>|) so that the space is not part of
|
|
|
|
|
// the link text.
|
|
|
|
|
node = range.endContainer;
|
2018-03-18 10:34:12 +00:00
|
|
|
|
if ( range.collapsed && range.endOffset === getLength( node ) ) {
|
2018-10-05 16:25:28 +10:00
|
|
|
|
do {
|
|
|
|
|
if ( node.nodeName === 'A' ) {
|
|
|
|
|
range.setStartAfter( node );
|
|
|
|
|
break;
|
|
|
|
|
}
|
|
|
|
|
} while ( !node.nextSibling &&
|
|
|
|
|
( node = node.parentNode ) && node !== root );
|
2015-04-29 11:16:11 +07:00
|
|
|
|
}
|
2016-07-14 11:32:38 +10:00
|
|
|
|
// Delete the selection if not collapsed
|
2018-03-18 10:34:12 +00:00
|
|
|
|
if ( !range.collapsed ) {
|
2018-10-05 16:25:28 +10:00
|
|
|
|
deleteContentsOfRange( range, root );
|
2016-07-14 11:32:38 +10:00
|
|
|
|
self._ensureBottomLine();
|
|
|
|
|
self.setSelection( range );
|
|
|
|
|
self._updatePath( range, true );
|
|
|
|
|
}
|
2015-04-29 11:16:11 +07:00
|
|
|
|
|
|
|
|
|
self.setSelection( range );
|
|
|
|
|
},
|
|
|
|
|
left: function ( self ) {
|
|
|
|
|
self._removeZWS();
|
|
|
|
|
},
|
|
|
|
|
right: function ( self ) {
|
|
|
|
|
self._removeZWS();
|
|
|
|
|
}
|
|
|
|
|
};
|
|
|
|
|
|
2016-01-06 14:38:53 +11:00
|
|
|
|
// Firefox pre v29 incorrectly handles Cmd-left/Cmd-right on Mac:
|
2015-04-29 11:16:11 +07:00
|
|
|
|
// it goes back/forward in history! Override to do the right
|
|
|
|
|
// thing.
|
|
|
|
|
// https://bugzilla.mozilla.org/show_bug.cgi?id=289384
|
2016-01-06 14:38:53 +11:00
|
|
|
|
if ( isMac && isGecko ) {
|
2015-04-29 11:16:11 +07:00
|
|
|
|
keyHandlers[ 'meta-left' ] = function ( self, event ) {
|
|
|
|
|
event.preventDefault();
|
2015-12-29 11:15:01 +11:00
|
|
|
|
var sel = getWindowSelection( self );
|
2016-01-06 14:38:53 +11:00
|
|
|
|
if ( sel && sel.modify ) {
|
2015-12-29 11:15:01 +11:00
|
|
|
|
sel.modify( 'move', 'backward', 'lineboundary' );
|
|
|
|
|
}
|
2015-04-29 11:16:11 +07:00
|
|
|
|
};
|
|
|
|
|
keyHandlers[ 'meta-right' ] = function ( self, event ) {
|
|
|
|
|
event.preventDefault();
|
2015-12-29 11:15:01 +11:00
|
|
|
|
var sel = getWindowSelection( self );
|
2016-01-06 14:38:53 +11:00
|
|
|
|
if ( sel && sel.modify ) {
|
2015-12-29 11:15:01 +11:00
|
|
|
|
sel.modify( 'move', 'forward', 'lineboundary' );
|
|
|
|
|
}
|
2015-04-29 11:16:11 +07:00
|
|
|
|
};
|
|
|
|
|
}
|
|
|
|
|
|
2015-06-25 10:54:53 +07:00
|
|
|
|
// System standard for page up/down on Mac is to just scroll, not move the
|
|
|
|
|
// cursor. On Linux/Windows, it should move the cursor, but some browsers don't
|
|
|
|
|
// implement this natively. Override to support it.
|
|
|
|
|
if ( !isMac ) {
|
|
|
|
|
keyHandlers.pageup = function ( self ) {
|
|
|
|
|
self.moveCursorToStart();
|
|
|
|
|
};
|
|
|
|
|
keyHandlers.pagedown = function ( self ) {
|
|
|
|
|
self.moveCursorToEnd();
|
|
|
|
|
};
|
|
|
|
|
}
|
|
|
|
|
|
2020-05-19 14:45:05 +10:00
|
|
|
|
const changeIndentationLevel = function ( methodIfInQuote, methodIfInList ) {
|
|
|
|
|
return function ( self, event ) {
|
|
|
|
|
event.preventDefault();
|
|
|
|
|
var path = self.getPath();
|
2020-06-04 14:59:59 +10:00
|
|
|
|
if ( /(?:^|>)BLOCKQUOTE/.test( path ) ||
|
2020-05-19 14:45:05 +10:00
|
|
|
|
!/(?:^|>)[OU]L/.test( path ) ) {
|
|
|
|
|
self[ methodIfInQuote ]();
|
|
|
|
|
} else {
|
|
|
|
|
self[ methodIfInList ]();
|
|
|
|
|
}
|
|
|
|
|
};
|
|
|
|
|
};
|
|
|
|
|
|
2020-06-30 13:32:22 +10:00
|
|
|
|
const toggleList = function ( listRegex, methodIfNotInList ) {
|
|
|
|
|
return function ( self, event ) {
|
|
|
|
|
event.preventDefault();
|
|
|
|
|
var path = self.getPath();
|
|
|
|
|
if ( !listRegex.test( path ) ) {
|
|
|
|
|
self[ methodIfNotInList ]();
|
|
|
|
|
} else {
|
|
|
|
|
self.removeList();
|
|
|
|
|
}
|
|
|
|
|
};
|
|
|
|
|
};
|
|
|
|
|
|
2015-04-29 11:16:11 +07:00
|
|
|
|
keyHandlers[ ctrlKey + 'b' ] = mapKeyToFormat( 'B' );
|
|
|
|
|
keyHandlers[ ctrlKey + 'i' ] = mapKeyToFormat( 'I' );
|
|
|
|
|
keyHandlers[ ctrlKey + 'u' ] = mapKeyToFormat( 'U' );
|
|
|
|
|
keyHandlers[ ctrlKey + 'shift-7' ] = mapKeyToFormat( 'S' );
|
|
|
|
|
keyHandlers[ ctrlKey + 'shift-5' ] = mapKeyToFormat( 'SUB', { tag: 'SUP' } );
|
|
|
|
|
keyHandlers[ ctrlKey + 'shift-6' ] = mapKeyToFormat( 'SUP', { tag: 'SUB' } );
|
2020-06-30 13:32:22 +10:00
|
|
|
|
keyHandlers[ ctrlKey + 'shift-8' ] =
|
|
|
|
|
toggleList( /(?:^|>)UL/, 'makeUnorderedList' );
|
|
|
|
|
keyHandlers[ ctrlKey + 'shift-9' ] =
|
|
|
|
|
toggleList( /(?:^|>)OL/, 'makeOrderedList' );
|
2020-06-04 14:59:59 +10:00
|
|
|
|
keyHandlers[ ctrlKey + '[' ] =
|
2020-05-19 14:45:05 +10:00
|
|
|
|
changeIndentationLevel( 'decreaseQuoteLevel', 'decreaseListLevel' );
|
2020-06-04 14:59:59 +10:00
|
|
|
|
keyHandlers[ ctrlKey + ']' ] =
|
2020-05-19 14:45:05 +10:00
|
|
|
|
changeIndentationLevel( 'increaseQuoteLevel', 'increaseListLevel' );
|
2018-07-27 10:29:49 +10:00
|
|
|
|
keyHandlers[ ctrlKey + 'd' ] = mapKeyTo( 'toggleCode' );
|
2015-04-29 11:16:11 +07:00
|
|
|
|
keyHandlers[ ctrlKey + 'y' ] = mapKeyTo( 'redo' );
|
|
|
|
|
keyHandlers[ ctrlKey + 'z' ] = mapKeyTo( 'undo' );
|
|
|
|
|
keyHandlers[ ctrlKey + 'shift-z' ] = mapKeyTo( 'redo' );
|
|
|
|
|
|
2015-06-18 15:54:37 +07:00
|
|
|
|
var fontSizes = {
|
|
|
|
|
1: 10,
|
|
|
|
|
2: 13,
|
|
|
|
|
3: 16,
|
|
|
|
|
4: 18,
|
|
|
|
|
5: 24,
|
|
|
|
|
6: 32,
|
|
|
|
|
7: 48
|
|
|
|
|
};
|
2015-04-29 11:16:11 +07:00
|
|
|
|
|
2016-05-11 08:16:11 -04:00
|
|
|
|
var styleToSemantic = {
|
2015-06-18 15:54:37 +07:00
|
|
|
|
fontWeight: {
|
2016-11-02 15:32:06 +11:00
|
|
|
|
regexp: /^bold|^700/i,
|
2015-06-18 15:54:37 +07:00
|
|
|
|
replace: function ( doc ) {
|
|
|
|
|
return createElement( doc, 'B' );
|
|
|
|
|
}
|
|
|
|
|
},
|
|
|
|
|
fontStyle: {
|
|
|
|
|
regexp: /^italic/i,
|
|
|
|
|
replace: function ( doc ) {
|
|
|
|
|
return createElement( doc, 'I' );
|
|
|
|
|
}
|
|
|
|
|
},
|
|
|
|
|
fontFamily: {
|
|
|
|
|
regexp: notWS,
|
2018-07-12 13:35:20 +10:00
|
|
|
|
replace: function ( doc, classNames, family ) {
|
2015-06-18 15:54:37 +07:00
|
|
|
|
return createElement( doc, 'SPAN', {
|
2018-07-12 13:35:20 +10:00
|
|
|
|
'class': classNames.fontFamily,
|
2015-06-18 15:54:37 +07:00
|
|
|
|
style: 'font-family:' + family
|
|
|
|
|
});
|
|
|
|
|
}
|
|
|
|
|
},
|
|
|
|
|
fontSize: {
|
|
|
|
|
regexp: notWS,
|
2018-07-12 13:35:20 +10:00
|
|
|
|
replace: function ( doc, classNames, size ) {
|
2015-06-18 15:54:37 +07:00
|
|
|
|
return createElement( doc, 'SPAN', {
|
2018-07-12 13:35:20 +10:00
|
|
|
|
'class': classNames.fontSize,
|
2015-06-18 15:54:37 +07:00
|
|
|
|
style: 'font-size:' + size
|
|
|
|
|
});
|
|
|
|
|
}
|
2016-05-11 08:16:11 -04:00
|
|
|
|
},
|
|
|
|
|
textDecoration: {
|
|
|
|
|
regexp: /^underline/i,
|
|
|
|
|
replace: function ( doc ) {
|
|
|
|
|
return createElement( doc, 'U' );
|
|
|
|
|
}
|
2015-04-29 11:16:11 +07:00
|
|
|
|
}
|
2015-06-18 15:54:37 +07:00
|
|
|
|
};
|
2015-04-29 11:16:11 +07:00
|
|
|
|
|
2015-06-18 15:54:37 +07:00
|
|
|
|
var replaceWithTag = function ( tag ) {
|
|
|
|
|
return function ( node, parent ) {
|
|
|
|
|
var el = createElement( node.ownerDocument, tag );
|
2020-01-23 10:42:44 +11:00
|
|
|
|
var attributes = node.attributes;
|
|
|
|
|
var i, l, attribute;
|
|
|
|
|
for ( i = 0, l = attributes.length; i < l; i += 1 ) {
|
|
|
|
|
attribute = attributes[i];
|
|
|
|
|
el.setAttribute( attribute.name, attribute.value );
|
|
|
|
|
}
|
2015-06-18 15:54:37 +07:00
|
|
|
|
parent.replaceChild( el, node );
|
|
|
|
|
el.appendChild( empty( node ) );
|
|
|
|
|
return el;
|
|
|
|
|
};
|
|
|
|
|
};
|
2015-04-29 11:16:11 +07:00
|
|
|
|
|
2018-07-12 13:35:20 +10:00
|
|
|
|
var replaceStyles = function ( node, parent, config ) {
|
2016-05-11 08:16:11 -04:00
|
|
|
|
var style = node.style;
|
|
|
|
|
var doc = node.ownerDocument;
|
|
|
|
|
var attr, converter, css, newTreeBottom, newTreeTop, el;
|
|
|
|
|
|
|
|
|
|
for ( attr in styleToSemantic ) {
|
|
|
|
|
converter = styleToSemantic[ attr ];
|
|
|
|
|
css = style[ attr ];
|
|
|
|
|
if ( css && converter.regexp.test( css ) ) {
|
2018-07-12 13:35:20 +10:00
|
|
|
|
el = converter.replace( doc, config.classNames, css );
|
2020-01-28 11:15:15 +11:00
|
|
|
|
if ( el.nodeName === node.nodeName &&
|
|
|
|
|
el.className === node.className ) {
|
|
|
|
|
continue;
|
|
|
|
|
}
|
2016-05-11 08:16:11 -04:00
|
|
|
|
if ( !newTreeTop ) {
|
|
|
|
|
newTreeTop = el;
|
2015-06-18 15:54:37 +07:00
|
|
|
|
}
|
2016-05-11 08:16:11 -04:00
|
|
|
|
if ( newTreeBottom ) {
|
|
|
|
|
newTreeBottom.appendChild( el );
|
|
|
|
|
}
|
|
|
|
|
newTreeBottom = el;
|
|
|
|
|
node.style[ attr ] = '';
|
2015-06-18 15:54:37 +07:00
|
|
|
|
}
|
2016-05-11 08:16:11 -04:00
|
|
|
|
}
|
2015-04-29 11:16:11 +07:00
|
|
|
|
|
2016-05-11 08:16:11 -04:00
|
|
|
|
if ( newTreeTop ) {
|
|
|
|
|
newTreeBottom.appendChild( empty( node ) );
|
2020-02-13 13:31:43 +11:00
|
|
|
|
node.appendChild( newTreeTop );
|
2016-05-11 08:16:11 -04:00
|
|
|
|
}
|
2015-04-29 11:16:11 +07:00
|
|
|
|
|
2016-05-11 08:16:11 -04:00
|
|
|
|
return newTreeBottom || node;
|
|
|
|
|
};
|
|
|
|
|
|
|
|
|
|
var stylesRewriters = {
|
|
|
|
|
SPAN: replaceStyles,
|
2015-06-18 15:54:37 +07:00
|
|
|
|
STRONG: replaceWithTag( 'B' ),
|
|
|
|
|
EM: replaceWithTag( 'I' ),
|
2016-05-11 08:16:11 -04:00
|
|
|
|
INS: replaceWithTag( 'U' ),
|
2015-06-18 15:54:37 +07:00
|
|
|
|
STRIKE: replaceWithTag( 'S' ),
|
2018-07-12 13:35:20 +10:00
|
|
|
|
FONT: function ( node, parent, config ) {
|
|
|
|
|
var face = node.face;
|
|
|
|
|
var size = node.size;
|
|
|
|
|
var colour = node.color;
|
|
|
|
|
var doc = node.ownerDocument;
|
|
|
|
|
var classNames = config.classNames;
|
|
|
|
|
var fontSpan, sizeSpan, colourSpan;
|
|
|
|
|
var newTreeBottom, newTreeTop;
|
2015-06-18 15:54:37 +07:00
|
|
|
|
if ( face ) {
|
|
|
|
|
fontSpan = createElement( doc, 'SPAN', {
|
2018-07-12 13:35:20 +10:00
|
|
|
|
'class': classNames.fontFamily,
|
2015-06-18 15:54:37 +07:00
|
|
|
|
style: 'font-family:' + face
|
|
|
|
|
});
|
|
|
|
|
newTreeTop = fontSpan;
|
|
|
|
|
newTreeBottom = fontSpan;
|
|
|
|
|
}
|
|
|
|
|
if ( size ) {
|
|
|
|
|
sizeSpan = createElement( doc, 'SPAN', {
|
2018-07-12 13:35:20 +10:00
|
|
|
|
'class': classNames.fontSize,
|
2015-06-18 15:54:37 +07:00
|
|
|
|
style: 'font-size:' + fontSizes[ size ] + 'px'
|
|
|
|
|
});
|
|
|
|
|
if ( !newTreeTop ) {
|
|
|
|
|
newTreeTop = sizeSpan;
|
2015-04-29 11:16:11 +07:00
|
|
|
|
}
|
2015-06-18 15:54:37 +07:00
|
|
|
|
if ( newTreeBottom ) {
|
|
|
|
|
newTreeBottom.appendChild( sizeSpan );
|
2015-04-29 11:16:11 +07:00
|
|
|
|
}
|
2015-06-18 15:54:37 +07:00
|
|
|
|
newTreeBottom = sizeSpan;
|
2013-04-08 13:27:06 +10:00
|
|
|
|
}
|
2015-06-18 15:54:37 +07:00
|
|
|
|
if ( colour && /^#?([\dA-F]{3}){1,2}$/i.test( colour ) ) {
|
|
|
|
|
if ( colour.charAt( 0 ) !== '#' ) {
|
|
|
|
|
colour = '#' + colour;
|
2013-04-08 13:27:06 +10:00
|
|
|
|
}
|
2015-06-18 15:54:37 +07:00
|
|
|
|
colourSpan = createElement( doc, 'SPAN', {
|
2018-07-12 13:35:20 +10:00
|
|
|
|
'class': classNames.colour,
|
2015-06-18 15:54:37 +07:00
|
|
|
|
style: 'color:' + colour
|
|
|
|
|
});
|
|
|
|
|
if ( !newTreeTop ) {
|
|
|
|
|
newTreeTop = colourSpan;
|
2013-04-08 13:27:06 +10:00
|
|
|
|
}
|
2015-06-18 15:54:37 +07:00
|
|
|
|
if ( newTreeBottom ) {
|
|
|
|
|
newTreeBottom.appendChild( colourSpan );
|
2013-04-08 13:27:06 +10:00
|
|
|
|
}
|
2015-06-18 15:54:37 +07:00
|
|
|
|
newTreeBottom = colourSpan;
|
2013-04-08 13:27:06 +10:00
|
|
|
|
}
|
2015-06-18 15:54:37 +07:00
|
|
|
|
if ( !newTreeTop ) {
|
|
|
|
|
newTreeTop = newTreeBottom = createElement( doc, 'SPAN' );
|
2013-04-08 13:27:06 +10:00
|
|
|
|
}
|
2015-06-18 15:54:37 +07:00
|
|
|
|
parent.replaceChild( newTreeTop, node );
|
|
|
|
|
newTreeBottom.appendChild( empty( node ) );
|
|
|
|
|
return newTreeBottom;
|
|
|
|
|
},
|
2018-07-12 13:35:20 +10:00
|
|
|
|
TT: function ( node, parent, config ) {
|
2015-06-18 15:54:37 +07:00
|
|
|
|
var el = createElement( node.ownerDocument, 'SPAN', {
|
2018-07-12 13:35:20 +10:00
|
|
|
|
'class': config.classNames.fontFamily,
|
2015-06-18 15:54:37 +07:00
|
|
|
|
style: 'font-family:menlo,consolas,"courier new",monospace'
|
|
|
|
|
});
|
|
|
|
|
parent.replaceChild( el, node );
|
|
|
|
|
el.appendChild( empty( node ) );
|
|
|
|
|
return el;
|
2013-04-08 13:27:06 +10:00
|
|
|
|
}
|
|
|
|
|
};
|
|
|
|
|
|
2016-11-02 15:32:06 +11:00
|
|
|
|
var allowedBlock = /^(?:A(?:DDRESS|RTICLE|SIDE|UDIO)|BLOCKQUOTE|CAPTION|D(?:[DLT]|IV)|F(?:IGURE|IGCAPTION|OOTER)|H[1-6]|HEADER|L(?:ABEL|EGEND|I)|O(?:L|UTPUT)|P(?:RE)?|SECTION|T(?:ABLE|BODY|D|FOOT|H|HEAD|R)|COL(?:GROUP)?|UL)$/;
|
2015-06-22 09:59:34 +07:00
|
|
|
|
|
|
|
|
|
var blacklist = /^(?:HEAD|META|STYLE)/;
|
|
|
|
|
|
2018-07-27 09:21:50 +10:00
|
|
|
|
var walker = new TreeWalker( null, SHOW_TEXT|SHOW_ELEMENT );
|
2015-06-22 09:59:34 +07:00
|
|
|
|
|
2015-06-18 15:54:37 +07:00
|
|
|
|
/*
|
|
|
|
|
Two purposes:
|
2013-04-08 13:27:06 +10:00
|
|
|
|
|
2015-06-18 15:54:37 +07:00
|
|
|
|
1. Remove nodes we don't want, such as weird <o:p> tags, comment nodes
|
|
|
|
|
and whitespace nodes.
|
|
|
|
|
2. Convert inline tags into our preferred format.
|
|
|
|
|
*/
|
2018-07-12 13:35:20 +10:00
|
|
|
|
var cleanTree = function cleanTree ( node, config, preserveWS ) {
|
2015-06-18 15:54:37 +07:00
|
|
|
|
var children = node.childNodes,
|
2015-06-22 09:59:34 +07:00
|
|
|
|
nonInlineParent, i, l, child, nodeName, nodeType, rewriter, childLength,
|
|
|
|
|
startsWithWS, endsWithWS, data, sibling;
|
2015-06-19 10:56:52 +07:00
|
|
|
|
|
2015-06-22 09:59:34 +07:00
|
|
|
|
nonInlineParent = node;
|
|
|
|
|
while ( isInline( nonInlineParent ) ) {
|
|
|
|
|
nonInlineParent = nonInlineParent.parentNode;
|
2015-06-19 10:56:52 +07:00
|
|
|
|
}
|
2015-06-22 09:59:34 +07:00
|
|
|
|
walker.root = nonInlineParent;
|
2015-06-19 10:56:52 +07:00
|
|
|
|
|
2015-06-18 15:54:37 +07:00
|
|
|
|
for ( i = 0, l = children.length; i < l; i += 1 ) {
|
|
|
|
|
child = children[i];
|
|
|
|
|
nodeName = child.nodeName;
|
|
|
|
|
nodeType = child.nodeType;
|
|
|
|
|
rewriter = stylesRewriters[ nodeName ];
|
|
|
|
|
if ( nodeType === ELEMENT_NODE ) {
|
|
|
|
|
childLength = child.childNodes.length;
|
|
|
|
|
if ( rewriter ) {
|
2018-07-12 13:35:20 +10:00
|
|
|
|
child = rewriter( child, node, config );
|
2015-06-22 09:59:34 +07:00
|
|
|
|
} else if ( blacklist.test( nodeName ) ) {
|
|
|
|
|
node.removeChild( child );
|
2015-06-18 15:54:37 +07:00
|
|
|
|
i -= 1;
|
2015-06-22 09:59:34 +07:00
|
|
|
|
l -= 1;
|
|
|
|
|
continue;
|
|
|
|
|
} else if ( !allowedBlock.test( nodeName ) && !isInline( child ) ) {
|
|
|
|
|
i -= 1;
|
|
|
|
|
l += childLength - 1;
|
|
|
|
|
node.replaceChild( empty( child ), child );
|
2015-06-18 15:54:37 +07:00
|
|
|
|
continue;
|
|
|
|
|
}
|
|
|
|
|
if ( childLength ) {
|
2018-07-12 13:35:20 +10:00
|
|
|
|
cleanTree( child, config,
|
|
|
|
|
preserveWS || ( nodeName === 'PRE' ) );
|
2015-06-18 15:54:37 +07:00
|
|
|
|
}
|
|
|
|
|
} else {
|
|
|
|
|
if ( nodeType === TEXT_NODE ) {
|
|
|
|
|
data = child.data;
|
2015-07-25 18:11:56 -07:00
|
|
|
|
startsWithWS = !notWS.test( data.charAt( 0 ) );
|
|
|
|
|
endsWithWS = !notWS.test( data.charAt( data.length - 1 ) );
|
2016-03-28 10:14:55 +11:00
|
|
|
|
if ( preserveWS || ( !startsWithWS && !endsWithWS ) ) {
|
2015-06-19 10:56:52 +07:00
|
|
|
|
continue;
|
|
|
|
|
}
|
2015-06-22 09:59:34 +07:00
|
|
|
|
// Iterate through the nodes; if we hit some other content
|
|
|
|
|
// before the start of a new block we don't trim
|
2015-06-19 10:56:52 +07:00
|
|
|
|
if ( startsWithWS ) {
|
2015-06-22 09:59:34 +07:00
|
|
|
|
walker.currentNode = child;
|
|
|
|
|
while ( sibling = walker.previousPONode() ) {
|
|
|
|
|
nodeName = sibling.nodeName;
|
|
|
|
|
if ( nodeName === 'IMG' ||
|
|
|
|
|
( nodeName === '#text' &&
|
2016-09-28 12:02:29 +10:00
|
|
|
|
notWS.test( sibling.data ) ) ) {
|
2015-06-22 09:59:34 +07:00
|
|
|
|
break;
|
|
|
|
|
}
|
|
|
|
|
if ( !isInline( sibling ) ) {
|
|
|
|
|
sibling = null;
|
|
|
|
|
break;
|
|
|
|
|
}
|
|
|
|
|
}
|
2016-09-28 12:02:29 +10:00
|
|
|
|
data = data.replace( /^[ \t\r\n]+/g, sibling ? ' ' : '' );
|
2015-06-19 10:56:52 +07:00
|
|
|
|
}
|
|
|
|
|
if ( endsWithWS ) {
|
2015-06-22 09:59:34 +07:00
|
|
|
|
walker.currentNode = child;
|
|
|
|
|
while ( sibling = walker.nextNode() ) {
|
|
|
|
|
if ( nodeName === 'IMG' ||
|
|
|
|
|
( nodeName === '#text' &&
|
2016-09-28 12:02:29 +10:00
|
|
|
|
notWS.test( sibling.data ) ) ) {
|
2015-06-22 09:59:34 +07:00
|
|
|
|
break;
|
|
|
|
|
}
|
|
|
|
|
if ( !isInline( sibling ) ) {
|
|
|
|
|
sibling = null;
|
|
|
|
|
break;
|
|
|
|
|
}
|
|
|
|
|
}
|
2016-09-28 12:02:29 +10:00
|
|
|
|
data = data.replace( /[ \t\r\n]+$/g, sibling ? ' ' : '' );
|
2015-06-18 15:54:37 +07:00
|
|
|
|
}
|
2015-06-19 10:56:52 +07:00
|
|
|
|
if ( data ) {
|
|
|
|
|
child.data = data;
|
2015-06-18 15:54:37 +07:00
|
|
|
|
continue;
|
|
|
|
|
}
|
2015-04-08 18:00:59 +07:00
|
|
|
|
}
|
2015-06-18 15:54:37 +07:00
|
|
|
|
node.removeChild( child );
|
|
|
|
|
i -= 1;
|
|
|
|
|
l -= 1;
|
2013-04-08 13:27:06 +10:00
|
|
|
|
}
|
|
|
|
|
}
|
2015-06-18 15:54:37 +07:00
|
|
|
|
return node;
|
2014-12-25 15:09:46 +07:00
|
|
|
|
};
|
2014-12-25 11:39:07 +07:00
|
|
|
|
|
2015-06-19 09:59:44 +07:00
|
|
|
|
// ---
|
|
|
|
|
|
2016-03-22 17:57:00 +11:00
|
|
|
|
var removeEmptyInlines = function removeEmptyInlines ( node ) {
|
|
|
|
|
var children = node.childNodes,
|
2015-06-19 09:59:44 +07:00
|
|
|
|
l = children.length,
|
|
|
|
|
child;
|
|
|
|
|
while ( l-- ) {
|
|
|
|
|
child = children[l];
|
|
|
|
|
if ( child.nodeType === ELEMENT_NODE && !isLeaf( child ) ) {
|
|
|
|
|
removeEmptyInlines( child );
|
|
|
|
|
if ( isInline( child ) && !child.firstChild ) {
|
2016-03-22 17:57:00 +11:00
|
|
|
|
node.removeChild( child );
|
2015-06-19 09:59:44 +07:00
|
|
|
|
}
|
|
|
|
|
} else if ( child.nodeType === TEXT_NODE && !child.data ) {
|
2016-03-22 17:57:00 +11:00
|
|
|
|
node.removeChild( child );
|
2015-06-19 09:59:44 +07:00
|
|
|
|
}
|
|
|
|
|
}
|
|
|
|
|
};
|
|
|
|
|
|
|
|
|
|
// ---
|
|
|
|
|
|
2015-06-18 15:54:37 +07:00
|
|
|
|
var notWSTextNode = function ( node ) {
|
|
|
|
|
return node.nodeType === ELEMENT_NODE ?
|
|
|
|
|
node.nodeName === 'BR' :
|
|
|
|
|
notWS.test( node.data );
|
2014-12-25 11:39:07 +07:00
|
|
|
|
};
|
2017-01-09 10:35:19 +11:00
|
|
|
|
var isLineBreak = function ( br, isLBIfEmptyBlock ) {
|
|
|
|
|
var block = br.parentNode;
|
|
|
|
|
var walker;
|
2015-06-18 15:54:37 +07:00
|
|
|
|
while ( isInline( block ) ) {
|
|
|
|
|
block = block.parentNode;
|
2014-12-25 11:39:07 +07:00
|
|
|
|
}
|
2015-06-18 15:54:37 +07:00
|
|
|
|
walker = new TreeWalker(
|
|
|
|
|
block, SHOW_ELEMENT|SHOW_TEXT, notWSTextNode );
|
|
|
|
|
walker.currentNode = br;
|
2017-01-09 10:35:19 +11:00
|
|
|
|
return !!walker.nextNode() ||
|
|
|
|
|
( isLBIfEmptyBlock && !walker.previousNode() );
|
2013-04-08 13:27:06 +10:00
|
|
|
|
};
|
|
|
|
|
|
2015-06-18 15:54:37 +07:00
|
|
|
|
// <br> elements are treated specially, and differently depending on the
|
|
|
|
|
// browser, when in rich text editor mode. When adding HTML from external
|
|
|
|
|
// sources, we must remove them, replacing the ones that actually affect
|
2015-06-19 09:59:44 +07:00
|
|
|
|
// line breaks by wrapping the inline text in a <div>. Browsers that want <br>
|
|
|
|
|
// elements at the end of each block will then have them added back in a later
|
|
|
|
|
// fixCursor method call.
|
2017-01-09 10:35:19 +11:00
|
|
|
|
var cleanupBRs = function ( node, root, keepForBlankLine ) {
|
|
|
|
|
var brs = node.querySelectorAll( 'BR' );
|
|
|
|
|
var brBreaksLine = [];
|
|
|
|
|
var l = brs.length;
|
|
|
|
|
var i, br, parent;
|
2013-04-08 13:27:06 +10:00
|
|
|
|
|
2015-06-18 15:54:37 +07:00
|
|
|
|
// Must calculate whether the <br> breaks a line first, because if we
|
|
|
|
|
// have two <br>s next to each other, after the first one is converted
|
|
|
|
|
// to a block split, the second will be at the end of a block and
|
|
|
|
|
// therefore seem to not be a line break. But in its original context it
|
|
|
|
|
// was, so we should also convert it to a block split.
|
|
|
|
|
for ( i = 0; i < l; i += 1 ) {
|
2017-01-09 10:35:19 +11:00
|
|
|
|
brBreaksLine[i] = isLineBreak( brs[i], keepForBlankLine );
|
2013-04-08 13:27:06 +10:00
|
|
|
|
}
|
2015-06-18 15:54:37 +07:00
|
|
|
|
while ( l-- ) {
|
|
|
|
|
br = brs[l];
|
|
|
|
|
// Cleanup may have removed it
|
|
|
|
|
parent = br.parentNode;
|
|
|
|
|
if ( !parent ) { continue; }
|
|
|
|
|
// If it doesn't break a line, just remove it; it's not doing
|
|
|
|
|
// anything useful. We'll add it back later if required by the
|
|
|
|
|
// browser. If it breaks a line, wrap the content in div tags
|
|
|
|
|
// and replace the brs.
|
|
|
|
|
if ( !brBreaksLine[l] ) {
|
|
|
|
|
detach( br );
|
|
|
|
|
} else if ( !isInline( parent ) ) {
|
2016-03-22 17:57:00 +11:00
|
|
|
|
fixContainer( parent, root );
|
2015-06-18 15:54:37 +07:00
|
|
|
|
}
|
2014-12-26 20:01:36 +07:00
|
|
|
|
}
|
2013-04-08 13:27:06 +10:00
|
|
|
|
};
|
|
|
|
|
|
2016-11-02 15:32:06 +11:00
|
|
|
|
// The (non-standard but supported enough) innerText property is based on the
|
|
|
|
|
// render tree in Firefox and possibly other browsers, so we must insert the
|
|
|
|
|
// DOM node into the document to ensure the text part is correct.
|
2020-07-08 16:12:32 +10:00
|
|
|
|
var setClipboardData =
|
|
|
|
|
function ( event, contents, root, willCutCopy, toPlainText, plainTextOnly ) {
|
|
|
|
|
var clipboardData = event.clipboardData;
|
|
|
|
|
var doc = event.target.ownerDocument;
|
|
|
|
|
var body = doc.body;
|
|
|
|
|
var node = createElement( doc, 'div' );
|
2016-12-11 12:30:09 +11:00
|
|
|
|
var html, text;
|
|
|
|
|
|
2020-07-08 16:12:32 +10:00
|
|
|
|
node.appendChild( contents );
|
2016-12-11 12:30:09 +11:00
|
|
|
|
// Firefox will add an extra new line for BRs at the end of block when
|
|
|
|
|
// calculating innerText, even though they don't actually affect display.
|
|
|
|
|
// So we need to remove them first.
|
2017-01-09 10:35:19 +11:00
|
|
|
|
cleanupBRs( node, root, true );
|
2020-07-14 19:58:50 +10:00
|
|
|
|
|
|
|
|
|
html = node.innerHTML;
|
|
|
|
|
if ( willCutCopy ) {
|
|
|
|
|
html = willCutCopy( html );
|
|
|
|
|
}
|
|
|
|
|
|
2020-07-08 16:12:32 +10:00
|
|
|
|
if ( toPlainText ) {
|
|
|
|
|
text = toPlainText( html );
|
|
|
|
|
} else {
|
2020-07-14 19:58:50 +10:00
|
|
|
|
node.setAttribute( 'style',
|
|
|
|
|
'position:fixed;overflow:hidden;bottom:100%;right:100%;' );
|
2020-07-08 16:12:32 +10:00
|
|
|
|
body.appendChild( node );
|
|
|
|
|
text = node.innerText || node.textContent;
|
|
|
|
|
text = text.replace( / /g, ' ' ); // Replace nbsp with regular space
|
|
|
|
|
body.removeChild( node );
|
|
|
|
|
}
|
2016-12-11 12:30:09 +11:00
|
|
|
|
// Firefox (and others?) returns unix line endings (\n) even on Windows.
|
|
|
|
|
// If on Windows, normalise to \r\n, since Notepad and some other crappy
|
|
|
|
|
// apps do not understand just \n.
|
|
|
|
|
if ( isWin ) {
|
|
|
|
|
text = text.replace( /\r?\n/g, '\r\n' );
|
|
|
|
|
}
|
|
|
|
|
|
2020-07-08 16:12:32 +10:00
|
|
|
|
if ( !plainTextOnly ) {
|
|
|
|
|
clipboardData.setData( 'text/html', html );
|
|
|
|
|
}
|
2016-12-11 12:30:09 +11:00
|
|
|
|
clipboardData.setData( 'text/plain', text );
|
2020-07-08 16:12:32 +10:00
|
|
|
|
event.preventDefault();
|
2016-11-02 15:32:06 +11:00
|
|
|
|
};
|
|
|
|
|
|
2016-03-11 15:22:49 +11:00
|
|
|
|
var onCut = function ( event ) {
|
2015-06-18 15:54:37 +07:00
|
|
|
|
var range = this.getSelection();
|
2016-03-22 17:57:00 +11:00
|
|
|
|
var root = this._root;
|
2015-06-18 15:54:37 +07:00
|
|
|
|
var self = this;
|
2020-07-08 16:12:32 +10:00
|
|
|
|
var startBlock, endBlock, copyRoot, contents, parent, newContents;
|
2016-03-11 15:22:49 +11:00
|
|
|
|
|
2017-01-09 14:25:34 +11:00
|
|
|
|
// Nothing to do
|
|
|
|
|
if ( range.collapsed ) {
|
|
|
|
|
event.preventDefault();
|
|
|
|
|
return;
|
|
|
|
|
}
|
|
|
|
|
|
2016-03-11 15:22:49 +11:00
|
|
|
|
// Save undo checkpoint
|
2016-03-11 15:56:34 +11:00
|
|
|
|
this.saveUndoState( range );
|
2016-03-11 15:22:49 +11:00
|
|
|
|
|
|
|
|
|
// Edge only seems to support setting plain text as of 2016-03-11.
|
2020-07-08 16:12:32 +10:00
|
|
|
|
if ( !isEdge && event.clipboardData ) {
|
2017-01-13 10:24:01 +11:00
|
|
|
|
// Clipboard content should include all parents within block, or all
|
|
|
|
|
// parents up to root if selection across blocks
|
|
|
|
|
startBlock = getStartBlockOfRange( range, root );
|
|
|
|
|
endBlock = getEndBlockOfRange( range, root );
|
|
|
|
|
copyRoot = ( ( startBlock === endBlock ) && startBlock ) || root;
|
|
|
|
|
// Extract the contents
|
|
|
|
|
contents = deleteContentsOfRange( range, root );
|
|
|
|
|
// Add any other parents not in extracted content, up to copy root
|
2017-01-19 17:08:39 +11:00
|
|
|
|
parent = range.commonAncestorContainer;
|
2017-01-13 10:24:01 +11:00
|
|
|
|
if ( parent.nodeType === TEXT_NODE ) {
|
|
|
|
|
parent = parent.parentNode;
|
|
|
|
|
}
|
|
|
|
|
while ( parent && parent !== copyRoot ) {
|
|
|
|
|
newContents = parent.cloneNode( false );
|
|
|
|
|
newContents.appendChild( contents );
|
|
|
|
|
contents = newContents;
|
|
|
|
|
parent = parent.parentNode;
|
|
|
|
|
}
|
|
|
|
|
// Set clipboard data
|
2020-07-08 16:12:32 +10:00
|
|
|
|
setClipboardData(
|
|
|
|
|
event, contents, root, this._config.willCutCopy, null, false );
|
2016-03-11 15:22:49 +11:00
|
|
|
|
} else {
|
|
|
|
|
setTimeout( function () {
|
|
|
|
|
try {
|
2016-03-22 17:57:00 +11:00
|
|
|
|
// If all content removed, ensure div at start of root.
|
2016-03-11 15:22:49 +11:00
|
|
|
|
self._ensureBottomLine();
|
|
|
|
|
} catch ( error ) {
|
|
|
|
|
self.didError( error );
|
|
|
|
|
}
|
|
|
|
|
}, 0 );
|
|
|
|
|
}
|
|
|
|
|
|
2015-06-18 15:54:37 +07:00
|
|
|
|
this.setSelection( range );
|
2016-03-11 15:22:49 +11:00
|
|
|
|
};
|
|
|
|
|
|
2020-07-08 16:12:32 +10:00
|
|
|
|
var _onCopy = function ( event, range, root, willCutCopy, toPlainText, plainTextOnly ) {
|
|
|
|
|
var startBlock, endBlock, copyRoot, contents, parent, newContents;
|
2016-03-11 15:22:49 +11:00
|
|
|
|
// Edge only seems to support setting plain text as of 2016-03-11.
|
2020-07-08 16:12:32 +10:00
|
|
|
|
if ( !isEdge && event.clipboardData ) {
|
2017-01-13 10:24:01 +11:00
|
|
|
|
// Clipboard content should include all parents within block, or all
|
|
|
|
|
// parents up to root if selection across blocks
|
2016-09-04 20:44:36 -04:00
|
|
|
|
startBlock = getStartBlockOfRange( range, root );
|
2016-09-09 12:14:57 -04:00
|
|
|
|
endBlock = getEndBlockOfRange( range, root );
|
2016-12-09 10:42:18 +11:00
|
|
|
|
copyRoot = ( ( startBlock === endBlock ) && startBlock ) || root;
|
2017-01-13 10:24:01 +11:00
|
|
|
|
// Clone range to mutate, then move up as high as possible without
|
|
|
|
|
// passing the copy root node.
|
|
|
|
|
range = range.cloneRange();
|
2016-09-09 12:14:57 -04:00
|
|
|
|
moveRangeBoundariesDownTree( range );
|
2017-01-13 15:15:59 +11:00
|
|
|
|
moveRangeBoundariesUpTree( range, copyRoot, copyRoot, root );
|
2017-01-13 10:24:01 +11:00
|
|
|
|
// Extract the contents
|
|
|
|
|
contents = range.cloneContents();
|
|
|
|
|
// Add any other parents not in extracted content, up to copy root
|
2017-01-19 17:08:39 +11:00
|
|
|
|
parent = range.commonAncestorContainer;
|
2016-09-10 09:24:14 -04:00
|
|
|
|
if ( parent.nodeType === TEXT_NODE ) {
|
|
|
|
|
parent = parent.parentNode;
|
|
|
|
|
}
|
2016-09-09 12:14:57 -04:00
|
|
|
|
while ( parent && parent !== copyRoot ) {
|
|
|
|
|
newContents = parent.cloneNode( false );
|
|
|
|
|
newContents.appendChild( contents );
|
|
|
|
|
contents = newContents;
|
|
|
|
|
parent = parent.parentNode;
|
2016-08-10 16:24:13 +10:00
|
|
|
|
}
|
2017-01-13 10:24:01 +11:00
|
|
|
|
// Set clipboard data
|
2020-07-08 16:12:32 +10:00
|
|
|
|
setClipboardData( event, contents, root, willCutCopy, toPlainText, plainTextOnly );
|
2016-03-11 15:22:49 +11:00
|
|
|
|
}
|
2013-04-08 13:27:06 +10:00
|
|
|
|
};
|
|
|
|
|
|
2020-07-08 16:12:32 +10:00
|
|
|
|
var onCopy = function ( event ) {
|
|
|
|
|
_onCopy(
|
|
|
|
|
event,
|
|
|
|
|
this.getSelection(),
|
|
|
|
|
this._root,
|
|
|
|
|
this._config.willCutCopy,
|
|
|
|
|
null,
|
|
|
|
|
false
|
|
|
|
|
);
|
|
|
|
|
};
|
|
|
|
|
|
2016-09-04 20:58:49 -04:00
|
|
|
|
// Need to monitor for shift key like this, as event.shiftKey is not available
|
|
|
|
|
// in paste event.
|
|
|
|
|
function monitorShiftKey ( event ) {
|
|
|
|
|
this.isShiftDown = event.shiftKey;
|
|
|
|
|
}
|
|
|
|
|
|
2015-06-18 15:54:37 +07:00
|
|
|
|
var onPaste = function ( event ) {
|
2016-09-04 20:58:49 -04:00
|
|
|
|
var clipboardData = event.clipboardData;
|
|
|
|
|
var items = clipboardData && clipboardData.items;
|
|
|
|
|
var choosePlain = this.isShiftDown;
|
|
|
|
|
var fireDrop = false;
|
2019-02-20 16:06:04 +11:00
|
|
|
|
var hasRTF = false;
|
2016-09-04 20:58:49 -04:00
|
|
|
|
var hasImage = false;
|
|
|
|
|
var plainItem = null;
|
2019-01-10 15:18:24 +11:00
|
|
|
|
var htmlItem = null;
|
2016-09-04 20:58:49 -04:00
|
|
|
|
var self = this;
|
|
|
|
|
var l, item, type, types, data;
|
2015-06-19 09:57:32 +07:00
|
|
|
|
|
|
|
|
|
// Current HTML5 Clipboard interface
|
2015-06-19 12:35:18 +07:00
|
|
|
|
// ---------------------------------
|
2015-06-19 09:57:32 +07:00
|
|
|
|
// https://html.spec.whatwg.org/multipage/interaction.html
|
2017-12-13 16:43:43 +11:00
|
|
|
|
if ( items ) {
|
2015-06-18 15:54:37 +07:00
|
|
|
|
l = items.length;
|
|
|
|
|
while ( l-- ) {
|
2015-06-19 09:57:32 +07:00
|
|
|
|
item = items[l];
|
|
|
|
|
type = item.type;
|
2019-01-10 15:18:24 +11:00
|
|
|
|
if ( type === 'text/html' ) {
|
|
|
|
|
htmlItem = item;
|
2019-03-25 15:40:41 -04:00
|
|
|
|
// iOS copy URL gives you type text/uri-list which is just a list
|
|
|
|
|
// of 1 or more URLs separated by new lines. Can just treat as
|
|
|
|
|
// plain text.
|
|
|
|
|
} else if ( type === 'text/plain' || type === 'text/uri-list' ) {
|
2015-06-19 09:57:32 +07:00
|
|
|
|
plainItem = item;
|
2019-02-20 16:06:04 +11:00
|
|
|
|
} else if ( type === 'text/rtf' ) {
|
|
|
|
|
hasRTF = true;
|
2019-01-10 15:18:24 +11:00
|
|
|
|
} else if ( /^image\/.*/.test( type ) ) {
|
2015-06-18 15:54:37 +07:00
|
|
|
|
hasImage = true;
|
|
|
|
|
}
|
|
|
|
|
}
|
2019-01-10 15:18:24 +11:00
|
|
|
|
|
|
|
|
|
// Treat image paste as a drop of an image file. When you copy
|
|
|
|
|
// an image in Chrome/Firefox (at least), it copies the image data
|
|
|
|
|
// but also an HTML version (referencing the original URL of the image)
|
2019-02-20 16:06:04 +11:00
|
|
|
|
// and a plain text version.
|
|
|
|
|
//
|
|
|
|
|
// However, when you copy in Excel, you get html, rtf, text, image;
|
|
|
|
|
// in this instance you want the html version! So let's try using
|
|
|
|
|
// the presence of text/rtf as an indicator to choose the html version
|
|
|
|
|
// over the image.
|
|
|
|
|
if ( hasImage && !( hasRTF && htmlItem ) ) {
|
2019-01-10 15:18:24 +11:00
|
|
|
|
event.preventDefault();
|
2015-06-19 09:57:32 +07:00
|
|
|
|
this.fireEvent( 'dragover', {
|
2015-06-18 15:54:37 +07:00
|
|
|
|
dataTransfer: clipboardData,
|
|
|
|
|
/*jshint loopfunc: true */
|
|
|
|
|
preventDefault: function () {
|
|
|
|
|
fireDrop = true;
|
|
|
|
|
}
|
|
|
|
|
/*jshint loopfunc: false */
|
|
|
|
|
});
|
|
|
|
|
if ( fireDrop ) {
|
|
|
|
|
this.fireEvent( 'drop', {
|
|
|
|
|
dataTransfer: clipboardData
|
|
|
|
|
});
|
|
|
|
|
}
|
2019-01-10 15:18:24 +11:00
|
|
|
|
return;
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
// Edge only provides access to plain text as of 2016-03-11 and gives no
|
|
|
|
|
// indication there should be an HTML part. However, it does support
|
|
|
|
|
// access to image data, so we check for that first. Otherwise though,
|
|
|
|
|
// fall through to fallback clipboard handling methods
|
|
|
|
|
if ( !isEdge ) {
|
|
|
|
|
event.preventDefault();
|
|
|
|
|
if ( htmlItem && ( !choosePlain || !plainItem ) ) {
|
|
|
|
|
htmlItem.getAsString( function ( html ) {
|
|
|
|
|
self.insertHTML( html, true );
|
|
|
|
|
});
|
|
|
|
|
} else if ( plainItem ) {
|
|
|
|
|
plainItem.getAsString( function ( text ) {
|
|
|
|
|
self.insertPlainText( text, true );
|
|
|
|
|
});
|
|
|
|
|
}
|
|
|
|
|
return;
|
2015-06-19 09:57:32 +07:00
|
|
|
|
}
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
// Old interface
|
2015-06-19 12:35:18 +07:00
|
|
|
|
// -------------
|
2015-07-11 19:22:46 +07:00
|
|
|
|
|
2015-07-14 12:59:27 -07:00
|
|
|
|
// Safari (and indeed many other OS X apps) copies stuff as text/rtf
|
|
|
|
|
// rather than text/html; even from a webpage in Safari. The only way
|
|
|
|
|
// to get an HTML version is to fallback to letting the browser insert
|
2015-07-16 10:46:29 -07:00
|
|
|
|
// the content. Same for getting image data. *Sigh*.
|
2016-03-07 18:38:07 +11:00
|
|
|
|
//
|
|
|
|
|
// Firefox is even worse: it doesn't even let you know that there might be
|
|
|
|
|
// an RTF version on the clipboard, but it will also convert to HTML if you
|
|
|
|
|
// let the browser insert the content. I've filed
|
|
|
|
|
// https://bugzilla.mozilla.org/show_bug.cgi?id=1254028
|
|
|
|
|
types = clipboardData && clipboardData.types;
|
2016-03-11 15:22:49 +11:00
|
|
|
|
if ( !isEdge && types && (
|
2016-03-07 18:38:07 +11:00
|
|
|
|
indexOf.call( types, 'text/html' ) > -1 || (
|
|
|
|
|
!isGecko &&
|
|
|
|
|
indexOf.call( types, 'text/plain' ) > -1 &&
|
|
|
|
|
indexOf.call( types, 'text/rtf' ) < 0 )
|
|
|
|
|
)) {
|
2015-07-14 12:59:27 -07:00
|
|
|
|
event.preventDefault();
|
|
|
|
|
// Abiword on Linux copies a plain text and html version, but the HTML
|
|
|
|
|
// version is the empty string! So always try to get HTML, but if none,
|
2016-04-11 16:58:13 +10:00
|
|
|
|
// insert plain text instead. On iOS, Facebook (and possibly other
|
|
|
|
|
// apps?) copy links as type text/uri-list, but also insert a **blank**
|
|
|
|
|
// text/plain item onto the clipboard. Why? Who knows.
|
2016-09-04 20:58:49 -04:00
|
|
|
|
if ( !choosePlain && ( data = clipboardData.getData( 'text/html' ) ) ) {
|
2015-07-14 12:59:27 -07:00
|
|
|
|
this.insertHTML( data, true );
|
2016-04-11 16:58:13 +10:00
|
|
|
|
} else if (
|
|
|
|
|
( data = clipboardData.getData( 'text/plain' ) ) ||
|
|
|
|
|
( data = clipboardData.getData( 'text/uri-list' ) ) ) {
|
2015-07-16 10:46:29 -07:00
|
|
|
|
this.insertPlainText( data, true );
|
2015-07-11 19:22:46 +07:00
|
|
|
|
}
|
2015-07-14 12:59:27 -07:00
|
|
|
|
return;
|
2015-07-11 19:22:46 +07:00
|
|
|
|
}
|
2013-04-08 13:27:06 +10:00
|
|
|
|
|
2016-03-11 15:22:49 +11:00
|
|
|
|
// No interface. Includes all versions of IE :(
|
|
|
|
|
// --------------------------------------------
|
2015-06-19 12:35:18 +07:00
|
|
|
|
|
2015-06-18 15:54:37 +07:00
|
|
|
|
this._awaitingPaste = true;
|
2013-04-08 13:27:06 +10:00
|
|
|
|
|
2016-03-22 17:57:00 +11:00
|
|
|
|
var body = this._doc.body,
|
2015-06-18 15:54:37 +07:00
|
|
|
|
range = this.getSelection(),
|
2015-06-19 09:57:32 +07:00
|
|
|
|
startContainer = range.startContainer,
|
|
|
|
|
startOffset = range.startOffset,
|
|
|
|
|
endContainer = range.endContainer,
|
2016-03-22 17:57:00 +11:00
|
|
|
|
endOffset = range.endOffset;
|
2013-04-08 13:27:06 +10:00
|
|
|
|
|
2015-06-18 15:54:37 +07:00
|
|
|
|
// We need to position the pasteArea in the visible portion of the screen
|
|
|
|
|
// to stop the browser auto-scrolling.
|
|
|
|
|
var pasteArea = this.createElement( 'DIV', {
|
2016-03-22 17:57:00 +11:00
|
|
|
|
contenteditable: 'true',
|
|
|
|
|
style: 'position:fixed; overflow:hidden; top:0; right:100%; width:1px; height:1px;'
|
2015-06-18 15:54:37 +07:00
|
|
|
|
});
|
|
|
|
|
body.appendChild( pasteArea );
|
|
|
|
|
range.selectNodeContents( pasteArea );
|
|
|
|
|
this.setSelection( range );
|
2013-04-08 13:27:06 +10:00
|
|
|
|
|
2015-06-18 15:54:37 +07:00
|
|
|
|
// A setTimeout of 0 means this is added to the back of the
|
|
|
|
|
// single javascript thread, so it will be executed after the
|
|
|
|
|
// paste event.
|
|
|
|
|
setTimeout( function () {
|
|
|
|
|
try {
|
2015-06-22 09:32:47 +07:00
|
|
|
|
// IE sometimes fires the beforepaste event twice; make sure it is
|
|
|
|
|
// not run again before our after paste function is called.
|
2015-06-19 09:57:32 +07:00
|
|
|
|
self._awaitingPaste = false;
|
|
|
|
|
|
2015-06-18 15:54:37 +07:00
|
|
|
|
// Get the pasted content and clean
|
2015-06-19 09:57:32 +07:00
|
|
|
|
var html = '',
|
2015-06-18 15:54:37 +07:00
|
|
|
|
next = pasteArea,
|
|
|
|
|
first, range;
|
2013-04-08 13:27:06 +10:00
|
|
|
|
|
2015-06-18 15:54:37 +07:00
|
|
|
|
// #88: Chrome can apparently split the paste area if certain
|
|
|
|
|
// content is inserted; gather them all up.
|
|
|
|
|
while ( pasteArea = next ) {
|
|
|
|
|
next = pasteArea.nextSibling;
|
2015-06-22 09:32:47 +07:00
|
|
|
|
detach( pasteArea );
|
2015-06-18 15:54:37 +07:00
|
|
|
|
// Safari and IE like putting extra divs around things.
|
2015-06-19 09:57:32 +07:00
|
|
|
|
first = pasteArea.firstChild;
|
|
|
|
|
if ( first && first === pasteArea.lastChild &&
|
2015-06-18 15:54:37 +07:00
|
|
|
|
first.nodeName === 'DIV' ) {
|
2015-06-19 09:57:32 +07:00
|
|
|
|
pasteArea = first;
|
2015-06-18 15:54:37 +07:00
|
|
|
|
}
|
2015-06-19 09:57:32 +07:00
|
|
|
|
html += pasteArea.innerHTML;
|
2015-06-18 15:54:37 +07:00
|
|
|
|
}
|
2013-04-08 13:27:06 +10:00
|
|
|
|
|
2018-05-09 17:43:11 +10:00
|
|
|
|
range = self.createRange(
|
2015-06-19 09:57:32 +07:00
|
|
|
|
startContainer, startOffset, endContainer, endOffset );
|
2015-06-18 15:54:37 +07:00
|
|
|
|
self.setSelection( range );
|
2013-04-08 13:27:06 +10:00
|
|
|
|
|
2015-06-19 09:57:32 +07:00
|
|
|
|
if ( html ) {
|
|
|
|
|
self.insertHTML( html, true );
|
|
|
|
|
}
|
2015-06-18 15:54:37 +07:00
|
|
|
|
} catch ( error ) {
|
|
|
|
|
self.didError( error );
|
2013-04-08 13:27:06 +10:00
|
|
|
|
}
|
2015-06-18 15:54:37 +07:00
|
|
|
|
}, 0 );
|
2013-04-08 13:27:06 +10:00
|
|
|
|
};
|
|
|
|
|
|
2016-05-26 13:42:54 +10:00
|
|
|
|
// On Windows you can drag an drop text. We can't handle this ourselves, because
|
|
|
|
|
// as far as I can see, there's no way to get the drop insertion point. So just
|
|
|
|
|
// save an undo state and hope for the best.
|
|
|
|
|
var onDrop = function ( event ) {
|
|
|
|
|
var types = event.dataTransfer.types;
|
|
|
|
|
var l = types.length;
|
|
|
|
|
var hasPlain = false;
|
|
|
|
|
var hasHTML = false;
|
|
|
|
|
while ( l-- ) {
|
|
|
|
|
switch ( types[l] ) {
|
|
|
|
|
case 'text/plain':
|
|
|
|
|
hasPlain = true;
|
|
|
|
|
break;
|
|
|
|
|
case 'text/html':
|
|
|
|
|
hasHTML = true;
|
|
|
|
|
break;
|
|
|
|
|
default:
|
|
|
|
|
return;
|
|
|
|
|
}
|
|
|
|
|
}
|
|
|
|
|
if ( hasHTML || hasPlain ) {
|
|
|
|
|
this.saveUndoState();
|
|
|
|
|
}
|
|
|
|
|
};
|
|
|
|
|
|
2016-07-16 16:50:59 +10:00
|
|
|
|
function mergeObjects ( base, extras, mayOverride ) {
|
2015-06-18 15:54:37 +07:00
|
|
|
|
var prop, value;
|
|
|
|
|
if ( !base ) {
|
|
|
|
|
base = {};
|
2013-04-08 13:27:06 +10:00
|
|
|
|
}
|
2016-07-16 16:50:59 +10:00
|
|
|
|
if ( extras ) {
|
|
|
|
|
for ( prop in extras ) {
|
|
|
|
|
if ( mayOverride || !( prop in base ) ) {
|
|
|
|
|
value = extras[ prop ];
|
|
|
|
|
base[ prop ] = ( value && value.constructor === Object ) ?
|
|
|
|
|
mergeObjects( base[ prop ], value, mayOverride ) :
|
|
|
|
|
value;
|
|
|
|
|
}
|
|
|
|
|
}
|
2015-06-18 15:54:37 +07:00
|
|
|
|
}
|
|
|
|
|
return base;
|
|
|
|
|
}
|
2013-04-08 13:27:06 +10:00
|
|
|
|
|
2016-03-22 17:57:00 +11:00
|
|
|
|
function Squire ( root, config ) {
|
|
|
|
|
if ( root.nodeType === DOCUMENT_NODE ) {
|
|
|
|
|
root = root.body;
|
|
|
|
|
}
|
|
|
|
|
var doc = root.ownerDocument;
|
2015-06-18 15:54:37 +07:00
|
|
|
|
var win = doc.defaultView;
|
|
|
|
|
var mutation;
|
2013-04-08 13:27:06 +10:00
|
|
|
|
|
2015-06-18 15:54:37 +07:00
|
|
|
|
this._win = win;
|
|
|
|
|
this._doc = doc;
|
2016-03-22 17:57:00 +11:00
|
|
|
|
this._root = root;
|
2013-04-08 13:27:06 +10:00
|
|
|
|
|
2015-06-18 15:54:37 +07:00
|
|
|
|
this._events = {};
|
2013-04-08 13:27:06 +10:00
|
|
|
|
|
2016-06-10 11:02:24 +10:00
|
|
|
|
this._isFocused = false;
|
2015-06-18 15:54:37 +07:00
|
|
|
|
this._lastSelection = null;
|
|
|
|
|
|
|
|
|
|
this._hasZWS = false;
|
2013-04-08 13:27:06 +10:00
|
|
|
|
|
2015-06-18 15:54:37 +07:00
|
|
|
|
this._lastAnchorNode = null;
|
|
|
|
|
this._lastFocusNode = null;
|
|
|
|
|
this._path = '';
|
2016-06-27 13:17:18 +10:00
|
|
|
|
this._willUpdatePath = false;
|
2013-04-08 13:27:06 +10:00
|
|
|
|
|
2016-05-26 11:07:11 +10:00
|
|
|
|
if ( 'onselectionchange' in doc ) {
|
|
|
|
|
this.addEventListener( 'selectionchange', this._updatePathOnEvent );
|
|
|
|
|
} else {
|
|
|
|
|
this.addEventListener( 'keyup', this._updatePathOnEvent );
|
|
|
|
|
this.addEventListener( 'mouseup', this._updatePathOnEvent );
|
|
|
|
|
}
|
2013-04-08 13:27:06 +10:00
|
|
|
|
|
2015-06-18 15:54:37 +07:00
|
|
|
|
this._undoIndex = -1;
|
|
|
|
|
this._undoStack = [];
|
|
|
|
|
this._undoStackLength = 0;
|
|
|
|
|
this._isInUndoState = false;
|
|
|
|
|
this._ignoreChange = false;
|
2016-04-28 12:01:20 -04:00
|
|
|
|
this._ignoreAllChanges = false;
|
2015-06-18 15:54:37 +07:00
|
|
|
|
|
|
|
|
|
if ( canObserveMutations ) {
|
|
|
|
|
mutation = new MutationObserver( this._docWasChanged.bind( this ) );
|
2016-03-22 17:57:00 +11:00
|
|
|
|
mutation.observe( root, {
|
2015-06-18 15:54:37 +07:00
|
|
|
|
childList: true,
|
|
|
|
|
attributes: true,
|
|
|
|
|
characterData: true,
|
|
|
|
|
subtree: true
|
|
|
|
|
});
|
|
|
|
|
this._mutation = mutation;
|
|
|
|
|
} else {
|
|
|
|
|
this.addEventListener( 'keyup', this._keyUpDetectChange );
|
2013-04-08 13:27:06 +10:00
|
|
|
|
}
|
|
|
|
|
|
2016-06-27 12:05:40 +10:00
|
|
|
|
// On blur, restore focus except if the user taps or clicks to focus a
|
|
|
|
|
// specific point. Can't actually use click event because focus happens
|
|
|
|
|
// before click, so use mousedown/touchstart
|
2016-03-28 09:43:14 +11:00
|
|
|
|
this._restoreSelection = false;
|
|
|
|
|
this.addEventListener( 'blur', enableRestoreSelection );
|
|
|
|
|
this.addEventListener( 'mousedown', disableRestoreSelection );
|
|
|
|
|
this.addEventListener( 'touchstart', disableRestoreSelection );
|
|
|
|
|
this.addEventListener( 'focus', restoreSelection );
|
|
|
|
|
|
2015-06-18 15:54:37 +07:00
|
|
|
|
// IE sometimes fires the beforepaste event twice; make sure it is not run
|
|
|
|
|
// again before our after paste function is called.
|
|
|
|
|
this._awaitingPaste = false;
|
2020-03-11 14:48:30 +11:00
|
|
|
|
this.addEventListener( 'cut', onCut );
|
2016-03-11 15:22:49 +11:00
|
|
|
|
this.addEventListener( 'copy', onCopy );
|
2016-09-04 20:58:49 -04:00
|
|
|
|
this.addEventListener( 'keydown', monitorShiftKey );
|
|
|
|
|
this.addEventListener( 'keyup', monitorShiftKey );
|
2020-03-11 14:48:30 +11:00
|
|
|
|
this.addEventListener( 'paste', onPaste );
|
2016-05-26 13:42:54 +10:00
|
|
|
|
this.addEventListener( 'drop', onDrop );
|
2020-03-11 14:40:10 +11:00
|
|
|
|
this.addEventListener( 'keydown', onKey );
|
2013-04-08 13:27:06 +10:00
|
|
|
|
|
2015-06-18 15:54:37 +07:00
|
|
|
|
// Add key handlers
|
|
|
|
|
this._keyHandlers = Object.create( keyHandlers );
|
2015-04-08 15:33:27 +07:00
|
|
|
|
|
2015-06-18 15:54:37 +07:00
|
|
|
|
// Override default properties
|
|
|
|
|
this.setConfig( config );
|
|
|
|
|
|
2016-03-22 17:57:00 +11:00
|
|
|
|
root.setAttribute( 'contenteditable', 'true' );
|
2020-04-27 16:18:10 +10:00
|
|
|
|
// Grammarly breaks the editor, *sigh*
|
2020-04-29 12:18:35 +10:00
|
|
|
|
root.setAttribute( 'data-gramm', 'false' );
|
2015-06-18 15:54:37 +07:00
|
|
|
|
|
|
|
|
|
// Remove Firefox's built-in controls
|
|
|
|
|
try {
|
|
|
|
|
doc.execCommand( 'enableObjectResizing', false, 'false' );
|
|
|
|
|
doc.execCommand( 'enableInlineTableEditing', false, 'false' );
|
|
|
|
|
} catch ( error ) {}
|
|
|
|
|
|
2016-12-07 17:54:13 +11:00
|
|
|
|
root.__squire__ = this;
|
2014-10-02 16:00:35 +07:00
|
|
|
|
|
2015-06-18 15:54:37 +07:00
|
|
|
|
// Need to register instance before calling setHTML, so that the fixCursor
|
|
|
|
|
// function can lookup any default block tag options set.
|
|
|
|
|
this.setHTML( '' );
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
var proto = Squire.prototype;
|
|
|
|
|
|
2016-12-13 12:15:35 +11:00
|
|
|
|
var sanitizeToDOMFragment = function ( html, isPaste, self ) {
|
|
|
|
|
var doc = self._doc;
|
|
|
|
|
var frag = html ? DOMPurify.sanitize( html, {
|
2017-06-28 16:03:31 +10:00
|
|
|
|
ALLOW_UNKNOWN_PROTOCOLS: true,
|
2016-12-07 17:24:16 +11:00
|
|
|
|
WHOLE_DOCUMENT: false,
|
|
|
|
|
RETURN_DOM: true,
|
|
|
|
|
RETURN_DOM_FRAGMENT: true
|
2016-12-13 12:15:35 +11:00
|
|
|
|
}) : null;
|
|
|
|
|
return frag ? doc.importNode( frag, true ) : doc.createDocumentFragment();
|
2016-12-07 17:24:16 +11:00
|
|
|
|
};
|
|
|
|
|
|
2015-06-18 15:54:37 +07:00
|
|
|
|
proto.setConfig = function ( config ) {
|
|
|
|
|
config = mergeObjects({
|
|
|
|
|
blockTag: 'DIV',
|
|
|
|
|
blockAttributes: null,
|
|
|
|
|
tagAttributes: {
|
|
|
|
|
blockquote: null,
|
|
|
|
|
ul: null,
|
|
|
|
|
ol: null,
|
2016-05-05 11:50:36 +10:00
|
|
|
|
li: null,
|
|
|
|
|
a: null
|
2016-05-31 15:26:22 -04:00
|
|
|
|
},
|
2018-07-12 13:35:20 +10:00
|
|
|
|
classNames: {
|
|
|
|
|
colour: 'colour',
|
|
|
|
|
fontFamily: 'font',
|
|
|
|
|
fontSize: 'size',
|
|
|
|
|
highlight: 'highlight'
|
|
|
|
|
},
|
2016-08-05 20:47:02 +10:00
|
|
|
|
leafNodeNames: leafNodeNames,
|
2016-05-31 15:26:22 -04:00
|
|
|
|
undo: {
|
|
|
|
|
documentSizeThreshold: -1, // -1 means no threshold
|
|
|
|
|
undoLimit: -1 // -1 means no limit
|
2016-12-07 17:24:16 +11:00
|
|
|
|
},
|
|
|
|
|
isInsertedHTMLSanitized: true,
|
|
|
|
|
isSetHTMLSanitized: true,
|
|
|
|
|
sanitizeToDOMFragment:
|
|
|
|
|
typeof DOMPurify !== 'undefined' && DOMPurify.isSupported ?
|
2018-07-12 15:40:01 +10:00
|
|
|
|
sanitizeToDOMFragment : null,
|
2019-12-23 01:04:10 +02:00
|
|
|
|
willCutCopy: null,
|
|
|
|
|
addLinks: true
|
2016-07-16 16:50:59 +10:00
|
|
|
|
}, config, true );
|
2015-04-29 11:16:11 +07:00
|
|
|
|
|
2015-06-18 15:54:37 +07:00
|
|
|
|
// Users may specify block tag in lower case
|
|
|
|
|
config.blockTag = config.blockTag.toUpperCase();
|
|
|
|
|
|
|
|
|
|
this._config = config;
|
|
|
|
|
|
|
|
|
|
return this;
|
2015-04-29 11:16:11 +07:00
|
|
|
|
};
|
2013-04-08 13:27:06 +10:00
|
|
|
|
|
2015-06-18 15:54:37 +07:00
|
|
|
|
proto.createElement = function ( tag, props, children ) {
|
|
|
|
|
return createElement( this._doc, tag, props, children );
|
|
|
|
|
};
|
2013-04-08 13:27:06 +10:00
|
|
|
|
|
2015-06-18 15:54:37 +07:00
|
|
|
|
proto.createDefaultBlock = function ( children ) {
|
|
|
|
|
var config = this._config;
|
|
|
|
|
return fixCursor(
|
2016-03-22 17:57:00 +11:00
|
|
|
|
this.createElement( config.blockTag, config.blockAttributes, children ),
|
|
|
|
|
this._root
|
2015-06-18 15:54:37 +07:00
|
|
|
|
);
|
|
|
|
|
};
|
2013-06-20 23:15:18 +10:00
|
|
|
|
|
2015-06-18 15:54:37 +07:00
|
|
|
|
proto.didError = function ( error ) {
|
|
|
|
|
console.log( error );
|
|
|
|
|
};
|
2013-04-08 13:27:06 +10:00
|
|
|
|
|
2015-06-18 15:54:37 +07:00
|
|
|
|
proto.getDocument = function () {
|
|
|
|
|
return this._doc;
|
|
|
|
|
};
|
2016-03-30 10:42:41 +11:00
|
|
|
|
proto.getRoot = function () {
|
|
|
|
|
return this._root;
|
|
|
|
|
};
|
2013-04-08 13:27:06 +10:00
|
|
|
|
|
2016-04-28 12:01:20 -04:00
|
|
|
|
proto.modifyDocument = function ( modificationCallback ) {
|
2016-12-12 10:35:14 +11:00
|
|
|
|
var mutation = this._mutation;
|
|
|
|
|
if ( mutation ) {
|
|
|
|
|
if ( mutation.takeRecords().length ) {
|
|
|
|
|
this._docWasChanged();
|
|
|
|
|
}
|
|
|
|
|
mutation.disconnect();
|
2016-04-28 12:01:20 -04:00
|
|
|
|
}
|
|
|
|
|
|
2016-12-12 10:35:14 +11:00
|
|
|
|
this._ignoreAllChanges = true;
|
2016-04-28 12:01:20 -04:00
|
|
|
|
modificationCallback();
|
|
|
|
|
this._ignoreAllChanges = false;
|
2016-12-12 10:35:14 +11:00
|
|
|
|
|
|
|
|
|
if ( mutation ) {
|
|
|
|
|
mutation.observe( this._root, {
|
2016-04-28 12:01:20 -04:00
|
|
|
|
childList: true,
|
|
|
|
|
attributes: true,
|
|
|
|
|
characterData: true,
|
|
|
|
|
subtree: true
|
|
|
|
|
});
|
2016-11-04 11:16:49 +11:00
|
|
|
|
this._ignoreChange = false;
|
2016-04-28 12:01:20 -04:00
|
|
|
|
}
|
|
|
|
|
};
|
|
|
|
|
|
2015-06-18 15:54:37 +07:00
|
|
|
|
// --- Events ---
|
2013-04-08 13:27:06 +10:00
|
|
|
|
|
2015-06-18 15:54:37 +07:00
|
|
|
|
// Subscribing to these events won't automatically add a listener to the
|
|
|
|
|
// document node, since these events are fired in a custom manner by the
|
|
|
|
|
// editor code.
|
|
|
|
|
var customEvents = {
|
2016-05-05 12:22:56 +10:00
|
|
|
|
pathChange: 1, select: 1, input: 1, undoStateChange: 1
|
2015-06-18 15:54:37 +07:00
|
|
|
|
};
|
2014-04-07 13:05:44 +10:00
|
|
|
|
|
2015-06-18 15:54:37 +07:00
|
|
|
|
proto.fireEvent = function ( type, event ) {
|
2016-06-10 11:02:24 +10:00
|
|
|
|
var handlers = this._events[ type ];
|
|
|
|
|
var isFocused, l, obj;
|
|
|
|
|
// UI code, especially modal views, may be monitoring for focus events and
|
|
|
|
|
// immediately removing focus. In certain conditions, this can cause the
|
|
|
|
|
// focus event to fire after the blur event, which can cause an infinite
|
|
|
|
|
// loop. So we detect whether we're actually focused/blurred before firing.
|
|
|
|
|
if ( /^(?:focus|blur)/.test( type ) ) {
|
2017-05-10 14:22:56 +10:00
|
|
|
|
isFocused = this._root === this._doc.activeElement;
|
2016-06-10 11:02:24 +10:00
|
|
|
|
if ( type === 'focus' ) {
|
|
|
|
|
if ( !isFocused || this._isFocused ) {
|
|
|
|
|
return this;
|
|
|
|
|
}
|
|
|
|
|
this._isFocused = true;
|
|
|
|
|
} else {
|
|
|
|
|
if ( isFocused || !this._isFocused ) {
|
|
|
|
|
return this;
|
|
|
|
|
}
|
|
|
|
|
this._isFocused = false;
|
|
|
|
|
}
|
|
|
|
|
}
|
2015-06-18 15:54:37 +07:00
|
|
|
|
if ( handlers ) {
|
|
|
|
|
if ( !event ) {
|
|
|
|
|
event = {};
|
|
|
|
|
}
|
|
|
|
|
if ( event.type !== type ) {
|
|
|
|
|
event.type = type;
|
|
|
|
|
}
|
|
|
|
|
// Clone handlers array, so any handlers added/removed do not affect it.
|
|
|
|
|
handlers = handlers.slice();
|
2015-07-13 09:25:44 +07:00
|
|
|
|
l = handlers.length;
|
|
|
|
|
while ( l-- ) {
|
|
|
|
|
obj = handlers[l];
|
2015-06-18 15:54:37 +07:00
|
|
|
|
try {
|
|
|
|
|
if ( obj.handleEvent ) {
|
|
|
|
|
obj.handleEvent( event );
|
|
|
|
|
} else {
|
|
|
|
|
obj.call( this, event );
|
2015-04-29 11:16:11 +07:00
|
|
|
|
}
|
2015-06-18 15:54:37 +07:00
|
|
|
|
} catch ( error ) {
|
|
|
|
|
error.details = 'Squire: fireEvent error. Event type: ' + type;
|
|
|
|
|
this.didError( error );
|
2013-04-08 13:27:06 +10:00
|
|
|
|
}
|
2015-06-18 15:54:37 +07:00
|
|
|
|
}
|
2015-04-29 11:16:11 +07:00
|
|
|
|
}
|
2015-06-18 15:54:37 +07:00
|
|
|
|
return this;
|
|
|
|
|
};
|
2013-04-08 13:27:06 +10:00
|
|
|
|
|
2015-06-18 15:54:37 +07:00
|
|
|
|
proto.destroy = function () {
|
2016-05-26 11:07:11 +10:00
|
|
|
|
var events = this._events;
|
|
|
|
|
var type;
|
2016-05-31 15:26:22 -04:00
|
|
|
|
|
2015-06-18 15:54:37 +07:00
|
|
|
|
for ( type in events ) {
|
2016-05-26 11:07:11 +10:00
|
|
|
|
this.removeEventListener( type );
|
2014-04-07 13:05:44 +10:00
|
|
|
|
}
|
2015-06-18 15:54:37 +07:00
|
|
|
|
if ( this._mutation ) {
|
|
|
|
|
this._mutation.disconnect();
|
|
|
|
|
}
|
2016-12-07 17:54:13 +11:00
|
|
|
|
delete this._root.__squire__;
|
2016-05-31 15:26:22 -04:00
|
|
|
|
|
|
|
|
|
// Destroy undo stack
|
|
|
|
|
this._undoIndex = -1;
|
|
|
|
|
this._undoStack = [];
|
|
|
|
|
this._undoStackLength = 0;
|
2015-06-18 15:54:37 +07:00
|
|
|
|
};
|
2014-04-07 13:05:44 +10:00
|
|
|
|
|
2015-06-18 15:54:37 +07:00
|
|
|
|
proto.handleEvent = function ( event ) {
|
|
|
|
|
this.fireEvent( event.type, event );
|
2013-04-08 13:27:06 +10:00
|
|
|
|
};
|
|
|
|
|
|
2015-06-18 15:54:37 +07:00
|
|
|
|
proto.addEventListener = function ( type, fn ) {
|
|
|
|
|
var handlers = this._events[ type ];
|
2016-05-26 11:07:11 +10:00
|
|
|
|
var target = this._root;
|
2015-06-18 15:54:37 +07:00
|
|
|
|
if ( !fn ) {
|
|
|
|
|
this.didError({
|
|
|
|
|
name: 'Squire: addEventListener with null or undefined fn',
|
|
|
|
|
message: 'Event type: ' + type
|
|
|
|
|
});
|
|
|
|
|
return this;
|
2015-04-29 11:16:11 +07:00
|
|
|
|
}
|
2015-06-18 15:54:37 +07:00
|
|
|
|
if ( !handlers ) {
|
|
|
|
|
handlers = this._events[ type ] = [];
|
|
|
|
|
if ( !customEvents[ type ] ) {
|
2016-05-26 11:07:11 +10:00
|
|
|
|
if ( type === 'selectionchange' ) {
|
|
|
|
|
target = this._doc;
|
|
|
|
|
}
|
|
|
|
|
target.addEventListener( type, this, true );
|
2015-06-18 15:54:37 +07:00
|
|
|
|
}
|
2015-04-29 11:16:11 +07:00
|
|
|
|
}
|
2015-06-18 15:54:37 +07:00
|
|
|
|
handlers.push( fn );
|
|
|
|
|
return this;
|
|
|
|
|
};
|
|
|
|
|
|
|
|
|
|
proto.removeEventListener = function ( type, fn ) {
|
2016-05-26 11:07:11 +10:00
|
|
|
|
var handlers = this._events[ type ];
|
|
|
|
|
var target = this._root;
|
|
|
|
|
var l;
|
2015-06-18 15:54:37 +07:00
|
|
|
|
if ( handlers ) {
|
2016-05-26 11:07:11 +10:00
|
|
|
|
if ( fn ) {
|
|
|
|
|
l = handlers.length;
|
|
|
|
|
while ( l-- ) {
|
|
|
|
|
if ( handlers[l] === fn ) {
|
|
|
|
|
handlers.splice( l, 1 );
|
|
|
|
|
}
|
2015-06-18 15:54:37 +07:00
|
|
|
|
}
|
2016-05-26 11:07:11 +10:00
|
|
|
|
} else {
|
|
|
|
|
handlers.length = 0;
|
2015-06-18 15:54:37 +07:00
|
|
|
|
}
|
|
|
|
|
if ( !handlers.length ) {
|
|
|
|
|
delete this._events[ type ];
|
|
|
|
|
if ( !customEvents[ type ] ) {
|
2016-05-26 11:07:11 +10:00
|
|
|
|
if ( type === 'selectionchange' ) {
|
|
|
|
|
target = this._doc;
|
|
|
|
|
}
|
|
|
|
|
target.removeEventListener( type, this, true );
|
2015-06-18 15:54:37 +07:00
|
|
|
|
}
|
|
|
|
|
}
|
2015-04-29 11:16:11 +07:00
|
|
|
|
}
|
2015-06-18 15:54:37 +07:00
|
|
|
|
return this;
|
|
|
|
|
};
|
2013-04-08 13:27:06 +10:00
|
|
|
|
|
2015-06-18 15:54:37 +07:00
|
|
|
|
// --- Selection and Path ---
|
2015-04-29 11:16:11 +07:00
|
|
|
|
|
2018-05-09 17:43:11 +10:00
|
|
|
|
proto.createRange =
|
2015-06-18 15:54:37 +07:00
|
|
|
|
function ( range, startOffset, endContainer, endOffset ) {
|
|
|
|
|
if ( range instanceof this._win.Range ) {
|
|
|
|
|
return range.cloneRange();
|
|
|
|
|
}
|
|
|
|
|
var domRange = this._doc.createRange();
|
|
|
|
|
domRange.setStart( range, startOffset );
|
|
|
|
|
if ( endContainer ) {
|
|
|
|
|
domRange.setEnd( endContainer, endOffset );
|
|
|
|
|
} else {
|
|
|
|
|
domRange.setEnd( range, startOffset );
|
2013-04-08 13:27:06 +10:00
|
|
|
|
}
|
2015-06-18 15:54:37 +07:00
|
|
|
|
return domRange;
|
|
|
|
|
};
|
2015-04-29 11:16:11 +07:00
|
|
|
|
|
2016-03-24 13:45:36 +11:00
|
|
|
|
proto.getCursorPosition = function ( range ) {
|
2016-03-24 21:25:14 +11:00
|
|
|
|
if ( ( !range && !( range = this.getSelection() ) ) ||
|
|
|
|
|
!range.getBoundingClientRect ) {
|
2016-03-24 13:45:36 +11:00
|
|
|
|
return null;
|
|
|
|
|
}
|
2015-12-09 17:29:47 +11:00
|
|
|
|
// Get the bounding rect
|
|
|
|
|
var rect = range.getBoundingClientRect();
|
2015-12-09 15:12:47 +11:00
|
|
|
|
var node, parent;
|
2016-03-07 17:16:08 +11:00
|
|
|
|
if ( rect && !rect.top ) {
|
2016-03-24 13:45:36 +11:00
|
|
|
|
this._ignoreChange = true;
|
2015-12-09 15:12:47 +11:00
|
|
|
|
node = this._doc.createElement( 'SPAN' );
|
2016-03-24 13:45:36 +11:00
|
|
|
|
node.textContent = ZWS;
|
2015-12-09 15:12:47 +11:00
|
|
|
|
insertNodeInRange( range, node );
|
2015-12-09 17:29:47 +11:00
|
|
|
|
rect = node.getBoundingClientRect();
|
2015-12-09 15:12:47 +11:00
|
|
|
|
parent = node.parentNode;
|
|
|
|
|
parent.removeChild( node );
|
2016-07-14 12:15:06 +10:00
|
|
|
|
mergeInlines( parent, range );
|
2016-03-22 17:57:00 +11:00
|
|
|
|
}
|
2016-03-24 13:45:36 +11:00
|
|
|
|
return rect;
|
2015-12-09 15:12:47 +11:00
|
|
|
|
};
|
|
|
|
|
|
2015-06-18 15:54:37 +07:00
|
|
|
|
proto._moveCursorTo = function ( toStart ) {
|
2016-03-22 17:57:00 +11:00
|
|
|
|
var root = this._root,
|
2018-05-09 17:43:11 +10:00
|
|
|
|
range = this.createRange( root, toStart ? 0 : root.childNodes.length );
|
2015-06-18 15:54:37 +07:00
|
|
|
|
moveRangeBoundariesDownTree( range );
|
|
|
|
|
this.setSelection( range );
|
2015-04-29 11:16:11 +07:00
|
|
|
|
return this;
|
2013-04-08 13:27:06 +10:00
|
|
|
|
};
|
2015-06-18 15:54:37 +07:00
|
|
|
|
proto.moveCursorToStart = function () {
|
|
|
|
|
return this._moveCursorTo( true );
|
|
|
|
|
};
|
|
|
|
|
proto.moveCursorToEnd = function () {
|
|
|
|
|
return this._moveCursorTo( false );
|
2013-04-08 13:27:06 +10:00
|
|
|
|
};
|
|
|
|
|
|
2015-12-29 11:15:01 +11:00
|
|
|
|
var getWindowSelection = function ( self ) {
|
|
|
|
|
return self._win.getSelection() || null;
|
|
|
|
|
};
|
|
|
|
|
|
2015-06-18 15:54:37 +07:00
|
|
|
|
proto.setSelection = function ( range ) {
|
|
|
|
|
if ( range ) {
|
2016-03-28 09:43:14 +11:00
|
|
|
|
this._lastSelection = range;
|
2016-06-16 14:45:19 +10:00
|
|
|
|
// If we're setting selection, that automatically, and synchronously, // triggers a focus event. So just store the selection and mark it as
|
|
|
|
|
// needing restore on focus.
|
|
|
|
|
if ( !this._isFocused ) {
|
|
|
|
|
enableRestoreSelection.call( this );
|
2016-07-14 11:32:38 +10:00
|
|
|
|
} else if ( isAndroid && !this._restoreSelection ) {
|
|
|
|
|
// Android closes the keyboard on removeAllRanges() and doesn't
|
|
|
|
|
// open it again when addRange() is called, sigh.
|
|
|
|
|
// Since Android doesn't trigger a focus event in setSelection(),
|
|
|
|
|
// use a blur/focus dance to work around this by letting the
|
|
|
|
|
// selection be restored on focus.
|
|
|
|
|
// Need to check for !this._restoreSelection to avoid infinite loop
|
|
|
|
|
enableRestoreSelection.call( this );
|
|
|
|
|
this.blur();
|
|
|
|
|
this.focus();
|
2016-06-16 14:45:19 +10:00
|
|
|
|
} else {
|
|
|
|
|
// iOS bug: if you don't focus the iframe before setting the
|
|
|
|
|
// selection, you can end up in a state where you type but the input
|
|
|
|
|
// doesn't get directed into the contenteditable area but is instead
|
|
|
|
|
// lost in a black hole. Very strange.
|
|
|
|
|
if ( isIOS ) {
|
|
|
|
|
this._win.focus();
|
|
|
|
|
}
|
|
|
|
|
var sel = getWindowSelection( this );
|
|
|
|
|
if ( sel ) {
|
|
|
|
|
sel.removeAllRanges();
|
|
|
|
|
sel.addRange( range );
|
|
|
|
|
}
|
2015-10-13 09:11:44 +02:00
|
|
|
|
}
|
2015-04-29 11:16:11 +07:00
|
|
|
|
}
|
2015-06-18 15:54:37 +07:00
|
|
|
|
return this;
|
|
|
|
|
};
|
2015-04-29 11:16:11 +07:00
|
|
|
|
|
2015-06-18 15:54:37 +07:00
|
|
|
|
proto.getSelection = function () {
|
2016-03-22 17:57:00 +11:00
|
|
|
|
var sel = getWindowSelection( this );
|
|
|
|
|
var root = this._root;
|
2017-07-07 11:48:42 +10:00
|
|
|
|
var selection, startContainer, endContainer, node;
|
2017-02-11 09:40:56 -08:00
|
|
|
|
// If not focused, always rely on cached selection; another function may
|
|
|
|
|
// have set it but the DOM is not modified until focus again
|
|
|
|
|
if ( this._isFocused && sel && sel.rangeCount ) {
|
2015-06-18 15:54:37 +07:00
|
|
|
|
selection = sel.getRangeAt( 0 ).cloneRange();
|
|
|
|
|
startContainer = selection.startContainer;
|
|
|
|
|
endContainer = selection.endContainer;
|
|
|
|
|
// FF can return the selection as being inside an <img>. WTF?
|
|
|
|
|
if ( startContainer && isLeaf( startContainer ) ) {
|
|
|
|
|
selection.setStartBefore( startContainer );
|
2013-04-08 13:27:06 +10:00
|
|
|
|
}
|
2015-06-18 15:54:37 +07:00
|
|
|
|
if ( endContainer && isLeaf( endContainer ) ) {
|
|
|
|
|
selection.setEndBefore( endContainer );
|
|
|
|
|
}
|
2016-03-22 17:57:00 +11:00
|
|
|
|
}
|
|
|
|
|
if ( selection &&
|
|
|
|
|
isOrContains( root, selection.commonAncestorContainer ) ) {
|
2015-06-18 15:54:37 +07:00
|
|
|
|
this._lastSelection = selection;
|
|
|
|
|
} else {
|
|
|
|
|
selection = this._lastSelection;
|
2017-07-07 11:48:42 +10:00
|
|
|
|
node = selection.commonAncestorContainer;
|
|
|
|
|
// Check the editor is in the live document; if not, the range has
|
|
|
|
|
// probably been rewritten by the browser and is bogus
|
|
|
|
|
if ( !isOrContains( node.ownerDocument, node ) ) {
|
|
|
|
|
selection = null;
|
|
|
|
|
}
|
2013-04-08 13:27:06 +10:00
|
|
|
|
}
|
2015-06-18 15:54:37 +07:00
|
|
|
|
if ( !selection ) {
|
2018-05-09 17:43:11 +10:00
|
|
|
|
selection = this.createRange( root.firstChild, 0 );
|
2015-06-18 15:54:37 +07:00
|
|
|
|
}
|
|
|
|
|
return selection;
|
2013-04-08 13:27:06 +10:00
|
|
|
|
};
|
|
|
|
|
|
2016-03-28 09:43:14 +11:00
|
|
|
|
function enableRestoreSelection () {
|
|
|
|
|
this._restoreSelection = true;
|
|
|
|
|
}
|
|
|
|
|
function disableRestoreSelection () {
|
|
|
|
|
this._restoreSelection = false;
|
|
|
|
|
}
|
|
|
|
|
function restoreSelection () {
|
|
|
|
|
if ( this._restoreSelection ) {
|
|
|
|
|
this.setSelection( this._lastSelection );
|
|
|
|
|
}
|
|
|
|
|
}
|
|
|
|
|
|
2015-06-18 15:54:37 +07:00
|
|
|
|
proto.getSelectedText = function () {
|
2017-07-07 11:48:42 +10:00
|
|
|
|
var range = this.getSelection();
|
|
|
|
|
if ( !range || range.collapsed ) {
|
|
|
|
|
return '';
|
|
|
|
|
}
|
|
|
|
|
var walker = new TreeWalker(
|
|
|
|
|
range.commonAncestorContainer,
|
|
|
|
|
SHOW_TEXT|SHOW_ELEMENT,
|
|
|
|
|
function ( node ) {
|
|
|
|
|
return isNodeContainedInRange( range, node, true );
|
|
|
|
|
}
|
|
|
|
|
);
|
|
|
|
|
var startContainer = range.startContainer;
|
|
|
|
|
var endContainer = range.endContainer;
|
|
|
|
|
var node = walker.currentNode = startContainer;
|
|
|
|
|
var textContent = '';
|
|
|
|
|
var addedTextInBlock = false;
|
|
|
|
|
var value;
|
2014-02-03 17:57:20 +11:00
|
|
|
|
|
2015-06-18 15:54:37 +07:00
|
|
|
|
if ( !walker.filter( node ) ) {
|
|
|
|
|
node = walker.nextNode();
|
2015-04-29 11:16:11 +07:00
|
|
|
|
}
|
2013-04-08 13:27:06 +10:00
|
|
|
|
|
2015-06-18 15:54:37 +07:00
|
|
|
|
while ( node ) {
|
|
|
|
|
if ( node.nodeType === TEXT_NODE ) {
|
|
|
|
|
value = node.data;
|
|
|
|
|
if ( value && ( /\S/.test( value ) ) ) {
|
|
|
|
|
if ( node === endContainer ) {
|
|
|
|
|
value = value.slice( 0, range.endOffset );
|
|
|
|
|
}
|
|
|
|
|
if ( node === startContainer ) {
|
|
|
|
|
value = value.slice( range.startOffset );
|
|
|
|
|
}
|
|
|
|
|
textContent += value;
|
|
|
|
|
addedTextInBlock = true;
|
|
|
|
|
}
|
|
|
|
|
} else if ( node.nodeName === 'BR' ||
|
|
|
|
|
addedTextInBlock && !isInline( node ) ) {
|
|
|
|
|
textContent += '\n';
|
|
|
|
|
addedTextInBlock = false;
|
|
|
|
|
}
|
|
|
|
|
node = walker.nextNode();
|
2015-04-29 11:16:11 +07:00
|
|
|
|
}
|
2013-04-08 13:27:06 +10:00
|
|
|
|
|
2015-06-18 15:54:37 +07:00
|
|
|
|
return textContent;
|
|
|
|
|
};
|
2013-04-08 13:27:06 +10:00
|
|
|
|
|
2015-06-18 15:54:37 +07:00
|
|
|
|
proto.getPath = function () {
|
|
|
|
|
return this._path;
|
|
|
|
|
};
|
2015-04-29 11:16:11 +07:00
|
|
|
|
|
2015-06-18 15:54:37 +07:00
|
|
|
|
// --- Workaround for browsers that can't focus empty text nodes ---
|
|
|
|
|
|
|
|
|
|
// WebKit bug: https://bugs.webkit.org/show_bug.cgi?id=15256
|
|
|
|
|
|
2016-08-05 20:47:02 +10:00
|
|
|
|
// Walk down the tree starting at the root and remove any ZWS. If the node only
|
|
|
|
|
// contained ZWS space then remove it too. We may want to keep one ZWS node at
|
|
|
|
|
// the bottom of the tree so the block can be selected. Define that node as the
|
|
|
|
|
// keepNode.
|
2016-07-25 12:24:51 -04:00
|
|
|
|
var removeZWS = function ( root, keepNode ) {
|
2018-07-27 09:21:50 +10:00
|
|
|
|
var walker = new TreeWalker( root, SHOW_TEXT );
|
|
|
|
|
var parent, node, index;
|
2015-06-18 15:54:37 +07:00
|
|
|
|
while ( node = walker.nextNode() ) {
|
2016-08-05 20:47:02 +10:00
|
|
|
|
while ( ( index = node.data.indexOf( ZWS ) ) > -1 &&
|
|
|
|
|
( !keepNode || node.parentNode !== keepNode ) ) {
|
2015-06-18 15:54:37 +07:00
|
|
|
|
if ( node.length === 1 ) {
|
|
|
|
|
do {
|
|
|
|
|
parent = node.parentNode;
|
|
|
|
|
parent.removeChild( node );
|
|
|
|
|
node = parent;
|
2015-12-03 17:55:22 +11:00
|
|
|
|
walker.currentNode = parent;
|
2015-06-18 15:54:37 +07:00
|
|
|
|
} while ( isInline( node ) && !getLength( node ) );
|
|
|
|
|
break;
|
|
|
|
|
} else {
|
|
|
|
|
node.deleteData( index, 1 );
|
|
|
|
|
}
|
2013-04-08 13:27:06 +10:00
|
|
|
|
}
|
|
|
|
|
}
|
2015-04-29 11:16:11 +07:00
|
|
|
|
};
|
|
|
|
|
|
2015-06-18 15:54:37 +07:00
|
|
|
|
proto._didAddZWS = function () {
|
|
|
|
|
this._hasZWS = true;
|
|
|
|
|
};
|
|
|
|
|
proto._removeZWS = function () {
|
|
|
|
|
if ( !this._hasZWS ) {
|
|
|
|
|
return;
|
2015-04-29 11:16:11 +07:00
|
|
|
|
}
|
2016-03-22 17:57:00 +11:00
|
|
|
|
removeZWS( this._root );
|
2015-06-18 15:54:37 +07:00
|
|
|
|
this._hasZWS = false;
|
|
|
|
|
};
|
2013-04-08 13:27:06 +10:00
|
|
|
|
|
2015-06-18 15:54:37 +07:00
|
|
|
|
// --- Path change events ---
|
2015-04-29 11:16:11 +07:00
|
|
|
|
|
2015-06-18 15:54:37 +07:00
|
|
|
|
proto._updatePath = function ( range, force ) {
|
2017-07-07 11:48:42 +10:00
|
|
|
|
if ( !range ) {
|
|
|
|
|
return;
|
|
|
|
|
}
|
2015-06-18 15:54:37 +07:00
|
|
|
|
var anchor = range.startContainer,
|
|
|
|
|
focus = range.endContainer,
|
|
|
|
|
newPath;
|
|
|
|
|
if ( force || anchor !== this._lastAnchorNode ||
|
|
|
|
|
focus !== this._lastFocusNode ) {
|
|
|
|
|
this._lastAnchorNode = anchor;
|
|
|
|
|
this._lastFocusNode = focus;
|
|
|
|
|
newPath = ( anchor && focus ) ? ( anchor === focus ) ?
|
2018-07-12 13:35:20 +10:00
|
|
|
|
getPath( focus, this._root, this._config ) : '(selection)' : '';
|
2015-06-18 15:54:37 +07:00
|
|
|
|
if ( this._path !== newPath ) {
|
|
|
|
|
this._path = newPath;
|
|
|
|
|
this.fireEvent( 'pathChange', { path: newPath } );
|
|
|
|
|
}
|
|
|
|
|
}
|
2017-06-28 16:03:31 +10:00
|
|
|
|
this.fireEvent( range.collapsed ? 'cursor' : 'select', {
|
|
|
|
|
range: range
|
|
|
|
|
});
|
2015-06-18 15:54:37 +07:00
|
|
|
|
};
|
2015-04-29 11:16:11 +07:00
|
|
|
|
|
2016-06-27 13:17:18 +10:00
|
|
|
|
// selectionchange is fired synchronously in IE when removing current selection
|
|
|
|
|
// and when setting new selection; keyup/mouseup may have processing we want
|
|
|
|
|
// to do first. Either way, send to next event loop.
|
2020-07-08 16:12:32 +10:00
|
|
|
|
proto._updatePathOnEvent = function () {
|
2016-06-27 13:17:18 +10:00
|
|
|
|
var self = this;
|
2017-07-07 11:48:42 +10:00
|
|
|
|
if ( self._isFocused && !self._willUpdatePath ) {
|
2016-06-27 13:17:18 +10:00
|
|
|
|
self._willUpdatePath = true;
|
|
|
|
|
setTimeout( function () {
|
|
|
|
|
self._willUpdatePath = false;
|
|
|
|
|
self._updatePath( self.getSelection() );
|
|
|
|
|
}, 0 );
|
|
|
|
|
}
|
2015-06-18 15:54:37 +07:00
|
|
|
|
};
|
2015-04-29 11:16:11 +07:00
|
|
|
|
|
2015-06-18 15:54:37 +07:00
|
|
|
|
// --- Focus ---
|
2015-04-29 11:16:11 +07:00
|
|
|
|
|
2015-06-18 15:54:37 +07:00
|
|
|
|
proto.focus = function () {
|
2019-09-25 20:49:56 -04:00
|
|
|
|
this._root.focus({ preventScroll: true });
|
2016-08-05 20:47:02 +10:00
|
|
|
|
|
|
|
|
|
if ( isIE ) {
|
|
|
|
|
this.fireEvent( 'focus' );
|
|
|
|
|
}
|
|
|
|
|
|
2015-06-18 15:54:37 +07:00
|
|
|
|
return this;
|
|
|
|
|
};
|
2015-04-29 11:16:11 +07:00
|
|
|
|
|
2015-06-18 15:54:37 +07:00
|
|
|
|
proto.blur = function () {
|
2016-03-22 17:57:00 +11:00
|
|
|
|
this._root.blur();
|
2016-08-05 20:47:02 +10:00
|
|
|
|
|
|
|
|
|
if ( isIE ) {
|
|
|
|
|
this.fireEvent( 'blur' );
|
|
|
|
|
}
|
|
|
|
|
|
2015-04-29 11:16:11 +07:00
|
|
|
|
return this;
|
2013-04-08 13:27:06 +10:00
|
|
|
|
};
|
|
|
|
|
|
2015-06-18 15:54:37 +07:00
|
|
|
|
// --- Bookmarking ---
|
2013-04-08 13:27:06 +10:00
|
|
|
|
|
2015-06-18 15:54:37 +07:00
|
|
|
|
var startSelectionId = 'squire-selection-start';
|
|
|
|
|
var endSelectionId = 'squire-selection-end';
|
2015-04-29 11:16:11 +07:00
|
|
|
|
|
2015-06-18 15:54:37 +07:00
|
|
|
|
proto._saveRangeToBookmark = function ( range ) {
|
|
|
|
|
var startNode = this.createElement( 'INPUT', {
|
2015-04-29 11:16:11 +07:00
|
|
|
|
id: startSelectionId,
|
|
|
|
|
type: 'hidden'
|
|
|
|
|
}),
|
2015-06-18 15:54:37 +07:00
|
|
|
|
endNode = this.createElement( 'INPUT', {
|
2015-04-29 11:16:11 +07:00
|
|
|
|
id: endSelectionId,
|
|
|
|
|
type: 'hidden'
|
2015-06-18 15:54:37 +07:00
|
|
|
|
}),
|
|
|
|
|
temp;
|
|
|
|
|
|
|
|
|
|
insertNodeInRange( range, startNode );
|
|
|
|
|
range.collapse( false );
|
|
|
|
|
insertNodeInRange( range, endNode );
|
|
|
|
|
|
|
|
|
|
// In a collapsed range, the start is sometimes inserted after the end!
|
|
|
|
|
if ( startNode.compareDocumentPosition( endNode ) &
|
|
|
|
|
DOCUMENT_POSITION_PRECEDING ) {
|
|
|
|
|
startNode.id = endSelectionId;
|
|
|
|
|
endNode.id = startSelectionId;
|
|
|
|
|
temp = startNode;
|
|
|
|
|
startNode = endNode;
|
|
|
|
|
endNode = temp;
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
range.setStartAfter( startNode );
|
|
|
|
|
range.setEndBefore( endNode );
|
2015-04-29 11:16:11 +07:00
|
|
|
|
};
|
|
|
|
|
|
2015-06-18 15:54:37 +07:00
|
|
|
|
proto._getRangeAndRemoveBookmark = function ( range ) {
|
2016-03-22 17:57:00 +11:00
|
|
|
|
var root = this._root,
|
|
|
|
|
start = root.querySelector( '#' + startSelectionId ),
|
|
|
|
|
end = root.querySelector( '#' + endSelectionId );
|
2015-04-29 11:16:11 +07:00
|
|
|
|
|
2015-06-18 15:54:37 +07:00
|
|
|
|
if ( start && end ) {
|
|
|
|
|
var startContainer = start.parentNode,
|
|
|
|
|
endContainer = end.parentNode,
|
2016-07-14 12:15:06 +10:00
|
|
|
|
startOffset = indexOf.call( startContainer.childNodes, start ),
|
|
|
|
|
endOffset = indexOf.call( endContainer.childNodes, end );
|
2015-06-18 15:54:37 +07:00
|
|
|
|
|
|
|
|
|
if ( startContainer === endContainer ) {
|
2016-07-14 12:15:06 +10:00
|
|
|
|
endOffset -= 1;
|
2015-06-18 15:54:37 +07:00
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
detach( start );
|
|
|
|
|
detach( end );
|
|
|
|
|
|
|
|
|
|
if ( !range ) {
|
2016-03-22 17:57:00 +11:00
|
|
|
|
range = this._doc.createRange();
|
2015-06-18 15:54:37 +07:00
|
|
|
|
}
|
2016-07-14 12:15:06 +10:00
|
|
|
|
range.setStart( startContainer, startOffset );
|
|
|
|
|
range.setEnd( endContainer, endOffset );
|
|
|
|
|
|
|
|
|
|
// Merge any text nodes we split
|
|
|
|
|
mergeInlines( startContainer, range );
|
|
|
|
|
if ( startContainer !== endContainer ) {
|
|
|
|
|
mergeInlines( endContainer, range );
|
|
|
|
|
}
|
2015-06-18 15:54:37 +07:00
|
|
|
|
|
2016-03-30 16:39:04 +11:00
|
|
|
|
// If we didn't split a text node, we should move into any adjacent
|
|
|
|
|
// text node to current selection point
|
2016-07-14 12:15:06 +10:00
|
|
|
|
if ( range.collapsed ) {
|
2016-03-30 16:39:04 +11:00
|
|
|
|
startContainer = range.startContainer;
|
|
|
|
|
if ( startContainer.nodeType === TEXT_NODE ) {
|
|
|
|
|
endContainer = startContainer.childNodes[ range.startOffset ];
|
|
|
|
|
if ( !endContainer || endContainer.nodeType !== TEXT_NODE ) {
|
|
|
|
|
endContainer =
|
|
|
|
|
startContainer.childNodes[ range.startOffset - 1 ];
|
|
|
|
|
}
|
|
|
|
|
if ( endContainer && endContainer.nodeType === TEXT_NODE ) {
|
|
|
|
|
range.setStart( endContainer, 0 );
|
|
|
|
|
range.collapse( true );
|
|
|
|
|
}
|
|
|
|
|
}
|
2013-04-08 13:27:06 +10:00
|
|
|
|
}
|
|
|
|
|
}
|
2015-06-18 15:54:37 +07:00
|
|
|
|
return range || null;
|
2013-04-08 13:27:06 +10:00
|
|
|
|
};
|
|
|
|
|
|
2015-06-18 15:54:37 +07:00
|
|
|
|
// --- Undo ---
|
2013-04-08 13:27:06 +10:00
|
|
|
|
|
2015-06-18 15:54:37 +07:00
|
|
|
|
proto._keyUpDetectChange = function ( event ) {
|
|
|
|
|
var code = event.keyCode;
|
|
|
|
|
// Presume document was changed if:
|
|
|
|
|
// 1. A modifier key (other than shift) wasn't held down
|
|
|
|
|
// 2. The key pressed is not in range 16<=x<=20 (control keys)
|
|
|
|
|
// 3. The key pressed is not in range 33<=x<=45 (navigation keys)
|
|
|
|
|
if ( !event.ctrlKey && !event.metaKey && !event.altKey &&
|
|
|
|
|
( code < 16 || code > 20 ) &&
|
2015-09-28 15:48:23 +02:00
|
|
|
|
( code < 33 || code > 45 ) ) {
|
2015-06-18 15:54:37 +07:00
|
|
|
|
this._docWasChanged();
|
|
|
|
|
}
|
2015-04-29 11:16:11 +07:00
|
|
|
|
};
|
2013-04-08 13:27:06 +10:00
|
|
|
|
|
2015-06-18 15:54:37 +07:00
|
|
|
|
proto._docWasChanged = function () {
|
2016-11-15 11:07:15 +11:00
|
|
|
|
if ( canWeakMap ) {
|
|
|
|
|
nodeCategoryCache = new WeakMap();
|
|
|
|
|
}
|
2016-04-28 12:01:20 -04:00
|
|
|
|
if ( this._ignoreAllChanges ) {
|
|
|
|
|
return;
|
|
|
|
|
}
|
|
|
|
|
|
2015-06-18 15:54:37 +07:00
|
|
|
|
if ( canObserveMutations && this._ignoreChange ) {
|
|
|
|
|
this._ignoreChange = false;
|
|
|
|
|
return;
|
2013-04-08 13:27:06 +10:00
|
|
|
|
}
|
2015-06-18 15:54:37 +07:00
|
|
|
|
if ( this._isInUndoState ) {
|
|
|
|
|
this._isInUndoState = false;
|
|
|
|
|
this.fireEvent( 'undoStateChange', {
|
|
|
|
|
canUndo: true,
|
|
|
|
|
canRedo: false
|
|
|
|
|
});
|
|
|
|
|
}
|
|
|
|
|
this.fireEvent( 'input' );
|
2013-04-08 13:27:06 +10:00
|
|
|
|
};
|
|
|
|
|
|
2015-06-18 15:54:37 +07:00
|
|
|
|
// Leaves bookmark
|
2017-07-16 12:43:17 +02:00
|
|
|
|
proto._recordUndoState = function ( range, replace ) {
|
2015-06-18 15:54:37 +07:00
|
|
|
|
// Don't record if we're already in an undo state
|
2017-07-16 12:43:17 +02:00
|
|
|
|
if ( !this._isInUndoState|| replace ) {
|
2015-06-18 15:54:37 +07:00
|
|
|
|
// Advance pointer to new position
|
2017-07-16 12:43:17 +02:00
|
|
|
|
var undoIndex = this._undoIndex;
|
2016-05-31 15:26:22 -04:00
|
|
|
|
var undoStack = this._undoStack;
|
|
|
|
|
var undoConfig = this._config.undo;
|
|
|
|
|
var undoThreshold = undoConfig.documentSizeThreshold;
|
|
|
|
|
var undoLimit = undoConfig.undoLimit;
|
|
|
|
|
var html;
|
2015-06-18 15:54:37 +07:00
|
|
|
|
|
2017-07-16 12:43:17 +02:00
|
|
|
|
if ( !replace ) {
|
|
|
|
|
undoIndex += 1;
|
|
|
|
|
}
|
|
|
|
|
|
2015-06-18 15:54:37 +07:00
|
|
|
|
// Truncate stack if longer (i.e. if has been previously undone)
|
2015-09-28 15:48:23 +02:00
|
|
|
|
if ( undoIndex < this._undoStackLength ) {
|
2015-06-18 15:54:37 +07:00
|
|
|
|
undoStack.length = this._undoStackLength = undoIndex;
|
2015-04-29 11:16:11 +07:00
|
|
|
|
}
|
2015-06-18 15:54:37 +07:00
|
|
|
|
|
2016-05-31 15:26:22 -04:00
|
|
|
|
// Get data
|
2015-06-18 15:54:37 +07:00
|
|
|
|
if ( range ) {
|
|
|
|
|
this._saveRangeToBookmark( range );
|
|
|
|
|
}
|
2016-05-31 15:26:22 -04:00
|
|
|
|
html = this._getHTML();
|
|
|
|
|
|
|
|
|
|
// If this document is above the configured size threshold,
|
|
|
|
|
// limit the number of saved undo states.
|
|
|
|
|
// Threshold is in bytes, JS uses 2 bytes per character
|
|
|
|
|
if ( undoThreshold > -1 && html.length * 2 > undoThreshold ) {
|
|
|
|
|
if ( undoLimit > -1 && undoIndex > undoLimit ) {
|
|
|
|
|
undoStack.splice( 0, undoIndex - undoLimit );
|
2017-07-16 12:43:17 +02:00
|
|
|
|
undoIndex = undoLimit;
|
2016-05-31 15:26:22 -04:00
|
|
|
|
this._undoStackLength = undoLimit;
|
|
|
|
|
}
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
// Save data
|
|
|
|
|
undoStack[ undoIndex ] = html;
|
2017-07-16 12:43:17 +02:00
|
|
|
|
this._undoIndex = undoIndex;
|
2015-06-18 15:54:37 +07:00
|
|
|
|
this._undoStackLength += 1;
|
|
|
|
|
this._isInUndoState = true;
|
2014-06-02 10:32:21 +10:00
|
|
|
|
}
|
|
|
|
|
};
|
|
|
|
|
|
2016-03-11 15:56:34 +11:00
|
|
|
|
proto.saveUndoState = function ( range ) {
|
|
|
|
|
if ( range === undefined ) {
|
|
|
|
|
range = this.getSelection();
|
|
|
|
|
}
|
2017-07-16 12:43:17 +02:00
|
|
|
|
this._recordUndoState( range, this._isInUndoState );
|
|
|
|
|
this._getRangeAndRemoveBookmark( range );
|
|
|
|
|
|
2016-03-11 15:56:34 +11:00
|
|
|
|
return this;
|
|
|
|
|
};
|
|
|
|
|
|
2015-06-18 15:54:37 +07:00
|
|
|
|
proto.undo = function () {
|
|
|
|
|
// Sanity check: must not be at beginning of the history stack
|
|
|
|
|
if ( this._undoIndex !== 0 || !this._isInUndoState ) {
|
|
|
|
|
// Make sure any changes since last checkpoint are saved.
|
2017-07-16 12:43:17 +02:00
|
|
|
|
this._recordUndoState( this.getSelection(), false );
|
2015-06-18 15:54:37 +07:00
|
|
|
|
|
|
|
|
|
this._undoIndex -= 1;
|
|
|
|
|
this._setHTML( this._undoStack[ this._undoIndex ] );
|
|
|
|
|
var range = this._getRangeAndRemoveBookmark();
|
|
|
|
|
if ( range ) {
|
|
|
|
|
this.setSelection( range );
|
2013-04-08 13:27:06 +10:00
|
|
|
|
}
|
2015-06-18 15:54:37 +07:00
|
|
|
|
this._isInUndoState = true;
|
|
|
|
|
this.fireEvent( 'undoStateChange', {
|
|
|
|
|
canUndo: this._undoIndex !== 0,
|
|
|
|
|
canRedo: true
|
|
|
|
|
});
|
|
|
|
|
this.fireEvent( 'input' );
|
|
|
|
|
}
|
|
|
|
|
return this;
|
|
|
|
|
};
|
|
|
|
|
|
|
|
|
|
proto.redo = function () {
|
|
|
|
|
// Sanity check: must not be at end of stack and must be in an undo
|
|
|
|
|
// state.
|
|
|
|
|
var undoIndex = this._undoIndex,
|
|
|
|
|
undoStackLength = this._undoStackLength;
|
|
|
|
|
if ( undoIndex + 1 < undoStackLength && this._isInUndoState ) {
|
|
|
|
|
this._undoIndex += 1;
|
|
|
|
|
this._setHTML( this._undoStack[ this._undoIndex ] );
|
|
|
|
|
var range = this._getRangeAndRemoveBookmark();
|
|
|
|
|
if ( range ) {
|
|
|
|
|
this.setSelection( range );
|
2015-04-29 11:16:11 +07:00
|
|
|
|
}
|
2015-06-18 15:54:37 +07:00
|
|
|
|
this.fireEvent( 'undoStateChange', {
|
|
|
|
|
canUndo: true,
|
|
|
|
|
canRedo: undoIndex + 2 < undoStackLength
|
|
|
|
|
});
|
|
|
|
|
this.fireEvent( 'input' );
|
|
|
|
|
}
|
|
|
|
|
return this;
|
2015-04-29 11:16:11 +07:00
|
|
|
|
};
|
2014-08-06 14:42:57 +02:00
|
|
|
|
|
2015-06-18 15:54:37 +07:00
|
|
|
|
// --- Inline formatting ---
|
2014-10-01 09:29:54 +07:00
|
|
|
|
|
2015-06-18 15:54:37 +07:00
|
|
|
|
// Looks for matching tag and attributes, so won't work
|
|
|
|
|
// if <strong> instead of <b> etc.
|
|
|
|
|
proto.hasFormat = function ( tag, attributes, range ) {
|
|
|
|
|
// 1. Normalise the arguments and get selection
|
|
|
|
|
tag = tag.toUpperCase();
|
|
|
|
|
if ( !attributes ) { attributes = {}; }
|
|
|
|
|
if ( !range && !( range = this.getSelection() ) ) {
|
|
|
|
|
return false;
|
|
|
|
|
}
|
2015-02-06 14:09:37 +07:00
|
|
|
|
|
2015-10-02 15:26:16 +02:00
|
|
|
|
// Sanitize range to prevent weird IE artifacts
|
|
|
|
|
if ( !range.collapsed &&
|
|
|
|
|
range.startContainer.nodeType === TEXT_NODE &&
|
|
|
|
|
range.startOffset === range.startContainer.length &&
|
|
|
|
|
range.startContainer.nextSibling ) {
|
|
|
|
|
range.setStartBefore( range.startContainer.nextSibling );
|
|
|
|
|
}
|
|
|
|
|
if ( !range.collapsed &&
|
|
|
|
|
range.endContainer.nodeType === TEXT_NODE &&
|
|
|
|
|
range.endOffset === 0 &&
|
2015-10-13 08:39:22 +02:00
|
|
|
|
range.endContainer.previousSibling ) {
|
2015-10-02 15:26:16 +02:00
|
|
|
|
range.setEndAfter( range.endContainer.previousSibling );
|
|
|
|
|
}
|
|
|
|
|
|
2015-06-18 15:54:37 +07:00
|
|
|
|
// If the common ancestor is inside the tag we require, we definitely
|
|
|
|
|
// have the format.
|
2016-03-22 17:57:00 +11:00
|
|
|
|
var root = this._root;
|
|
|
|
|
var common = range.commonAncestorContainer;
|
|
|
|
|
var walker, node;
|
|
|
|
|
if ( getNearest( common, root, tag, attributes ) ) {
|
2015-06-18 15:54:37 +07:00
|
|
|
|
return true;
|
2015-04-29 11:16:11 +07:00
|
|
|
|
}
|
2013-04-08 13:27:06 +10:00
|
|
|
|
|
2015-06-18 15:54:37 +07:00
|
|
|
|
// If common ancestor is a text node and doesn't have the format, we
|
|
|
|
|
// definitely don't have it.
|
2016-03-22 17:57:00 +11:00
|
|
|
|
if ( common.nodeType === TEXT_NODE ) {
|
2015-06-18 15:54:37 +07:00
|
|
|
|
return false;
|
|
|
|
|
}
|
2013-04-08 13:27:06 +10:00
|
|
|
|
|
2015-06-18 15:54:37 +07:00
|
|
|
|
// Otherwise, check each text node at least partially contained within
|
|
|
|
|
// the selection and make sure all of them have the format we want.
|
2016-03-22 17:57:00 +11:00
|
|
|
|
walker = new TreeWalker( common, SHOW_TEXT, function ( node ) {
|
2015-06-18 15:54:37 +07:00
|
|
|
|
return isNodeContainedInRange( range, node, true );
|
2018-07-27 09:21:50 +10:00
|
|
|
|
});
|
2013-04-08 13:27:06 +10:00
|
|
|
|
|
2015-06-18 15:54:37 +07:00
|
|
|
|
var seenNode = false;
|
|
|
|
|
while ( node = walker.nextNode() ) {
|
2016-03-22 17:57:00 +11:00
|
|
|
|
if ( !getNearest( node, root, tag, attributes ) ) {
|
2015-06-18 15:54:37 +07:00
|
|
|
|
return false;
|
2015-04-29 11:16:11 +07:00
|
|
|
|
}
|
2015-06-18 15:54:37 +07:00
|
|
|
|
seenNode = true;
|
2015-04-29 11:16:11 +07:00
|
|
|
|
}
|
2013-04-08 13:27:06 +10:00
|
|
|
|
|
2015-06-18 15:54:37 +07:00
|
|
|
|
return seenNode;
|
2015-04-29 11:16:11 +07:00
|
|
|
|
};
|
2013-04-08 13:27:06 +10:00
|
|
|
|
|
2015-08-27 11:48:54 +02:00
|
|
|
|
// Extracts the font-family and font-size (if any) of the element
|
|
|
|
|
// holding the cursor. If there's a selection, returns an empty object.
|
|
|
|
|
proto.getFontInfo = function ( range ) {
|
|
|
|
|
var fontInfo = {
|
2015-12-03 18:14:49 +11:00
|
|
|
|
color: undefined,
|
|
|
|
|
backgroundColor: undefined,
|
|
|
|
|
family: undefined,
|
|
|
|
|
size: undefined
|
|
|
|
|
};
|
|
|
|
|
var seenAttributes = 0;
|
2016-02-03 15:35:13 +11:00
|
|
|
|
var element, style, attr;
|
2015-08-27 11:48:54 +02:00
|
|
|
|
|
|
|
|
|
if ( !range && !( range = this.getSelection() ) ) {
|
|
|
|
|
return fontInfo;
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
element = range.commonAncestorContainer;
|
|
|
|
|
if ( range.collapsed || element.nodeType === TEXT_NODE ) {
|
|
|
|
|
if ( element.nodeType === TEXT_NODE ) {
|
|
|
|
|
element = element.parentNode;
|
|
|
|
|
}
|
2016-03-18 18:55:21 +11:00
|
|
|
|
while ( seenAttributes < 4 && element ) {
|
|
|
|
|
if ( style = element.style ) {
|
|
|
|
|
if ( !fontInfo.color && ( attr = style.color ) ) {
|
|
|
|
|
fontInfo.color = attr;
|
|
|
|
|
seenAttributes += 1;
|
|
|
|
|
}
|
|
|
|
|
if ( !fontInfo.backgroundColor &&
|
|
|
|
|
( attr = style.backgroundColor ) ) {
|
|
|
|
|
fontInfo.backgroundColor = attr;
|
|
|
|
|
seenAttributes += 1;
|
|
|
|
|
}
|
|
|
|
|
if ( !fontInfo.family && ( attr = style.fontFamily ) ) {
|
|
|
|
|
fontInfo.family = attr;
|
|
|
|
|
seenAttributes += 1;
|
|
|
|
|
}
|
|
|
|
|
if ( !fontInfo.size && ( attr = style.fontSize ) ) {
|
|
|
|
|
fontInfo.size = attr;
|
|
|
|
|
seenAttributes += 1;
|
|
|
|
|
}
|
2015-08-27 11:48:54 +02:00
|
|
|
|
}
|
|
|
|
|
element = element.parentNode;
|
|
|
|
|
}
|
|
|
|
|
}
|
|
|
|
|
return fontInfo;
|
2016-03-18 18:55:21 +11:00
|
|
|
|
};
|
2015-08-27 11:48:54 +02:00
|
|
|
|
|
2015-06-18 15:54:37 +07:00
|
|
|
|
proto._addFormat = function ( tag, attributes, range ) {
|
|
|
|
|
// If the range is collapsed we simply insert the node by wrapping
|
|
|
|
|
// it round the range and focus it.
|
2016-03-22 17:57:00 +11:00
|
|
|
|
var root = this._root;
|
2015-06-18 15:54:37 +07:00
|
|
|
|
var el, walker, startContainer, endContainer, startOffset, endOffset,
|
2016-08-05 20:47:02 +10:00
|
|
|
|
node, needsFormat, block;
|
2013-04-08 13:27:06 +10:00
|
|
|
|
|
2015-06-18 15:54:37 +07:00
|
|
|
|
if ( range.collapsed ) {
|
2016-03-22 17:57:00 +11:00
|
|
|
|
el = fixCursor( this.createElement( tag, attributes ), root );
|
2015-06-18 15:54:37 +07:00
|
|
|
|
insertNodeInRange( range, el );
|
|
|
|
|
range.setStart( el.firstChild, el.firstChild.length );
|
|
|
|
|
range.collapse( true );
|
2016-07-25 12:24:51 -04:00
|
|
|
|
|
|
|
|
|
// Clean up any previous formats that may have been set on this block
|
|
|
|
|
// that are unused.
|
2016-08-05 20:47:02 +10:00
|
|
|
|
block = el;
|
2016-07-25 12:24:51 -04:00
|
|
|
|
while ( isInline( block ) ) {
|
|
|
|
|
block = block.parentNode;
|
|
|
|
|
}
|
|
|
|
|
removeZWS( block, el );
|
2015-06-18 15:54:37 +07:00
|
|
|
|
}
|
|
|
|
|
// Otherwise we find all the textnodes in the range (splitting
|
|
|
|
|
// partially selected nodes) and if they're not already formatted
|
|
|
|
|
// correctly we wrap them in the appropriate tag.
|
|
|
|
|
else {
|
|
|
|
|
// Create an iterator to walk over all the text nodes under this
|
|
|
|
|
// ancestor which are in the range and not already formatted
|
|
|
|
|
// correctly.
|
|
|
|
|
//
|
|
|
|
|
// In Blink/WebKit, empty blocks may have no text nodes, just a <br>.
|
|
|
|
|
// Therefore we wrap this in the tag as well, as this will then cause it
|
|
|
|
|
// to apply when the user types something in the block, which is
|
|
|
|
|
// presumably what was intended.
|
2015-08-26 15:00:50 +02:00
|
|
|
|
//
|
2016-08-05 20:47:02 +10:00
|
|
|
|
// IMG tags are included because we may want to create a link around
|
|
|
|
|
// them, and adding other styles is harmless.
|
2015-06-18 15:54:37 +07:00
|
|
|
|
walker = new TreeWalker(
|
|
|
|
|
range.commonAncestorContainer,
|
|
|
|
|
SHOW_TEXT|SHOW_ELEMENT,
|
|
|
|
|
function ( node ) {
|
|
|
|
|
return ( node.nodeType === TEXT_NODE ||
|
2015-09-28 15:48:23 +02:00
|
|
|
|
node.nodeName === 'BR' ||
|
|
|
|
|
node.nodeName === 'IMG'
|
|
|
|
|
) && isNodeContainedInRange( range, node, true );
|
2018-07-27 09:21:50 +10:00
|
|
|
|
}
|
2015-06-18 15:54:37 +07:00
|
|
|
|
);
|
2013-04-08 13:27:06 +10:00
|
|
|
|
|
2015-06-18 15:54:37 +07:00
|
|
|
|
// Start at the beginning node of the range and iterate through
|
|
|
|
|
// all the nodes in the range that need formatting.
|
|
|
|
|
startContainer = range.startContainer;
|
|
|
|
|
startOffset = range.startOffset;
|
|
|
|
|
endContainer = range.endContainer;
|
|
|
|
|
endOffset = range.endOffset;
|
2013-04-08 13:27:06 +10:00
|
|
|
|
|
2015-06-18 15:54:37 +07:00
|
|
|
|
// Make sure we start with a valid node.
|
|
|
|
|
walker.currentNode = startContainer;
|
|
|
|
|
if ( !walker.filter( startContainer ) ) {
|
|
|
|
|
startContainer = walker.nextNode();
|
|
|
|
|
startOffset = 0;
|
2013-04-08 13:27:06 +10:00
|
|
|
|
}
|
2015-06-18 15:54:37 +07:00
|
|
|
|
|
|
|
|
|
// If there are no interesting nodes in the selection, abort
|
|
|
|
|
if ( !startContainer ) {
|
|
|
|
|
return range;
|
2015-04-29 11:16:11 +07:00
|
|
|
|
}
|
2015-06-18 15:54:37 +07:00
|
|
|
|
|
|
|
|
|
do {
|
|
|
|
|
node = walker.currentNode;
|
2016-03-22 17:57:00 +11:00
|
|
|
|
needsFormat = !getNearest( node, root, tag, attributes );
|
2015-06-18 15:54:37 +07:00
|
|
|
|
if ( needsFormat ) {
|
|
|
|
|
// <br> can never be a container node, so must have a text node
|
|
|
|
|
// if node == (end|start)Container
|
|
|
|
|
if ( node === endContainer && node.length > endOffset ) {
|
|
|
|
|
node.splitText( endOffset );
|
|
|
|
|
}
|
|
|
|
|
if ( node === startContainer && startOffset ) {
|
|
|
|
|
node = node.splitText( startOffset );
|
|
|
|
|
if ( endContainer === startContainer ) {
|
|
|
|
|
endContainer = node;
|
|
|
|
|
endOffset -= startOffset;
|
|
|
|
|
}
|
|
|
|
|
startContainer = node;
|
|
|
|
|
startOffset = 0;
|
|
|
|
|
}
|
|
|
|
|
el = this.createElement( tag, attributes );
|
|
|
|
|
replaceWith( node, el );
|
|
|
|
|
el.appendChild( node );
|
2015-04-29 11:16:11 +07:00
|
|
|
|
}
|
2015-06-18 15:54:37 +07:00
|
|
|
|
} while ( walker.nextNode() );
|
|
|
|
|
|
|
|
|
|
// If we don't finish inside a text node, offset may have changed.
|
|
|
|
|
if ( endContainer.nodeType !== TEXT_NODE ) {
|
|
|
|
|
if ( node.nodeType === TEXT_NODE ) {
|
|
|
|
|
endContainer = node;
|
|
|
|
|
endOffset = node.length;
|
|
|
|
|
} else {
|
|
|
|
|
// If <br>, we must have just wrapped it, so it must have only
|
|
|
|
|
// one child
|
|
|
|
|
endContainer = node.parentNode;
|
|
|
|
|
endOffset = 1;
|
2015-04-29 11:16:11 +07:00
|
|
|
|
}
|
|
|
|
|
}
|
2015-06-18 15:54:37 +07:00
|
|
|
|
|
|
|
|
|
// Now set the selection to as it was before
|
2018-05-09 17:43:11 +10:00
|
|
|
|
range = this.createRange(
|
2015-06-18 15:54:37 +07:00
|
|
|
|
startContainer, startOffset, endContainer, endOffset );
|
2015-04-29 11:16:11 +07:00
|
|
|
|
}
|
2015-06-18 15:54:37 +07:00
|
|
|
|
return range;
|
2013-06-20 23:15:18 +10:00
|
|
|
|
};
|
2013-04-08 13:27:06 +10:00
|
|
|
|
|
2015-06-18 15:54:37 +07:00
|
|
|
|
proto._removeFormat = function ( tag, attributes, range, partial ) {
|
|
|
|
|
// Add bookmark
|
|
|
|
|
this._saveRangeToBookmark( range );
|
|
|
|
|
|
|
|
|
|
// We need a node in the selection to break the surrounding
|
|
|
|
|
// formatted text.
|
|
|
|
|
var doc = this._doc,
|
|
|
|
|
fixer;
|
|
|
|
|
if ( range.collapsed ) {
|
|
|
|
|
if ( cantFocusEmptyTextNodes ) {
|
|
|
|
|
fixer = doc.createTextNode( ZWS );
|
|
|
|
|
this._didAddZWS();
|
|
|
|
|
} else {
|
|
|
|
|
fixer = doc.createTextNode( '' );
|
2015-04-29 11:16:11 +07:00
|
|
|
|
}
|
2015-06-18 15:54:37 +07:00
|
|
|
|
insertNodeInRange( range, fixer );
|
2015-04-29 11:16:11 +07:00
|
|
|
|
}
|
2013-04-08 13:27:06 +10:00
|
|
|
|
|
2015-06-18 15:54:37 +07:00
|
|
|
|
// Find block-level ancestor of selection
|
|
|
|
|
var root = range.commonAncestorContainer;
|
|
|
|
|
while ( isInline( root ) ) {
|
|
|
|
|
root = root.parentNode;
|
|
|
|
|
}
|
2013-04-08 13:27:06 +10:00
|
|
|
|
|
2015-06-18 15:54:37 +07:00
|
|
|
|
// Find text nodes inside formatTags that are not in selection and
|
|
|
|
|
// add an extra tag with the same formatting.
|
|
|
|
|
var startContainer = range.startContainer,
|
|
|
|
|
startOffset = range.startOffset,
|
|
|
|
|
endContainer = range.endContainer,
|
|
|
|
|
endOffset = range.endOffset,
|
|
|
|
|
toWrap = [],
|
|
|
|
|
examineNode = function ( node, exemplar ) {
|
|
|
|
|
// If the node is completely contained by the range then
|
|
|
|
|
// we're going to remove all formatting so ignore it.
|
|
|
|
|
if ( isNodeContainedInRange( range, node, false ) ) {
|
|
|
|
|
return;
|
2015-04-29 11:16:11 +07:00
|
|
|
|
}
|
2015-06-18 15:54:37 +07:00
|
|
|
|
|
|
|
|
|
var isText = ( node.nodeType === TEXT_NODE ),
|
|
|
|
|
child, next;
|
|
|
|
|
|
|
|
|
|
// If not at least partially contained, wrap entire contents
|
|
|
|
|
// in a clone of the tag we're removing and we're done.
|
|
|
|
|
if ( !isNodeContainedInRange( range, node, true ) ) {
|
|
|
|
|
// Ignore bookmarks and empty text nodes
|
|
|
|
|
if ( node.nodeName !== 'INPUT' &&
|
|
|
|
|
( !isText || node.data ) ) {
|
|
|
|
|
toWrap.push([ exemplar, node ]);
|
|
|
|
|
}
|
|
|
|
|
return;
|
2015-04-29 11:16:11 +07:00
|
|
|
|
}
|
2015-06-18 15:54:37 +07:00
|
|
|
|
|
|
|
|
|
// Split any partially selected text nodes.
|
|
|
|
|
if ( isText ) {
|
|
|
|
|
if ( node === endContainer && endOffset !== node.length ) {
|
|
|
|
|
toWrap.push([ exemplar, node.splitText( endOffset ) ]);
|
2015-04-29 11:16:11 +07:00
|
|
|
|
}
|
2015-06-18 15:54:37 +07:00
|
|
|
|
if ( node === startContainer && startOffset ) {
|
|
|
|
|
node.splitText( startOffset );
|
|
|
|
|
toWrap.push([ exemplar, node ]);
|
2015-04-29 11:16:11 +07:00
|
|
|
|
}
|
2013-04-08 13:27:06 +10:00
|
|
|
|
}
|
2015-06-18 15:54:37 +07:00
|
|
|
|
// If not a text node, recurse onto all children.
|
|
|
|
|
// Beware, the tree may be rewritten with each call
|
|
|
|
|
// to examineNode, hence find the next sibling first.
|
|
|
|
|
else {
|
|
|
|
|
for ( child = node.firstChild; child; child = next ) {
|
|
|
|
|
next = child.nextSibling;
|
|
|
|
|
examineNode( child, exemplar );
|
|
|
|
|
}
|
|
|
|
|
}
|
|
|
|
|
},
|
|
|
|
|
formatTags = Array.prototype.filter.call(
|
|
|
|
|
root.getElementsByTagName( tag ), function ( el ) {
|
|
|
|
|
return isNodeContainedInRange( range, el, true ) &&
|
|
|
|
|
hasTagAttributes( el, tag, attributes );
|
|
|
|
|
}
|
|
|
|
|
);
|
|
|
|
|
|
|
|
|
|
if ( !partial ) {
|
|
|
|
|
formatTags.forEach( function ( node ) {
|
|
|
|
|
examineNode( node, node );
|
|
|
|
|
});
|
2013-04-08 13:27:06 +10:00
|
|
|
|
}
|
|
|
|
|
|
2015-06-18 15:54:37 +07:00
|
|
|
|
// Now wrap unselected nodes in the tag
|
|
|
|
|
toWrap.forEach( function ( item ) {
|
|
|
|
|
// [ exemplar, node ] tuple
|
|
|
|
|
var el = item[0].cloneNode( false ),
|
|
|
|
|
node = item[1];
|
|
|
|
|
replaceWith( node, el );
|
|
|
|
|
el.appendChild( node );
|
|
|
|
|
});
|
|
|
|
|
// and remove old formatting tags.
|
|
|
|
|
formatTags.forEach( function ( el ) {
|
|
|
|
|
replaceWith( el, empty( el ) );
|
|
|
|
|
});
|
|
|
|
|
|
|
|
|
|
// Merge adjacent inlines:
|
|
|
|
|
this._getRangeAndRemoveBookmark( range );
|
|
|
|
|
if ( fixer ) {
|
|
|
|
|
range.collapse( false );
|
|
|
|
|
}
|
2016-07-14 12:15:06 +10:00
|
|
|
|
mergeInlines( root, range );
|
2015-06-18 15:54:37 +07:00
|
|
|
|
|
|
|
|
|
return range;
|
2015-04-29 11:16:11 +07:00
|
|
|
|
};
|
2015-06-18 15:54:37 +07:00
|
|
|
|
|
|
|
|
|
proto.changeFormat = function ( add, remove, range, partial ) {
|
|
|
|
|
// Normalise the arguments and get selection
|
|
|
|
|
if ( !range && !( range = this.getSelection() ) ) {
|
2016-06-07 16:27:04 +10:00
|
|
|
|
return this;
|
2015-04-29 11:16:11 +07:00
|
|
|
|
}
|
2013-04-08 13:27:06 +10:00
|
|
|
|
|
2015-06-18 15:54:37 +07:00
|
|
|
|
// Save undo checkpoint
|
2016-03-11 15:56:34 +11:00
|
|
|
|
this.saveUndoState( range );
|
2013-04-08 13:27:06 +10:00
|
|
|
|
|
2015-06-18 15:54:37 +07:00
|
|
|
|
if ( remove ) {
|
|
|
|
|
range = this._removeFormat( remove.tag.toUpperCase(),
|
|
|
|
|
remove.attributes || {}, range, partial );
|
2015-04-29 11:16:11 +07:00
|
|
|
|
}
|
2015-06-18 15:54:37 +07:00
|
|
|
|
if ( add ) {
|
|
|
|
|
range = this._addFormat( add.tag.toUpperCase(),
|
|
|
|
|
add.attributes || {}, range );
|
2015-04-29 11:16:11 +07:00
|
|
|
|
}
|
2013-04-08 13:27:06 +10:00
|
|
|
|
|
2015-06-18 15:54:37 +07:00
|
|
|
|
this.setSelection( range );
|
|
|
|
|
this._updatePath( range, true );
|
|
|
|
|
|
|
|
|
|
// We're not still in an undo state
|
|
|
|
|
if ( !canObserveMutations ) {
|
|
|
|
|
this._docWasChanged();
|
2015-04-29 11:16:11 +07:00
|
|
|
|
}
|
2015-06-18 15:54:37 +07:00
|
|
|
|
|
|
|
|
|
return this;
|
2015-04-29 11:16:11 +07:00
|
|
|
|
};
|
2013-04-08 13:27:06 +10:00
|
|
|
|
|
2015-06-18 15:54:37 +07:00
|
|
|
|
// --- Block formatting ---
|
2013-04-08 13:27:06 +10:00
|
|
|
|
|
2015-06-18 15:54:37 +07:00
|
|
|
|
var tagAfterSplit = {
|
|
|
|
|
DT: 'DD',
|
|
|
|
|
DD: 'DT',
|
2017-09-05 11:42:54 +10:00
|
|
|
|
LI: 'LI',
|
|
|
|
|
PRE: 'PRE'
|
2015-04-29 11:16:11 +07:00
|
|
|
|
};
|
2013-04-08 13:27:06 +10:00
|
|
|
|
|
2015-06-18 15:54:37 +07:00
|
|
|
|
var splitBlock = function ( self, block, node, offset ) {
|
|
|
|
|
var splitTag = tagAfterSplit[ block.nodeName ],
|
|
|
|
|
splitProperties = null,
|
2016-03-22 17:57:00 +11:00
|
|
|
|
nodeAfterSplit = split( node, offset, block.parentNode, self._root ),
|
2015-06-18 15:54:37 +07:00
|
|
|
|
config = self._config;
|
2015-04-29 11:16:11 +07:00
|
|
|
|
|
2015-06-18 15:54:37 +07:00
|
|
|
|
if ( !splitTag ) {
|
|
|
|
|
splitTag = config.blockTag;
|
|
|
|
|
splitProperties = config.blockAttributes;
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
// Make sure the new node is the correct type.
|
|
|
|
|
if ( !hasTagAttributes( nodeAfterSplit, splitTag, splitProperties ) ) {
|
|
|
|
|
block = createElement( nodeAfterSplit.ownerDocument,
|
|
|
|
|
splitTag, splitProperties );
|
|
|
|
|
if ( nodeAfterSplit.dir ) {
|
|
|
|
|
block.dir = nodeAfterSplit.dir;
|
2013-04-08 13:27:06 +10:00
|
|
|
|
}
|
2015-06-18 15:54:37 +07:00
|
|
|
|
replaceWith( nodeAfterSplit, block );
|
|
|
|
|
block.appendChild( empty( nodeAfterSplit ) );
|
|
|
|
|
nodeAfterSplit = block;
|
2015-04-29 11:16:11 +07:00
|
|
|
|
}
|
2015-06-18 15:54:37 +07:00
|
|
|
|
return nodeAfterSplit;
|
|
|
|
|
};
|
2015-04-29 11:16:11 +07:00
|
|
|
|
|
2015-06-18 15:54:37 +07:00
|
|
|
|
proto.forEachBlock = function ( fn, mutates, range ) {
|
|
|
|
|
if ( !range && !( range = this.getSelection() ) ) {
|
|
|
|
|
return this;
|
|
|
|
|
}
|
2015-04-29 11:16:11 +07:00
|
|
|
|
|
2015-06-18 15:54:37 +07:00
|
|
|
|
// Save undo checkpoint
|
|
|
|
|
if ( mutates ) {
|
2016-03-11 15:56:34 +11:00
|
|
|
|
this.saveUndoState( range );
|
2015-06-18 15:54:37 +07:00
|
|
|
|
}
|
2015-04-29 11:16:11 +07:00
|
|
|
|
|
2016-03-22 17:57:00 +11:00
|
|
|
|
var root = this._root;
|
|
|
|
|
var start = getStartBlockOfRange( range, root );
|
|
|
|
|
var end = getEndBlockOfRange( range, root );
|
2015-06-18 15:54:37 +07:00
|
|
|
|
if ( start && end ) {
|
|
|
|
|
do {
|
|
|
|
|
if ( fn( start ) || start === end ) { break; }
|
2016-03-22 17:57:00 +11:00
|
|
|
|
} while ( start = getNextBlock( start, root ) );
|
2015-06-18 15:54:37 +07:00
|
|
|
|
}
|
2015-04-29 11:16:11 +07:00
|
|
|
|
|
2015-06-18 15:54:37 +07:00
|
|
|
|
if ( mutates ) {
|
|
|
|
|
this.setSelection( range );
|
2015-04-29 11:16:11 +07:00
|
|
|
|
|
2015-06-18 15:54:37 +07:00
|
|
|
|
// Path may have changed
|
|
|
|
|
this._updatePath( range, true );
|
2015-04-29 11:16:11 +07:00
|
|
|
|
|
2015-06-18 15:54:37 +07:00
|
|
|
|
// We're not still in an undo state
|
|
|
|
|
if ( !canObserveMutations ) {
|
|
|
|
|
this._docWasChanged();
|
|
|
|
|
}
|
|
|
|
|
}
|
|
|
|
|
return this;
|
|
|
|
|
};
|
2015-06-09 16:32:20 +07:00
|
|
|
|
|
2015-06-18 15:54:37 +07:00
|
|
|
|
proto.modifyBlocks = function ( modify, range ) {
|
|
|
|
|
if ( !range && !( range = this.getSelection() ) ) {
|
|
|
|
|
return this;
|
|
|
|
|
}
|
2015-06-09 16:32:20 +07:00
|
|
|
|
|
2015-06-18 15:54:37 +07:00
|
|
|
|
// 1. Save undo checkpoint and bookmark selection
|
2017-07-16 12:43:17 +02:00
|
|
|
|
this._recordUndoState( range, this._isInUndoState );
|
2015-04-29 11:16:11 +07:00
|
|
|
|
|
2016-03-22 17:57:00 +11:00
|
|
|
|
var root = this._root;
|
|
|
|
|
var frag;
|
|
|
|
|
|
2015-06-18 15:54:37 +07:00
|
|
|
|
// 2. Expand range to block boundaries
|
2016-03-22 17:57:00 +11:00
|
|
|
|
expandRangeToBlockBoundaries( range, root );
|
2015-04-29 11:16:11 +07:00
|
|
|
|
|
2015-06-18 15:54:37 +07:00
|
|
|
|
// 3. Remove range.
|
2017-01-13 15:15:59 +11:00
|
|
|
|
moveRangeBoundariesUpTree( range, root, root, root );
|
2016-03-22 17:57:00 +11:00
|
|
|
|
frag = extractContentsOfRange( range, root, root );
|
2015-04-29 11:16:11 +07:00
|
|
|
|
|
2015-06-18 15:54:37 +07:00
|
|
|
|
// 4. Modify tree of fragment and reinsert.
|
|
|
|
|
insertNodeInRange( range, modify.call( this, frag ) );
|
2015-04-29 11:16:11 +07:00
|
|
|
|
|
2015-06-18 15:54:37 +07:00
|
|
|
|
// 5. Merge containers at edges
|
|
|
|
|
if ( range.endOffset < range.endContainer.childNodes.length ) {
|
2016-03-22 17:57:00 +11:00
|
|
|
|
mergeContainers( range.endContainer.childNodes[ range.endOffset ], root );
|
2015-06-18 15:54:37 +07:00
|
|
|
|
}
|
2016-03-22 17:57:00 +11:00
|
|
|
|
mergeContainers( range.startContainer.childNodes[ range.startOffset ], root );
|
2015-04-29 11:16:11 +07:00
|
|
|
|
|
2015-06-18 15:54:37 +07:00
|
|
|
|
// 6. Restore selection
|
|
|
|
|
this._getRangeAndRemoveBookmark( range );
|
|
|
|
|
this.setSelection( range );
|
|
|
|
|
this._updatePath( range, true );
|
2014-12-24 14:24:25 +07:00
|
|
|
|
|
2015-06-18 15:54:37 +07:00
|
|
|
|
// 7. We're not still in an undo state
|
|
|
|
|
if ( !canObserveMutations ) {
|
|
|
|
|
this._docWasChanged();
|
|
|
|
|
}
|
2014-12-24 14:24:25 +07:00
|
|
|
|
|
2015-06-18 15:54:37 +07:00
|
|
|
|
return this;
|
2013-04-08 13:27:06 +10:00
|
|
|
|
};
|
2014-10-01 09:29:54 +07:00
|
|
|
|
|
2015-06-18 15:54:37 +07:00
|
|
|
|
var increaseBlockQuoteLevel = function ( frag ) {
|
|
|
|
|
return this.createElement( 'BLOCKQUOTE',
|
|
|
|
|
this._config.tagAttributes.blockquote, [
|
|
|
|
|
frag
|
|
|
|
|
]);
|
|
|
|
|
};
|
2013-05-20 16:14:28 +10:00
|
|
|
|
|
2015-06-18 15:54:37 +07:00
|
|
|
|
var decreaseBlockQuoteLevel = function ( frag ) {
|
2016-03-22 17:57:00 +11:00
|
|
|
|
var root = this._root;
|
2015-06-18 15:54:37 +07:00
|
|
|
|
var blockquotes = frag.querySelectorAll( 'blockquote' );
|
|
|
|
|
Array.prototype.filter.call( blockquotes, function ( el ) {
|
2016-03-22 17:57:00 +11:00
|
|
|
|
return !getNearest( el.parentNode, root, 'BLOCKQUOTE' );
|
2015-06-18 15:54:37 +07:00
|
|
|
|
}).forEach( function ( el ) {
|
|
|
|
|
replaceWith( el, empty( el ) );
|
|
|
|
|
});
|
|
|
|
|
return frag;
|
2015-04-29 11:16:11 +07:00
|
|
|
|
};
|
2013-04-08 13:27:06 +10:00
|
|
|
|
|
2015-06-18 15:54:37 +07:00
|
|
|
|
var removeBlockQuote = function (/* frag */) {
|
|
|
|
|
return this.createDefaultBlock([
|
|
|
|
|
this.createElement( 'INPUT', {
|
|
|
|
|
id: startSelectionId,
|
|
|
|
|
type: 'hidden'
|
|
|
|
|
}),
|
|
|
|
|
this.createElement( 'INPUT', {
|
|
|
|
|
id: endSelectionId,
|
|
|
|
|
type: 'hidden'
|
|
|
|
|
})
|
|
|
|
|
]);
|
|
|
|
|
};
|
2013-04-08 13:27:06 +10:00
|
|
|
|
|
2015-06-18 15:54:37 +07:00
|
|
|
|
var makeList = function ( self, frag, type ) {
|
2016-03-22 17:57:00 +11:00
|
|
|
|
var walker = getBlockWalker( frag, self._root ),
|
2015-06-18 15:54:37 +07:00
|
|
|
|
node, tag, prev, newLi,
|
|
|
|
|
tagAttributes = self._config.tagAttributes,
|
|
|
|
|
listAttrs = tagAttributes[ type.toLowerCase() ],
|
|
|
|
|
listItemAttrs = tagAttributes.li;
|
|
|
|
|
|
|
|
|
|
while ( node = walker.nextNode() ) {
|
2016-10-19 14:44:39 +11:00
|
|
|
|
if ( node.parentNode.nodeName === 'LI' ) {
|
|
|
|
|
node = node.parentNode;
|
|
|
|
|
walker.currentNode = node.lastChild;
|
|
|
|
|
}
|
|
|
|
|
if ( node.nodeName !== 'LI' ) {
|
2015-06-18 15:54:37 +07:00
|
|
|
|
newLi = self.createElement( 'LI', listItemAttrs );
|
|
|
|
|
if ( node.dir ) {
|
|
|
|
|
newLi.dir = node.dir;
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
// Have we replaced the previous block with a new <ul>/<ol>?
|
2016-09-27 06:06:42 +04:00
|
|
|
|
if ( ( prev = node.previousSibling ) && prev.nodeName === type ) {
|
2015-06-18 15:54:37 +07:00
|
|
|
|
prev.appendChild( newLi );
|
2016-09-27 06:06:42 +04:00
|
|
|
|
detach( node );
|
2015-06-18 15:54:37 +07:00
|
|
|
|
}
|
|
|
|
|
// Otherwise, replace this block with the <ul>/<ol>
|
|
|
|
|
else {
|
|
|
|
|
replaceWith(
|
|
|
|
|
node,
|
|
|
|
|
self.createElement( type, listAttrs, [
|
|
|
|
|
newLi
|
|
|
|
|
])
|
|
|
|
|
);
|
|
|
|
|
}
|
2016-09-27 06:06:42 +04:00
|
|
|
|
newLi.appendChild( empty( node ) );
|
|
|
|
|
walker.currentNode = newLi;
|
2015-06-18 15:54:37 +07:00
|
|
|
|
} else {
|
2016-10-19 14:44:39 +11:00
|
|
|
|
node = node.parentNode;
|
2015-06-18 15:54:37 +07:00
|
|
|
|
tag = node.nodeName;
|
|
|
|
|
if ( tag !== type && ( /^[OU]L$/.test( tag ) ) ) {
|
|
|
|
|
replaceWith( node,
|
|
|
|
|
self.createElement( type, listAttrs, [ empty( node ) ] )
|
|
|
|
|
);
|
|
|
|
|
}
|
2014-04-07 14:44:44 +10:00
|
|
|
|
}
|
|
|
|
|
}
|
2015-06-18 15:54:37 +07:00
|
|
|
|
};
|
2014-04-07 14:44:44 +10:00
|
|
|
|
|
2015-06-18 15:54:37 +07:00
|
|
|
|
var makeUnorderedList = function ( frag ) {
|
|
|
|
|
makeList( this, frag, 'UL' );
|
|
|
|
|
return frag;
|
|
|
|
|
};
|
2013-04-08 13:27:06 +10:00
|
|
|
|
|
2015-06-18 15:54:37 +07:00
|
|
|
|
var makeOrderedList = function ( frag ) {
|
|
|
|
|
makeList( this, frag, 'OL' );
|
|
|
|
|
return frag;
|
|
|
|
|
};
|
|
|
|
|
|
|
|
|
|
var removeList = function ( frag ) {
|
|
|
|
|
var lists = frag.querySelectorAll( 'UL, OL' ),
|
2016-06-28 21:31:31 -07:00
|
|
|
|
items = frag.querySelectorAll( 'LI' ),
|
2016-09-27 06:06:42 +04:00
|
|
|
|
root = this._root,
|
2016-06-28 21:31:31 -07:00
|
|
|
|
i, l, list, listFrag, item;
|
2015-06-18 15:54:37 +07:00
|
|
|
|
for ( i = 0, l = lists.length; i < l; i += 1 ) {
|
|
|
|
|
list = lists[i];
|
|
|
|
|
listFrag = empty( list );
|
2016-09-27 06:06:42 +04:00
|
|
|
|
fixContainer( listFrag, root );
|
2015-06-18 15:54:37 +07:00
|
|
|
|
replaceWith( list, listFrag );
|
2013-04-08 13:27:06 +10:00
|
|
|
|
}
|
2016-06-28 21:31:31 -07:00
|
|
|
|
|
|
|
|
|
for ( i = 0, l = items.length; i < l; i += 1 ) {
|
|
|
|
|
item = items[i];
|
2016-09-27 06:06:42 +04:00
|
|
|
|
if ( isBlock( item ) ) {
|
|
|
|
|
replaceWith( item,
|
|
|
|
|
this.createDefaultBlock([ empty( item ) ])
|
|
|
|
|
);
|
|
|
|
|
} else {
|
|
|
|
|
fixContainer( item, root );
|
|
|
|
|
replaceWith( item, empty( item ) );
|
|
|
|
|
}
|
2016-06-28 21:31:31 -07:00
|
|
|
|
}
|
2015-06-18 15:54:37 +07:00
|
|
|
|
return frag;
|
|
|
|
|
};
|
2013-04-08 13:27:06 +10:00
|
|
|
|
|
2017-08-15 11:11:48 +10:00
|
|
|
|
var getListSelection = function ( range, root ) {
|
|
|
|
|
// Get start+end li in single common ancestor
|
|
|
|
|
var list = range.commonAncestorContainer;
|
|
|
|
|
var startLi = range.startContainer;
|
|
|
|
|
var endLi = range.endContainer;
|
|
|
|
|
while ( list && list !== root && !/^[OU]L$/.test( list.nodeName ) ) {
|
|
|
|
|
list = list.parentNode;
|
|
|
|
|
}
|
|
|
|
|
if ( !list || list === root ) {
|
|
|
|
|
return null;
|
|
|
|
|
}
|
|
|
|
|
if ( startLi === list ) {
|
|
|
|
|
startLi = startLi.childNodes[ range.startOffset ];
|
|
|
|
|
}
|
|
|
|
|
if ( endLi === list ) {
|
|
|
|
|
endLi = endLi.childNodes[ range.endOffset ];
|
|
|
|
|
}
|
|
|
|
|
while ( startLi && startLi.parentNode !== list ) {
|
|
|
|
|
startLi = startLi.parentNode;
|
|
|
|
|
}
|
|
|
|
|
while ( endLi && endLi.parentNode !== list ) {
|
|
|
|
|
endLi = endLi.parentNode;
|
|
|
|
|
}
|
|
|
|
|
return [ list, startLi, endLi ];
|
|
|
|
|
};
|
2016-06-28 21:31:31 -07:00
|
|
|
|
|
2017-08-15 11:11:48 +10:00
|
|
|
|
proto.increaseListLevel = function ( range ) {
|
|
|
|
|
if ( !range && !( range = this.getSelection() ) ) {
|
|
|
|
|
return this.focus();
|
2014-10-01 09:29:54 +07:00
|
|
|
|
}
|
2017-08-15 11:11:48 +10:00
|
|
|
|
|
2017-08-17 14:12:29 +10:00
|
|
|
|
var root = this._root;
|
2017-08-15 11:11:48 +10:00
|
|
|
|
var listSelection = getListSelection( range, root );
|
|
|
|
|
if ( !listSelection ) {
|
|
|
|
|
return this.focus();
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
var list = listSelection[0];
|
|
|
|
|
var startLi = listSelection[1];
|
|
|
|
|
var endLi = listSelection[2];
|
|
|
|
|
if ( !startLi || startLi === list.firstChild ) {
|
|
|
|
|
return this.focus();
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
// Save undo checkpoint and bookmark selection
|
|
|
|
|
this._recordUndoState( range, this._isInUndoState );
|
|
|
|
|
|
|
|
|
|
// Increase list depth
|
|
|
|
|
var type = list.nodeName;
|
|
|
|
|
var newParent = startLi.previousSibling;
|
|
|
|
|
var listAttrs, next;
|
|
|
|
|
if ( newParent.nodeName !== type ) {
|
|
|
|
|
listAttrs = this._config.tagAttributes[ type.toLowerCase() ];
|
|
|
|
|
newParent = this.createElement( type, listAttrs );
|
|
|
|
|
list.insertBefore( newParent, startLi );
|
|
|
|
|
}
|
|
|
|
|
do {
|
|
|
|
|
next = startLi === endLi ? null : startLi.nextSibling;
|
|
|
|
|
newParent.appendChild( startLi );
|
|
|
|
|
} while ( ( startLi = next ) );
|
|
|
|
|
next = newParent.nextSibling;
|
|
|
|
|
if ( next ) {
|
2017-08-17 14:12:29 +10:00
|
|
|
|
mergeContainers( next, root );
|
2017-08-15 11:11:48 +10:00
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
// Restore selection
|
|
|
|
|
this._getRangeAndRemoveBookmark( range );
|
|
|
|
|
this.setSelection( range );
|
|
|
|
|
this._updatePath( range, true );
|
|
|
|
|
|
|
|
|
|
// We're not still in an undo state
|
|
|
|
|
if ( !canObserveMutations ) {
|
|
|
|
|
this._docWasChanged();
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
return this.focus();
|
2015-06-18 15:54:37 +07:00
|
|
|
|
};
|
2014-11-20 12:45:36 +07:00
|
|
|
|
|
2017-08-15 11:11:48 +10:00
|
|
|
|
proto.decreaseListLevel = function ( range ) {
|
|
|
|
|
if ( !range && !( range = this.getSelection() ) ) {
|
|
|
|
|
return this.focus();
|
|
|
|
|
}
|
|
|
|
|
|
2016-03-22 17:57:00 +11:00
|
|
|
|
var root = this._root;
|
2017-08-15 11:11:48 +10:00
|
|
|
|
var listSelection = getListSelection( range, root );
|
|
|
|
|
if ( !listSelection ) {
|
|
|
|
|
return this.focus();
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
var list = listSelection[0];
|
|
|
|
|
var startLi = listSelection[1];
|
|
|
|
|
var endLi = listSelection[2];
|
2018-10-19 11:54:05 +11:00
|
|
|
|
var newParent, next, insertBefore, makeNotList;
|
2017-08-15 11:11:48 +10:00
|
|
|
|
if ( !startLi ) {
|
|
|
|
|
startLi = list.firstChild;
|
|
|
|
|
}
|
|
|
|
|
if ( !endLi ) {
|
|
|
|
|
endLi = list.lastChild;
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
// Save undo checkpoint and bookmark selection
|
|
|
|
|
this._recordUndoState( range, this._isInUndoState );
|
|
|
|
|
|
2018-10-19 11:54:05 +11:00
|
|
|
|
if ( startLi ) {
|
|
|
|
|
// Find the new parent list node
|
|
|
|
|
newParent = list.parentNode;
|
|
|
|
|
|
|
|
|
|
// Split list if necesary
|
|
|
|
|
insertBefore = !endLi.nextSibling ?
|
|
|
|
|
list.nextSibling :
|
|
|
|
|
split( list, endLi.nextSibling, newParent, root );
|
|
|
|
|
|
|
|
|
|
if ( newParent !== root && newParent.nodeName === 'LI' ) {
|
|
|
|
|
newParent = newParent.parentNode;
|
|
|
|
|
while ( insertBefore ) {
|
|
|
|
|
next = insertBefore.nextSibling;
|
|
|
|
|
endLi.appendChild( insertBefore );
|
|
|
|
|
insertBefore = next;
|
|
|
|
|
}
|
|
|
|
|
insertBefore = list.parentNode.nextSibling;
|
2015-06-18 15:54:37 +07:00
|
|
|
|
}
|
2017-08-15 11:11:48 +10:00
|
|
|
|
|
2018-10-19 11:54:05 +11:00
|
|
|
|
makeNotList = !/^[OU]L$/.test( newParent.nodeName );
|
|
|
|
|
do {
|
|
|
|
|
next = startLi === endLi ? null : startLi.nextSibling;
|
|
|
|
|
list.removeChild( startLi );
|
|
|
|
|
if ( makeNotList && startLi.nodeName === 'LI' ) {
|
|
|
|
|
startLi = this.createDefaultBlock([ empty( startLi ) ]);
|
|
|
|
|
}
|
|
|
|
|
newParent.insertBefore( startLi, insertBefore );
|
|
|
|
|
} while (( startLi = next ));
|
|
|
|
|
}
|
2017-08-15 11:11:48 +10:00
|
|
|
|
|
|
|
|
|
if ( !list.firstChild ) {
|
|
|
|
|
detach( list );
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
if ( insertBefore ) {
|
|
|
|
|
mergeContainers( insertBefore, root );
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
// Restore selection
|
|
|
|
|
this._getRangeAndRemoveBookmark( range );
|
|
|
|
|
this.setSelection( range );
|
|
|
|
|
this._updatePath( range, true );
|
|
|
|
|
|
|
|
|
|
// We're not still in an undo state
|
|
|
|
|
if ( !canObserveMutations ) {
|
|
|
|
|
this._docWasChanged();
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
return this.focus();
|
2015-06-18 15:54:37 +07:00
|
|
|
|
};
|
2013-04-08 13:27:06 +10:00
|
|
|
|
|
2015-06-18 15:54:37 +07:00
|
|
|
|
proto._ensureBottomLine = function () {
|
2016-03-22 17:57:00 +11:00
|
|
|
|
var root = this._root;
|
|
|
|
|
var last = root.lastElementChild;
|
2015-06-18 15:54:37 +07:00
|
|
|
|
if ( !last ||
|
|
|
|
|
last.nodeName !== this._config.blockTag || !isBlock( last ) ) {
|
2016-03-22 17:57:00 +11:00
|
|
|
|
root.appendChild( this.createDefaultBlock() );
|
2013-04-08 13:27:06 +10:00
|
|
|
|
}
|
2013-06-20 23:15:18 +10:00
|
|
|
|
};
|
2013-04-08 13:27:06 +10:00
|
|
|
|
|
2015-06-18 15:54:37 +07:00
|
|
|
|
// --- Keyboard interaction ---
|
|
|
|
|
|
2015-04-29 11:16:11 +07:00
|
|
|
|
proto.setKeyHandler = function ( key, fn ) {
|
|
|
|
|
this._keyHandlers[ key ] = fn;
|
|
|
|
|
return this;
|
|
|
|
|
};
|
|
|
|
|
|
2013-06-20 23:15:18 +10:00
|
|
|
|
// --- Get/Set data ---
|
2013-04-08 13:27:06 +10:00
|
|
|
|
|
2013-06-20 23:15:18 +10:00
|
|
|
|
proto._getHTML = function () {
|
2016-03-22 17:57:00 +11:00
|
|
|
|
return this._root.innerHTML;
|
2013-04-08 13:27:06 +10:00
|
|
|
|
};
|
|
|
|
|
|
2013-06-20 23:15:18 +10:00
|
|
|
|
proto._setHTML = function ( html ) {
|
2016-03-22 17:57:00 +11:00
|
|
|
|
var root = this._root;
|
|
|
|
|
var node = root;
|
2013-06-20 23:15:18 +10:00
|
|
|
|
node.innerHTML = html;
|
|
|
|
|
do {
|
2016-03-22 17:57:00 +11:00
|
|
|
|
fixCursor( node, root );
|
|
|
|
|
} while ( node = getNextBlock( node, root ) );
|
2014-12-30 10:22:39 +07:00
|
|
|
|
this._ignoreChange = true;
|
2013-04-08 13:27:06 +10:00
|
|
|
|
};
|
|
|
|
|
|
2013-06-20 23:15:18 +10:00
|
|
|
|
proto.getHTML = function ( withBookMark ) {
|
2020-07-08 16:12:32 +10:00
|
|
|
|
var html, range;
|
2013-06-20 23:15:18 +10:00
|
|
|
|
if ( withBookMark && ( range = this.getSelection() ) ) {
|
|
|
|
|
this._saveRangeToBookmark( range );
|
|
|
|
|
}
|
2014-04-16 18:06:10 +10:00
|
|
|
|
html = this._getHTML().replace( /\u200B/g, '' );
|
2013-06-20 23:15:18 +10:00
|
|
|
|
if ( range ) {
|
|
|
|
|
this._getRangeAndRemoveBookmark( range );
|
|
|
|
|
}
|
|
|
|
|
return html;
|
|
|
|
|
};
|
2013-04-08 13:27:06 +10:00
|
|
|
|
|
2013-06-20 23:15:18 +10:00
|
|
|
|
proto.setHTML = function ( html ) {
|
2016-12-07 17:24:16 +11:00
|
|
|
|
var config = this._config;
|
|
|
|
|
var sanitizeToDOMFragment = config.isSetHTMLSanitized ?
|
|
|
|
|
config.sanitizeToDOMFragment : null;
|
2016-03-22 17:57:00 +11:00
|
|
|
|
var root = this._root;
|
2016-12-07 17:24:16 +11:00
|
|
|
|
var div, frag, child;
|
2013-04-08 13:27:06 +10:00
|
|
|
|
|
2013-06-20 23:15:18 +10:00
|
|
|
|
// Parse HTML into DOM tree
|
2016-12-07 17:24:16 +11:00
|
|
|
|
if ( typeof sanitizeToDOMFragment === 'function' ) {
|
2016-12-13 12:15:35 +11:00
|
|
|
|
frag = sanitizeToDOMFragment( html, false, this );
|
2016-12-07 17:24:16 +11:00
|
|
|
|
} else {
|
|
|
|
|
div = this.createElement( 'DIV' );
|
|
|
|
|
div.innerHTML = html;
|
|
|
|
|
frag = this._doc.createDocumentFragment();
|
|
|
|
|
frag.appendChild( empty( div ) );
|
|
|
|
|
}
|
2013-04-08 13:27:06 +10:00
|
|
|
|
|
2018-07-12 13:35:20 +10:00
|
|
|
|
cleanTree( frag, config );
|
2017-01-09 10:35:19 +11:00
|
|
|
|
cleanupBRs( frag, root, false );
|
2013-04-08 13:27:06 +10:00
|
|
|
|
|
2016-03-22 17:57:00 +11:00
|
|
|
|
fixContainer( frag, root );
|
2013-04-08 13:27:06 +10:00
|
|
|
|
|
2013-06-20 23:15:18 +10:00
|
|
|
|
// Fix cursor
|
|
|
|
|
var node = frag;
|
2016-03-22 17:57:00 +11:00
|
|
|
|
while ( node = getNextBlock( node, root ) ) {
|
|
|
|
|
fixCursor( node, root );
|
2013-06-20 23:15:18 +10:00
|
|
|
|
}
|
2013-04-08 13:27:06 +10:00
|
|
|
|
|
2014-12-30 10:22:39 +07:00
|
|
|
|
// Don't fire an input event
|
|
|
|
|
this._ignoreChange = true;
|
|
|
|
|
|
2016-03-22 17:57:00 +11:00
|
|
|
|
// Remove existing root children
|
|
|
|
|
while ( child = root.lastChild ) {
|
|
|
|
|
root.removeChild( child );
|
2013-06-20 23:15:18 +10:00
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
// And insert new content
|
2016-03-22 17:57:00 +11:00
|
|
|
|
root.appendChild( frag );
|
|
|
|
|
fixCursor( root, root );
|
2013-06-20 23:15:18 +10:00
|
|
|
|
|
|
|
|
|
// Reset the undo stack
|
|
|
|
|
this._undoIndex = -1;
|
|
|
|
|
this._undoStack.length = 0;
|
|
|
|
|
this._undoStackLength = 0;
|
|
|
|
|
this._isInUndoState = false;
|
|
|
|
|
|
|
|
|
|
// Record undo state
|
|
|
|
|
var range = this._getRangeAndRemoveBookmark() ||
|
2018-05-09 17:43:11 +10:00
|
|
|
|
this.createRange( root.firstChild, 0 );
|
2016-03-11 15:56:34 +11:00
|
|
|
|
this.saveUndoState( range );
|
2013-06-20 23:15:18 +10:00
|
|
|
|
// IE will also set focus when selecting text so don't use
|
|
|
|
|
// setSelection. Instead, just store it in lastSelection, so if
|
|
|
|
|
// anything calls getSelection before first focus, we have a range
|
|
|
|
|
// to return.
|
2016-03-22 17:57:00 +11:00
|
|
|
|
this._lastSelection = range;
|
2016-06-16 14:45:19 +10:00
|
|
|
|
enableRestoreSelection.call( this );
|
2013-06-20 23:15:18 +10:00
|
|
|
|
this._updatePath( range, true );
|
|
|
|
|
|
|
|
|
|
return this;
|
|
|
|
|
};
|
2013-04-08 13:27:06 +10:00
|
|
|
|
|
2013-06-20 23:15:18 +10:00
|
|
|
|
proto.insertElement = function ( el, range ) {
|
2017-07-07 11:48:42 +10:00
|
|
|
|
if ( !range ) {
|
|
|
|
|
range = this.getSelection();
|
|
|
|
|
}
|
2013-06-20 23:15:18 +10:00
|
|
|
|
range.collapse( true );
|
|
|
|
|
if ( isInline( el ) ) {
|
|
|
|
|
insertNodeInRange( range, el );
|
|
|
|
|
range.setStartAfter( el );
|
|
|
|
|
} else {
|
|
|
|
|
// Get containing block node.
|
2016-03-22 17:57:00 +11:00
|
|
|
|
var root = this._root;
|
|
|
|
|
var splitNode = getStartBlockOfRange( range, root ) || root;
|
|
|
|
|
var parent, nodeAfterSplit;
|
2013-06-20 23:15:18 +10:00
|
|
|
|
// While at end of container node, move up DOM tree.
|
2016-03-22 17:57:00 +11:00
|
|
|
|
while ( splitNode !== root && !splitNode.nextSibling ) {
|
2013-06-20 23:15:18 +10:00
|
|
|
|
splitNode = splitNode.parentNode;
|
|
|
|
|
}
|
2016-03-22 17:57:00 +11:00
|
|
|
|
// If in the middle of a container node, split up to root.
|
|
|
|
|
if ( splitNode !== root ) {
|
2013-06-20 23:15:18 +10:00
|
|
|
|
parent = splitNode.parentNode;
|
2016-03-22 17:57:00 +11:00
|
|
|
|
nodeAfterSplit = split( parent, splitNode.nextSibling, root, root );
|
2013-06-20 23:15:18 +10:00
|
|
|
|
}
|
|
|
|
|
if ( nodeAfterSplit ) {
|
2016-03-22 17:57:00 +11:00
|
|
|
|
root.insertBefore( el, nodeAfterSplit );
|
2013-04-08 13:27:06 +10:00
|
|
|
|
} else {
|
2016-03-22 17:57:00 +11:00
|
|
|
|
root.appendChild( el );
|
2013-06-20 23:15:18 +10:00
|
|
|
|
// Insert blank line below block.
|
2015-06-02 21:15:14 +07:00
|
|
|
|
nodeAfterSplit = this.createDefaultBlock();
|
2016-03-22 17:57:00 +11:00
|
|
|
|
root.appendChild( nodeAfterSplit );
|
2013-04-08 13:27:06 +10:00
|
|
|
|
}
|
2015-06-02 21:15:14 +07:00
|
|
|
|
range.setStart( nodeAfterSplit, 0 );
|
|
|
|
|
range.setEnd( nodeAfterSplit, 0 );
|
|
|
|
|
moveRangeBoundariesDownTree( range );
|
2013-06-20 23:15:18 +10:00
|
|
|
|
}
|
2015-06-02 21:15:14 +07:00
|
|
|
|
this.focus();
|
|
|
|
|
this.setSelection( range );
|
|
|
|
|
this._updatePath( range );
|
2016-05-02 13:47:39 -04:00
|
|
|
|
|
|
|
|
|
if ( !canObserveMutations ) {
|
|
|
|
|
this._docWasChanged();
|
|
|
|
|
}
|
2016-05-05 11:50:36 +10:00
|
|
|
|
|
2013-06-20 23:15:18 +10:00
|
|
|
|
return this;
|
|
|
|
|
};
|
2013-04-08 13:27:06 +10:00
|
|
|
|
|
2015-05-09 17:14:51 +07:00
|
|
|
|
proto.insertImage = function ( src, attributes ) {
|
|
|
|
|
var img = this.createElement( 'IMG', mergeObjects({
|
2013-06-20 23:15:18 +10:00
|
|
|
|
src: src
|
2016-07-16 16:50:59 +10:00
|
|
|
|
}, attributes, true ));
|
2013-06-20 23:15:18 +10:00
|
|
|
|
this.insertElement( img );
|
|
|
|
|
return img;
|
|
|
|
|
};
|
2013-04-08 13:27:06 +10:00
|
|
|
|
|
2020-01-28 11:08:15 +11:00
|
|
|
|
/*
|
|
|
|
|
const linkRegExp = new RegExp(
|
|
|
|
|
// Only look on boundaries
|
|
|
|
|
'\\b(?:' +
|
|
|
|
|
// Capture group 1: URLs
|
|
|
|
|
'(' +
|
|
|
|
|
// Add links to URLS
|
|
|
|
|
// Starts with:
|
|
|
|
|
'(?:' +
|
|
|
|
|
// http(s):// or ftp://
|
|
|
|
|
'(?:ht|f)tps?:\\/\\/' +
|
|
|
|
|
// or
|
|
|
|
|
'|' +
|
|
|
|
|
// www.
|
|
|
|
|
'www\\d{0,3}[.]' +
|
|
|
|
|
// or
|
|
|
|
|
'|' +
|
|
|
|
|
// foo90.com/
|
|
|
|
|
'[a-z0-9][a-z0-9.\\-]*[.][a-z]{2,}\\/' +
|
|
|
|
|
')' +
|
|
|
|
|
// Then we get one or more:
|
|
|
|
|
'(?:' +
|
|
|
|
|
// Run of non-spaces, non ()<>
|
|
|
|
|
'[^\\s()<>]+' +
|
|
|
|
|
// or
|
|
|
|
|
'|' +
|
|
|
|
|
// balanced parentheses (one level deep only)
|
|
|
|
|
'\\([^\\s()<>]+\\)' +
|
|
|
|
|
')+' +
|
|
|
|
|
// And we finish with
|
|
|
|
|
'(?:' +
|
|
|
|
|
// Not a space or punctuation character
|
|
|
|
|
'[^\\s?&`!()\\[\\]{};:\'".,<>«»“”‘’]' +
|
|
|
|
|
// or
|
|
|
|
|
'|' +
|
|
|
|
|
// Balanced parentheses.
|
|
|
|
|
'\\([^\\s()<>]+\\)' +
|
|
|
|
|
')' +
|
|
|
|
|
// Capture group 2: Emails
|
|
|
|
|
')|(' +
|
|
|
|
|
// Add links to emails
|
|
|
|
|
'[\\w\\-.%+]+@(?:[\\w\\-]+\\.)+[a-z]{2,}\\b' +
|
|
|
|
|
// Allow query parameters in the mailto: style
|
|
|
|
|
'(?:' +
|
|
|
|
|
'[?][^&?\\s]+=[^\\s?&`!()\\[\\]{};:\'".,<>«»“”‘’]+' +
|
|
|
|
|
'(?:&[^&?\\s]+=[^\\s?&`!()\\[\\]{};:\'".,<>«»“”‘’]+)*' +
|
|
|
|
|
')?' +
|
|
|
|
|
'))', 'i' );
|
|
|
|
|
*/
|
|
|
|
|
|
|
|
|
|
proto.linkRegExp = /\b(?:((?:(?:ht|f)tps?:\/\/|www\d{0,3}[.]|[a-z0-9][a-z0-9.\-]*[.][a-z]{2,}\/)(?:[^\s()<>]+|\([^\s()<>]+\))+(?:[^\s?&`!()\[\]{};:'".,<>«»“”‘’]|\([^\s()<>]+\)))|([\w\-.%+]+@(?:[\w\-]+\.)+[a-z]{2,}\b(?:[?][^&?\s]+=[^\s?&`!()\[\]{};:'".,<>«»“”‘’]+(?:&[^&?\s]+=[^\s?&`!()\[\]{};:'".,<>«»“”‘’]+)*)?))/i;
|
2015-06-19 09:59:01 +07:00
|
|
|
|
|
2016-05-05 11:50:36 +10:00
|
|
|
|
var addLinks = function ( frag, root, self ) {
|
2018-10-05 17:00:09 +10:00
|
|
|
|
var doc = frag.ownerDocument;
|
|
|
|
|
var walker = new TreeWalker( frag, SHOW_TEXT, function ( node ) {
|
|
|
|
|
return !getNearest( node, root, 'A' );
|
|
|
|
|
});
|
|
|
|
|
var linkRegExp = self.linkRegExp;
|
|
|
|
|
var defaultAttributes = self._config.tagAttributes.a;
|
|
|
|
|
var node, data, parent, match, index, endIndex, child;
|
|
|
|
|
if ( !linkRegExp ) {
|
|
|
|
|
return;
|
|
|
|
|
}
|
|
|
|
|
while (( node = walker.nextNode() )) {
|
2015-06-19 09:59:01 +07:00
|
|
|
|
data = node.data;
|
|
|
|
|
parent = node.parentNode;
|
2018-10-05 17:00:09 +10:00
|
|
|
|
while (( match = linkRegExp.exec( data ) )) {
|
2015-06-19 09:59:01 +07:00
|
|
|
|
index = match.index;
|
|
|
|
|
endIndex = index + match[0].length;
|
|
|
|
|
if ( index ) {
|
|
|
|
|
child = doc.createTextNode( data.slice( 0, index ) );
|
|
|
|
|
parent.insertBefore( child, node );
|
|
|
|
|
}
|
2016-05-05 11:50:36 +10:00
|
|
|
|
child = self.createElement( 'A', mergeObjects({
|
|
|
|
|
href: match[1] ?
|
2018-06-23 14:39:32 +10:00
|
|
|
|
/^(?:ht|f)tps?:/i.test( match[1] ) ?
|
2016-05-05 11:50:36 +10:00
|
|
|
|
match[1] :
|
|
|
|
|
'http://' + match[1] :
|
2018-05-29 14:20:31 +02:00
|
|
|
|
'mailto:' + match[0]
|
2016-07-16 16:50:59 +10:00
|
|
|
|
}, defaultAttributes, false ));
|
2015-06-19 09:59:01 +07:00
|
|
|
|
child.textContent = data.slice( index, endIndex );
|
|
|
|
|
parent.insertBefore( child, node );
|
|
|
|
|
node.data = data = data.slice( endIndex );
|
|
|
|
|
}
|
|
|
|
|
}
|
|
|
|
|
};
|
|
|
|
|
|
2015-04-14 21:20:36 -04:00
|
|
|
|
// Insert HTML at the cursor location. If the selection is not collapsed
|
|
|
|
|
// insertTreeFragmentIntoRange will delete the selection so that it is replaced
|
|
|
|
|
// by the html being inserted.
|
2015-06-19 09:57:32 +07:00
|
|
|
|
proto.insertHTML = function ( html, isPaste ) {
|
2016-12-07 17:24:16 +11:00
|
|
|
|
var config = this._config;
|
|
|
|
|
var sanitizeToDOMFragment = config.isInsertedHTMLSanitized ?
|
|
|
|
|
config.sanitizeToDOMFragment : null;
|
2016-05-26 13:09:19 +10:00
|
|
|
|
var range = this.getSelection();
|
2016-06-06 11:32:15 +10:00
|
|
|
|
var doc = this._doc;
|
2016-05-26 13:09:19 +10:00
|
|
|
|
var startFragmentIndex, endFragmentIndex;
|
2016-06-06 11:32:15 +10:00
|
|
|
|
var div, frag, root, node, event;
|
2016-05-26 13:09:19 +10:00
|
|
|
|
|
|
|
|
|
// Edge doesn't just copy the fragment, but includes the surrounding guff
|
2016-06-06 11:32:15 +10:00
|
|
|
|
// including the full <head> of the page. Need to strip this out. If
|
|
|
|
|
// available use DOMPurify to parse and sanitise.
|
2016-12-07 17:24:16 +11:00
|
|
|
|
if ( typeof sanitizeToDOMFragment === 'function' ) {
|
2016-12-13 12:15:35 +11:00
|
|
|
|
frag = sanitizeToDOMFragment( html, isPaste, this );
|
2016-06-06 11:32:15 +10:00
|
|
|
|
} else {
|
|
|
|
|
if ( isPaste ) {
|
|
|
|
|
startFragmentIndex = html.indexOf( '<!--StartFragment-->' );
|
|
|
|
|
endFragmentIndex = html.lastIndexOf( '<!--EndFragment-->' );
|
|
|
|
|
if ( startFragmentIndex > -1 && endFragmentIndex > -1 ) {
|
|
|
|
|
html = html.slice( startFragmentIndex + 20, endFragmentIndex );
|
|
|
|
|
}
|
2016-05-26 13:09:19 +10:00
|
|
|
|
}
|
2017-05-10 14:00:28 +10:00
|
|
|
|
// Wrap with <tr> if html contains dangling <td> tags
|
|
|
|
|
if ( /<\/td>((?!<\/tr>)[\s\S])*$/i.test( html ) ) {
|
|
|
|
|
html = '<TR>' + html + '</TR>';
|
|
|
|
|
}
|
|
|
|
|
// Wrap with <table> if html contains dangling <tr> tags
|
|
|
|
|
if ( /<\/tr>((?!<\/table>)[\s\S])*$/i.test( html ) ) {
|
|
|
|
|
html = '<TABLE>' + html + '</TABLE>';
|
|
|
|
|
}
|
2016-06-06 11:32:15 +10:00
|
|
|
|
// Parse HTML into DOM tree
|
|
|
|
|
div = this.createElement( 'DIV' );
|
|
|
|
|
div.innerHTML = html;
|
|
|
|
|
frag = doc.createDocumentFragment();
|
|
|
|
|
frag.appendChild( empty( div ) );
|
2016-05-26 13:09:19 +10:00
|
|
|
|
}
|
2015-04-14 21:20:36 -04:00
|
|
|
|
|
|
|
|
|
// Record undo checkpoint
|
2016-03-11 15:56:34 +11:00
|
|
|
|
this.saveUndoState( range );
|
2015-04-14 21:20:36 -04:00
|
|
|
|
|
|
|
|
|
try {
|
2016-05-26 13:09:19 +10:00
|
|
|
|
root = this._root;
|
|
|
|
|
node = frag;
|
|
|
|
|
event = {
|
2015-06-19 09:57:32 +07:00
|
|
|
|
fragment: frag,
|
|
|
|
|
preventDefault: function () {
|
|
|
|
|
this.defaultPrevented = true;
|
|
|
|
|
},
|
|
|
|
|
defaultPrevented: false
|
|
|
|
|
};
|
|
|
|
|
|
2016-05-05 11:50:36 +10:00
|
|
|
|
addLinks( frag, frag, this );
|
2018-07-12 13:35:20 +10:00
|
|
|
|
cleanTree( frag, config );
|
2017-01-09 10:35:19 +11:00
|
|
|
|
cleanupBRs( frag, root, false );
|
2015-04-14 21:20:36 -04:00
|
|
|
|
removeEmptyInlines( frag );
|
2015-06-19 09:57:32 +07:00
|
|
|
|
frag.normalize();
|
2015-04-14 21:20:36 -04:00
|
|
|
|
|
2016-05-05 11:50:36 +10:00
|
|
|
|
while ( node = getNextBlock( node, frag ) ) {
|
2016-12-07 19:08:52 +11:00
|
|
|
|
fixCursor( node, root );
|
2015-04-14 21:20:36 -04:00
|
|
|
|
}
|
|
|
|
|
|
2015-06-19 09:57:32 +07:00
|
|
|
|
if ( isPaste ) {
|
|
|
|
|
this.fireEvent( 'willPaste', event );
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
if ( !event.defaultPrevented ) {
|
2016-03-22 17:57:00 +11:00
|
|
|
|
insertTreeFragmentIntoRange( range, event.fragment, root );
|
2015-06-19 09:57:32 +07:00
|
|
|
|
if ( !canObserveMutations ) {
|
|
|
|
|
this._docWasChanged();
|
|
|
|
|
}
|
|
|
|
|
range.collapse( false );
|
2020-06-03 14:53:05 +10:00
|
|
|
|
|
|
|
|
|
// After inserting the fragment, check whether the cursor is inside
|
|
|
|
|
// an <a> element and if so if there is an equivalent cursor
|
|
|
|
|
// position after the <a> element. If there is, move it there.
|
|
|
|
|
moveRangeBoundaryOutOf( range, 'A', root );
|
|
|
|
|
|
2015-06-19 09:57:32 +07:00
|
|
|
|
this._ensureBottomLine();
|
2015-04-14 21:20:36 -04:00
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
this.setSelection( range );
|
|
|
|
|
this._updatePath( range, true );
|
2016-07-14 10:39:20 +10:00
|
|
|
|
// Safari sometimes loses focus after paste. Weird.
|
|
|
|
|
if ( isPaste ) {
|
|
|
|
|
this.focus();
|
|
|
|
|
}
|
2015-04-14 21:20:36 -04:00
|
|
|
|
} catch ( error ) {
|
|
|
|
|
this.didError( error );
|
|
|
|
|
}
|
|
|
|
|
return this;
|
|
|
|
|
};
|
|
|
|
|
|
2020-03-02 10:23:24 +11:00
|
|
|
|
var escapeHTML = function ( text ) {
|
2016-05-25 12:41:57 +10:00
|
|
|
|
return text.split( '&' ).join( '&' )
|
2020-03-02 10:23:24 +11:00
|
|
|
|
.split( '<' ).join( '<' )
|
|
|
|
|
.split( '>' ).join( '>' )
|
|
|
|
|
.split( '"' ).join( '"' );
|
2016-05-25 12:41:57 +10:00
|
|
|
|
};
|
|
|
|
|
|
2015-06-19 09:57:32 +07:00
|
|
|
|
proto.insertPlainText = function ( plainText, isPaste ) {
|
2018-07-27 10:29:49 +10:00
|
|
|
|
var range = this.getSelection();
|
|
|
|
|
if ( range.collapsed &&
|
|
|
|
|
getNearest( range.startContainer, this._root, 'PRE' ) ) {
|
|
|
|
|
var node = range.startContainer;
|
|
|
|
|
var offset = range.startOffset;
|
|
|
|
|
var text, event;
|
|
|
|
|
if ( !node || node.nodeType !== TEXT_NODE ) {
|
|
|
|
|
text = this._doc.createTextNode( '' );
|
|
|
|
|
node.insertBefore( text, node.childNodes[ offset ] );
|
|
|
|
|
node = text;
|
|
|
|
|
offset = 0;
|
|
|
|
|
}
|
|
|
|
|
event = {
|
|
|
|
|
text: plainText,
|
|
|
|
|
preventDefault: function () {
|
|
|
|
|
this.defaultPrevented = true;
|
|
|
|
|
},
|
|
|
|
|
defaultPrevented: false
|
|
|
|
|
};
|
|
|
|
|
if ( isPaste ) {
|
|
|
|
|
this.fireEvent( 'willPaste', event );
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
if ( !event.defaultPrevented ) {
|
|
|
|
|
plainText = event.text;
|
|
|
|
|
node.insertData( offset, plainText );
|
|
|
|
|
range.setStart( node, offset + plainText.length );
|
|
|
|
|
range.collapse( true );
|
|
|
|
|
}
|
|
|
|
|
this.setSelection( range );
|
|
|
|
|
return this;
|
|
|
|
|
}
|
2016-05-25 12:41:57 +10:00
|
|
|
|
var lines = plainText.split( '\n' );
|
|
|
|
|
var config = this._config;
|
|
|
|
|
var tag = config.blockTag;
|
|
|
|
|
var attributes = config.blockAttributes;
|
|
|
|
|
var closeBlock = '</' + tag + '>';
|
|
|
|
|
var openBlock = '<' + tag;
|
|
|
|
|
var attr, i, l, line;
|
|
|
|
|
|
|
|
|
|
for ( attr in attributes ) {
|
|
|
|
|
openBlock += ' ' + attr + '="' +
|
2020-03-02 10:23:24 +11:00
|
|
|
|
escapeHTML( attributes[ attr ] ) +
|
2016-05-25 12:41:57 +10:00
|
|
|
|
'"';
|
|
|
|
|
}
|
|
|
|
|
openBlock += '>';
|
|
|
|
|
|
2020-03-10 18:46:40 +11:00
|
|
|
|
for ( i = 0, l = lines.length; i < l; i += 1 ) {
|
2015-11-20 13:55:57 +11:00
|
|
|
|
line = lines[i];
|
2020-03-02 10:23:24 +11:00
|
|
|
|
line = escapeHTML( line ).replace( / (?= )/g, ' ' );
|
2020-03-10 18:46:40 +11:00
|
|
|
|
// We don't wrap the first line in the block, so if it gets inserted
|
|
|
|
|
// into a blank line it keeps that line's formatting.
|
2017-07-15 20:33:32 +10:00
|
|
|
|
// Wrap each line in <div></div>
|
2020-03-10 18:46:40 +11:00
|
|
|
|
if ( i ) {
|
|
|
|
|
line = openBlock + ( line || '<BR>' ) + closeBlock;
|
|
|
|
|
}
|
|
|
|
|
lines[i] = line;
|
2015-06-19 09:57:32 +07:00
|
|
|
|
}
|
|
|
|
|
return this.insertHTML( lines.join( '' ), isPaste );
|
|
|
|
|
};
|
|
|
|
|
|
2013-06-20 23:15:18 +10:00
|
|
|
|
// --- Formatting ---
|
2013-04-08 13:27:06 +10:00
|
|
|
|
|
2013-06-20 23:15:18 +10:00
|
|
|
|
var command = function ( method, arg, arg2 ) {
|
|
|
|
|
return function () {
|
|
|
|
|
this[ method ]( arg, arg2 );
|
|
|
|
|
return this.focus();
|
|
|
|
|
};
|
|
|
|
|
};
|
2013-04-08 13:27:06 +10:00
|
|
|
|
|
2013-06-20 23:15:18 +10:00
|
|
|
|
proto.addStyles = function ( styles ) {
|
|
|
|
|
if ( styles ) {
|
|
|
|
|
var head = this._doc.documentElement.firstChild,
|
|
|
|
|
style = this.createElement( 'STYLE', {
|
|
|
|
|
type: 'text/css'
|
|
|
|
|
});
|
2015-06-11 15:53:12 +07:00
|
|
|
|
style.appendChild( this._doc.createTextNode( styles ) );
|
|
|
|
|
head.appendChild( style );
|
2013-06-20 23:15:18 +10:00
|
|
|
|
}
|
|
|
|
|
return this;
|
|
|
|
|
};
|
2013-04-08 13:27:06 +10:00
|
|
|
|
|
2013-06-20 23:15:18 +10:00
|
|
|
|
proto.bold = command( 'changeFormat', { tag: 'B' } );
|
|
|
|
|
proto.italic = command( 'changeFormat', { tag: 'I' } );
|
|
|
|
|
proto.underline = command( 'changeFormat', { tag: 'U' } );
|
2014-02-03 18:19:13 +11:00
|
|
|
|
proto.strikethrough = command( 'changeFormat', { tag: 'S' } );
|
2014-01-13 14:31:19 +11:00
|
|
|
|
proto.subscript = command( 'changeFormat', { tag: 'SUB' }, { tag: 'SUP' } );
|
|
|
|
|
proto.superscript = command( 'changeFormat', { tag: 'SUP' }, { tag: 'SUB' } );
|
2013-04-08 13:27:06 +10:00
|
|
|
|
|
2013-06-20 23:15:18 +10:00
|
|
|
|
proto.removeBold = command( 'changeFormat', null, { tag: 'B' } );
|
|
|
|
|
proto.removeItalic = command( 'changeFormat', null, { tag: 'I' } );
|
|
|
|
|
proto.removeUnderline = command( 'changeFormat', null, { tag: 'U' } );
|
2014-02-03 18:19:13 +11:00
|
|
|
|
proto.removeStrikethrough = command( 'changeFormat', null, { tag: 'S' } );
|
2014-01-13 14:31:19 +11:00
|
|
|
|
proto.removeSubscript = command( 'changeFormat', null, { tag: 'SUB' } );
|
|
|
|
|
proto.removeSuperscript = command( 'changeFormat', null, { tag: 'SUP' } );
|
2013-04-08 13:27:06 +10:00
|
|
|
|
|
2014-12-20 17:07:11 +11:00
|
|
|
|
proto.makeLink = function ( url, attributes ) {
|
2013-06-20 23:15:18 +10:00
|
|
|
|
var range = this.getSelection();
|
|
|
|
|
if ( range.collapsed ) {
|
|
|
|
|
var protocolEnd = url.indexOf( ':' ) + 1;
|
|
|
|
|
if ( protocolEnd ) {
|
|
|
|
|
while ( url[ protocolEnd ] === '/' ) { protocolEnd += 1; }
|
|
|
|
|
}
|
2013-07-17 11:35:34 +10:00
|
|
|
|
insertNodeInRange(
|
|
|
|
|
range,
|
2013-06-20 23:15:18 +10:00
|
|
|
|
this._doc.createTextNode( url.slice( protocolEnd ) )
|
|
|
|
|
);
|
|
|
|
|
}
|
2016-07-16 16:50:59 +10:00
|
|
|
|
attributes = mergeObjects(
|
|
|
|
|
mergeObjects({
|
|
|
|
|
href: url
|
|
|
|
|
}, attributes, true ),
|
|
|
|
|
this._config.tagAttributes.a,
|
|
|
|
|
false
|
|
|
|
|
);
|
2014-12-20 17:07:11 +11:00
|
|
|
|
|
2013-06-20 23:15:18 +10:00
|
|
|
|
this.changeFormat({
|
|
|
|
|
tag: 'A',
|
2014-12-20 17:07:11 +11:00
|
|
|
|
attributes: attributes
|
2013-06-20 23:15:18 +10:00
|
|
|
|
}, {
|
|
|
|
|
tag: 'A'
|
|
|
|
|
}, range );
|
|
|
|
|
return this.focus();
|
|
|
|
|
};
|
|
|
|
|
proto.removeLink = function () {
|
|
|
|
|
this.changeFormat( null, {
|
|
|
|
|
tag: 'A'
|
|
|
|
|
}, this.getSelection(), true );
|
|
|
|
|
return this.focus();
|
2013-04-08 13:27:06 +10:00
|
|
|
|
};
|
|
|
|
|
|
2013-06-20 23:15:18 +10:00
|
|
|
|
proto.setFontFace = function ( name ) {
|
2018-07-12 13:35:20 +10:00
|
|
|
|
var className = this._config.classNames.fontFamily;
|
2016-06-08 15:38:13 +10:00
|
|
|
|
this.changeFormat( name ? {
|
2013-06-20 23:15:18 +10:00
|
|
|
|
tag: 'SPAN',
|
|
|
|
|
attributes: {
|
2018-07-12 13:35:20 +10:00
|
|
|
|
'class': className,
|
2013-06-20 23:15:18 +10:00
|
|
|
|
style: 'font-family: ' + name + ', sans-serif;'
|
|
|
|
|
}
|
2016-06-08 15:38:13 +10:00
|
|
|
|
} : null, {
|
2013-06-20 23:15:18 +10:00
|
|
|
|
tag: 'SPAN',
|
2018-07-12 13:35:20 +10:00
|
|
|
|
attributes: { 'class': className }
|
2013-06-20 23:15:18 +10:00
|
|
|
|
});
|
|
|
|
|
return this.focus();
|
|
|
|
|
};
|
|
|
|
|
proto.setFontSize = function ( size ) {
|
2018-07-12 13:35:20 +10:00
|
|
|
|
var className = this._config.classNames.fontSize;
|
2016-06-08 15:38:13 +10:00
|
|
|
|
this.changeFormat( size ? {
|
2013-06-20 23:15:18 +10:00
|
|
|
|
tag: 'SPAN',
|
|
|
|
|
attributes: {
|
2018-07-12 13:35:20 +10:00
|
|
|
|
'class': className,
|
2013-06-20 23:15:18 +10:00
|
|
|
|
style: 'font-size: ' +
|
|
|
|
|
( typeof size === 'number' ? size + 'px' : size )
|
|
|
|
|
}
|
2016-06-08 15:38:13 +10:00
|
|
|
|
} : null, {
|
2013-06-20 23:15:18 +10:00
|
|
|
|
tag: 'SPAN',
|
2018-07-12 13:35:20 +10:00
|
|
|
|
attributes: { 'class': className }
|
2013-06-20 23:15:18 +10:00
|
|
|
|
});
|
|
|
|
|
return this.focus();
|
|
|
|
|
};
|
2013-04-08 13:27:06 +10:00
|
|
|
|
|
2013-06-20 23:15:18 +10:00
|
|
|
|
proto.setTextColour = function ( colour ) {
|
2018-07-12 13:35:20 +10:00
|
|
|
|
var className = this._config.classNames.colour;
|
2016-06-08 15:38:13 +10:00
|
|
|
|
this.changeFormat( colour ? {
|
2013-06-20 23:15:18 +10:00
|
|
|
|
tag: 'SPAN',
|
|
|
|
|
attributes: {
|
2018-07-12 13:35:20 +10:00
|
|
|
|
'class': className,
|
2015-12-03 17:55:22 +11:00
|
|
|
|
style: 'color:' + colour
|
2013-06-20 23:15:18 +10:00
|
|
|
|
}
|
2016-06-08 15:38:13 +10:00
|
|
|
|
} : null, {
|
2013-06-20 23:15:18 +10:00
|
|
|
|
tag: 'SPAN',
|
2018-07-12 13:35:20 +10:00
|
|
|
|
attributes: { 'class': className }
|
2013-06-20 23:15:18 +10:00
|
|
|
|
});
|
|
|
|
|
return this.focus();
|
|
|
|
|
};
|
2013-04-08 13:27:06 +10:00
|
|
|
|
|
2013-06-20 23:15:18 +10:00
|
|
|
|
proto.setHighlightColour = function ( colour ) {
|
2018-07-12 13:35:20 +10:00
|
|
|
|
var className = this._config.classNames.highlight;
|
2016-06-08 15:38:13 +10:00
|
|
|
|
this.changeFormat( colour ? {
|
2013-06-20 23:15:18 +10:00
|
|
|
|
tag: 'SPAN',
|
|
|
|
|
attributes: {
|
2018-07-12 13:35:20 +10:00
|
|
|
|
'class': className,
|
2015-12-03 17:55:22 +11:00
|
|
|
|
style: 'background-color:' + colour
|
2013-06-20 23:15:18 +10:00
|
|
|
|
}
|
2016-06-08 15:38:13 +10:00
|
|
|
|
} : colour, {
|
2013-06-20 23:15:18 +10:00
|
|
|
|
tag: 'SPAN',
|
2018-07-12 13:35:20 +10:00
|
|
|
|
attributes: { 'class': className }
|
2013-06-20 23:15:18 +10:00
|
|
|
|
});
|
|
|
|
|
return this.focus();
|
|
|
|
|
};
|
|
|
|
|
|
|
|
|
|
proto.setTextAlignment = function ( alignment ) {
|
|
|
|
|
this.forEachBlock( function ( block ) {
|
2016-12-07 19:22:28 +11:00
|
|
|
|
var className = block.className
|
2013-06-20 23:15:18 +10:00
|
|
|
|
.split( /\s+/ )
|
|
|
|
|
.filter( function ( klass ) {
|
2016-12-07 19:22:28 +11:00
|
|
|
|
return !!klass && !/^align/.test( klass );
|
2013-06-20 23:15:18 +10:00
|
|
|
|
})
|
2016-12-07 19:22:28 +11:00
|
|
|
|
.join( ' ' );
|
|
|
|
|
if ( alignment ) {
|
|
|
|
|
block.className = className + ' align-' + alignment;
|
|
|
|
|
block.style.textAlign = alignment;
|
|
|
|
|
} else {
|
|
|
|
|
block.className = className;
|
|
|
|
|
block.style.textAlign = '';
|
|
|
|
|
}
|
2013-06-20 23:15:18 +10:00
|
|
|
|
}, true );
|
|
|
|
|
return this.focus();
|
|
|
|
|
};
|
|
|
|
|
|
|
|
|
|
proto.setTextDirection = function ( direction ) {
|
|
|
|
|
this.forEachBlock( function ( block ) {
|
2016-12-07 19:22:28 +11:00
|
|
|
|
if ( direction ) {
|
|
|
|
|
block.dir = direction;
|
|
|
|
|
} else {
|
|
|
|
|
block.removeAttribute( 'dir' );
|
|
|
|
|
}
|
2013-06-20 23:15:18 +10:00
|
|
|
|
}, true );
|
|
|
|
|
return this.focus();
|
|
|
|
|
};
|
|
|
|
|
|
2018-07-27 10:29:49 +10:00
|
|
|
|
// ---
|
|
|
|
|
|
|
|
|
|
var addPre = function ( frag ) {
|
|
|
|
|
var root = this._root;
|
|
|
|
|
var document = this._doc;
|
|
|
|
|
var output = document.createDocumentFragment();
|
|
|
|
|
var walker = getBlockWalker( frag, root );
|
|
|
|
|
var node;
|
|
|
|
|
// 1. Extract inline content; drop all blocks and contains.
|
|
|
|
|
while (( node = walker.nextNode() )) {
|
|
|
|
|
// 2. Replace <br> with \n in content
|
|
|
|
|
var nodes = node.querySelectorAll( 'BR' );
|
|
|
|
|
var brBreaksLine = [];
|
|
|
|
|
var l = nodes.length;
|
|
|
|
|
var i, br;
|
|
|
|
|
|
|
|
|
|
// Must calculate whether the <br> breaks a line first, because if we
|
|
|
|
|
// have two <br>s next to each other, after the first one is converted
|
|
|
|
|
// to a block split, the second will be at the end of a block and
|
|
|
|
|
// therefore seem to not be a line break. But in its original context it
|
|
|
|
|
// was, so we should also convert it to a block split.
|
|
|
|
|
for ( i = 0; i < l; i += 1 ) {
|
|
|
|
|
brBreaksLine[i] = isLineBreak( nodes[i], false );
|
|
|
|
|
}
|
|
|
|
|
while ( l-- ) {
|
|
|
|
|
br = nodes[l];
|
|
|
|
|
if ( !brBreaksLine[l] ) {
|
|
|
|
|
detach( br );
|
|
|
|
|
} else {
|
|
|
|
|
replaceWith( br, document.createTextNode( '\n' ) );
|
|
|
|
|
}
|
|
|
|
|
}
|
|
|
|
|
// 3. Remove <code>; its format clashes with <pre>
|
|
|
|
|
nodes = node.querySelectorAll( 'CODE' );
|
|
|
|
|
l = nodes.length;
|
|
|
|
|
while ( l-- ) {
|
|
|
|
|
detach( nodes[l] );
|
|
|
|
|
}
|
|
|
|
|
if ( output.childNodes.length ) {
|
|
|
|
|
output.appendChild( document.createTextNode( '\n' ) );
|
|
|
|
|
}
|
|
|
|
|
output.appendChild( empty( node ) );
|
|
|
|
|
}
|
|
|
|
|
// 4. Replace nbsp with regular sp
|
|
|
|
|
walker = new TreeWalker( output, SHOW_TEXT );
|
|
|
|
|
while (( node = walker.nextNode() )) {
|
|
|
|
|
node.data = node.data.replace( / /g, ' ' ); // nbsp -> sp
|
|
|
|
|
}
|
|
|
|
|
output.normalize();
|
|
|
|
|
return fixCursor( this.createElement( 'PRE',
|
|
|
|
|
this._config.tagAttributes.pre, [
|
|
|
|
|
output
|
|
|
|
|
]), root );
|
|
|
|
|
};
|
|
|
|
|
|
|
|
|
|
var removePre = function ( frag ) {
|
|
|
|
|
var document = this._doc;
|
|
|
|
|
var root = this._root;
|
|
|
|
|
var pres = frag.querySelectorAll( 'PRE' );
|
|
|
|
|
var l = pres.length;
|
|
|
|
|
var pre, walker, node, value, contents, index;
|
|
|
|
|
while ( l-- ) {
|
|
|
|
|
pre = pres[l];
|
|
|
|
|
walker = new TreeWalker( pre, SHOW_TEXT );
|
|
|
|
|
while (( node = walker.nextNode() )) {
|
|
|
|
|
value = node.data;
|
|
|
|
|
value = value.replace( / (?= )/g, ' ' ); // sp -> nbsp
|
|
|
|
|
contents = document.createDocumentFragment();
|
|
|
|
|
while (( index = value.indexOf( '\n' ) ) > -1 ) {
|
|
|
|
|
contents.appendChild(
|
|
|
|
|
document.createTextNode( value.slice( 0, index ) )
|
|
|
|
|
);
|
|
|
|
|
contents.appendChild( document.createElement( 'BR' ) );
|
|
|
|
|
value = value.slice( index + 1 );
|
|
|
|
|
}
|
|
|
|
|
node.parentNode.insertBefore( contents, node );
|
|
|
|
|
node.data = value;
|
|
|
|
|
}
|
|
|
|
|
fixContainer( pre, root );
|
|
|
|
|
replaceWith( pre, empty( pre ) );
|
|
|
|
|
}
|
|
|
|
|
return frag;
|
|
|
|
|
};
|
|
|
|
|
|
|
|
|
|
proto.code = function () {
|
|
|
|
|
var range = this.getSelection();
|
|
|
|
|
if ( range.collapsed || isContainer( range.commonAncestorContainer ) ) {
|
|
|
|
|
this.modifyBlocks( addPre, range );
|
|
|
|
|
} else {
|
|
|
|
|
this.changeFormat({
|
|
|
|
|
tag: 'CODE',
|
|
|
|
|
attributes: this._config.tagAttributes.code
|
|
|
|
|
}, null, range );
|
|
|
|
|
}
|
|
|
|
|
return this.focus();
|
|
|
|
|
};
|
|
|
|
|
|
|
|
|
|
proto.removeCode = function () {
|
|
|
|
|
var range = this.getSelection();
|
|
|
|
|
var ancestor = range.commonAncestorContainer;
|
|
|
|
|
var inPre = getNearest( ancestor, this._root, 'PRE' );
|
|
|
|
|
if ( inPre ) {
|
|
|
|
|
this.modifyBlocks( removePre, range );
|
|
|
|
|
} else {
|
|
|
|
|
this.changeFormat( null, { tag: 'CODE' }, range );
|
|
|
|
|
}
|
|
|
|
|
return this.focus();
|
|
|
|
|
};
|
|
|
|
|
|
|
|
|
|
proto.toggleCode = function () {
|
|
|
|
|
if ( this.hasFormat( 'PRE' ) || this.hasFormat( 'CODE' ) ) {
|
|
|
|
|
this.removeCode();
|
|
|
|
|
} else {
|
|
|
|
|
this.code();
|
|
|
|
|
}
|
|
|
|
|
return this;
|
|
|
|
|
};
|
|
|
|
|
|
|
|
|
|
// ---
|
|
|
|
|
|
2015-06-17 15:38:12 +07:00
|
|
|
|
function removeFormatting ( self, root, clean ) {
|
|
|
|
|
var node, next;
|
|
|
|
|
for ( node = root.firstChild; node; node = next ) {
|
|
|
|
|
next = node.nextSibling;
|
|
|
|
|
if ( isInline( node ) ) {
|
2015-08-26 15:42:01 +02:00
|
|
|
|
if ( node.nodeType === TEXT_NODE || node.nodeName === 'BR' || node.nodeName === 'IMG' ) {
|
2015-06-17 15:38:12 +07:00
|
|
|
|
clean.appendChild( node );
|
|
|
|
|
continue;
|
|
|
|
|
}
|
|
|
|
|
} else if ( isBlock( node ) ) {
|
|
|
|
|
clean.appendChild( self.createDefaultBlock([
|
|
|
|
|
removeFormatting(
|
|
|
|
|
self, node, self._doc.createDocumentFragment() )
|
|
|
|
|
]));
|
|
|
|
|
continue;
|
2015-06-17 14:12:23 +07:00
|
|
|
|
}
|
2015-06-17 15:38:12 +07:00
|
|
|
|
removeFormatting( self, node, clean );
|
2015-06-03 17:03:10 +02:00
|
|
|
|
}
|
2015-06-17 15:38:12 +07:00
|
|
|
|
return clean;
|
2015-06-03 17:03:10 +02:00
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
proto.removeAllFormatting = function ( range ) {
|
|
|
|
|
if ( !range && !( range = this.getSelection() ) || range.collapsed ) {
|
2015-06-17 15:38:12 +07:00
|
|
|
|
return this;
|
2015-06-03 17:03:10 +02:00
|
|
|
|
}
|
|
|
|
|
|
2016-03-22 17:57:00 +11:00
|
|
|
|
var root = this._root;
|
2015-06-03 17:03:10 +02:00
|
|
|
|
var stopNode = range.commonAncestorContainer;
|
2015-06-17 15:38:12 +07:00
|
|
|
|
while ( stopNode && !isBlock( stopNode ) ) {
|
2015-06-03 17:03:10 +02:00
|
|
|
|
stopNode = stopNode.parentNode;
|
|
|
|
|
}
|
2015-06-17 15:38:12 +07:00
|
|
|
|
if ( !stopNode ) {
|
2016-03-22 17:57:00 +11:00
|
|
|
|
expandRangeToBlockBoundaries( range, root );
|
|
|
|
|
stopNode = root;
|
2015-06-03 17:03:10 +02:00
|
|
|
|
}
|
2015-06-17 15:38:12 +07:00
|
|
|
|
if ( stopNode.nodeType === TEXT_NODE ) {
|
|
|
|
|
return this;
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
// Record undo point
|
2016-03-11 15:56:34 +11:00
|
|
|
|
this.saveUndoState( range );
|
2015-06-03 17:03:10 +02:00
|
|
|
|
|
2015-06-17 15:38:12 +07:00
|
|
|
|
// Avoid splitting where we're already at edges.
|
2017-01-13 15:15:59 +11:00
|
|
|
|
moveRangeBoundariesUpTree( range, stopNode, stopNode, root );
|
2015-06-03 17:03:10 +02:00
|
|
|
|
|
2015-06-17 15:38:12 +07:00
|
|
|
|
// Split the selection up to the block, or if whole selection in same
|
2016-03-22 17:57:00 +11:00
|
|
|
|
// block, expand range boundaries to ends of block and split up to root.
|
2015-06-03 17:03:10 +02:00
|
|
|
|
var doc = stopNode.ownerDocument;
|
|
|
|
|
var startContainer = range.startContainer;
|
|
|
|
|
var startOffset = range.startOffset;
|
|
|
|
|
var endContainer = range.endContainer;
|
|
|
|
|
var endOffset = range.endOffset;
|
|
|
|
|
|
2015-06-17 15:38:12 +07:00
|
|
|
|
// Split end point first to avoid problems when end and start
|
|
|
|
|
// in same container.
|
|
|
|
|
var formattedNodes = doc.createDocumentFragment();
|
|
|
|
|
var cleanNodes = doc.createDocumentFragment();
|
2016-03-22 17:57:00 +11:00
|
|
|
|
var nodeAfterSplit = split( endContainer, endOffset, stopNode, root );
|
|
|
|
|
var nodeInSplit = split( startContainer, startOffset, stopNode, root );
|
2016-07-14 12:15:06 +10:00
|
|
|
|
var nextNode, childNodes;
|
2015-06-17 15:38:12 +07:00
|
|
|
|
|
|
|
|
|
// Then replace contents in split with a cleaned version of the same:
|
|
|
|
|
// blocks become default blocks, text and leaf nodes survive, everything
|
|
|
|
|
// else is obliterated.
|
|
|
|
|
while ( nodeInSplit !== nodeAfterSplit ) {
|
|
|
|
|
nextNode = nodeInSplit.nextSibling;
|
|
|
|
|
formattedNodes.appendChild( nodeInSplit );
|
|
|
|
|
nodeInSplit = nextNode;
|
|
|
|
|
}
|
|
|
|
|
removeFormatting( this, formattedNodes, cleanNodes );
|
2015-06-17 19:07:29 +07:00
|
|
|
|
cleanNodes.normalize();
|
2015-06-17 15:38:12 +07:00
|
|
|
|
nodeInSplit = cleanNodes.firstChild;
|
|
|
|
|
nextNode = cleanNodes.lastChild;
|
|
|
|
|
|
2015-06-17 19:07:29 +07:00
|
|
|
|
// Restore selection
|
|
|
|
|
childNodes = stopNode.childNodes;
|
2015-06-17 15:38:12 +07:00
|
|
|
|
if ( nodeInSplit ) {
|
|
|
|
|
stopNode.insertBefore( cleanNodes, nodeAfterSplit );
|
2015-06-17 19:07:29 +07:00
|
|
|
|
startOffset = indexOf.call( childNodes, nodeInSplit );
|
|
|
|
|
endOffset = indexOf.call( childNodes, nextNode ) + 1;
|
2015-06-17 15:38:12 +07:00
|
|
|
|
} else {
|
2015-06-17 19:07:29 +07:00
|
|
|
|
startOffset = indexOf.call( childNodes, nodeAfterSplit );
|
|
|
|
|
endOffset = startOffset;
|
2015-06-17 15:38:12 +07:00
|
|
|
|
}
|
2015-06-17 19:07:29 +07:00
|
|
|
|
|
|
|
|
|
// Merge text nodes at edges, if possible
|
2016-07-14 12:15:06 +10:00
|
|
|
|
range.setStart( stopNode, startOffset );
|
|
|
|
|
range.setEnd( stopNode, endOffset );
|
|
|
|
|
mergeInlines( stopNode, range );
|
2015-06-17 19:07:29 +07:00
|
|
|
|
|
|
|
|
|
// And move back down the tree
|
2015-06-17 15:38:12 +07:00
|
|
|
|
moveRangeBoundariesDownTree( range );
|
2015-06-03 17:03:10 +02:00
|
|
|
|
|
2015-06-17 15:38:12 +07:00
|
|
|
|
this.setSelection( range );
|
2015-06-17 19:07:29 +07:00
|
|
|
|
this._updatePath( range, true );
|
2015-06-16 15:27:31 +02:00
|
|
|
|
|
2015-06-17 19:07:29 +07:00
|
|
|
|
return this.focus();
|
2015-06-03 17:03:10 +02:00
|
|
|
|
};
|
|
|
|
|
|
2013-06-20 23:15:18 +10:00
|
|
|
|
proto.increaseQuoteLevel = command( 'modifyBlocks', increaseBlockQuoteLevel );
|
|
|
|
|
proto.decreaseQuoteLevel = command( 'modifyBlocks', decreaseBlockQuoteLevel );
|
|
|
|
|
|
|
|
|
|
proto.makeUnorderedList = command( 'modifyBlocks', makeUnorderedList );
|
|
|
|
|
proto.makeOrderedList = command( 'modifyBlocks', makeOrderedList );
|
2014-04-07 13:05:44 +10:00
|
|
|
|
proto.removeList = command( 'modifyBlocks', removeList );
|
|
|
|
|
|
2016-12-07 18:42:20 +11:00
|
|
|
|
// Node.js exports
|
|
|
|
|
Squire.isInline = isInline;
|
|
|
|
|
Squire.isBlock = isBlock;
|
|
|
|
|
Squire.isContainer = isContainer;
|
|
|
|
|
Squire.getBlockWalker = getBlockWalker;
|
|
|
|
|
Squire.getPreviousBlock = getPreviousBlock;
|
|
|
|
|
Squire.getNextBlock = getNextBlock;
|
|
|
|
|
Squire.areAlike = areAlike;
|
|
|
|
|
Squire.hasTagAttributes = hasTagAttributes;
|
|
|
|
|
Squire.getNearest = getNearest;
|
|
|
|
|
Squire.isOrContains = isOrContains;
|
|
|
|
|
Squire.detach = detach;
|
|
|
|
|
Squire.replaceWith = replaceWith;
|
|
|
|
|
Squire.empty = empty;
|
|
|
|
|
|
2016-08-23 20:52:46 +10:00
|
|
|
|
// Range.js exports
|
|
|
|
|
Squire.getNodeBefore = getNodeBefore;
|
|
|
|
|
Squire.getNodeAfter = getNodeAfter;
|
|
|
|
|
Squire.insertNodeInRange = insertNodeInRange;
|
|
|
|
|
Squire.extractContentsOfRange = extractContentsOfRange;
|
|
|
|
|
Squire.deleteContentsOfRange = deleteContentsOfRange;
|
|
|
|
|
Squire.insertTreeFragmentIntoRange = insertTreeFragmentIntoRange;
|
|
|
|
|
Squire.isNodeContainedInRange = isNodeContainedInRange;
|
|
|
|
|
Squire.moveRangeBoundariesDownTree = moveRangeBoundariesDownTree;
|
|
|
|
|
Squire.moveRangeBoundariesUpTree = moveRangeBoundariesUpTree;
|
|
|
|
|
Squire.getStartBlockOfRange = getStartBlockOfRange;
|
|
|
|
|
Squire.getEndBlockOfRange = getEndBlockOfRange;
|
|
|
|
|
Squire.contentWalker = contentWalker;
|
|
|
|
|
Squire.rangeDoesStartAtBlockBoundary = rangeDoesStartAtBlockBoundary;
|
|
|
|
|
Squire.rangeDoesEndAtBlockBoundary = rangeDoesEndAtBlockBoundary;
|
|
|
|
|
Squire.expandRangeToBlockBoundaries = expandRangeToBlockBoundaries;
|
|
|
|
|
|
|
|
|
|
// Clipboard.js exports
|
2020-07-08 16:12:32 +10:00
|
|
|
|
Squire.onCopy = _onCopy;
|
2016-08-23 20:52:46 +10:00
|
|
|
|
Squire.onPaste = onPaste;
|
|
|
|
|
|
|
|
|
|
// Editor.js exports
|
|
|
|
|
Squire.addLinks = addLinks;
|
|
|
|
|
Squire.splitBlock = splitBlock;
|
|
|
|
|
Squire.startSelectionId = startSelectionId;
|
|
|
|
|
Squire.endSelectionId = endSelectionId;
|
|
|
|
|
|
2015-06-25 10:54:53 +07:00
|
|
|
|
if ( typeof exports === 'object' ) {
|
|
|
|
|
module.exports = Squire;
|
2015-07-14 00:16:16 +08:00
|
|
|
|
} else if ( typeof define === 'function' && define.amd ) {
|
2015-07-14 00:32:34 +08:00
|
|
|
|
define( function () {
|
2015-07-14 00:16:16 +08:00
|
|
|
|
return Squire;
|
2015-07-14 00:32:34 +08:00
|
|
|
|
});
|
2013-06-20 23:15:18 +10:00
|
|
|
|
} else {
|
2015-07-06 12:19:21 +07:00
|
|
|
|
win.Squire = Squire;
|
|
|
|
|
|
|
|
|
|
if ( top !== win &&
|
|
|
|
|
doc.documentElement.getAttribute( 'data-squireinit' ) === 'true' ) {
|
2015-06-25 10:54:53 +07:00
|
|
|
|
win.editor = new Squire( doc );
|
|
|
|
|
if ( win.onEditorLoad ) {
|
|
|
|
|
win.onEditorLoad( win.editor );
|
|
|
|
|
win.onEditorLoad = null;
|
|
|
|
|
}
|
2015-05-07 10:52:32 +08:00
|
|
|
|
}
|
2013-04-08 13:27:06 +10:00
|
|
|
|
}
|
2013-06-20 23:15:18 +10:00
|
|
|
|
|
2013-04-08 13:27:06 +10:00
|
|
|
|
}( document ) );
|