0
Fork 0
mirror of https://github.com/fastmail/Squire.git synced 2025-01-18 12:42:37 -05:00
Squire/source/Node.js

519 lines
15 KiB
JavaScript
Raw Normal View History

/*jshint strict:false, undef:false, unused:false */
2011-10-29 14:15:21 +11: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)$/;
var leafNodeNames = {
BR: 1,
2016-05-01 16:33:47 +10:00
HR: 1,
IFRAME: 1,
IMG: 1,
INPUT: 1
2011-10-29 14:15:21 +11:00
};
function every ( nodeList, fn ) {
2011-10-29 14:15:21 +11:00
var l = nodeList.length;
while ( l-- ) {
if ( !fn( nodeList[l] ) ) {
return false;
}
}
return true;
}
2011-10-29 14:15:21 +11:00
// ---
var UNKNOWN = 0;
var INLINE = 1;
var BLOCK = 2;
var CONTAINER = 3;
var nodeCategoryCache = canWeakMap ? new WeakMap() : null;
function isLeaf ( node ) {
return node.nodeType === ELEMENT_NODE && !!leafNodeNames[ node.nodeName ];
}
function getNodeCategory ( node ) {
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;
}
var nodeCategory;
if ( !every( node.childNodes, isInline ) ) {
// Malformed HTML can have block tags inside inline tags. Need to treat
// these as containers rather than inline. See #239.
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;
}
function isBlock ( node ) {
return getNodeCategory( node ) === BLOCK;
}
function isContainer ( node ) {
return getNodeCategory( node ) === CONTAINER;
}
2011-10-29 14:15:21 +11:00
2016-03-22 17:57:00 +11:00
function getBlockWalker ( node, root ) {
var walker = new TreeWalker( root, SHOW_ELEMENT, isBlock );
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;
}
2011-10-29 14:15:21 +11: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 )
);
}
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;
}
2016-03-22 17:57:00 +11:00
function getNearest ( node, root, tag, attributes ) {
while ( node && node !== root ) {
if ( hasTagAttributes( node, tag, attributes ) ) {
return node;
2011-10-29 14:15:21 +11:00
}
2016-03-22 17:57:00 +11:00
node = node.parentNode;
}
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;
}
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,'' ) + ']';
}
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,'' ) + ']';
}
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,'' ) + ']';
}
2018-07-12 13:35:20 +10:00
if ( indexOf.call( classNames, styleNames.fontSize ) > -1 ) {
path += '[fontSize=' + node.style.fontSize + ']';
}
}
}
}
return path;
}
function getLength ( node ) {
var nodeType = node.nodeType;
return nodeType === ELEMENT_NODE || nodeType === DOCUMENT_FRAGMENT_NODE ?
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;
}
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 ) {
el.setAttribute( attr, value );
}
}
}
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 ) {
// In Webkit and Gecko, block level elements are collapsed and
// unfocusable if they have no content. To remedy this, a <BR> must be
// inserted. In Opera and IE, we just need a textnode in order for the
// cursor to appear.
var self = root.__squire__;
var doc = node.ownerDocument;
var originalNode = node;
var fixer, child;
2016-03-22 17:57:00 +11:00
if ( node === root ) {
if ( !( child = node.firstChild ) || child.nodeName === 'BR' ) {
fixer = self.createDefaultBlock();
if ( child ) {
node.replaceChild( fixer, child );
}
else {
node.appendChild( fixer );
2011-10-29 14:15:21 +11:00
}
node = fixer;
fixer = null;
2011-10-29 14:15:21 +11:00
}
}
if ( node.nodeType === TEXT_NODE ) {
return originalNode;
}
if ( isInline( node ) ) {
child = node.firstChild;
while ( cantFocusEmptyTextNodes && child &&
child.nodeType === TEXT_NODE && !child.data ) {
node.removeChild( child );
child = node.firstChild;
}
if ( !child ) {
if ( cantFocusEmptyTextNodes ) {
fixer = doc.createTextNode( ZWS );
self._didAddZWS();
} else {
fixer = doc.createTextNode( '' );
2011-10-29 14:15:21 +11:00
}
}
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;
2011-10-29 14:15:21 +11:00
}
}
if ( fixer ) {
try {
node.appendChild( fixer );
} catch ( error ) {
self.didError({
name: 'Squire: fixCursor  ' + error,
message: 'Parent: ' + node.nodeName + '/' + node.innerHTML +
' appendChild: ' + fixer.nodeName
});
}
}
2012-01-25 11:47:26 +11:00
2016-03-22 17:57:00 +11:00
return originalNode;
}
2012-01-25 11:47:26 +11:00
// Recursively examine container nodes and wrap any inline children.
2016-03-22 17:57:00 +11:00
function fixContainer ( container, root ) {
var children = container.childNodes;
var doc = container.ownerDocument;
var wrapper = null;
var i, l, child, isBR;
for ( i = 0, l = children.length; i < l; i += 1 ) {
child = children[i];
isBR = child.nodeName === 'BR';
if ( !isBR && isInline( child ) ) {
if ( !wrapper ) {
wrapper = createElement( doc, 'div' );
}
wrapper.appendChild( child );
i -= 1;
l -= 1;
} else if ( isBR || wrapper ) {
if ( !wrapper ) {
wrapper = createElement( doc, 'div' );
}
2016-03-22 17:57:00 +11:00
fixCursor( wrapper, root );
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 );
}
}
if ( wrapper ) {
2016-03-22 17:57:00 +11:00
container.appendChild( fixCursor( wrapper, root ) );
}
return container;
}
2016-03-22 17:57:00 +11:00
function split ( node, offset, stopNode, root ) {
var nodeType = node.nodeType,
parent, clone, next;
if ( nodeType === TEXT_NODE && node !== stopNode ) {
2016-03-22 17:57:00 +11:00
return split(
node.parentNode, node.splitText( offset ), stopNode, root );
}
if ( nodeType === ELEMENT_NODE ) {
if ( typeof( offset ) === 'number' ) {
offset = offset < node.childNodes.length ?
node.childNodes[ offset ] : null;
2011-10-29 14:15:21 +11:00
}
2011-11-04 16:53:12 +11:00
if ( node === stopNode ) {
return offset;
2011-10-29 14:15:21 +11:00
}
2012-01-25 11:47:26 +11:00
2011-10-29 14:15:21 +11:00
// Clone node without children
parent = node.parentNode;
clone = node.cloneNode( false );
2012-01-25 11:47:26 +11:00
2011-10-29 14:15:21 +11:00
// Add right-hand siblings to the clone
while ( offset ) {
next = offset.nextSibling;
clone.appendChild( offset );
offset = next;
2011-10-29 14:15:21 +11:00
}
2012-01-25 11:47:26 +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' ) ) {
clone.start = ( +node.start || 1 ) + node.childNodes.length - 1;
}
2011-10-29 14:15:21 +11:00
// DO NOT NORMALISE. This may undo the fixCursor() call
// of a node lower down the tree!
2012-01-25 11:47:26 +11:00
2011-10-29 14:15:21 +11:00
// 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 );
2012-01-25 11:47:26 +11:00
2011-10-29 14:15:21 +11:00
// Inject clone after original node
if ( next = node.nextSibling ) {
parent.insertBefore( clone, next );
} else {
parent.appendChild( clone );
}
2012-01-25 11:47:26 +11:00
2011-10-29 14:15:21 +11:00
// Keep on splitting up the tree
2016-03-22 17:57:00 +11:00
return split( parent, clone, stopNode, root );
}
return offset;
}
2012-01-25 11:47:26 +11:00
function _mergeInlines ( node, fakeRange ) {
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 ] ) {
if ( fakeRange.startContainer === child ) {
fakeRange.startContainer = prev;
fakeRange.startOffset += getLength( prev );
2011-10-29 14:15:21 +11:00
}
if ( fakeRange.endContainer === child ) {
fakeRange.endContainer = prev;
fakeRange.endOffset += getLength( prev );
}
if ( fakeRange.startContainer === node ) {
if ( fakeRange.startOffset > l ) {
fakeRange.startOffset -= 1;
2011-10-29 14:15:21 +11:00
}
else if ( fakeRange.startOffset === l ) {
fakeRange.startContainer = prev;
fakeRange.startOffset = getLength( prev );
}
2011-10-29 14:15:21 +11:00
}
if ( fakeRange.endContainer === node ) {
if ( fakeRange.endOffset > l ) {
fakeRange.endOffset -= 1;
2011-10-29 14:15:21 +11:00
}
else if ( fakeRange.endOffset === l ) {
fakeRange.endContainer = prev;
fakeRange.endOffset = getLength( prev );
}
}
detach( child );
if ( child.nodeType === TEXT_NODE ) {
prev.appendData( child.data );
}
else {
frags.push( empty( child ) );
2011-10-29 14:15:21 +11:00
}
}
else if ( child.nodeType === ELEMENT_NODE ) {
len = frags.length;
while ( len-- ) {
child.appendChild( frags.pop() );
}
_mergeInlines( child, fakeRange );
2011-10-29 14:15:21 +11: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 );
}
}
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;
}
detach( container );
offset = block.childNodes.length;
2012-01-25 11:47:26 +11:00
// Remove extra <BR> fixer if present.
last = block.lastChild;
if ( last && last.nodeName === 'BR' ) {
block.removeChild( last );
offset -= 1;
2011-10-29 14:15:21 +11:00
}
block.appendChild( empty( next ) );
range.setStart( block, offset );
range.collapse( true );
mergeInlines( block, range );
}
2016-03-22 17:57:00 +11:00
function mergeContainers ( node, root ) {
var prev = node.previousSibling,
first = node.firstChild,
doc = node.ownerDocument,
2014-05-23 15:26:47 +10:00
isListItem = ( node.nodeName === 'LI' ),
needsFix, block;
// 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 ) {
block = createElement( doc, 'DIV' );
2014-05-23 15:26:47 +10:00
block.appendChild( empty( prev ) );
prev.appendChild( block );
} else {
return;
}
}
detach( node );
needsFix = !isContainer( node );
prev.appendChild( empty( node ) );
if ( needsFix ) {
2016-03-22 17:57:00 +11:00
fixContainer( prev, root );
}
if ( first ) {
2016-03-22 17:57:00 +11:00
mergeContainers( first, root );
}
} else if ( isListItem ) {
prev = createElement( doc, 'DIV' );
node.insertBefore( prev, first );
2016-03-22 17:57:00 +11:00
fixCursor( prev, root );
}
}