mirror of
https://github.com/fastmail/Squire.git
synced 2025-01-21 14:02:28 -05:00
If selection is empty, expand to current line when removing PREs.
This commit is contained in:
parent
6ba03f6989
commit
8a8285560e
4 changed files with 210 additions and 108 deletions
|
@ -3551,9 +3551,9 @@ var getTextFromHTMLFragment = function ( self, frag ) {
|
||||||
var makePreformatted = function ( frag ) {
|
var makePreformatted = function ( frag ) {
|
||||||
return this.createElement( 'PRE',
|
return this.createElement( 'PRE',
|
||||||
this._config.tagAttributes.pre, [
|
this._config.tagAttributes.pre, [
|
||||||
this.createElement( 'DIV', { id: startSelectionId } ),
|
this.createElement( 'INPUT', { id: startSelectionId, type: 'hidden' } ),
|
||||||
getTextFromHTMLFragment( this, frag ),
|
getTextFromHTMLFragment( this, frag ),
|
||||||
this.createElement( 'DIV', { id: endSelectionId } )
|
this.createElement( 'INPUT', { id: endSelectionId, type: 'hidden' } )
|
||||||
] );
|
] );
|
||||||
};
|
};
|
||||||
|
|
||||||
|
@ -3561,14 +3561,13 @@ var removePreformatted = function ( frag ) {
|
||||||
var range = this._doc.createRange();
|
var range = this._doc.createRange();
|
||||||
var startRangeMarker = frag.querySelector( '#' + startSelectionId );
|
var startRangeMarker = frag.querySelector( '#' + startSelectionId );
|
||||||
var endRangeMarker = frag.querySelector( '#' + endSelectionId );
|
var endRangeMarker = frag.querySelector( '#' + endSelectionId );
|
||||||
if (!startRangeMarker || !endRangeMarker) { return frag; }
|
if ( !startRangeMarker || !endRangeMarker ) {
|
||||||
|
return frag;
|
||||||
|
}
|
||||||
|
|
||||||
range.setStartBefore( startRangeMarker );
|
range.setStartBefore( startRangeMarker );
|
||||||
range.setEndAfter( endRangeMarker );
|
range.setEndAfter( endRangeMarker );
|
||||||
|
|
||||||
if (!range || range.collapsed) {
|
|
||||||
return frag; // Maybe remove whole element instead? Single line?
|
|
||||||
} else {
|
|
||||||
var preElems = frag.querySelectorAll('pre');
|
var preElems = frag.querySelectorAll('pre');
|
||||||
if ( preElems.length === 0 ) {
|
if ( preElems.length === 0 ) {
|
||||||
return frag;
|
return frag;
|
||||||
|
@ -3580,6 +3579,41 @@ var removePreformatted = function ( frag ) {
|
||||||
endContainer = range.endContainer,
|
endContainer = range.endContainer,
|
||||||
endOffset = range.endOffset,
|
endOffset = range.endOffset,
|
||||||
splitElems = [];
|
splitElems = [];
|
||||||
|
if ( startRangeMarker.nextSibling === endRangeMarker ) {
|
||||||
|
// Collapsed selection - expand to the whole line
|
||||||
|
var beforeStart = startRangeMarker.previousSibling,
|
||||||
|
newNode;
|
||||||
|
if ( beforeStart && beforeStart.nodeType === TEXT_NODE ) {
|
||||||
|
var lineStartOffset = beforeStart.nodeValue.lastIndexOf( '\n' );
|
||||||
|
if ( lineStartOffset !== -1 ) {
|
||||||
|
newNode = beforeStart.splitText( lineStartOffset );
|
||||||
|
beforeStart.parentNode.insertBefore( startRangeMarker, newNode );
|
||||||
|
} else {
|
||||||
|
beforeStart.parentNode.insertBefore( startRangeMarker, beforeStart );
|
||||||
|
}
|
||||||
|
range.setStartBefore( startRangeMarker );
|
||||||
|
startOffset = range.startOffset;
|
||||||
|
startContainer = range.startContainer;
|
||||||
|
}
|
||||||
|
var afterEnd = endRangeMarker.nextSibling;
|
||||||
|
if ( afterEnd && afterEnd.nodeType === TEXT_NODE ) {
|
||||||
|
var lineEndOffset = afterEnd.nodeValue.indexOf( '\n' );
|
||||||
|
if ( lineEndOffset !== -1 ) {
|
||||||
|
newNode = afterEnd.splitText( lineEndOffset + 1 );
|
||||||
|
afterEnd.parentNode.insertBefore( endRangeMarker, newNode );
|
||||||
|
} else {
|
||||||
|
if ( afterEnd.nextSibling ) {
|
||||||
|
afterEnd.parentNode.insertBefore( endRangeMarker, afterEnd.nextSibling );
|
||||||
|
} else {
|
||||||
|
afterEnd.parentNode.appendChild( endRangeMarker );
|
||||||
|
}
|
||||||
|
}
|
||||||
|
range.setEndBefore( endRangeMarker );
|
||||||
|
endOffset = range.endOffset;
|
||||||
|
endContainer = range.endContainer;
|
||||||
|
}
|
||||||
|
startContainer.normalize();
|
||||||
|
}
|
||||||
if ( getNearest( endContainer, 'PRE' ) === lastPre &&
|
if ( getNearest( endContainer, 'PRE' ) === lastPre &&
|
||||||
endOffset < endContainer.childNodes.length - 1 ) {
|
endOffset < endContainer.childNodes.length - 1 ) {
|
||||||
splitElems.push( split( endContainer, endOffset, lastPre.parentNode ) );
|
splitElems.push( split( endContainer, endOffset, lastPre.parentNode ) );
|
||||||
|
@ -3605,6 +3639,7 @@ var removePreformatted = function ( frag ) {
|
||||||
var nodeLines = childNode.nodeValue.split( '\n' );
|
var nodeLines = childNode.nodeValue.split( '\n' );
|
||||||
/*jshint loopfunc: true*/
|
/*jshint loopfunc: true*/
|
||||||
nodeLines.forEach( function (line) {
|
nodeLines.forEach( function (line) {
|
||||||
|
if ( !line ) { return; }
|
||||||
var div = self.createDefaultBlock( [ self._doc.createTextNode ( line ) ] );
|
var div = self.createDefaultBlock( [ self._doc.createTextNode ( line ) ] );
|
||||||
replacement.appendChild( div );
|
replacement.appendChild( div );
|
||||||
});
|
});
|
||||||
|
@ -3619,7 +3654,6 @@ var removePreformatted = function ( frag ) {
|
||||||
}
|
}
|
||||||
return frag;
|
return frag;
|
||||||
}
|
}
|
||||||
}
|
|
||||||
};
|
};
|
||||||
|
|
||||||
proto._ensureBottomLine = function () {
|
proto._ensureBottomLine = function () {
|
||||||
|
|
File diff suppressed because one or more lines are too long
|
@ -1352,9 +1352,9 @@ var getTextFromHTMLFragment = function ( self, frag ) {
|
||||||
var makePreformatted = function ( frag ) {
|
var makePreformatted = function ( frag ) {
|
||||||
return this.createElement( 'PRE',
|
return this.createElement( 'PRE',
|
||||||
this._config.tagAttributes.pre, [
|
this._config.tagAttributes.pre, [
|
||||||
this.createElement( 'DIV', { id: startSelectionId } ),
|
this.createElement( 'INPUT', { id: startSelectionId, type: 'hidden' } ),
|
||||||
getTextFromHTMLFragment( this, frag ),
|
getTextFromHTMLFragment( this, frag ),
|
||||||
this.createElement( 'DIV', { id: endSelectionId } )
|
this.createElement( 'INPUT', { id: endSelectionId, type: 'hidden' } )
|
||||||
] );
|
] );
|
||||||
};
|
};
|
||||||
|
|
||||||
|
@ -1362,14 +1362,13 @@ var removePreformatted = function ( frag ) {
|
||||||
var range = this._doc.createRange();
|
var range = this._doc.createRange();
|
||||||
var startRangeMarker = frag.querySelector( '#' + startSelectionId );
|
var startRangeMarker = frag.querySelector( '#' + startSelectionId );
|
||||||
var endRangeMarker = frag.querySelector( '#' + endSelectionId );
|
var endRangeMarker = frag.querySelector( '#' + endSelectionId );
|
||||||
if (!startRangeMarker || !endRangeMarker) { return frag; }
|
if ( !startRangeMarker || !endRangeMarker ) {
|
||||||
|
return frag;
|
||||||
|
}
|
||||||
|
|
||||||
range.setStartBefore( startRangeMarker );
|
range.setStartBefore( startRangeMarker );
|
||||||
range.setEndAfter( endRangeMarker );
|
range.setEndAfter( endRangeMarker );
|
||||||
|
|
||||||
if (!range || range.collapsed) {
|
|
||||||
return frag; // Maybe remove whole element instead? Single line?
|
|
||||||
} else {
|
|
||||||
var preElems = frag.querySelectorAll('pre');
|
var preElems = frag.querySelectorAll('pre');
|
||||||
if ( preElems.length === 0 ) {
|
if ( preElems.length === 0 ) {
|
||||||
return frag;
|
return frag;
|
||||||
|
@ -1381,6 +1380,41 @@ var removePreformatted = function ( frag ) {
|
||||||
endContainer = range.endContainer,
|
endContainer = range.endContainer,
|
||||||
endOffset = range.endOffset,
|
endOffset = range.endOffset,
|
||||||
splitElems = [];
|
splitElems = [];
|
||||||
|
if ( startRangeMarker.nextSibling === endRangeMarker ) {
|
||||||
|
// Collapsed selection - expand to the whole line
|
||||||
|
var beforeStart = startRangeMarker.previousSibling,
|
||||||
|
newNode;
|
||||||
|
if ( beforeStart && beforeStart.nodeType === TEXT_NODE ) {
|
||||||
|
var lineStartOffset = beforeStart.nodeValue.lastIndexOf( '\n' );
|
||||||
|
if ( lineStartOffset !== -1 ) {
|
||||||
|
newNode = beforeStart.splitText( lineStartOffset );
|
||||||
|
beforeStart.parentNode.insertBefore( startRangeMarker, newNode );
|
||||||
|
} else {
|
||||||
|
beforeStart.parentNode.insertBefore( startRangeMarker, beforeStart );
|
||||||
|
}
|
||||||
|
range.setStartBefore( startRangeMarker );
|
||||||
|
startOffset = range.startOffset;
|
||||||
|
startContainer = range.startContainer;
|
||||||
|
}
|
||||||
|
var afterEnd = endRangeMarker.nextSibling;
|
||||||
|
if ( afterEnd && afterEnd.nodeType === TEXT_NODE ) {
|
||||||
|
var lineEndOffset = afterEnd.nodeValue.indexOf( '\n' );
|
||||||
|
if ( lineEndOffset !== -1 ) {
|
||||||
|
newNode = afterEnd.splitText( lineEndOffset + 1 );
|
||||||
|
afterEnd.parentNode.insertBefore( endRangeMarker, newNode );
|
||||||
|
} else {
|
||||||
|
if ( afterEnd.nextSibling ) {
|
||||||
|
afterEnd.parentNode.insertBefore( endRangeMarker, afterEnd.nextSibling );
|
||||||
|
} else {
|
||||||
|
afterEnd.parentNode.appendChild( endRangeMarker );
|
||||||
|
}
|
||||||
|
}
|
||||||
|
range.setEndBefore( endRangeMarker );
|
||||||
|
endOffset = range.endOffset;
|
||||||
|
endContainer = range.endContainer;
|
||||||
|
}
|
||||||
|
startContainer.normalize();
|
||||||
|
}
|
||||||
if ( getNearest( endContainer, 'PRE' ) === lastPre &&
|
if ( getNearest( endContainer, 'PRE' ) === lastPre &&
|
||||||
endOffset < endContainer.childNodes.length - 1 ) {
|
endOffset < endContainer.childNodes.length - 1 ) {
|
||||||
splitElems.push( split( endContainer, endOffset, lastPre.parentNode ) );
|
splitElems.push( split( endContainer, endOffset, lastPre.parentNode ) );
|
||||||
|
@ -1406,6 +1440,7 @@ var removePreformatted = function ( frag ) {
|
||||||
var nodeLines = childNode.nodeValue.split( '\n' );
|
var nodeLines = childNode.nodeValue.split( '\n' );
|
||||||
/*jshint loopfunc: true*/
|
/*jshint loopfunc: true*/
|
||||||
nodeLines.forEach( function (line) {
|
nodeLines.forEach( function (line) {
|
||||||
|
if ( !line ) { return; }
|
||||||
var div = self.createDefaultBlock( [ self._doc.createTextNode ( line ) ] );
|
var div = self.createDefaultBlock( [ self._doc.createTextNode ( line ) ] );
|
||||||
replacement.appendChild( div );
|
replacement.appendChild( div );
|
||||||
});
|
});
|
||||||
|
@ -1420,7 +1455,6 @@ var removePreformatted = function ( frag ) {
|
||||||
}
|
}
|
||||||
return frag;
|
return frag;
|
||||||
}
|
}
|
||||||
}
|
|
||||||
};
|
};
|
||||||
|
|
||||||
proto._ensureBottomLine = function () {
|
proto._ensureBottomLine = function () {
|
||||||
|
|
|
@ -242,7 +242,7 @@ describe('Squire RTE', function () {
|
||||||
expect(editor, 'to contain HTML', startHTML);
|
expect(editor, 'to contain HTML', startHTML);
|
||||||
selectAll(editor);
|
selectAll(editor);
|
||||||
editor.removePreformatted();
|
editor.removePreformatted();
|
||||||
expect(editor, 'to contain HTML', '<div>abc</div><div></div><div>one two three four five</div><div></div>');
|
expect(editor, 'to contain HTML', '<div>abc</div><div>one two three four five</div>');
|
||||||
});
|
});
|
||||||
|
|
||||||
it('cuts the beginning off PRE tags', function () {
|
it('cuts the beginning off PRE tags', function () {
|
||||||
|
@ -254,7 +254,7 @@ describe('Squire RTE', function () {
|
||||||
range.setEnd(doc.querySelector('pre').childNodes[0], 18);
|
range.setEnd(doc.querySelector('pre').childNodes[0], 18);
|
||||||
editor.setSelection(range);
|
editor.setSelection(range);
|
||||||
editor.removePreformatted();
|
editor.removePreformatted();
|
||||||
expect(editor, 'to contain HTML', '<div>abc</div><div></div><div>one two three</div><pre> four five\n</pre>');
|
expect(editor, 'to contain HTML', '<div>abc</div><div>one two three</div><pre> four five\n</pre>');
|
||||||
});
|
});
|
||||||
|
|
||||||
it('cuts the end off PRE tags', function () {
|
it('cuts the end off PRE tags', function () {
|
||||||
|
@ -268,6 +268,40 @@ describe('Squire RTE', function () {
|
||||||
editor.removePreformatted();
|
editor.removePreformatted();
|
||||||
expect(editor, 'to contain HTML', '<pre>abc\n\none two three</pre><div> four five</div>');
|
expect(editor, 'to contain HTML', '<pre>abc\n\none two three</pre><div> four five</div>');
|
||||||
});
|
});
|
||||||
|
|
||||||
|
describe('with collapsed selection', function () {
|
||||||
|
beforeEach(function () {
|
||||||
|
var startHTML = '<pre>abc\n\none two three four five\nxyz</pre>';
|
||||||
|
editor.setHTML(startHTML);
|
||||||
|
});
|
||||||
|
|
||||||
|
it('expands selection to the whole line', function () {
|
||||||
|
var range = doc.createRange();
|
||||||
|
range.setStart(doc.querySelector('pre').childNodes[0], 24);
|
||||||
|
range.setEnd(doc.querySelector('pre').childNodes[0], 24);
|
||||||
|
editor.setSelection(range);
|
||||||
|
editor.removePreformatted();
|
||||||
|
expect(editor, 'to contain HTML', '<pre>abc\n</pre><div>one two three four five</div><pre>xyz</pre>');
|
||||||
|
});
|
||||||
|
|
||||||
|
it('... even when on first line of tag', function () {
|
||||||
|
var range = doc.createRange();
|
||||||
|
range.setStart(doc.querySelector('pre').childNodes[0], 2);
|
||||||
|
range.setEnd(doc.querySelector('pre').childNodes[0], 2);
|
||||||
|
editor.setSelection(range);
|
||||||
|
editor.removePreformatted();
|
||||||
|
expect(editor, 'to contain HTML', '<div>abc</div><pre>\none two three four five\nxyz</pre>');
|
||||||
|
});
|
||||||
|
|
||||||
|
it('... or on last line of tag', function () {
|
||||||
|
var range = doc.createRange();
|
||||||
|
range.setStart(doc.querySelector('pre').childNodes[0], 31);
|
||||||
|
range.setEnd(doc.querySelector('pre').childNodes[0], 31);
|
||||||
|
editor.setSelection(range);
|
||||||
|
editor.removePreformatted();
|
||||||
|
expect(editor, 'to contain HTML', '<pre>abc\n\none two three four five</pre><div>xyz</div>');
|
||||||
|
});
|
||||||
|
});
|
||||||
});
|
});
|
||||||
|
|
||||||
afterEach(function () {
|
afterEach(function () {
|
||||||
|
|
Loading…
Add table
Reference in a new issue