diff --git a/src/table.js b/src/table.js
index 64574d0e..e8428e45 100644
--- a/src/table.js
+++ b/src/table.js
@@ -13,6 +13,7 @@ import TableEditing from './tableediting';
import TableUI from './tableui';
import TableSelection from './tableselection';
import TableClipboard from './tableclipboard';
+import TableNavigation from './tablenavigation';
import Widget from '@ckeditor/ckeditor5-widget/src/widget';
import '../theme/table.css';
@@ -26,6 +27,7 @@ import '../theme/table.css';
*
* * {@link module:table/tableediting~TableEditing editing feature},
* * {@link module:table/tableselection~TableSelection selection feature},
+ * * {@link module:table/tablenavigation~TableNavigation keyboard navigation feature},
* * {@link module:table/tableclipboard~TableClipboard clipboard feature},
* * {@link module:table/tableui~TableUI UI feature}.
*
@@ -36,7 +38,7 @@ export default class Table extends Plugin {
* @inheritDoc
*/
static get requires() {
- return [ TableEditing, TableUI, TableSelection, TableClipboard, Widget ];
+ return [ TableEditing, TableUI, TableSelection, TableClipboard, TableNavigation, Widget ];
}
/**
diff --git a/src/tableediting.js b/src/tableediting.js
index 4e2375ea..9aaf4286 100644
--- a/src/tableediting.js
+++ b/src/tableediting.js
@@ -31,7 +31,6 @@ import SetHeaderColumnCommand from './commands/setheadercolumncommand';
import MergeCellsCommand from './commands/mergecellscommand';
import SelectRowCommand from './commands/selectrowcommand';
import SelectColumnCommand from './commands/selectcolumncommand';
-import { getTableCellsContainingSelection } from './utils';
import TableUtils from '../src/tableutils';
import injectTableLayoutPostFixer from './converters/table-layout-post-fixer';
@@ -150,11 +149,6 @@ export default class TableEditing extends Plugin {
injectTableLayoutPostFixer( model );
injectTableCellRefreshPostFixer( model );
injectTableCellParagraphPostFixer( model );
-
- // Handle Tab key navigation.
- this.editor.keystrokes.set( 'Tab', ( ...args ) => this._handleTabOnSelectedTable( ...args ), { priority: 'low' } );
- this.editor.keystrokes.set( 'Tab', this._getTabHandler( true ), { priority: 'low' } );
- this.editor.keystrokes.set( 'Shift+Tab', this._getTabHandler( false ), { priority: 'low' } );
}
/**
@@ -163,102 +157,4 @@ export default class TableEditing extends Plugin {
static get requires() {
return [ TableUtils ];
}
-
- /**
- * Handles {@link module:engine/view/document~Document#event:keydown keydown} events for the Tab key executed
- * when the table widget is selected.
- *
- * @private
- * @param {module:utils/eventinfo~EventInfo} eventInfo
- * @param {module:engine/view/observer/domeventdata~DomEventData} domEventData
- */
- _handleTabOnSelectedTable( domEventData, cancel ) {
- const editor = this.editor;
- const selection = editor.model.document.selection;
-
- if ( !selection.isCollapsed && selection.rangeCount === 1 && selection.getFirstRange().isFlat ) {
- const selectedElement = selection.getSelectedElement();
-
- if ( !selectedElement || !selectedElement.is( 'table' ) ) {
- return;
- }
-
- cancel();
-
- editor.model.change( writer => {
- writer.setSelection( writer.createRangeIn( selectedElement.getChild( 0 ).getChild( 0 ) ) );
- } );
- }
- }
-
- /**
- * Returns a handler for {@link module:engine/view/document~Document#event:keydown keydown} events for the Tab key executed
- * inside table cell.
- *
- * @private
- * @param {Boolean} isForward Whether this handler will move the selection to the next or the previous cell.
- */
- _getTabHandler( isForward ) {
- const editor = this.editor;
-
- return ( domEventData, cancel ) => {
- const selection = editor.model.document.selection;
- const tableCell = getTableCellsContainingSelection( selection )[ 0 ];
-
- if ( !tableCell ) {
- return;
- }
-
- cancel();
-
- const tableRow = tableCell.parent;
- const table = tableRow.parent;
-
- const currentRowIndex = table.getChildIndex( tableRow );
- const currentCellIndex = tableRow.getChildIndex( tableCell );
-
- const isFirstCellInRow = currentCellIndex === 0;
-
- if ( !isForward && isFirstCellInRow && currentRowIndex === 0 ) {
- // It's the first cell of the table - don't do anything (stay in the current position).
- return;
- }
-
- const isLastCellInRow = currentCellIndex === tableRow.childCount - 1;
- const isLastRow = currentRowIndex === table.childCount - 1;
-
- if ( isForward && isLastRow && isLastCellInRow ) {
- editor.execute( 'insertTableRowBelow' );
-
- // Check if the command actually added a row. If `insertTableRowBelow` execution didn't add a row (because it was disabled
- // or it got overwritten) do not change the selection.
- if ( currentRowIndex === table.childCount - 1 ) {
- return;
- }
- }
-
- let cellToFocus;
-
- // Move to first cell in next row.
- if ( isForward && isLastCellInRow ) {
- const nextRow = table.getChild( currentRowIndex + 1 );
-
- cellToFocus = nextRow.getChild( 0 );
- }
- // Move to last cell in a previous row.
- else if ( !isForward && isFirstCellInRow ) {
- const previousRow = table.getChild( currentRowIndex - 1 );
-
- cellToFocus = previousRow.getChild( previousRow.childCount - 1 );
- }
- // Move to next/previous cell.
- else {
- cellToFocus = tableRow.getChild( currentCellIndex + ( isForward ? 1 : -1 ) );
- }
-
- editor.model.change( writer => {
- writer.setSelection( writer.createRangeIn( cellToFocus ) );
- } );
- };
- }
}
diff --git a/src/tablenavigation.js b/src/tablenavigation.js
new file mode 100644
index 00000000..26ad8064
--- /dev/null
+++ b/src/tablenavigation.js
@@ -0,0 +1,516 @@
+/**
+ * @license Copyright (c) 2003-2020, CKSource - Frederico Knabben. All rights reserved.
+ * For licensing, see LICENSE.md or https://ckeditor.com/legal/ckeditor-oss-license
+ */
+
+/**
+ * @module table/tablenavigation
+ */
+
+import Plugin from '@ckeditor/ckeditor5-core/src/plugin';
+import { getSelectedTableCells, getTableCellsContainingSelection } from './utils';
+import { findAncestor } from './commands/utils';
+import TableWalker from './tablewalker';
+import Rect from '@ckeditor/ckeditor5-utils/src/dom/rect';
+import { keyCodes } from '@ckeditor/ckeditor5-utils/src/keyboard';
+import priorities from '@ckeditor/ckeditor5-utils/src/priorities';
+
+/**
+ * This plugin enables keyboard navigation for tables.
+ * It is loaded automatically by the {@link module:table/table~Table} plugin.
+ *
+ * @extends module:core/plugin~Plugin
+ */
+export default class TableNavigation extends Plugin {
+ /**
+ * @inheritDoc
+ */
+ static get pluginName() {
+ return 'TableNavigation';
+ }
+
+ /**
+ * @inheritDoc
+ */
+ init() {
+ const view = this.editor.editing.view;
+ const viewDocument = view.document;
+
+ // Handle Tab key navigation.
+ this.editor.keystrokes.set( 'Tab', ( ...args ) => this._handleTabOnSelectedTable( ...args ), { priority: 'low' } );
+ this.editor.keystrokes.set( 'Tab', this._getTabHandler( true ), { priority: 'low' } );
+ this.editor.keystrokes.set( 'Shift+Tab', this._getTabHandler( false ), { priority: 'low' } );
+
+ // Note: This listener has the "high+1" priority because we would like to avoid collisions with other features
+ // (like Widgets), which take over the keydown events with the "high" priority. Table navigation takes precedence
+ // over Widgets in that matter (widget arrow handler stops propagation of event if object element was selected
+ // but getNearestSelectionRange didn't returned any range).
+ this.listenTo( viewDocument, 'keydown', ( ...args ) => this._onKeydown( ...args ), { priority: priorities.get( 'high' ) + 1 } );
+ }
+
+ /**
+ * Handles {@link module:engine/view/document~Document#event:keydown keydown} events for the Tab key executed
+ * when the table widget is selected.
+ *
+ * @private
+ * @param {module:engine/view/observer/keyobserver~KeyEventData} data Key event data.
+ * @param {Function} cancel The stop/stopPropagation/preventDefault function.
+ */
+ _handleTabOnSelectedTable( data, cancel ) {
+ const editor = this.editor;
+ const selection = editor.model.document.selection;
+
+ if ( !selection.isCollapsed && selection.rangeCount === 1 && selection.getFirstRange().isFlat ) {
+ const selectedElement = selection.getSelectedElement();
+
+ if ( !selectedElement || !selectedElement.is( 'table' ) ) {
+ return;
+ }
+
+ cancel();
+
+ editor.model.change( writer => {
+ writer.setSelection( writer.createRangeIn( selectedElement.getChild( 0 ).getChild( 0 ) ) );
+ } );
+ }
+ }
+
+ /**
+ * Returns a handler for {@link module:engine/view/document~Document#event:keydown keydown} events for the Tab key executed
+ * inside table cell.
+ *
+ * @private
+ * @param {Boolean} isForward Whether this handler will move the selection to the next or the previous cell.
+ */
+ _getTabHandler( isForward ) {
+ const editor = this.editor;
+
+ return ( domEventData, cancel ) => {
+ const selection = editor.model.document.selection;
+ const tableCell = getTableCellsContainingSelection( selection )[ 0 ];
+
+ if ( !tableCell ) {
+ return;
+ }
+
+ cancel();
+
+ const tableRow = tableCell.parent;
+ const table = tableRow.parent;
+
+ const currentRowIndex = table.getChildIndex( tableRow );
+ const currentCellIndex = tableRow.getChildIndex( tableCell );
+
+ const isFirstCellInRow = currentCellIndex === 0;
+
+ if ( !isForward && isFirstCellInRow && currentRowIndex === 0 ) {
+ // It's the first cell of the table - don't do anything (stay in the current position).
+ return;
+ }
+
+ const isLastCellInRow = currentCellIndex === tableRow.childCount - 1;
+ const isLastRow = currentRowIndex === table.childCount - 1;
+
+ if ( isForward && isLastRow && isLastCellInRow ) {
+ editor.execute( 'insertTableRowBelow' );
+
+ // Check if the command actually added a row. If `insertTableRowBelow` execution didn't add a row (because it was disabled
+ // or it got overwritten) do not change the selection.
+ if ( currentRowIndex === table.childCount - 1 ) {
+ return;
+ }
+ }
+
+ let cellToFocus;
+
+ // Move to the first cell in the next row.
+ if ( isForward && isLastCellInRow ) {
+ const nextRow = table.getChild( currentRowIndex + 1 );
+
+ cellToFocus = nextRow.getChild( 0 );
+ }
+ // Move to the last cell in the previous row.
+ else if ( !isForward && isFirstCellInRow ) {
+ const previousRow = table.getChild( currentRowIndex - 1 );
+
+ cellToFocus = previousRow.getChild( previousRow.childCount - 1 );
+ }
+ // Move to the next/previous cell.
+ else {
+ cellToFocus = tableRow.getChild( currentCellIndex + ( isForward ? 1 : -1 ) );
+ }
+
+ editor.model.change( writer => {
+ writer.setSelection( writer.createRangeIn( cellToFocus ) );
+ } );
+ };
+ }
+
+ /**
+ * Handles {@link module:engine/view/document~Document#event:keydown keydown} events.
+ *
+ * @private
+ * @param {module:utils/eventinfo~EventInfo} eventInfo
+ * @param {module:engine/view/observer/domeventdata~DomEventData} domEventData
+ */
+ _onKeydown( eventInfo, domEventData ) {
+ const keyCode = domEventData.keyCode;
+
+ if ( !isArrowKeyCode( keyCode ) ) {
+ return;
+ }
+
+ const wasHandled = this._handleArrowKeys( getDirectionFromKeyCode( keyCode, this.editor.locale.contentLanguageDirection ) );
+
+ if ( wasHandled ) {
+ domEventData.preventDefault();
+ domEventData.stopPropagation();
+ eventInfo.stop();
+ }
+ }
+
+ /**
+ * Handles arrow keys to move the selection around a table.
+ *
+ * @private
+ * @param {'left'|'up'|'right'|'down'} direction The direction of the arrow key.
+ * @returns {Boolean} Returns `true` if key was handled.
+ */
+ _handleArrowKeys( direction ) {
+ const model = this.editor.model;
+ const selection = model.document.selection;
+ const isForward = [ 'right', 'down' ].includes( direction );
+
+ // In case one or more table cells are selected (from outside),
+ // move the selection to a cell adjacent to the selected table fragment.
+ const selectedCells = getSelectedTableCells( selection );
+
+ if ( selectedCells.length ) {
+ const tableCell = isForward ? selectedCells[ selectedCells.length - 1 ] : selectedCells[ 0 ];
+
+ this._navigateFromCellInDirection( tableCell, direction );
+
+ return true;
+ }
+
+ // Abort if we're not in a table cell.
+ const tableCell = findAncestor( 'tableCell', selection.focus );
+
+ if ( !tableCell ) {
+ return false;
+ }
+
+ const cellRange = model.createRangeIn( tableCell );
+
+ // Let's check if the selection is at the beginning/end of the cell.
+ if ( this._isSelectionAtCellEdge( selection, isForward ) ) {
+ this._navigateFromCellInDirection( tableCell, direction );
+
+ return true;
+ }
+
+ // If this is an object selected and it's not at the start or the end of cell content
+ // then let's allow widget handler to take care of it.
+ const objectElement = selection.getSelectedElement();
+
+ if ( objectElement && model.schema.isObject( objectElement ) ) {
+ return false;
+ }
+
+ // If next to the selection there is an object then this is not the cell boundary (widget handler should handle this).
+ if ( this._isObjectElementNextToSelection( selection, isForward ) ) {
+ return false;
+ }
+
+ // If there isn't any $text position between cell edge and selection then we shall move the selection to next cell.
+ const textRange = this._findTextRangeFromSelection( cellRange, selection, isForward );
+
+ if ( !textRange ) {
+ this._navigateFromCellInDirection( tableCell, direction );
+
+ return true;
+ }
+
+ // If the navigation is horizontal then we have no more custom cases.
+ if ( [ 'left', 'right' ].includes( direction ) ) {
+ return false;
+ }
+
+ // If the range is a single line then move the selection to the beginning/end of a cell content.
+ //
+ // We can't move the selection directly to the another cell because of dual position at the end/beginning
+ // of wrapped line (it's at the same time at the end of one line and at the start of the next line).
+ if ( this._isSingleLineRange( textRange, isForward ) ) {
+ model.change( writer => {
+ writer.setSelection( isForward ? cellRange.end : cellRange.start );
+ } );
+
+ return true;
+ }
+ }
+
+ /**
+ * Returns true if the selection is at the boundary of a table cell according to the navigation direction.
+ *
+ * @private
+ * @param {module:engine/model/selection~Selection} selection The current selection.
+ * @param {Boolean} isForward The expected navigation direction.
+ * @returns {Boolean}
+ */
+ _isSelectionAtCellEdge( selection, isForward ) {
+ const model = this.editor.model;
+ const schema = this.editor.model.schema;
+
+ const focus = isForward ? selection.getLastPosition() : selection.getFirstPosition();
+
+ // If the current limit element is not table cell we are for sure not at the cell edge.
+ // Also `modifySelection` will not let us out of it.
+ if ( !schema.getLimitElement( focus ).is( 'tableCell' ) ) {
+ return false;
+ }
+
+ const probe = model.createSelection( focus );
+
+ model.modifySelection( probe, { direction: isForward ? 'forward' : 'backward' } );
+
+ // If there was no change in the focus position, then it's not possible to move the selection there.
+ return focus.isEqual( probe.focus );
+ }
+
+ /**
+ * Checks if there is an {@link module:engine/model/element~Element element} next to the current
+ * {@link module:engine/model/selection~Selection model selection} marked in
+ * {@link module:engine/model/schema~Schema schema} as an `object`.
+ *
+ * @private
+ * @param {module:engine/model/selection~Selection} modelSelection The selection.
+ * @param {Boolean} isForward Direction of checking.
+ * @returns {Boolean}
+ */
+ _isObjectElementNextToSelection( modelSelection, isForward ) {
+ const model = this.editor.model;
+ const schema = model.schema;
+
+ const probe = model.createSelection( modelSelection );
+ model.modifySelection( probe, { direction: isForward ? 'forward' : 'backward' } );
+ const objectElement = isForward ? probe.focus.nodeBefore : probe.focus.nodeAfter;
+
+ return objectElement && schema.isObject( objectElement );
+ }
+
+ /**
+ * Truncates the range so that it spans from the last selection position
+ * to the last allowed $text position (mirrored if isForward is false).
+ *
+ * Returns `null` if resulting range can't contain $text element (according to schema).
+ *
+ * @private
+ * @param {module:engine/model/range~Range} range Current table cell content range.
+ * @param {module:engine/model/selection~Selection} selection The current selection.
+ * @param {Boolean} isForward The expected navigation direction.
+ * @returns {module:engine/model/range~Range|null}
+ */
+ _findTextRangeFromSelection( range, selection, isForward ) {
+ const model = this.editor.model;
+
+ if ( isForward ) {
+ const position = selection.getLastPosition();
+ const lastRangePosition = this._getNearestVisibleTextPosition( range, 'backward' );
+
+ if ( lastRangePosition && position.isBefore( lastRangePosition ) ) {
+ return model.createRange( position, lastRangePosition );
+ }
+
+ return null;
+ } else {
+ const position = selection.getFirstPosition();
+ const firstRangePosition = this._getNearestVisibleTextPosition( range, 'forward' );
+
+ if ( firstRangePosition && position.isAfter( firstRangePosition ) ) {
+ return model.createRange( firstRangePosition, position );
+ }
+
+ return null;
+ }
+ }
+
+ /**
+ * Basing on provided range, finds first/last (depending on `direction`) position inside the range
+ * that can contain `$text` (according to schema) and is visible in the view.
+ *
+ * @param {module:engine/model/range~Range} range The range to find position in.
+ * @param {'forward'|'backward'} direction Search direction.
+ * @returns {module:engine/model/position~Position} Nearest selection range.
+ */
+ _getNearestVisibleTextPosition( range, direction ) {
+ const schema = this.editor.model.schema;
+ const mapper = this.editor.editing.mapper;
+
+ for ( const { nextPosition, item } of range.getWalker( { direction } ) ) {
+ if ( schema.checkChild( nextPosition, '$text' ) ) {
+ const viewElement = mapper.toViewElement( item );
+
+ if ( viewElement && !viewElement.hasClass( 'ck-hidden' ) ) {
+ return nextPosition;
+ }
+ }
+ }
+ }
+
+ /**
+ * Checks if the DOM range corresponding to provided model range renders as a single line by analyzing DOMRects
+ * (verifying if they visually wrap content to the next line).
+ *
+ * @private
+ * @param {module:engine/model/range~Range} modelRange Current table cell content range.
+ * @param {Boolean} isForward The expected navigation direction.
+ * @returns {Boolean}
+ */
+ _isSingleLineRange( modelRange, isForward ) {
+ const model = this.editor.model;
+ const editing = this.editor.editing;
+ const domConverter = editing.view.domConverter;
+
+ // Wrapped lines contain exactly the same position at the end of current line
+ // and at the beginning of next line. That position's client rect is at the end
+ // of current line. In case of caret at first position of the last line that 'dual'
+ // position would be detected as it's not the last line.
+ if ( isForward ) {
+ const probe = model.createSelection( modelRange.start );
+
+ model.modifySelection( probe );
+
+ // If the new position is at the end of the container then we can't use this position
+ // because it would provide incorrect result for eg caption of image and selection
+ // just before end of it. Also in this case there is no "dual" position.
+ if ( !probe.focus.isAtEnd && !modelRange.start.isEqual( probe.focus ) ) {
+ modelRange = model.createRange( probe.focus, modelRange.end );
+ }
+ }
+
+ const viewRange = editing.mapper.toViewRange( modelRange );
+ const domRange = domConverter.viewRangeToDom( viewRange );
+ const rects = Rect.getDomRangeRects( domRange );
+
+ let boundaryVerticalPosition;
+
+ for ( const rect of rects ) {
+ if ( boundaryVerticalPosition === undefined ) {
+ boundaryVerticalPosition = Math.round( rect.bottom );
+ continue;
+ }
+
+ // Let's check if this rect is in new line.
+ if ( Math.round( rect.top ) >= boundaryVerticalPosition ) {
+ return false;
+ }
+
+ boundaryVerticalPosition = Math.max( boundaryVerticalPosition, Math.round( rect.bottom ) );
+ }
+
+ return true;
+ }
+
+ /**
+ * Moves the selection from the given table cell in the specified direction.
+ *
+ * @private
+ * @param {module:engine/model/element~Element} tableCell The table cell to start the selection navigation.
+ * @param {'left'|'up'|'right'|'down'} direction Direction in which selection should move.
+ */
+ _navigateFromCellInDirection( tableCell, direction ) {
+ const model = this.editor.model;
+
+ const table = findAncestor( 'table', tableCell );
+ const tableMap = [ ...new TableWalker( table, { includeSpanned: true } ) ];
+ const { row: lastRow, column: lastColumn } = tableMap[ tableMap.length - 1 ];
+
+ const currentCellInfo = tableMap.find( ( { cell } ) => cell == tableCell );
+ let { row, column } = currentCellInfo;
+
+ switch ( direction ) {
+ case 'left':
+ column--;
+ break;
+
+ case 'up':
+ row--;
+ break;
+
+ case 'right':
+ column += currentCellInfo.colspan;
+ break;
+
+ case 'down':
+ row += currentCellInfo.rowspan;
+ break;
+ }
+
+ const isOutsideVertically = row < 0 || row > lastRow;
+ const isBeforeFirstCell = column < 0 && row <= 0;
+ const isAfterLastCell = column > lastColumn && row >= lastRow;
+
+ // Note that if the table cell at the end of a row is row-spanned then isAfterLastCell will never be true.
+ // However, we don't know if user was navigating on the last row or not, so let's stay in the table.
+
+ if ( isOutsideVertically || isBeforeFirstCell || isAfterLastCell ) {
+ model.change( writer => {
+ writer.setSelection( writer.createRangeOn( table ) );
+ } );
+
+ return;
+ }
+
+ if ( column < 0 ) {
+ column = lastColumn;
+ row--;
+ } else if ( column > lastColumn ) {
+ column = 0;
+ row++;
+ }
+
+ const cellToSelect = tableMap.find( cellInfo => cellInfo.row == row && cellInfo.column == column ).cell;
+ const isForward = [ 'right', 'down' ].includes( direction );
+ const positionToSelect = model.createPositionAt( cellToSelect, isForward ? 0 : 'end' );
+
+ model.change( writer => {
+ writer.setSelection( positionToSelect );
+ } );
+ }
+}
+
+// Returns 'true' if provided key code represents one of the arrow keys.
+//
+// @private
+// @param {Number} keyCode
+// @returns {Boolean}
+function isArrowKeyCode( keyCode ) {
+ return keyCode == keyCodes.arrowright ||
+ keyCode == keyCodes.arrowleft ||
+ keyCode == keyCodes.arrowup ||
+ keyCode == keyCodes.arrowdown;
+}
+
+// Returns direction name from `keyCode`.
+//
+// @private
+// @param {Number} keyCode
+// @param {String} contentLanguageDirection The content language direction.
+// @returns {'left'|'up'|'right'|'down'} Arrow direction.
+function getDirectionFromKeyCode( keyCode, contentLanguageDirection ) {
+ const isLtrContent = contentLanguageDirection === 'ltr';
+
+ switch ( keyCode ) {
+ case keyCodes.arrowleft:
+ return isLtrContent ? 'left' : 'right';
+
+ case keyCodes.arrowright:
+ return isLtrContent ? 'right' : 'left';
+
+ case keyCodes.arrowup:
+ return 'up';
+
+ case keyCodes.arrowdown:
+ return 'down';
+ }
+}
diff --git a/tests/table.js b/tests/table.js
index b8c0ed45..d37ae22e 100644
--- a/tests/table.js
+++ b/tests/table.js
@@ -8,11 +8,12 @@ import TableEditing from '../src/tableediting';
import TableUI from '../src/tableui';
import TableSelection from '../src/tableselection';
import TableClipboard from '../src/tableclipboard';
+import TableNavigation from '../src/tablenavigation';
import Widget from '@ckeditor/ckeditor5-widget/src/widget';
describe( 'Table', () => {
- it( 'requires TableEditing, TableUI, TableSelection, TableClipboard, and Widget', () => {
- expect( Table.requires ).to.deep.equal( [ TableEditing, TableUI, TableSelection, TableClipboard, Widget ] );
+ it( 'requires TableEditing, TableUI, TableSelection, TableClipboard, TableNavigation and Widget', () => {
+ expect( Table.requires ).to.deep.equal( [ TableEditing, TableUI, TableSelection, TableClipboard, TableNavigation, Widget ] );
} );
it( 'has proper name', () => {
diff --git a/tests/tableediting.js b/tests/tableediting.js
index df6731c2..955cfb47 100644
--- a/tests/tableediting.js
+++ b/tests/tableediting.js
@@ -6,7 +6,6 @@
import Paragraph from '@ckeditor/ckeditor5-paragraph/src/paragraph';
import VirtualTestEditor from '@ckeditor/ckeditor5-core/tests/_utils/virtualtesteditor';
import { getData as getModelData, setData as setModelData } from '@ckeditor/ckeditor5-engine/src/dev-utils/model';
-import { getCode } from '@ckeditor/ckeditor5-utils/src/keyboard';
import ImageEditing from '@ckeditor/ckeditor5-image/src/image/imageediting';
import TableEditing from '../src/tableediting';
@@ -224,335 +223,6 @@ describe( 'TableEditing', () => {
} );
} );
- describe( 'caret movement', () => {
- let domEvtDataStub;
-
- beforeEach( () => {
- domEvtDataStub = {
- keyCode: getCode( 'Tab' ),
- preventDefault: sinon.spy(),
- stopPropagation: sinon.spy()
- };
- } );
-
- it( 'should do nothing if not tab pressed', () => {
- setModelData( model, modelTable( [
- [ '11', '12[]' ]
- ] ) );
-
- domEvtDataStub.keyCode = getCode( 'a' );
-
- editor.editing.view.document.fire( 'keydown', domEvtDataStub );
-
- sinon.assert.notCalled( domEvtDataStub.preventDefault );
- sinon.assert.notCalled( domEvtDataStub.stopPropagation );
- assertEqualMarkup( getModelData( model ), modelTable( [
- [ '11', '12[]' ]
- ] ) );
- } );
-
- it( 'should do nothing if Ctrl+Tab is pressed', () => {
- setModelData( model, modelTable( [
- [ '11', '12[]' ]
- ] ) );
-
- domEvtDataStub.ctrlKey = true;
-
- editor.editing.view.document.fire( 'keydown', domEvtDataStub );
-
- sinon.assert.notCalled( domEvtDataStub.preventDefault );
- sinon.assert.notCalled( domEvtDataStub.stopPropagation );
- assertEqualMarkup( getModelData( model ), modelTable( [
- [ '11', '12[]' ]
- ] ) );
- } );
-
- describe( 'on TAB', () => {
- it( 'should do nothing if selection is not in a table', () => {
- setModelData( model, '[]' + modelTable( [ [ '11', '12' ] ] ) );
-
- editor.editing.view.document.fire( 'keydown', domEvtDataStub );
-
- sinon.assert.notCalled( domEvtDataStub.preventDefault );
- sinon.assert.notCalled( domEvtDataStub.stopPropagation );
- assertEqualMarkup( getModelData( model ), '[]' + modelTable( [
- [ '11', '12' ]
- ] ) );
- } );
-
- it( 'should move to next cell', () => {
- setModelData( model, modelTable( [
- [ '11[]', '12' ]
- ] ) );
-
- editor.editing.view.document.fire( 'keydown', domEvtDataStub );
-
- sinon.assert.calledOnce( domEvtDataStub.preventDefault );
- sinon.assert.calledOnce( domEvtDataStub.stopPropagation );
- assertEqualMarkup( getModelData( model ), modelTable( [
- [ '11', '[12]' ]
- ] ) );
- } );
-
- it( 'should create another row and move to first cell in new row', () => {
- setModelData( model, modelTable( [
- [ '11', '[12]' ]
- ] ) );
-
- editor.editing.view.document.fire( 'keydown', domEvtDataStub );
-
- assertEqualMarkup( getModelData( model ), modelTable( [
- [ '11', '12' ],
- [ '[]', '' ]
- ] ) );
- } );
-
- it( 'should not create another row and not move the caret if insertTableRowBelow command is disabled', () => {
- setModelData( model, modelTable( [
- [ '11', '12[]' ]
- ] ) );
-
- const insertTableRowBelowCommand = editor.commands.get( 'insertTableRowBelow' );
-
- insertTableRowBelowCommand.forceDisabled( 'test' );
-
- editor.editing.view.document.fire( 'keydown', domEvtDataStub );
-
- assertEqualMarkup( getModelData( model ), modelTable( [
- [ '11', '12[]' ]
- ] ) );
- } );
-
- it( 'should move to the first cell of next row if on end of a row', () => {
- setModelData( model, modelTable( [
- [ '11', '12[]' ],
- [ '21', '22' ]
- ] ) );
-
- editor.editing.view.document.fire( 'keydown', domEvtDataStub );
-
- assertEqualMarkup( getModelData( model ), modelTable( [
- [ '11', '12' ],
- [ '[21]', '22' ]
- ] ) );
- } );
-
- it( 'should move to the next table cell if part of block content is selected', () => {
- setModelData( model, modelTable( [
- [ '11', '12[foo]bar', '13' ]
- ] ) );
-
- editor.editing.view.document.fire( 'keydown', domEvtDataStub );
-
- assertEqualMarkup( getModelData( model ), modelTable( [
- [
- '11',
- '12foobar',
- '[13]'
- ]
- ] ) );
- } );
-
- it( 'should move to next cell with an image', () => {
- setModelData( model, modelTable( [
- [ '11[]', 'foo' ]
- ] ) );
-
- editor.editing.view.document.fire( 'keydown', domEvtDataStub );
-
- sinon.assert.calledOnce( domEvtDataStub.preventDefault );
- sinon.assert.calledOnce( domEvtDataStub.stopPropagation );
- assertEqualMarkup( getModelData( model ), modelTable( [
- [ '11', '[foo]' ]
- ] ) );
- } );
-
- it( 'should move to next cell with an blockQuote', () => {
- model.schema.register( 'blockQuote', {
- allowWhere: '$block',
- allowContentOf: '$root'
- } );
- editor.conversion.elementToElement( { model: 'blockQuote', view: 'blockquote' } );
-
- setModelData( model, modelTable( [
- [ '11[]', '
foo
' ]
- ] ) );
-
- editor.editing.view.document.fire( 'keydown', domEvtDataStub );
-
- sinon.assert.calledOnce( domEvtDataStub.preventDefault );
- sinon.assert.calledOnce( domEvtDataStub.stopPropagation );
- assertEqualMarkup( getModelData( model ), modelTable( [
- [ '11', '[foo]
' ]
- ] ) );
- } );
-
- it( 'should listen with lower priority then its children', () => {
- // Cancel TAB event.
- editor.keystrokes.set( 'Tab', ( data, cancel ) => cancel() );
-
- setModelData( model, modelTable( [
- [ '11[]', '12' ]
- ] ) );
-
- editor.editing.view.document.fire( 'keydown', domEvtDataStub );
-
- sinon.assert.calledOnce( domEvtDataStub.preventDefault );
- sinon.assert.calledOnce( domEvtDataStub.stopPropagation );
-
- assertEqualMarkup( getModelData( model ), modelTable( [
- [ '11[]', '12' ]
- ] ) );
- } );
-
- describe( 'on table widget selected', () => {
- beforeEach( () => {
- editor.model.schema.register( 'block', {
- allowWhere: '$block',
- allowContentOf: '$block',
- isObject: true
- } );
-
- editor.conversion.elementToElement( { model: 'block', view: 'block' } );
- } );
-
- it( 'should move caret to the first table cell on TAB', () => {
- const spy = sinon.spy();
-
- editor.keystrokes.set( 'Tab', spy, { priority: 'lowest' } );
-
- setModelData( model, '[' + modelTable( [
- [ '11', '12' ]
- ] ) + ']' );
-
- editor.editing.view.document.fire( 'keydown', domEvtDataStub );
-
- sinon.assert.calledOnce( domEvtDataStub.preventDefault );
- sinon.assert.calledOnce( domEvtDataStub.stopPropagation );
-
- assertEqualMarkup( getModelData( model ), modelTable( [
- [ '[11]', '12' ]
- ] ) );
-
- // Should cancel event - so no other tab handler is called.
- sinon.assert.notCalled( spy );
- } );
-
- it( 'shouldn\'t do anything on other blocks', () => {
- const spy = sinon.spy();
-
- editor.editing.view.document.on( 'keydown', spy );
-
- setModelData( model, '[foo]' );
-
- editor.editing.view.document.fire( 'keydown', domEvtDataStub );
-
- sinon.assert.notCalled( domEvtDataStub.preventDefault );
- sinon.assert.notCalled( domEvtDataStub.stopPropagation );
-
- assertEqualMarkup( getModelData( model ), '[foo]' );
-
- // Should not cancel event.
- sinon.assert.calledOnce( spy );
- } );
- } );
- } );
-
- describe( 'on SHIFT+TAB', () => {
- beforeEach( () => {
- domEvtDataStub.shiftKey = true;
- } );
-
- it( 'should do nothing if selection is not in a table', () => {
- setModelData( model, '[]' + modelTable( [
- [ '11', '12' ]
- ] ) );
-
- domEvtDataStub.keyCode = getCode( 'Tab' );
- domEvtDataStub.shiftKey = true;
-
- editor.editing.view.document.fire( 'keydown', domEvtDataStub );
-
- sinon.assert.notCalled( domEvtDataStub.preventDefault );
- sinon.assert.notCalled( domEvtDataStub.stopPropagation );
- assertEqualMarkup( getModelData( model ), '[]' + modelTable( [
- [ '11', '12' ]
- ] ) );
- } );
-
- it( 'should move to previous cell', () => {
- setModelData( model, modelTable( [
- [ '11', '12[]' ]
- ] ) );
-
- editor.editing.view.document.fire( 'keydown', domEvtDataStub );
-
- sinon.assert.calledOnce( domEvtDataStub.preventDefault );
- sinon.assert.calledOnce( domEvtDataStub.stopPropagation );
-
- assertEqualMarkup( getModelData( model ), modelTable( [
- [ '[11]', '12' ]
- ] ) );
- } );
-
- it( 'should not move if caret is in first table cell', () => {
- setModelData( model, 'foo' + modelTable( [
- [ '[]11', '12' ]
- ] ) );
-
- editor.editing.view.document.fire( 'keydown', domEvtDataStub );
-
- assertEqualMarkup( getModelData( model ),
- 'foo' + modelTable( [ [ '[]11', '12' ] ] )
- );
- } );
-
- it( 'should move to the last cell of previous row if on beginning of a row', () => {
- setModelData( model, modelTable( [
- [ '11', '12' ],
- [ '[]21', '22' ]
- ] ) );
-
- editor.editing.view.document.fire( 'keydown', domEvtDataStub );
-
- assertEqualMarkup( getModelData( model ), modelTable( [
- [ '11', '[12]' ],
- [ '21', '22' ]
- ] ) );
- } );
-
- it( 'should move to the previous table cell if part of block content is selected', () => {
- setModelData( model, modelTable( [
- [ '11', '12[foo]bar', '13' ]
- ] ) );
-
- editor.editing.view.document.fire( 'keydown', domEvtDataStub );
-
- assertEqualMarkup( getModelData( model ), modelTable( [
- [
- '[11]',
- '12foobar',
- '13'
- ]
- ] ) );
- } );
-
- it( 'should move to previous cell with an image', () => {
- setModelData( model, modelTable( [
- [ 'foo', 'bar[]' ]
- ] ) );
-
- editor.editing.view.document.fire( 'keydown', domEvtDataStub );
-
- sinon.assert.calledOnce( domEvtDataStub.preventDefault );
- sinon.assert.calledOnce( domEvtDataStub.stopPropagation );
- assertEqualMarkup( getModelData( model ), modelTable( [
- [ '[foo]', 'bar' ]
- ] ) );
- } );
- } );
- } );
-
describe( 'enter key', () => {
let evtDataStub, viewDocument;
diff --git a/tests/tablenavigation.js b/tests/tablenavigation.js
new file mode 100644
index 00000000..81aefcc9
--- /dev/null
+++ b/tests/tablenavigation.js
@@ -0,0 +1,2284 @@
+/**
+ * @license Copyright (c) 2003-2020, CKSource - Frederico Knabben. All rights reserved.
+ * For licensing, see LICENSE.md or https://ckeditor.com/legal/ckeditor-oss-license
+ */
+
+import TableNavigation from '../src/tablenavigation';
+import Table from '../src/table';
+import TableEditing from '../src/tableediting';
+import TableSelection from '../src/tableselection';
+import { modelTable } from './_utils/utils';
+
+import VirtualTestEditor from '@ckeditor/ckeditor5-core/tests/_utils/virtualtesteditor';
+import Paragraph from '@ckeditor/ckeditor5-paragraph/src/paragraph';
+import ImageEditing from '@ckeditor/ckeditor5-image/src/image/imageediting';
+import MediaEmbedEditing from '@ckeditor/ckeditor5-media-embed/src/mediaembedediting';
+import ImageCaptionEditing from '@ckeditor/ckeditor5-image/src/imagecaption/imagecaptionediting';
+import HorizontalLineEditing from '@ckeditor/ckeditor5-horizontal-line/src/horizontallineediting';
+import ClassicTestEditor from '@ckeditor/ckeditor5-core/tests/_utils/classictesteditor';
+import Image from '@ckeditor/ckeditor5-image/src/image';
+import ImageCaption from '@ckeditor/ckeditor5-image/src/imagecaption';
+import HorizontalLine from '@ckeditor/ckeditor5-horizontal-line/src/horizontalline';
+
+import { getCode } from '@ckeditor/ckeditor5-utils/src/keyboard';
+import { getData as getModelData, setData as setModelData } from '@ckeditor/ckeditor5-engine/src/dev-utils/model';
+import { assertEqualMarkup } from '@ckeditor/ckeditor5-utils/tests/_utils/utils';
+import global from '@ckeditor/ckeditor5-utils/src/dom/global';
+import env from '@ckeditor/ckeditor5-utils/src/env';
+
+describe( 'TableNavigation', () => {
+ let editor, model, modelRoot, tableSelection;
+
+ const imageUrl = 'data:image/png;base64,iVBORw0KGgoAAAANSUhEUgAAAGQAAAAUCAQAAADRyVAeAAAAKklEQVR42u3PAQ0AAAwCI' +
+ 'O0f+u/hoAHNZUJFRERERERERERERERERLYiD9N4FAFj2iK6AAAAAElFTkSuQmCC';
+
+ beforeEach( () => {
+ return VirtualTestEditor
+ .create( {
+ plugins: [ TableEditing, TableNavigation, TableSelection, Paragraph, ImageEditing, ImageCaptionEditing, MediaEmbedEditing,
+ HorizontalLineEditing ]
+ } )
+ .then( newEditor => {
+ editor = newEditor;
+
+ model = editor.model;
+ modelRoot = model.document.getRoot();
+ tableSelection = editor.plugins.get( TableSelection );
+ } );
+ } );
+
+ afterEach( () => {
+ editor.destroy();
+ } );
+
+ it( 'should have pluginName', () => {
+ expect( TableNavigation.pluginName ).to.equal( 'TableNavigation' );
+ } );
+
+ describe( 'Tab key handling', () => {
+ let domEvtDataStub;
+
+ beforeEach( () => {
+ domEvtDataStub = {
+ keyCode: getCode( 'Tab' ),
+ preventDefault: sinon.spy(),
+ stopPropagation: sinon.spy()
+ };
+ } );
+
+ it( 'should do nothing if pressed other key', () => {
+ setModelData( model, modelTable( [
+ [ '11', '12[]' ]
+ ] ) );
+
+ domEvtDataStub.keyCode = getCode( 'a' );
+
+ editor.editing.view.document.fire( 'keydown', domEvtDataStub );
+
+ sinon.assert.notCalled( domEvtDataStub.preventDefault );
+ sinon.assert.notCalled( domEvtDataStub.stopPropagation );
+ assertEqualMarkup( getModelData( model ), modelTable( [
+ [ '11', '12[]' ]
+ ] ) );
+ } );
+
+ it( 'should do nothing if Ctrl+Tab is pressed', () => {
+ setModelData( model, modelTable( [
+ [ '11', '12[]' ]
+ ] ) );
+
+ domEvtDataStub.ctrlKey = true;
+
+ editor.editing.view.document.fire( 'keydown', domEvtDataStub );
+
+ sinon.assert.notCalled( domEvtDataStub.preventDefault );
+ sinon.assert.notCalled( domEvtDataStub.stopPropagation );
+ assertEqualMarkup( getModelData( model ), modelTable( [
+ [ '11', '12[]' ]
+ ] ) );
+ } );
+
+ describe( 'on Tab key press', () => {
+ it( 'should do nothing if the selection is not in a table', () => {
+ setModelData( model, '[]' + modelTable( [ [ '11', '12' ] ] ) );
+
+ editor.editing.view.document.fire( 'keydown', domEvtDataStub );
+
+ sinon.assert.notCalled( domEvtDataStub.preventDefault );
+ sinon.assert.notCalled( domEvtDataStub.stopPropagation );
+ assertEqualMarkup( getModelData( model ), '[]' + modelTable( [
+ [ '11', '12' ]
+ ] ) );
+ } );
+
+ it( 'should move to the next cell', () => {
+ setModelData( model, modelTable( [
+ [ '11[]', '12' ]
+ ] ) );
+
+ editor.editing.view.document.fire( 'keydown', domEvtDataStub );
+
+ sinon.assert.calledOnce( domEvtDataStub.preventDefault );
+ sinon.assert.calledOnce( domEvtDataStub.stopPropagation );
+ assertEqualMarkup( getModelData( model ), modelTable( [
+ [ '11', '[12]' ]
+ ] ) );
+ } );
+
+ it( 'should create another row and move to the first cell in a new row', () => {
+ setModelData( model, modelTable( [
+ [ '11', '[12]' ]
+ ] ) );
+
+ editor.editing.view.document.fire( 'keydown', domEvtDataStub );
+
+ assertEqualMarkup( getModelData( model ), modelTable( [
+ [ '11', '12' ],
+ [ '[]', '' ]
+ ] ) );
+ } );
+
+ it( 'should not create another row and not move the caret if the "insertTableRowBelow" command is disabled', () => {
+ setModelData( model, modelTable( [
+ [ '11', '12[]' ]
+ ] ) );
+
+ const insertTableRowBelowCommand = editor.commands.get( 'insertTableRowBelow' );
+
+ insertTableRowBelowCommand.forceDisabled( 'test' );
+
+ editor.editing.view.document.fire( 'keydown', domEvtDataStub );
+
+ assertEqualMarkup( getModelData( model ), modelTable( [
+ [ '11', '12[]' ]
+ ] ) );
+ } );
+
+ it( 'should move to the first cell of the next row if at the end of a row', () => {
+ setModelData( model, modelTable( [
+ [ '11', '12[]' ],
+ [ '21', '22' ]
+ ] ) );
+
+ editor.editing.view.document.fire( 'keydown', domEvtDataStub );
+
+ assertEqualMarkup( getModelData( model ), modelTable( [
+ [ '11', '12' ],
+ [ '[21]', '22' ]
+ ] ) );
+ } );
+
+ it( 'should move to the next table cell if the block content is partially selected', () => {
+ setModelData( model, modelTable( [
+ [ '11', '12[foo]bar', '13' ]
+ ] ) );
+
+ editor.editing.view.document.fire( 'keydown', domEvtDataStub );
+
+ assertEqualMarkup( getModelData( model ), modelTable( [
+ [
+ '11',
+ '12foobar',
+ '[13]'
+ ]
+ ] ) );
+ } );
+
+ it( 'should move to the next cell containing an image', () => {
+ setModelData( model, modelTable( [
+ [ '11[]', 'foo' ]
+ ] ) );
+
+ editor.editing.view.document.fire( 'keydown', domEvtDataStub );
+
+ sinon.assert.calledOnce( domEvtDataStub.preventDefault );
+ sinon.assert.calledOnce( domEvtDataStub.stopPropagation );
+ assertEqualMarkup( getModelData( model ), modelTable( [
+ [ '11', '[foo]' ]
+ ] ) );
+ } );
+
+ it( 'should move to the next cell containing a block quote', () => {
+ model.schema.register( 'blockQuote', {
+ allowWhere: '$block',
+ allowContentOf: '$root'
+ } );
+ editor.conversion.elementToElement( { model: 'blockQuote', view: 'blockquote' } );
+
+ setModelData( model, modelTable( [
+ [ '11[]', 'foo
' ]
+ ] ) );
+
+ editor.editing.view.document.fire( 'keydown', domEvtDataStub );
+
+ sinon.assert.calledOnce( domEvtDataStub.preventDefault );
+ sinon.assert.calledOnce( domEvtDataStub.stopPropagation );
+ assertEqualMarkup( getModelData( model ), modelTable( [
+ [ '11', '[foo]
' ]
+ ] ) );
+ } );
+
+ it( 'should listen with the lower priority than its children', () => {
+ // Cancel TAB event.
+ editor.keystrokes.set( 'Tab', ( data, cancel ) => cancel() );
+
+ setModelData( model, modelTable( [
+ [ '11[]', '12' ]
+ ] ) );
+
+ editor.editing.view.document.fire( 'keydown', domEvtDataStub );
+
+ sinon.assert.calledOnce( domEvtDataStub.preventDefault );
+ sinon.assert.calledOnce( domEvtDataStub.stopPropagation );
+
+ assertEqualMarkup( getModelData( model ), modelTable( [
+ [ '11[]', '12' ]
+ ] ) );
+ } );
+
+ describe( 'on table widget selected', () => {
+ beforeEach( () => {
+ editor.model.schema.register( 'block', {
+ allowWhere: '$block',
+ allowContentOf: '$block',
+ isObject: true
+ } );
+
+ editor.conversion.elementToElement( { model: 'block', view: 'block' } );
+ } );
+
+ it( 'should move caret to the first table cell on TAB', () => {
+ const spy = sinon.spy();
+
+ editor.keystrokes.set( 'Tab', spy, { priority: 'lowest' } );
+
+ setModelData( model, '[' + modelTable( [
+ [ '11', '12' ]
+ ] ) + ']' );
+
+ editor.editing.view.document.fire( 'keydown', domEvtDataStub );
+
+ sinon.assert.calledOnce( domEvtDataStub.preventDefault );
+ sinon.assert.calledOnce( domEvtDataStub.stopPropagation );
+
+ assertEqualMarkup( getModelData( model ), modelTable( [
+ [ '[11]', '12' ]
+ ] ) );
+
+ // Should cancel event - so no other tab handler is called.
+ sinon.assert.notCalled( spy );
+ } );
+
+ it( 'shouldn\'t do anything on other blocks', () => {
+ const spy = sinon.spy();
+
+ editor.editing.view.document.on( 'keydown', spy );
+
+ setModelData( model, '[foo]' );
+
+ editor.editing.view.document.fire( 'keydown', domEvtDataStub );
+
+ sinon.assert.notCalled( domEvtDataStub.preventDefault );
+ sinon.assert.notCalled( domEvtDataStub.stopPropagation );
+
+ assertEqualMarkup( getModelData( model ), '[foo]' );
+
+ // Should not cancel event.
+ sinon.assert.calledOnce( spy );
+ } );
+ } );
+ } );
+
+ describe( 'on Shift+Tab key press', () => {
+ beforeEach( () => {
+ domEvtDataStub.shiftKey = true;
+ } );
+
+ it( 'should do nothing if the selection is not in a table', () => {
+ setModelData( model, '[]' + modelTable( [
+ [ '11', '12' ]
+ ] ) );
+
+ domEvtDataStub.keyCode = getCode( 'Tab' );
+ domEvtDataStub.shiftKey = true;
+
+ editor.editing.view.document.fire( 'keydown', domEvtDataStub );
+
+ sinon.assert.notCalled( domEvtDataStub.preventDefault );
+ sinon.assert.notCalled( domEvtDataStub.stopPropagation );
+ assertEqualMarkup( getModelData( model ), '[]' + modelTable( [
+ [ '11', '12' ]
+ ] ) );
+ } );
+
+ it( 'should move to the previous cell', () => {
+ setModelData( model, modelTable( [
+ [ '11', '12[]' ]
+ ] ) );
+
+ editor.editing.view.document.fire( 'keydown', domEvtDataStub );
+
+ sinon.assert.calledOnce( domEvtDataStub.preventDefault );
+ sinon.assert.calledOnce( domEvtDataStub.stopPropagation );
+
+ assertEqualMarkup( getModelData( model ), modelTable( [
+ [ '[11]', '12' ]
+ ] ) );
+ } );
+
+ it( 'should not move if the caret is in the first table cell', () => {
+ setModelData( model, 'foo' + modelTable( [
+ [ '[]11', '12' ]
+ ] ) );
+
+ editor.editing.view.document.fire( 'keydown', domEvtDataStub );
+
+ assertEqualMarkup( getModelData( model ),
+ 'foo' + modelTable( [ [ '[]11', '12' ] ] )
+ );
+ } );
+
+ it( 'should move to the last cell of a previous row if at the beginning of a row', () => {
+ setModelData( model, modelTable( [
+ [ '11', '12' ],
+ [ '[]21', '22' ]
+ ] ) );
+
+ editor.editing.view.document.fire( 'keydown', domEvtDataStub );
+
+ assertEqualMarkup( getModelData( model ), modelTable( [
+ [ '11', '[12]' ],
+ [ '21', '22' ]
+ ] ) );
+ } );
+
+ it( 'should move to the previous table cell if the block content is partially selected', () => {
+ setModelData( model, modelTable( [
+ [ '11', '12[foo]bar', '13' ]
+ ] ) );
+
+ editor.editing.view.document.fire( 'keydown', domEvtDataStub );
+
+ assertEqualMarkup( getModelData( model ), modelTable( [
+ [
+ '[11]',
+ '12foobar',
+ '13'
+ ]
+ ] ) );
+ } );
+
+ it( 'should move to the previous cell containing an image', () => {
+ setModelData( model, modelTable( [
+ [ 'foo', 'bar[]' ]
+ ] ) );
+
+ editor.editing.view.document.fire( 'keydown', domEvtDataStub );
+
+ sinon.assert.calledOnce( domEvtDataStub.preventDefault );
+ sinon.assert.calledOnce( domEvtDataStub.stopPropagation );
+ assertEqualMarkup( getModelData( model ), modelTable( [
+ [ '[foo]', 'bar' ]
+ ] ) );
+ } );
+ } );
+ } );
+
+ describe( 'Arrow keys handling', () => {
+ let leftArrowDomEvtDataStub, rightArrowDomEvtDataStub, upArrowDomEvtDataStub, downArrowDomEvtDataStub;
+
+ beforeEach( () => {
+ leftArrowDomEvtDataStub = {
+ keyCode: getCode( 'ArrowLeft' ),
+ preventDefault: sinon.spy(),
+ stopPropagation: sinon.spy(),
+ domTarget: global.document.body
+ };
+ rightArrowDomEvtDataStub = {
+ keyCode: getCode( 'ArrowRight' ),
+ preventDefault: sinon.spy(),
+ stopPropagation: sinon.spy(),
+ domTarget: global.document.body
+ };
+ upArrowDomEvtDataStub = {
+ keyCode: getCode( 'ArrowUp' ),
+ preventDefault: sinon.spy(),
+ stopPropagation: sinon.spy(),
+ domTarget: global.document.body
+ };
+ downArrowDomEvtDataStub = {
+ keyCode: getCode( 'ArrowDown' ),
+ preventDefault: sinon.spy(),
+ stopPropagation: sinon.spy(),
+ domTarget: global.document.body
+ };
+ } );
+
+ it( 'should do nothing if pressed some non-arrow key', () => {
+ setModelData( model, modelTable( [
+ [ '00', '01[]' ]
+ ] ) );
+
+ leftArrowDomEvtDataStub.keyCode = getCode( 'a' );
+
+ editor.editing.view.document.fire( 'keydown', leftArrowDomEvtDataStub );
+
+ sinon.assert.notCalled( leftArrowDomEvtDataStub.preventDefault );
+ sinon.assert.notCalled( leftArrowDomEvtDataStub.stopPropagation );
+
+ assertEqualMarkup( getModelData( model ), modelTable( [
+ [ '00', '01[]' ]
+ ] ) );
+ } );
+
+ it( 'should do nothing if the selection is not in a table', () => {
+ const modelData = '[]foobar' + modelTable( [ [ '00', '01' ] ] );
+
+ setModelData( model, modelData );
+
+ editor.editing.view.document.fire( 'keydown', upArrowDomEvtDataStub );
+
+ sinon.assert.notCalled( upArrowDomEvtDataStub.preventDefault );
+ sinon.assert.notCalled( upArrowDomEvtDataStub.stopPropagation );
+
+ assertEqualMarkup( getModelData( model ), modelData );
+ } );
+
+ describe( '#_navigateFromCellInDirection (finding a proper cell to move the selection to)', () => {
+ let tableNavigation;
+
+ beforeEach( () => {
+ tableNavigation = editor.plugins.get( TableNavigation );
+ } );
+
+ describe( 'with no col/row-spanned cells', () => {
+ beforeEach( () => {
+ setModelData( model, 'foo' + modelTable( [
+ [ '00', '01', '02' ],
+ [ '10', '11', '12' ],
+ [ '20', '21', '22' ]
+ ] ) + 'bar' );
+ } );
+
+ describe( 'from the first table cell', () => {
+ let tableCell;
+
+ beforeEach( () => {
+ tableCell = modelRoot.getNodeByPath( [ 1, 0, 0 ] );
+ } );
+
+ it( 'should navigate to the start position of the cell on the right when the direction is "right"', () => {
+ tableNavigation._navigateFromCellInDirection( tableCell, 'right' );
+
+ assertEqualMarkup( getModelData( model ), 'foo' + modelTable( [
+ [ '00', '[]01', '02' ],
+ [ '10', '11', '12' ],
+ [ '20', '21', '22' ]
+ ] ) + 'bar' );
+ } );
+
+ it( 'should navigate to the start position of the cell below when the direction is "down"', () => {
+ tableNavigation._navigateFromCellInDirection( tableCell, 'down' );
+
+ assertEqualMarkup( getModelData( model ), 'foo' + modelTable( [
+ [ '00', '01', '02' ],
+ [ '[]10', '11', '12' ],
+ [ '20', '21', '22' ]
+ ] ) + 'bar' );
+ } );
+
+ it( 'should select a whole table when the direction is "up"', () => {
+ tableNavigation._navigateFromCellInDirection( tableCell, 'up' );
+
+ assertEqualMarkup( getModelData( model ), 'foo[' + modelTable( [
+ [ '00', '01', '02' ],
+ [ '10', '11', '12' ],
+ [ '20', '21', '22' ]
+ ] ) + ']bar' );
+ } );
+
+ it( 'should select a whole table when the direction is "left"', () => {
+ tableNavigation._navigateFromCellInDirection( tableCell, 'left' );
+
+ assertEqualMarkup( getModelData( model ), 'foo[' + modelTable( [
+ [ '00', '01', '02' ],
+ [ '10', '11', '12' ],
+ [ '20', '21', '22' ]
+ ] ) + ']bar' );
+ } );
+ } );
+
+ describe( 'from the last table cell', () => {
+ let tableCell;
+
+ beforeEach( () => {
+ tableCell = modelRoot.getNodeByPath( [ 1, 2, 2 ] );
+ } );
+
+ it( 'should navigate to the end position of the cell on the left when the direction is "left"', () => {
+ tableNavigation._navigateFromCellInDirection( tableCell, 'left' );
+
+ assertEqualMarkup( getModelData( model ), 'foo' + modelTable( [
+ [ '00', '01', '02' ],
+ [ '10', '11', '12' ],
+ [ '20', '21[]', '22' ]
+ ] ) + 'bar' );
+ } );
+
+ it( 'should navigate to the end position of the cell above when the direction is "up"', () => {
+ tableNavigation._navigateFromCellInDirection( tableCell, 'up' );
+
+ assertEqualMarkup( getModelData( model ), 'foo' + modelTable( [
+ [ '00', '01', '02' ],
+ [ '10', '11', '12[]' ],
+ [ '20', '21', '22' ]
+ ] ) + 'bar' );
+ } );
+
+ it( 'should select a whole table when the direction is "down"', () => {
+ tableNavigation._navigateFromCellInDirection( tableCell, 'down' );
+
+ assertEqualMarkup( getModelData( model ), 'foo[' + modelTable( [
+ [ '00', '01', '02' ],
+ [ '10', '11', '12' ],
+ [ '20', '21', '22' ]
+ ] ) + ']bar' );
+ } );
+
+ it( 'should select a whole table when the direction is "right"', () => {
+ tableNavigation._navigateFromCellInDirection( tableCell, 'right' );
+
+ assertEqualMarkup( getModelData( model ), 'foo[' + modelTable( [
+ [ '00', '01', '02' ],
+ [ '10', '11', '12' ],
+ [ '20', '21', '22' ]
+ ] ) + ']bar' );
+ } );
+ } );
+
+ describe( 'from a cell in the first column (but not first row)', () => {
+ let tableCell;
+
+ beforeEach( () => {
+ tableCell = modelRoot.getNodeByPath( [ 1, 1, 0 ] );
+ } );
+
+ it( 'should navigate to start position of the cell on the right when the direction is "right"', () => {
+ tableNavigation._navigateFromCellInDirection( tableCell, 'right' );
+
+ assertEqualMarkup( getModelData( model ), 'foo' + modelTable( [
+ [ '00', '01', '02' ],
+ [ '10', '[]11', '12' ],
+ [ '20', '21', '22' ]
+ ] ) + 'bar' );
+ } );
+
+ it( 'should navigate to the end position of the cell above when the direction is "up"', () => {
+ tableNavigation._navigateFromCellInDirection( tableCell, 'up' );
+
+ assertEqualMarkup( getModelData( model ), 'foo' + modelTable( [
+ [ '00[]', '01', '02' ],
+ [ '10', '11', '12' ],
+ [ '20', '21', '22' ]
+ ] ) + 'bar' );
+ } );
+
+ it( 'should navigate to the start position of the cell below when the direction is "down"', () => {
+ tableNavigation._navigateFromCellInDirection( tableCell, 'down' );
+
+ assertEqualMarkup( getModelData( model ), 'foo' + modelTable( [
+ [ '00', '01', '02' ],
+ [ '10', '11', '12' ],
+ [ '[]20', '21', '22' ]
+ ] ) + 'bar' );
+ } );
+
+ it( 'should navigate to the end position of the last cell in the previous row when the direction is "left"', () => {
+ tableNavigation._navigateFromCellInDirection( tableCell, 'left' );
+
+ assertEqualMarkup( getModelData( model ), 'foo' + modelTable( [
+ [ '00', '01', '02[]' ],
+ [ '10', '11', '12' ],
+ [ '20', '21', '22' ]
+ ] ) + 'bar' );
+ } );
+ } );
+
+ describe( 'from a cell in the last column (but not the last row)', () => {
+ let tableCell;
+
+ beforeEach( () => {
+ tableCell = modelRoot.getNodeByPath( [ 1, 1, 2 ] );
+ } );
+
+ it( 'should navigate to the end position of the cell on the left when the direction is "left"', () => {
+ tableNavigation._navigateFromCellInDirection( tableCell, 'left' );
+
+ assertEqualMarkup( getModelData( model ), 'foo' + modelTable( [
+ [ '00', '01', '02' ],
+ [ '10', '11[]', '12' ],
+ [ '20', '21', '22' ]
+ ] ) + 'bar' );
+ } );
+
+ it( 'should navigate to the end position the cell above when the direction is "up"', () => {
+ tableNavigation._navigateFromCellInDirection( tableCell, 'up' );
+
+ assertEqualMarkup( getModelData( model ), 'foo' + modelTable( [
+ [ '00', '01', '02[]' ],
+ [ '10', '11', '12' ],
+ [ '20', '21', '22' ]
+ ] ) + 'bar' );
+ } );
+
+ it( 'should navigate to the start position of the cell below when the direction is "down"', () => {
+ tableNavigation._navigateFromCellInDirection( tableCell, 'down' );
+
+ assertEqualMarkup( getModelData( model ), 'foo' + modelTable( [
+ [ '00', '01', '02' ],
+ [ '10', '11', '12' ],
+ [ '20', '21', '[]22' ]
+ ] ) + 'bar' );
+ } );
+
+ it( 'should navigate to the start position of the first cell in the next row when the direction is "right"', () => {
+ tableNavigation._navigateFromCellInDirection( tableCell, 'right' );
+
+ assertEqualMarkup( getModelData( model ), 'foo' + modelTable( [
+ [ '00', '01', '02' ],
+ [ '10', '11', '12' ],
+ [ '[]20', '21', '22' ]
+ ] ) + 'bar' );
+ } );
+ } );
+ } );
+
+ describe( 'with col/row-spanned cells', () => {
+ beforeEach( () => {
+ // +----+----+----+----+----+
+ // | 00 | 01 | 02 | 03 | 04 |
+ // +----+----+----+----+----+
+ // | 10 | 11 | 13 | 14 |
+ // +----+ + +----+
+ // | 20 | | | 24 |
+ // +----+----+----+----+----+
+ // | 30 | 31 | 33 | 34 |
+ // +----+----+----+----+----+
+ // | 40 | 41 | 42 | 43 | 44 |
+ // +----+----+----+----+----+
+ setModelData( model, modelTable( [
+ [ '00', '01', '02', '03', '04' ],
+ [ '10', { contents: '11', colspan: 2, rowspan: 2 }, { contents: '13', rowspan: 2 }, '14' ],
+ [ '20', '24' ],
+ [ '30', { contents: '31', colspan: 2 }, '33', '34' ],
+ [ '40', '41', '42', '43', '44' ]
+ ] ) );
+ } );
+
+ describe( 'when navigating to the right', () => {
+ it( 'should navigate to the row-col-spanned cell when approaching from the upper-spanned row', () => {
+ const tableCell = modelRoot.getNodeByPath( [ 0, 1, 0 ] );
+
+ tableNavigation._navigateFromCellInDirection( tableCell, 'right' );
+
+ assertEqualMarkup( getModelData( model ), modelTable( [
+ [ '00', '01', '02', '03', '04' ],
+ [ '10', { contents: '[]11', colspan: 2, rowspan: 2 }, { contents: '13', rowspan: 2 }, '14' ],
+ [ '20', '24' ],
+ [ '30', { contents: '31', colspan: 2 }, '33', '34' ],
+ [ '40', '41', '42', '43', '44' ]
+ ] ) );
+ } );
+
+ it( 'should navigate to the row-col-spanned cell when approaching from the lower-spanned row', () => {
+ const tableCell = modelRoot.getNodeByPath( [ 0, 2, 0 ] );
+
+ tableNavigation._navigateFromCellInDirection( tableCell, 'right' );
+
+ assertEqualMarkup( getModelData( model ), modelTable( [
+ [ '00', '01', '02', '03', '04' ],
+ [ '10', { contents: '[]11', colspan: 2, rowspan: 2 }, { contents: '13', rowspan: 2 }, '14' ],
+ [ '20', '24' ],
+ [ '30', { contents: '31', colspan: 2 }, '33', '34' ],
+ [ '40', '41', '42', '43', '44' ]
+ ] ) );
+ } );
+
+ it( 'should navigate to the row-spanned cell when approaching from the other row-spanned cell', () => {
+ const tableCell = modelRoot.getNodeByPath( [ 0, 1, 1 ] );
+
+ tableNavigation._navigateFromCellInDirection( tableCell, 'right' );
+
+ assertEqualMarkup( getModelData( model ), modelTable( [
+ [ '00', '01', '02', '03', '04' ],
+ [ '10', { contents: '11', colspan: 2, rowspan: 2 }, { contents: '[]13', rowspan: 2 }, '14' ],
+ [ '20', '24' ],
+ [ '30', { contents: '31', colspan: 2 }, '33', '34' ],
+ [ '40', '41', '42', '43', '44' ]
+ ] ) );
+ } );
+
+ it( 'should navigate to the cell in the upper-spanned row when approaching from the row-spanned cell', () => {
+ const tableCell = modelRoot.getNodeByPath( [ 0, 1, 2 ] ); // Cell 13.
+
+ tableNavigation._navigateFromCellInDirection( tableCell, 'right' );
+
+ assertEqualMarkup( getModelData( model ), modelTable( [
+ [ '00', '01', '02', '03', '04' ],
+ [ '10', { contents: '11', colspan: 2, rowspan: 2 }, { contents: '13', rowspan: 2 }, '[]14' ],
+ [ '20', '24' ],
+ [ '30', { contents: '31', colspan: 2 }, '33', '34' ],
+ [ '40', '41', '42', '43', '44' ]
+ ] ) );
+ } );
+
+ it( 'should navigate to the col-spanned cell', () => {
+ const tableCell = modelRoot.getNodeByPath( [ 0, 3, 0 ] );
+
+ tableNavigation._navigateFromCellInDirection( tableCell, 'right' );
+
+ assertEqualMarkup( getModelData( model ), modelTable( [
+ [ '00', '01', '02', '03', '04' ],
+ [ '10', { contents: '11', colspan: 2, rowspan: 2 }, { contents: '13', rowspan: 2 }, '14' ],
+ [ '20', '24' ],
+ [ '30', { contents: '[]31', colspan: 2 }, '33', '34' ],
+ [ '40', '41', '42', '43', '44' ]
+ ] ) );
+ } );
+
+ it( 'should navigate from the col-spanned cell', () => {
+ const tableCell = modelRoot.getNodeByPath( [ 0, 3, 1 ] );
+
+ tableNavigation._navigateFromCellInDirection( tableCell, 'right' );
+
+ assertEqualMarkup( getModelData( model ), modelTable( [
+ [ '00', '01', '02', '03', '04' ],
+ [ '10', { contents: '11', colspan: 2, rowspan: 2 }, { contents: '13', rowspan: 2 }, '14' ],
+ [ '20', '24' ],
+ [ '30', { contents: '31', colspan: 2 }, '[]33', '34' ],
+ [ '40', '41', '42', '43', '44' ]
+ ] ) );
+ } );
+ } );
+
+ describe( 'when navigating to the left', () => {
+ it( 'should navigate to the row-spanned cell when approaching from the upper-spanned row', () => {
+ const tableCell = modelRoot.getNodeByPath( [ 0, 1, 3 ] ); // Cell 14.
+
+ tableNavigation._navigateFromCellInDirection( tableCell, 'left' );
+
+ assertEqualMarkup( getModelData( model ), modelTable( [
+ [ '00', '01', '02', '03', '04' ],
+ [ '10', { contents: '11', colspan: 2, rowspan: 2 }, { contents: '13[]', rowspan: 2 }, '14' ],
+ [ '20', '24' ],
+ [ '30', { contents: '31', colspan: 2 }, '33', '34' ],
+ [ '40', '41', '42', '43', '44' ]
+ ] ) );
+ } );
+
+ it( 'should navigate to the row-spanned cell when approaching from the lower-spanned row', () => {
+ const tableCell = modelRoot.getNodeByPath( [ 0, 2, 1 ] ); // Cell 24.
+
+ tableNavigation._navigateFromCellInDirection( tableCell, 'left' );
+
+ assertEqualMarkup( getModelData( model ), modelTable( [
+ [ '00', '01', '02', '03', '04' ],
+ [ '10', { contents: '11', colspan: 2, rowspan: 2 }, { contents: '13[]', rowspan: 2 }, '14' ],
+ [ '20', '24' ],
+ [ '30', { contents: '31', colspan: 2 }, '33', '34' ],
+ [ '40', '41', '42', '43', '44' ]
+ ] ) );
+ } );
+
+ it( 'should navigate to the row-spanned cell when approaching from the other row-spanned cell', () => {
+ const tableCell = modelRoot.getNodeByPath( [ 0, 1, 2 ] ); // Cell 13.
+
+ tableNavigation._navigateFromCellInDirection( tableCell, 'left' );
+
+ assertEqualMarkup( getModelData( model ), modelTable( [
+ [ '00', '01', '02', '03', '04' ],
+ [ '10', { contents: '11[]', colspan: 2, rowspan: 2 }, { contents: '13', rowspan: 2 }, '14' ],
+ [ '20', '24' ],
+ [ '30', { contents: '31', colspan: 2 }, '33', '34' ],
+ [ '40', '41', '42', '43', '44' ]
+ ] ) );
+ } );
+
+ it( 'should navigate to the cell in the upper-spanned row when approaching from the row-spanned cell', () => {
+ const tableCell = modelRoot.getNodeByPath( [ 0, 1, 1 ] ); // Cell 11.
+
+ tableNavigation._navigateFromCellInDirection( tableCell, 'left' );
+
+ assertEqualMarkup( getModelData( model ), modelTable( [
+ [ '00', '01', '02', '03', '04' ],
+ [ '10[]', { contents: '11', colspan: 2, rowspan: 2 }, { contents: '13', rowspan: 2 }, '14' ],
+ [ '20', '24' ],
+ [ '30', { contents: '31', colspan: 2 }, '33', '34' ],
+ [ '40', '41', '42', '43', '44' ]
+ ] ) );
+ } );
+
+ it( 'should navigate to the col-spanned cell', () => {
+ const tableCell = modelRoot.getNodeByPath( [ 0, 3, 2 ] ); // Cell 33.
+
+ tableNavigation._navigateFromCellInDirection( tableCell, 'left' );
+
+ assertEqualMarkup( getModelData( model ), modelTable( [
+ [ '00', '01', '02', '03', '04' ],
+ [ '10', { contents: '11', colspan: 2, rowspan: 2 }, { contents: '13', rowspan: 2 }, '14' ],
+ [ '20', '24' ],
+ [ '30', { contents: '31[]', colspan: 2 }, '33', '34' ],
+ [ '40', '41', '42', '43', '44' ]
+ ] ) );
+ } );
+
+ it( 'should navigate from the col-spanned cell', () => {
+ const tableCell = modelRoot.getNodeByPath( [ 0, 3, 1 ] );
+
+ tableNavigation._navigateFromCellInDirection( tableCell, 'left' );
+
+ assertEqualMarkup( getModelData( model ), modelTable( [
+ [ '00', '01', '02', '03', '04' ],
+ [ '10', { contents: '11', colspan: 2, rowspan: 2 }, { contents: '13', rowspan: 2 }, '14' ],
+ [ '20', '24' ],
+ [ '30[]', { contents: '31', colspan: 2 }, '33', '34' ],
+ [ '40', '41', '42', '43', '44' ]
+ ] ) );
+ } );
+ } );
+
+ describe( 'when navigating down', () => {
+ it( 'should navigate to the row-col-spanned cell when approaching from the first spanned column', () => {
+ const tableCell = modelRoot.getNodeByPath( [ 0, 0, 1 ] );
+
+ tableNavigation._navigateFromCellInDirection( tableCell, 'down' );
+
+ assertEqualMarkup( getModelData( model ), modelTable( [
+ [ '00', '01', '02', '03', '04' ],
+ [ '10', { contents: '[]11', colspan: 2, rowspan: 2 }, { contents: '13', rowspan: 2 }, '14' ],
+ [ '20', '24' ],
+ [ '30', { contents: '31', colspan: 2 }, '33', '34' ],
+ [ '40', '41', '42', '43', '44' ]
+ ] ) );
+ } );
+
+ it( 'should navigate to the row-col-spanned cell when approaching from the last spanned column', () => {
+ const tableCell = modelRoot.getNodeByPath( [ 0, 0, 2 ] );
+
+ tableNavigation._navigateFromCellInDirection( tableCell, 'down' );
+
+ assertEqualMarkup( getModelData( model ), modelTable( [
+ [ '00', '01', '02', '03', '04' ],
+ [ '10', { contents: '[]11', colspan: 2, rowspan: 2 }, { contents: '13', rowspan: 2 }, '14' ],
+ [ '20', '24' ],
+ [ '30', { contents: '31', colspan: 2 }, '33', '34' ],
+ [ '40', '41', '42', '43', '44' ]
+ ] ) );
+ } );
+
+ it( 'should navigate to the row-spanned cell when approaching from the other col-spanned cell', () => {
+ const tableCell = modelRoot.getNodeByPath( [ 0, 1, 1 ] );
+
+ tableNavigation._navigateFromCellInDirection( tableCell, 'down' );
+
+ assertEqualMarkup( getModelData( model ), modelTable( [
+ [ '00', '01', '02', '03', '04' ],
+ [ '10', { contents: '11', colspan: 2, rowspan: 2 }, { contents: '13', rowspan: 2 }, '14' ],
+ [ '20', '24' ],
+ [ '30', { contents: '[]31', colspan: 2 }, '33', '34' ],
+ [ '40', '41', '42', '43', '44' ]
+ ] ) );
+ } );
+
+ it( 'should navigate to the cell in the first spanned column when approaching from the col-spanned cell', () => {
+ const tableCell = modelRoot.getNodeByPath( [ 0, 1, 1 ] ); // Cell 11.
+
+ tableNavigation._navigateFromCellInDirection( tableCell, 'down' );
+
+ assertEqualMarkup( getModelData( model ), modelTable( [
+ [ '00', '01', '02', '03', '04' ],
+ [ '10', { contents: '11', colspan: 2, rowspan: 2 }, { contents: '13', rowspan: 2 }, '14' ],
+ [ '20', '24' ],
+ [ '30', { contents: '[]31', colspan: 2 }, '33', '34' ],
+ [ '40', '41', '42', '43', '44' ]
+ ] ) );
+ } );
+
+ it( 'should navigate to the row-spanned cell', () => {
+ const tableCell = modelRoot.getNodeByPath( [ 0, 0, 3 ] );
+
+ tableNavigation._navigateFromCellInDirection( tableCell, 'down' );
+
+ assertEqualMarkup( getModelData( model ), modelTable( [
+ [ '00', '01', '02', '03', '04' ],
+ [ '10', { contents: '11', colspan: 2, rowspan: 2 }, { contents: '[]13', rowspan: 2 }, '14' ],
+ [ '20', '24' ],
+ [ '30', { contents: '31', colspan: 2 }, '33', '34' ],
+ [ '40', '41', '42', '43', '44' ]
+ ] ) );
+ } );
+
+ it( 'should navigate from the row-spanned cell', () => {
+ const tableCell = modelRoot.getNodeByPath( [ 0, 1, 2 ] ); // Cell 13.
+
+ tableNavigation._navigateFromCellInDirection( tableCell, 'down' );
+
+ assertEqualMarkup( getModelData( model ), modelTable( [
+ [ '00', '01', '02', '03', '04' ],
+ [ '10', { contents: '11', colspan: 2, rowspan: 2 }, { contents: '13', rowspan: 2 }, '14' ],
+ [ '20', '24' ],
+ [ '30', { contents: '31', colspan: 2 }, '[]33', '34' ],
+ [ '40', '41', '42', '43', '44' ]
+ ] ) );
+ } );
+ } );
+
+ describe( 'when navigating up', () => {
+ it( 'should navigate to the col-spanned cell when approaching from the first spanned column', () => {
+ const tableCell = modelRoot.getNodeByPath( [ 0, 4, 1 ] );
+
+ tableNavigation._navigateFromCellInDirection( tableCell, 'up' );
+
+ assertEqualMarkup( getModelData( model ), modelTable( [
+ [ '00', '01', '02', '03', '04' ],
+ [ '10', { contents: '11', colspan: 2, rowspan: 2 }, { contents: '13', rowspan: 2 }, '14' ],
+ [ '20', '24' ],
+ [ '30', { contents: '31[]', colspan: 2 }, '33', '34' ],
+ [ '40', '41', '42', '43', '44' ]
+ ] ) );
+ } );
+
+ it( 'should navigate to the col-spanned cell when approaching from the last spanned column', () => {
+ const tableCell = modelRoot.getNodeByPath( [ 0, 4, 2 ] );
+
+ tableNavigation._navigateFromCellInDirection( tableCell, 'up' );
+
+ assertEqualMarkup( getModelData( model ), modelTable( [
+ [ '00', '01', '02', '03', '04' ],
+ [ '10', { contents: '11', colspan: 2, rowspan: 2 }, { contents: '13', rowspan: 2 }, '14' ],
+ [ '20', '24' ],
+ [ '30', { contents: '31[]', colspan: 2 }, '33', '34' ],
+ [ '40', '41', '42', '43', '44' ]
+ ] ) );
+ } );
+
+ it( 'should navigate to the row-col-spanned cell when approaching from the other col-spanned cell', () => {
+ const tableCell = modelRoot.getNodeByPath( [ 0, 3, 1 ] );
+
+ tableNavigation._navigateFromCellInDirection( tableCell, 'up' );
+
+ assertEqualMarkup( getModelData( model ), modelTable( [
+ [ '00', '01', '02', '03', '04' ],
+ [ '10', { contents: '11[]', colspan: 2, rowspan: 2 }, { contents: '13', rowspan: 2 }, '14' ],
+ [ '20', '24' ],
+ [ '30', { contents: '31', colspan: 2 }, '33', '34' ],
+ [ '40', '41', '42', '43', '44' ]
+ ] ) );
+ } );
+
+ it( 'should navigate to the cell in the first spanned column when approaching from the col-spanned cell', () => {
+ const tableCell = modelRoot.getNodeByPath( [ 0, 1, 1 ] );
+
+ tableNavigation._navigateFromCellInDirection( tableCell, 'up' );
+
+ assertEqualMarkup( getModelData( model ), modelTable( [
+ [ '00', '01[]', '02', '03', '04' ],
+ [ '10', { contents: '11', colspan: 2, rowspan: 2 }, { contents: '13', rowspan: 2 }, '14' ],
+ [ '20', '24' ],
+ [ '30', { contents: '31', colspan: 2 }, '33', '34' ],
+ [ '40', '41', '42', '43', '44' ]
+ ] ) );
+ } );
+
+ it( 'should navigate to the row-spanned cell', () => {
+ const tableCell = modelRoot.getNodeByPath( [ 0, 3, 2 ] ); // Cell 33.
+
+ tableNavigation._navigateFromCellInDirection( tableCell, 'up' );
+
+ assertEqualMarkup( getModelData( model ), modelTable( [
+ [ '00', '01', '02', '03', '04' ],
+ [ '10', { contents: '11', colspan: 2, rowspan: 2 }, { contents: '13[]', rowspan: 2 }, '14' ],
+ [ '20', '24' ],
+ [ '30', { contents: '31', colspan: 2 }, '33', '34' ],
+ [ '40', '41', '42', '43', '44' ]
+ ] ) );
+ } );
+
+ it( 'should navigate from the row-spanned cell', () => {
+ const tableCell = modelRoot.getNodeByPath( [ 0, 1, 2 ] ); // Cell 13.
+
+ tableNavigation._navigateFromCellInDirection( tableCell, 'up' );
+
+ assertEqualMarkup( getModelData( model ), modelTable( [
+ [ '00', '01', '02', '03[]', '04' ],
+ [ '10', { contents: '11', colspan: 2, rowspan: 2 }, { contents: '13', rowspan: 2 }, '14' ],
+ [ '20', '24' ],
+ [ '30', { contents: '31', colspan: 2 }, '33', '34' ],
+ [ '40', '41', '42', '43', '44' ]
+ ] ) );
+ } );
+ } );
+ } );
+ } );
+
+ describe( 'with the table cells selected from outside', () => {
+ describe( 'on a single table cell selected', () => {
+ beforeEach( () => {
+ setModelData( model, modelTable( [
+ [ '00', '01', '02' ],
+ [ '10', '11', '12' ],
+ [ '20', '21', '22' ]
+ ] ) );
+
+ tableSelection._setCellSelection(
+ modelRoot.getNodeByPath( [ 0, 1, 1 ] ),
+ modelRoot.getNodeByPath( [ 0, 1, 1 ] )
+ );
+ } );
+
+ it( 'should move to the cell on the left', () => {
+ editor.editing.view.document.fire( 'keydown', leftArrowDomEvtDataStub );
+
+ sinon.assert.calledOnce( leftArrowDomEvtDataStub.preventDefault );
+ sinon.assert.calledOnce( leftArrowDomEvtDataStub.stopPropagation );
+
+ assertEqualMarkup( getModelData( model ), modelTable( [
+ [ '00', '01', '02' ],
+ [ '10[]', '11', '12' ],
+ [ '20', '21', '22' ]
+ ] ) );
+ } );
+
+ it( 'should move to the cell on the right', () => {
+ editor.editing.view.document.fire( 'keydown', rightArrowDomEvtDataStub );
+
+ sinon.assert.calledOnce( rightArrowDomEvtDataStub.preventDefault );
+ sinon.assert.calledOnce( rightArrowDomEvtDataStub.stopPropagation );
+
+ assertEqualMarkup( getModelData( model ), modelTable( [
+ [ '00', '01', '02' ],
+ [ '10', '11', '[]12' ],
+ [ '20', '21', '22' ]
+ ] ) );
+ } );
+
+ it( 'should move to the cell above the selection', () => {
+ editor.editing.view.document.fire( 'keydown', upArrowDomEvtDataStub );
+
+ sinon.assert.calledOnce( upArrowDomEvtDataStub.preventDefault );
+ sinon.assert.calledOnce( upArrowDomEvtDataStub.stopPropagation );
+
+ assertEqualMarkup( getModelData( model ), modelTable( [
+ [ '00', '01[]', '02' ],
+ [ '10', '11', '12' ],
+ [ '20', '21', '22' ]
+ ] ) );
+ } );
+
+ it( 'should move to the cell below the selection', () => {
+ editor.editing.view.document.fire( 'keydown', downArrowDomEvtDataStub );
+
+ sinon.assert.calledOnce( downArrowDomEvtDataStub.preventDefault );
+ sinon.assert.calledOnce( downArrowDomEvtDataStub.stopPropagation );
+
+ assertEqualMarkup( getModelData( model ), modelTable( [
+ [ '00', '01', '02' ],
+ [ '10', '11', '12' ],
+ [ '20', '[]21', '22' ]
+ ] ) );
+ } );
+ } );
+
+ describe( 'on multiple table cells selected vertically', () => {
+ beforeEach( () => {
+ setModelData( model, modelTable( [
+ [ '00', '01', '02', '03' ],
+ [ '10', '11', '12', '13' ],
+ [ '20', '21', '22', '23' ],
+ [ '30', '31', '32', '33' ]
+ ] ) );
+
+ tableSelection._setCellSelection(
+ modelRoot.getNodeByPath( [ 0, 1, 1 ] ),
+ modelRoot.getNodeByPath( [ 0, 2, 1 ] )
+ );
+ } );
+
+ it( 'should move to the cell on the top left of the selection', () => {
+ editor.editing.view.document.fire( 'keydown', leftArrowDomEvtDataStub );
+
+ sinon.assert.calledOnce( leftArrowDomEvtDataStub.preventDefault );
+ sinon.assert.calledOnce( leftArrowDomEvtDataStub.stopPropagation );
+
+ assertEqualMarkup( getModelData( model ), modelTable( [
+ [ '00', '01', '02', '03' ],
+ [ '10[]', '11', '12', '13' ],
+ [ '20', '21', '22', '23' ],
+ [ '30', '31', '32', '33' ]
+ ] ) );
+ } );
+
+ it( 'should move to the cell on the bottom right of the selection', () => {
+ editor.editing.view.document.fire( 'keydown', rightArrowDomEvtDataStub );
+
+ sinon.assert.calledOnce( rightArrowDomEvtDataStub.preventDefault );
+ sinon.assert.calledOnce( rightArrowDomEvtDataStub.stopPropagation );
+
+ assertEqualMarkup( getModelData( model ), modelTable( [
+ [ '00', '01', '02', '03' ],
+ [ '10', '11', '12', '13' ],
+ [ '20', '21', '[]22', '23' ],
+ [ '30', '31', '32', '33' ]
+ ] ) );
+ } );
+
+ it( 'should move to the cell above the selection', () => {
+ editor.editing.view.document.fire( 'keydown', upArrowDomEvtDataStub );
+
+ sinon.assert.calledOnce( upArrowDomEvtDataStub.preventDefault );
+ sinon.assert.calledOnce( upArrowDomEvtDataStub.stopPropagation );
+
+ assertEqualMarkup( getModelData( model ), modelTable( [
+ [ '00', '01[]', '02', '03' ],
+ [ '10', '11', '12', '13' ],
+ [ '20', '21', '22', '23' ],
+ [ '30', '31', '32', '33' ]
+ ] ) );
+ } );
+
+ it( 'should move to the cell below the selection', () => {
+ editor.editing.view.document.fire( 'keydown', downArrowDomEvtDataStub );
+
+ sinon.assert.calledOnce( downArrowDomEvtDataStub.preventDefault );
+ sinon.assert.calledOnce( downArrowDomEvtDataStub.stopPropagation );
+
+ assertEqualMarkup( getModelData( model ), modelTable( [
+ [ '00', '01', '02', '03' ],
+ [ '10', '11', '12', '13' ],
+ [ '20', '21', '22', '23' ],
+ [ '30', '[]31', '32', '33' ]
+ ] ) );
+ } );
+ } );
+
+ describe( 'on multiple table cell selected horizontally', () => {
+ beforeEach( () => {
+ setModelData( model, modelTable( [
+ [ '00', '01', '02', '03' ],
+ [ '10', '11', '12', '13' ],
+ [ '20', '21', '22', '23' ],
+ [ '30', '31', '32', '33' ]
+ ] ) );
+
+ // Note that this also tests that selection direction doesn't matter.
+
+ tableSelection._setCellSelection(
+ modelRoot.getNodeByPath( [ 0, 1, 2 ] ),
+ modelRoot.getNodeByPath( [ 0, 1, 1 ] )
+ );
+ } );
+
+ it( 'should move to the cell on the top left of the selection', () => {
+ editor.editing.view.document.fire( 'keydown', leftArrowDomEvtDataStub );
+
+ sinon.assert.calledOnce( leftArrowDomEvtDataStub.preventDefault );
+ sinon.assert.calledOnce( leftArrowDomEvtDataStub.stopPropagation );
+
+ assertEqualMarkup( getModelData( model ), modelTable( [
+ [ '00', '01', '02', '03' ],
+ [ '10[]', '11', '12', '13' ],
+ [ '20', '21', '22', '23' ],
+ [ '30', '31', '32', '33' ]
+ ] ) );
+ } );
+
+ it( 'should move to the cell on the bottom right of the selection', () => {
+ editor.editing.view.document.fire( 'keydown', rightArrowDomEvtDataStub );
+
+ sinon.assert.calledOnce( rightArrowDomEvtDataStub.preventDefault );
+ sinon.assert.calledOnce( rightArrowDomEvtDataStub.stopPropagation );
+
+ assertEqualMarkup( getModelData( model ), modelTable( [
+ [ '00', '01', '02', '03' ],
+ [ '10', '11', '12', '[]13' ],
+ [ '20', '21', '22', '23' ],
+ [ '30', '31', '32', '33' ]
+ ] ) );
+ } );
+
+ it( 'should move to the cell above the selection', () => {
+ editor.editing.view.document.fire( 'keydown', upArrowDomEvtDataStub );
+
+ sinon.assert.calledOnce( upArrowDomEvtDataStub.preventDefault );
+ sinon.assert.calledOnce( upArrowDomEvtDataStub.stopPropagation );
+
+ assertEqualMarkup( getModelData( model ), modelTable( [
+ [ '00', '01[]', '02', '03' ],
+ [ '10', '11', '12', '13' ],
+ [ '20', '21', '22', '23' ],
+ [ '30', '31', '32', '33' ]
+ ] ) );
+ } );
+
+ it( 'should move to the cell below the selection', () => {
+ editor.editing.view.document.fire( 'keydown', downArrowDomEvtDataStub );
+
+ sinon.assert.calledOnce( downArrowDomEvtDataStub.preventDefault );
+ sinon.assert.calledOnce( downArrowDomEvtDataStub.stopPropagation );
+
+ assertEqualMarkup( getModelData( model ), modelTable( [
+ [ '00', '01', '02', '03' ],
+ [ '10', '11', '12', '13' ],
+ [ '20', '21', '[]22', '23' ],
+ [ '30', '31', '32', '33' ]
+ ] ) );
+ } );
+ } );
+
+ describe( 'on multiple table cell selected diagonally', () => {
+ beforeEach( () => {
+ setModelData( model, modelTable( [
+ [ '00', '01', '02', '03' ],
+ [ '10', '11', '12', '13' ],
+ [ '20', '21', '22', '23' ],
+ [ '30', '31', '32', '33' ]
+ ] ) );
+
+ tableSelection._setCellSelection(
+ modelRoot.getNodeByPath( [ 0, 1, 1 ] ),
+ modelRoot.getNodeByPath( [ 0, 2, 2 ] )
+ );
+ } );
+
+ it( 'should move to the cell on the top left of selection', () => {
+ editor.editing.view.document.fire( 'keydown', leftArrowDomEvtDataStub );
+
+ sinon.assert.calledOnce( leftArrowDomEvtDataStub.preventDefault );
+ sinon.assert.calledOnce( leftArrowDomEvtDataStub.stopPropagation );
+
+ assertEqualMarkup( getModelData( model ), modelTable( [
+ [ '00', '01', '02', '03' ],
+ [ '10[]', '11', '12', '13' ],
+ [ '20', '21', '22', '23' ],
+ [ '30', '31', '32', '33' ]
+ ] ) );
+ } );
+
+ it( 'should move to the cell on the bottom right of selection', () => {
+ editor.editing.view.document.fire( 'keydown', rightArrowDomEvtDataStub );
+
+ sinon.assert.calledOnce( rightArrowDomEvtDataStub.preventDefault );
+ sinon.assert.calledOnce( rightArrowDomEvtDataStub.stopPropagation );
+
+ assertEqualMarkup( getModelData( model ), modelTable( [
+ [ '00', '01', '02', '03' ],
+ [ '10', '11', '12', '13' ],
+ [ '20', '21', '22', '[]23' ],
+ [ '30', '31', '32', '33' ]
+ ] ) );
+ } );
+
+ it( 'should move to the cell above selection', () => {
+ editor.editing.view.document.fire( 'keydown', upArrowDomEvtDataStub );
+
+ sinon.assert.calledOnce( upArrowDomEvtDataStub.preventDefault );
+ sinon.assert.calledOnce( upArrowDomEvtDataStub.stopPropagation );
+
+ assertEqualMarkup( getModelData( model ), modelTable( [
+ [ '00', '01[]', '02', '03' ],
+ [ '10', '11', '12', '13' ],
+ [ '20', '21', '22', '23' ],
+ [ '30', '31', '32', '33' ]
+ ] ) );
+ } );
+
+ it( 'should move to the cell below selection', () => {
+ editor.editing.view.document.fire( 'keydown', downArrowDomEvtDataStub );
+
+ sinon.assert.calledOnce( downArrowDomEvtDataStub.preventDefault );
+ sinon.assert.calledOnce( downArrowDomEvtDataStub.stopPropagation );
+
+ assertEqualMarkup( getModelData( model ), modelTable( [
+ [ '00', '01', '02', '03' ],
+ [ '10', '11', '12', '13' ],
+ [ '20', '21', '22', '23' ],
+ [ '30', '31', '[]32', '33' ]
+ ] ) );
+ } );
+ } );
+ } );
+
+ describe( 'with the selection inside a table cell', () => {
+ describe( 'with the selection at the boundary of a cell', () => {
+ describe( 'simple cell text content', () => {
+ it( 'should navigate to the cell on the left', () => {
+ setModelData( model, modelTable( [
+ [ '00', '01', '02' ],
+ [ '10', '[]11', '12' ],
+ [ '20', '21', '22' ]
+ ] ) );
+
+ editor.editing.view.document.fire( 'keydown', leftArrowDomEvtDataStub );
+
+ sinon.assert.calledOnce( leftArrowDomEvtDataStub.preventDefault );
+ sinon.assert.calledOnce( leftArrowDomEvtDataStub.stopPropagation );
+
+ assertEqualMarkup( getModelData( model ), modelTable( [
+ [ '00', '01', '02' ],
+ [ '10[]', '11', '12' ],
+ [ '20', '21', '22' ]
+ ] ) );
+ } );
+
+ it( 'should navigate to the cell on the right', () => {
+ setModelData( model, modelTable( [
+ [ '00', '01', '02' ],
+ [ '10', '11[]', '12' ],
+ [ '20', '21', '22' ]
+ ] ) );
+
+ editor.editing.view.document.fire( 'keydown', rightArrowDomEvtDataStub );
+
+ sinon.assert.calledOnce( rightArrowDomEvtDataStub.preventDefault );
+ sinon.assert.calledOnce( rightArrowDomEvtDataStub.stopPropagation );
+
+ assertEqualMarkup( getModelData( model ), modelTable( [
+ [ '00', '01', '02' ],
+ [ '10', '11', '[]12' ],
+ [ '20', '21', '22' ]
+ ] ) );
+ } );
+
+ it( 'should navigate to the cell above', () => {
+ setModelData( model, modelTable( [
+ [ '00', '01', '02' ],
+ [ '10', '[]11', '12' ],
+ [ '20', '21', '22' ]
+ ] ) );
+
+ editor.editing.view.document.fire( 'keydown', upArrowDomEvtDataStub );
+
+ sinon.assert.calledOnce( upArrowDomEvtDataStub.preventDefault );
+ sinon.assert.calledOnce( upArrowDomEvtDataStub.stopPropagation );
+
+ assertEqualMarkup( getModelData( model ), modelTable( [
+ [ '00', '01[]', '02' ],
+ [ '10', '11', '12' ],
+ [ '20', '21', '22' ]
+ ] ) );
+ } );
+
+ it( 'should navigate to the cell below', () => {
+ setModelData( model, modelTable( [
+ [ '00', '01', '02' ],
+ [ '10', '11[]', '12' ],
+ [ '20', '21', '22' ]
+ ] ) );
+
+ editor.editing.view.document.fire( 'keydown', downArrowDomEvtDataStub );
+
+ sinon.assert.calledOnce( downArrowDomEvtDataStub.preventDefault );
+ sinon.assert.calledOnce( downArrowDomEvtDataStub.stopPropagation );
+
+ assertEqualMarkup( getModelData( model ), modelTable( [
+ [ '00', '01', '02' ],
+ [ '10', '11', '12' ],
+ [ '20', '[]21', '22' ]
+ ] ) );
+ } );
+ } );
+
+ describe( 'multiple paragraphs in the cell content', () => {
+ it( 'should navigate to the cell on the left', () => {
+ setModelData( model, modelTable( [
+ [ '00', '01', '02' ],
+ [ '10', '[]11x', '12' ],
+ [ '20', '21', '22' ]
+ ] ) );
+
+ editor.editing.view.document.fire( 'keydown', leftArrowDomEvtDataStub );
+
+ sinon.assert.calledOnce( leftArrowDomEvtDataStub.preventDefault );
+ sinon.assert.calledOnce( leftArrowDomEvtDataStub.stopPropagation );
+
+ assertEqualMarkup( getModelData( model ), modelTable( [
+ [ '00', '01', '02' ],
+ [ '10[]', '11x', '12' ],
+ [ '20', '21', '22' ]
+ ] ) );
+ } );
+
+ it( 'should navigate to the cell on the right', () => {
+ setModelData( model, modelTable( [
+ [ '00', '01', '02' ],
+ [ '10', '11x[]', '12' ],
+ [ '20', '21', '22' ]
+ ] ) );
+
+ editor.editing.view.document.fire( 'keydown', rightArrowDomEvtDataStub );
+
+ sinon.assert.calledOnce( rightArrowDomEvtDataStub.preventDefault );
+ sinon.assert.calledOnce( rightArrowDomEvtDataStub.stopPropagation );
+
+ assertEqualMarkup( getModelData( model ), modelTable( [
+ [ '00', '01', '02' ],
+ [ '10', '11x', '[]12' ],
+ [ '20', '21', '22' ]
+ ] ) );
+ } );
+
+ it( 'should navigate to the cell above', () => {
+ setModelData( model, modelTable( [
+ [ '00', '01', '02' ],
+ [ '10', '[]11x', '12' ],
+ [ '20', '21', '22' ]
+ ] ) );
+
+ editor.editing.view.document.fire( 'keydown', upArrowDomEvtDataStub );
+
+ sinon.assert.calledOnce( upArrowDomEvtDataStub.preventDefault );
+ sinon.assert.calledOnce( upArrowDomEvtDataStub.stopPropagation );
+
+ assertEqualMarkup( getModelData( model ), modelTable( [
+ [ '00', '01[]', '02' ],
+ [ '10', '11x', '12' ],
+ [ '20', '21', '22' ]
+ ] ) );
+ } );
+
+ it( 'should navigate to the cell below', () => {
+ setModelData( model, modelTable( [
+ [ '00', '01', '02' ],
+ [ '10', '11x[]', '12' ],
+ [ '20', '21', '22' ]
+ ] ) );
+
+ editor.editing.view.document.fire( 'keydown', downArrowDomEvtDataStub );
+
+ sinon.assert.calledOnce( downArrowDomEvtDataStub.preventDefault );
+ sinon.assert.calledOnce( downArrowDomEvtDataStub.stopPropagation );
+
+ assertEqualMarkup( getModelData( model ), modelTable( [
+ [ '00', '01', '02' ],
+ [ '10', '11x', '12' ],
+ [ '20', '[]21', '22' ]
+ ] ) );
+ } );
+ } );
+
+ describe( 'image widget with caption as only cell content', () => {
+ it( 'should navigate to the cell on the left', () => {
+ setModelData( model, modelTable( [
+ [ '00', '01', '02' ],
+ [ '10', `[]11`, '12' ],
+ [ '20', '21', '22' ]
+ ] ) );
+
+ editor.editing.view.document.fire( 'keydown', leftArrowDomEvtDataStub );
+
+ sinon.assert.calledOnce( leftArrowDomEvtDataStub.preventDefault );
+ sinon.assert.calledOnce( leftArrowDomEvtDataStub.stopPropagation );
+
+ assertEqualMarkup( getModelData( model ), modelTable( [
+ [ '00', '01', '02' ],
+ [ '10[]', `11`, '12' ],
+ [ '20', '21', '22' ]
+ ] ) );
+ } );
+
+ it( 'should navigate to the cell on the right', () => {
+ setModelData( model, modelTable( [
+ [ '00', '01', '02' ],
+ [ '10', `11[]`, '12' ],
+ [ '20', '21', '22' ]
+ ] ) );
+
+ editor.editing.view.document.fire( 'keydown', rightArrowDomEvtDataStub );
+
+ sinon.assert.calledOnce( rightArrowDomEvtDataStub.preventDefault );
+ sinon.assert.calledOnce( rightArrowDomEvtDataStub.stopPropagation );
+
+ assertEqualMarkup( getModelData( model ), modelTable( [
+ [ '00', '01', '02' ],
+ [ '10', `11`, '[]12' ],
+ [ '20', '21', '22' ]
+ ] ) );
+ } );
+
+ it( 'should navigate to the cell above', () => {
+ setModelData( model, modelTable( [
+ [ '00', '01', '02' ],
+ [ '10', `[]11`, '12' ],
+ [ '20', '21', '22' ]
+ ] ) );
+
+ editor.editing.view.document.fire( 'keydown', upArrowDomEvtDataStub );
+
+ sinon.assert.calledOnce( upArrowDomEvtDataStub.preventDefault );
+ sinon.assert.calledOnce( upArrowDomEvtDataStub.stopPropagation );
+
+ assertEqualMarkup( getModelData( model ), modelTable( [
+ [ '00', '01[]', '02' ],
+ [ '10', `11`, '12' ],
+ [ '20', '21', '22' ]
+ ] ) );
+ } );
+
+ it( 'should navigate to the cell below', () => {
+ setModelData( model, modelTable( [
+ [ '00', '01', '02' ],
+ [ '10', `11[]`, '12' ],
+ [ '20', '21', '22' ]
+ ] ) );
+
+ editor.editing.view.document.fire( 'keydown', downArrowDomEvtDataStub );
+
+ sinon.assert.calledOnce( downArrowDomEvtDataStub.preventDefault );
+ sinon.assert.calledOnce( downArrowDomEvtDataStub.stopPropagation );
+
+ assertEqualMarkup( getModelData( model ), modelTable( [
+ [ '00', '01', '02' ],
+ [ '10', `11`, '12' ],
+ [ '20', '[]21', '22' ]
+ ] ) );
+ } );
+ } );
+
+ describe( 'horizontal line as only cell content (widget without $text positions)', () => {
+ beforeEach( () => {
+ setModelData( model, modelTable( [
+ [ '00', '01', '02' ],
+ [ '10', '[]', '12' ],
+ [ '20', '21', '22' ]
+ ] ) );
+ } );
+
+ it( 'should navigate to the cell on the left', () => {
+ editor.editing.view.document.fire( 'keydown', leftArrowDomEvtDataStub );
+
+ sinon.assert.calledOnce( leftArrowDomEvtDataStub.preventDefault );
+ sinon.assert.calledOnce( leftArrowDomEvtDataStub.stopPropagation );
+
+ assertEqualMarkup( getModelData( model ), modelTable( [
+ [ '00', '01', '02' ],
+ [ '10[]', '', '12' ],
+ [ '20', '21', '22' ]
+ ] ) );
+ } );
+
+ it( 'should navigate to the cell on the right', () => {
+ editor.editing.view.document.fire( 'keydown', rightArrowDomEvtDataStub );
+
+ sinon.assert.calledOnce( rightArrowDomEvtDataStub.preventDefault );
+ sinon.assert.calledOnce( rightArrowDomEvtDataStub.stopPropagation );
+
+ assertEqualMarkup( getModelData( model ), modelTable( [
+ [ '00', '01', '02' ],
+ [ '10', '', '[]12' ],
+ [ '20', '21', '22' ]
+ ] ) );
+ } );
+
+ it( 'should navigate to the cell above', () => {
+ editor.editing.view.document.fire( 'keydown', upArrowDomEvtDataStub );
+
+ sinon.assert.calledOnce( upArrowDomEvtDataStub.preventDefault );
+ sinon.assert.calledOnce( upArrowDomEvtDataStub.stopPropagation );
+
+ assertEqualMarkup( getModelData( model ), modelTable( [
+ [ '00', '01[]', '02' ],
+ [ '10', '', '12' ],
+ [ '20', '21', '22' ]
+ ] ) );
+ } );
+
+ it( 'should navigate to the cell below', () => {
+ editor.editing.view.document.fire( 'keydown', downArrowDomEvtDataStub );
+
+ sinon.assert.calledOnce( downArrowDomEvtDataStub.preventDefault );
+ sinon.assert.calledOnce( downArrowDomEvtDataStub.stopPropagation );
+
+ assertEqualMarkup( getModelData( model ), modelTable( [
+ [ '00', '01', '02' ],
+ [ '10', '', '12' ],
+ [ '20', '[]21', '22' ]
+ ] ) );
+ } );
+ } );
+
+ describe( 'two horizontal lines as only cell content (widget without $text positions)', () => {
+ it( 'should navigate to the cell on the left', () => {
+ setModelData( model, modelTable( [
+ [ '00', '01', '02' ],
+ [ '10', '[]', '12' ],
+ [ '20', '21', '22' ]
+ ] ) );
+
+ editor.editing.view.document.fire( 'keydown', leftArrowDomEvtDataStub );
+
+ sinon.assert.calledOnce( leftArrowDomEvtDataStub.preventDefault );
+ sinon.assert.calledOnce( leftArrowDomEvtDataStub.stopPropagation );
+
+ assertEqualMarkup( getModelData( model ), modelTable( [
+ [ '00', '01', '02' ],
+ [ '10[]', '', '12' ],
+ [ '20', '21', '22' ]
+ ] ) );
+ } );
+
+ it( 'should navigate to the cell on the right', () => {
+ setModelData( model, modelTable( [
+ [ '00', '01', '02' ],
+ [ '10', '[]', '12' ],
+ [ '20', '21', '22' ]
+ ] ) );
+
+ editor.editing.view.document.fire( 'keydown', rightArrowDomEvtDataStub );
+
+ sinon.assert.calledOnce( rightArrowDomEvtDataStub.preventDefault );
+ sinon.assert.calledOnce( rightArrowDomEvtDataStub.stopPropagation );
+
+ assertEqualMarkup( getModelData( model ), modelTable( [
+ [ '00', '01', '02' ],
+ [ '10', '', '[]12' ],
+ [ '20', '21', '22' ]
+ ] ) );
+ } );
+
+ it( 'should navigate to the cell above', () => {
+ setModelData( model, modelTable( [
+ [ '00', '01', '02' ],
+ [ '10', '[]', '12' ],
+ [ '20', '21', '22' ]
+ ] ) );
+
+ editor.editing.view.document.fire( 'keydown', upArrowDomEvtDataStub );
+
+ sinon.assert.calledOnce( upArrowDomEvtDataStub.preventDefault );
+ sinon.assert.calledOnce( upArrowDomEvtDataStub.stopPropagation );
+
+ assertEqualMarkup( getModelData( model ), modelTable( [
+ [ '00', '01[]', '02' ],
+ [ '10', '', '12' ],
+ [ '20', '21', '22' ]
+ ] ) );
+ } );
+
+ it( 'should navigate to the cell below', () => {
+ setModelData( model, modelTable( [
+ [ '00', '01', '02' ],
+ [ '10', '[]', '12' ],
+ [ '20', '21', '22' ]
+ ] ) );
+
+ editor.editing.view.document.fire( 'keydown', downArrowDomEvtDataStub );
+
+ sinon.assert.calledOnce( downArrowDomEvtDataStub.preventDefault );
+ sinon.assert.calledOnce( downArrowDomEvtDataStub.stopPropagation );
+
+ assertEqualMarkup( getModelData( model ), modelTable( [
+ [ '00', '01', '02' ],
+ [ '10', '', '12' ],
+ [ '20', '[]21', '22' ]
+ ] ) );
+ } );
+ } );
+ } );
+
+ describe( 'with selection not at the boundary of a cell', () => {
+ let editorElement, editor, model, styleElement;
+
+ beforeEach( async () => {
+ editorElement = global.document.createElement( 'div' );
+ global.document.body.appendChild( editorElement );
+
+ editor = await ClassicTestEditor.create( editorElement, {
+ plugins: [ Table, Paragraph, Image, ImageCaption, HorizontalLine ]
+ } );
+
+ model = editor.model;
+
+ styleElement = global.document.createElement( 'style' );
+ styleElement.type = 'text/css';
+ styleElement.appendChild( global.document.createTextNode(
+ `
+ * {
+ font-size: 12px !important;
+ font-family: serif !important;
+ margin: 0 !important;
+ padding: 0 !important;
+ border: 0 !important
+ }
+ td { width: 30px !important; }
+ tr:nth-child(2) td:nth-child(2) { width: 300px !important; }
+ `
+ ) );
+ global.document.querySelector( 'head' ).appendChild( styleElement );
+ } );
+
+ afterEach( async () => {
+ editorElement.remove();
+ styleElement.remove();
+ await editor.destroy();
+ } );
+
+ describe( 'simple cell text content', () => {
+ it( 'should not navigate to the cell on the left', () => {
+ setModelData( model, modelTable( [
+ [ '00', '01', '02' ],
+ [ '10', '1[]1', '12' ],
+ [ '20', '21', '22' ]
+ ] ) );
+
+ editor.editing.view.document.fire( 'keydown', leftArrowDomEvtDataStub );
+
+ sinon.assert.notCalled( leftArrowDomEvtDataStub.preventDefault );
+ sinon.assert.notCalled( leftArrowDomEvtDataStub.stopPropagation );
+ } );
+
+ it( 'should not navigate to the cell on the right', () => {
+ setModelData( model, modelTable( [
+ [ '00', '01', '02' ],
+ [ '10', '1[]1', '12' ],
+ [ '20', '21', '22' ]
+ ] ) );
+
+ editor.editing.view.document.fire( 'keydown', rightArrowDomEvtDataStub );
+
+ sinon.assert.notCalled( rightArrowDomEvtDataStub.preventDefault );
+ sinon.assert.notCalled( rightArrowDomEvtDataStub.stopPropagation );
+ } );
+
+ it( 'should not navigate to the cell above', () => {
+ setModelData( model, modelTable( [
+ [ '00', '01', '02' ],
+ [ '10', '1[]1', '12' ],
+ [ '20', '21', '22' ]
+ ] ) );
+
+ editor.editing.view.document.fire( 'keydown', upArrowDomEvtDataStub );
+
+ sinon.assert.calledOnce( upArrowDomEvtDataStub.preventDefault );
+ sinon.assert.calledOnce( upArrowDomEvtDataStub.stopPropagation );
+
+ assertEqualMarkup( getModelData( model ), modelTable( [
+ [ '00', '01', '02' ],
+ [ '10', '[]11', '12' ],
+ [ '20', '21', '22' ]
+ ] ) );
+ } );
+
+ it( 'should not navigate to the cell below', () => {
+ setModelData( model, modelTable( [
+ [ '00', '01', '02' ],
+ [ '10', '1[]1', '12' ],
+ [ '20', '21', '22' ]
+ ] ) );
+
+ editor.editing.view.document.fire( 'keydown', downArrowDomEvtDataStub );
+
+ sinon.assert.calledOnce( downArrowDomEvtDataStub.preventDefault );
+ sinon.assert.calledOnce( downArrowDomEvtDataStub.stopPropagation );
+
+ assertEqualMarkup( getModelData( model ), modelTable( [
+ [ '00', '01', '02' ],
+ [ '10', '11[]', '12' ],
+ [ '20', '21', '22' ]
+ ] ) );
+ } );
+ } );
+
+ describe( 'selection inside paragraph', () => {
+ const text = new Array( 20 ).fill( 0 ).map( () => 'word' ).join( ' ' );
+
+ it( 'should not navigate if caret is in the middle line of a text', () => {
+ setModelData( model, modelTable( [
+ [ '00', '01', '02' ],
+ [ '10', text + '[] ' + text, '12' ],
+ [ '20', '21', '22' ]
+ ] ) );
+
+ editor.editing.view.document.fire( 'keydown', upArrowDomEvtDataStub );
+
+ sinon.assert.notCalled( upArrowDomEvtDataStub.preventDefault );
+ sinon.assert.notCalled( upArrowDomEvtDataStub.stopPropagation );
+ } );
+
+ it( 'should move caret to beginning of cell content if caret is in the first line of a text', () => {
+ setModelData( model, modelTable( [
+ [ '00', '01', '02' ],
+ [ '10', 'word[] word' + text, '12' ],
+ [ '20', '21', '22' ]
+ ] ) );
+
+ editor.editing.view.document.fire( 'keydown', upArrowDomEvtDataStub );
+
+ sinon.assert.calledOnce( upArrowDomEvtDataStub.preventDefault );
+ sinon.assert.calledOnce( upArrowDomEvtDataStub.stopPropagation );
+
+ assertEqualMarkup( getModelData( model ), modelTable( [
+ [ '00', '01', '02' ],
+ [ '10', '[]word word' + text, '12' ],
+ [ '20', '21', '22' ]
+ ] ) );
+ } );
+
+ it( 'should move caret to end of cell content if caret is in the last line of a text', () => {
+ setModelData( model, modelTable( [
+ [ '00', '01', '02' ],
+ [ '10', text + 'word[] word', '12' ],
+ [ '20', '21', '22' ]
+ ] ) );
+
+ editor.editing.view.document.fire( 'keydown', downArrowDomEvtDataStub );
+
+ sinon.assert.calledOnce( downArrowDomEvtDataStub.preventDefault );
+ sinon.assert.calledOnce( downArrowDomEvtDataStub.stopPropagation );
+
+ assertEqualMarkup( getModelData( model ), modelTable( [
+ [ '00', '01', '02' ],
+ [ '10', text + 'word word[]', '12' ],
+ [ '20', '21', '22' ]
+ ] ) );
+ } );
+ } );
+
+ if ( !env.isGecko ) {
+ // These tests fails on Travis. They work correctly when started on local machine.
+ // Issue is probably related to text rendering and wrapping.
+
+ describe( 'with selection in the wrap area', () => {
+ const text = new Array( 10 ).fill( 0 ).map( () => 'word' ).join( ' ' );
+
+ it( 'should move the caret to end if the caret is after the last space in the line next to the last one', () => {
+ // This is also first position in the last line.
+ setModelData( model, modelTable( [
+ [ '00', '01', '02' ],
+ [ '10', text + ' []word word word', '12' ],
+ [ '20', '21', '22' ]
+ ] ) );
+
+ editor.editing.view.document.fire( 'keydown', downArrowDomEvtDataStub );
+
+ sinon.assert.calledOnce( downArrowDomEvtDataStub.preventDefault );
+ sinon.assert.calledOnce( downArrowDomEvtDataStub.stopPropagation );
+
+ assertEqualMarkup( getModelData( model ), modelTable( [
+ [ '00', '01', '02' ],
+ [ '10', text + ' word word word[]', '12' ],
+ [ '20', '21', '22' ]
+ ] ) );
+ } );
+
+ it( 'should move the caret to end if ther caret is at the last space in the line next to last one', () => {
+ setModelData( model, modelTable( [
+ [ '00', '01', '02' ],
+ [ '10', text + '[] word word word', '12' ],
+ [ '20', '21', '22' ]
+ ] ) );
+
+ editor.editing.view.document.fire( 'keydown', downArrowDomEvtDataStub );
+
+ sinon.assert.calledOnce( downArrowDomEvtDataStub.preventDefault );
+ sinon.assert.calledOnce( downArrowDomEvtDataStub.stopPropagation );
+
+ assertEqualMarkup( getModelData( model ), modelTable( [
+ [ '00', '01', '02' ],
+ [ '10', text + ' word word word[]', '12' ],
+ [ '20', '21', '22' ]
+ ] ) );
+ } );
+
+ it( 'should not move the caret if it\'s just before the last space in the line next to last one', () => {
+ setModelData( model, modelTable( [
+ [ '00', '01', '02' ],
+ [ '10', text.substring( 0, text.length - 1 ) + '[]d word word word', '12' ],
+ [ '20', '21', '22' ]
+ ] ) );
+
+ editor.editing.view.document.fire( 'keydown', downArrowDomEvtDataStub );
+
+ sinon.assert.notCalled( downArrowDomEvtDataStub.preventDefault );
+ sinon.assert.notCalled( downArrowDomEvtDataStub.stopPropagation );
+ } );
+ } );
+ }
+
+ describe( 'with multiple paragraphs of text', () => {
+ const text = new Array( 100 ).fill( 0 ).map( () => 'word' ).join( ' ' );
+
+ it( 'should not navigate if caret is in the middle of a line of text', () => {
+ setModelData( model, modelTable( [
+ [ '00', '01', '02' ],
+ [ '10', `${ text }[]${ text }foobar`, '12' ],
+ [ '20', '21', '22' ]
+ ] ) );
+
+ editor.editing.view.document.fire( 'keydown', upArrowDomEvtDataStub );
+
+ sinon.assert.notCalled( upArrowDomEvtDataStub.preventDefault );
+ sinon.assert.notCalled( upArrowDomEvtDataStub.stopPropagation );
+ } );
+
+ it( 'should move the caret to the beginning of a cell content if the caret is in the first line of text', () => {
+ setModelData( model, modelTable( [
+ [ '00', '01', '02' ],
+ [ '10', `word[]${ text }foobar`, '12' ],
+ [ '20', '21', '22' ]
+ ] ) );
+
+ editor.editing.view.document.fire( 'keydown', upArrowDomEvtDataStub );
+
+ sinon.assert.calledOnce( upArrowDomEvtDataStub.preventDefault );
+ sinon.assert.calledOnce( upArrowDomEvtDataStub.stopPropagation );
+
+ assertEqualMarkup( getModelData( model ), modelTable( [
+ [ '00', '01', '02' ],
+ [ '10', `[]word${ text }foobar`, '12' ],
+ [ '20', '21', '22' ]
+ ] ) );
+ } );
+
+ it( 'should not move the caret to the end of a cell content if the caret is not in the last line of text', () => {
+ setModelData( model, modelTable( [
+ [ '00', '01', '02' ],
+ [ '10', `${ text }word []wordfoobar`, '12' ],
+ [ '20', '21', '22' ]
+ ] ) );
+
+ editor.editing.view.document.fire( 'keydown', downArrowDomEvtDataStub );
+
+ sinon.assert.notCalled( downArrowDomEvtDataStub.preventDefault );
+ sinon.assert.notCalled( downArrowDomEvtDataStub.stopPropagation );
+ } );
+
+ it( 'should move the caret to end of a cell content if the caret is in the last line of text', () => {
+ setModelData( model, modelTable( [
+ [ '00', '01', '02' ],
+ [ '10', `foobar${ text }word []word`, '12' ],
+ [ '20', '21', '22' ]
+ ] ) );
+
+ editor.editing.view.document.fire( 'keydown', downArrowDomEvtDataStub );
+
+ sinon.assert.calledOnce( downArrowDomEvtDataStub.preventDefault );
+ sinon.assert.calledOnce( downArrowDomEvtDataStub.stopPropagation );
+
+ assertEqualMarkup( getModelData( model ), modelTable( [
+ [ '00', '01', '02' ],
+ [ '10', `foobar${ text }word word[]`, '12' ],
+ [ '20', '21', '22' ]
+ ] ) );
+ } );
+ } );
+
+ describe( 'with horizontal line widget', () => {
+ const text = new Array( 100 ).fill( 0 ).map( () => 'word' ).join( ' ' );
+
+ it( 'should not navigate if the caret is in the middle line of text', () => {
+ setModelData( model, modelTable( [
+ [ '00', '01', '02' ],
+ [ '10', `${ text }[]${ text }`, '12' ],
+ [ '20', '21', '22' ]
+ ] ) );
+
+ editor.editing.view.document.fire( 'keydown', upArrowDomEvtDataStub );
+
+ sinon.assert.notCalled( upArrowDomEvtDataStub.preventDefault );
+ sinon.assert.notCalled( upArrowDomEvtDataStub.stopPropagation );
+ } );
+
+ it( 'should move the caret to the beginning of cell content if the caret is in the first line of text', () => {
+ setModelData( model, modelTable( [
+ [ '00', '01', '02' ],
+ [ '10', `word[] ${ text }`, '12' ],
+ [ '20', '21', '22' ]
+ ] ) );
+
+ editor.editing.view.document.fire( 'keydown', upArrowDomEvtDataStub );
+
+ sinon.assert.calledOnce( upArrowDomEvtDataStub.preventDefault );
+ sinon.assert.calledOnce( upArrowDomEvtDataStub.stopPropagation );
+
+ assertEqualMarkup( getModelData( model ), modelTable( [
+ [ '00', '01', '02' ],
+ [ '10', `[]word ${ text }`, '12' ],
+ [ '20', '21', '22' ]
+ ] ) );
+ } );
+
+ it( 'should move the caret to the end of cell content if the caret is in the last line of text', () => {
+ setModelData( model, modelTable( [
+ [ '00', '01', '02' ],
+ [ '10', `${ text } word []word`, '12' ],
+ [ '20', '21', '22' ]
+ ] ) );
+
+ editor.editing.view.document.fire( 'keydown', downArrowDomEvtDataStub );
+
+ sinon.assert.calledOnce( downArrowDomEvtDataStub.preventDefault );
+ sinon.assert.calledOnce( downArrowDomEvtDataStub.stopPropagation );
+
+ assertEqualMarkup( getModelData( model ), modelTable( [
+ [ '00', '01', '02' ],
+ [ '10', `${ text } word word[]`, '12' ],
+ [ '20', '21', '22' ]
+ ] ) );
+ } );
+
+ it( 'should not move the caret to the end of cell content if widget is selected in middle of a cell content', () => {
+ const paragraph = `${ text }`;
+ const hr = '';
+
+ setModelData( model, modelTable( [
+ [ '00', '01', '02' ],
+ [ '10', `${ paragraph }[${ hr }]${ paragraph }${ hr }`, '12' ],
+ [ '20', '21', '22' ]
+ ] ) );
+
+ editor.editing.view.document.fire( 'keydown', downArrowDomEvtDataStub );
+
+ assertEqualMarkup( getModelData( model ), modelTable( [
+ [ '00', '01', '02' ],
+ [ '10', `${ paragraph }${ hr }[]${ text }${ hr }`, '12' ],
+ [ '20', '21', '22' ]
+ ] ) );
+ } );
+
+ it( 'should not move the caret to the end of cell content if widget is next to the selection', () => {
+ const paragraph = `${ text }`;
+ const hr = '';
+
+ setModelData( model, modelTable( [
+ [ '00', '01', '02' ],
+ [ '10', `${ paragraph }[]${ hr }`, '12' ],
+ [ '20', '21', '22' ]
+ ] ) );
+
+ editor.editing.view.document.fire( 'keydown', downArrowDomEvtDataStub );
+
+ assertEqualMarkup( getModelData( model ), modelTable( [
+ [ '00', '01', '02' ],
+ [ '10', `${ paragraph }[${ hr }]`, '12' ],
+ [ '20', '21', '22' ]
+ ] ) );
+ } );
+ } );
+
+ describe( 'contains image widget with caption and selection inside the caption', () => {
+ it( 'should not navigate to the cell on the left', () => {
+ setModelData( model, modelTable( [
+ [ '00', '01', '02' ],
+ [ '10', `foo[]11`, '12' ],
+ [ '20', '21', '22' ]
+ ] ) );
+
+ editor.editing.view.document.fire( 'keydown', leftArrowDomEvtDataStub );
+
+ sinon.assert.notCalled( leftArrowDomEvtDataStub.preventDefault );
+ sinon.assert.notCalled( leftArrowDomEvtDataStub.stopPropagation );
+ } );
+
+ it( 'should not navigate to the cell on the right', () => {
+ setModelData( model, modelTable( [
+ [ '00', '01', '02' ],
+ [ '10', `11[]foo`, '12' ],
+ [ '20', '21', '22' ]
+ ] ) );
+
+ editor.editing.view.document.fire( 'keydown', rightArrowDomEvtDataStub );
+
+ sinon.assert.notCalled( rightArrowDomEvtDataStub.preventDefault );
+ sinon.assert.notCalled( rightArrowDomEvtDataStub.stopPropagation );
+ } );
+
+ it( 'should not navigate to the cell above', () => {
+ setModelData( model, modelTable( [
+ [ '00', '01', '02' ],
+ [ '10', `foo1[]1`, '12' ],
+ [ '20', '21', '22' ]
+ ] ) );
+
+ editor.editing.view.document.fire( 'keydown', upArrowDomEvtDataStub );
+
+ sinon.assert.notCalled( upArrowDomEvtDataStub.preventDefault );
+ sinon.assert.notCalled( upArrowDomEvtDataStub.stopPropagation );
+ } );
+
+ it( 'should not navigate to the cell above but should select the image widget', () => {
+ setModelData( model, modelTable( [
+ [ '00', '01', '02' ],
+ [ '10', `1[]1foo`, '12' ],
+ [ '20', '21', '22' ]
+ ] ) );
+
+ editor.editing.view.document.fire( 'keydown', upArrowDomEvtDataStub );
+
+ sinon.assert.calledOnce( upArrowDomEvtDataStub.preventDefault );
+ sinon.assert.calledOnce( upArrowDomEvtDataStub.stopPropagation );
+
+ assertEqualMarkup( getModelData( model ), modelTable( [
+ [ '00', '01', '02' ],
+ [ '10', `[11]foo`, '12' ],
+ [ '20', '21', '22' ]
+ ] ) );
+ } );
+
+ it( 'should not navigate to the cell below when followed by a paragraph', () => {
+ setModelData( model, modelTable( [
+ [ '00', '01', '02' ],
+ [ '10', `1[]1foo`, '12' ],
+ [ '20', '21', '22' ]
+ ] ) );
+
+ editor.editing.view.document.fire( 'keydown', downArrowDomEvtDataStub );
+
+ sinon.assert.notCalled( downArrowDomEvtDataStub.preventDefault );
+ sinon.assert.notCalled( downArrowDomEvtDataStub.stopPropagation );
+ } );
+
+ it( 'should not navigate to the cell below but should select the image widget', () => {
+ setModelData( model, modelTable( [
+ [ '00', '01', '02' ],
+ [ '10', `foo1[]1`, '12' ],
+ [ '20', '21', '22' ]
+ ] ) );
+
+ editor.editing.view.document.fire( 'keydown', downArrowDomEvtDataStub );
+
+ sinon.assert.calledOnce( downArrowDomEvtDataStub.preventDefault );
+ sinon.assert.calledOnce( downArrowDomEvtDataStub.stopPropagation );
+
+ assertEqualMarkup( getModelData( model ), modelTable( [
+ [ '00', '01', '02' ],
+ [ '10', `foo[11]`, '12' ],
+ [ '20', '21', '22' ]
+ ] ) );
+ } );
+
+ it( 'should not navigate to the cell above but should select the image widget without caption', () => {
+ setModelData( model, modelTable( [
+ [ '00', '01', '02' ],
+ [ '10', `f[]oo`, '12' ],
+ [ '20', '21', '22' ]
+ ] ) );
+
+ editor.editing.view.document.fire( 'keydown', upArrowDomEvtDataStub );
+
+ sinon.assert.calledOnce( upArrowDomEvtDataStub.preventDefault );
+ sinon.assert.calledOnce( upArrowDomEvtDataStub.stopPropagation );
+
+ assertEqualMarkup( getModelData( model ), modelTable( [
+ [ '00', '01', '02' ],
+ [ '10', `[]foo`, '12' ],
+ [ '20', '21', '22' ]
+ ] ) );
+ } );
+
+ it( 'should not navigate to the cell below but should select the image widget without caption', () => {
+ setModelData( model, modelTable( [
+ [ '00', '01', '02' ],
+ [ '10', `f[]oo`, '12' ],
+ [ '20', '21', '22' ]
+ ] ) );
+
+ editor.editing.view.document.fire( 'keydown', downArrowDomEvtDataStub );
+
+ sinon.assert.calledOnce( downArrowDomEvtDataStub.preventDefault );
+ sinon.assert.calledOnce( downArrowDomEvtDataStub.stopPropagation );
+
+ assertEqualMarkup( getModelData( model ), modelTable( [
+ [ '00', '01', '02' ],
+ [ '10', `foo[]`, '12' ],
+ [ '20', '21', '22' ]
+ ] ) );
+ } );
+ } );
+ } );
+ } );
+
+ describe( 'for right-to-left content language', () => {
+ beforeEach( () => {
+ return VirtualTestEditor
+ .create( {
+ plugins: [ TableEditing, TableNavigation, TableSelection, Paragraph, ImageEditing, MediaEmbedEditing ],
+ language: 'ar'
+ } )
+ .then( newEditor => {
+ editor = newEditor;
+
+ model = editor.model;
+ modelRoot = model.document.getRoot();
+ tableSelection = editor.plugins.get( TableSelection );
+ } );
+ } );
+
+ describe( 'with the table cell selected from outside', () => {
+ beforeEach( () => {
+ setModelData( model, modelTable( [
+ [ '00', '01', '02' ],
+ [ '10', '11', '12' ],
+ [ '20', '21', '22' ]
+ ] ) );
+
+ tableSelection._setCellSelection(
+ modelRoot.getNodeByPath( [ 0, 1, 1 ] ),
+ modelRoot.getNodeByPath( [ 0, 1, 1 ] )
+ );
+ } );
+
+ it( 'should move to the cell on the right (visually flipped by the browser)', () => {
+ editor.editing.view.document.fire( 'keydown', leftArrowDomEvtDataStub );
+
+ sinon.assert.calledOnce( leftArrowDomEvtDataStub.preventDefault );
+ sinon.assert.calledOnce( leftArrowDomEvtDataStub.stopPropagation );
+
+ assertEqualMarkup( getModelData( model ), modelTable( [
+ [ '00', '01', '02' ],
+ [ '10', '11', '[]12' ],
+ [ '20', '21', '22' ]
+ ] ) );
+ } );
+
+ it( 'should move to the cell on the left (visually flipped by the browser)', () => {
+ editor.editing.view.document.fire( 'keydown', rightArrowDomEvtDataStub );
+
+ sinon.assert.calledOnce( rightArrowDomEvtDataStub.preventDefault );
+ sinon.assert.calledOnce( rightArrowDomEvtDataStub.stopPropagation );
+
+ assertEqualMarkup( getModelData( model ), modelTable( [
+ [ '00', '01', '02' ],
+ [ '10[]', '11', '12' ],
+ [ '20', '21', '22' ]
+ ] ) );
+ } );
+
+ it( 'should move to the cell above the selection', () => {
+ editor.editing.view.document.fire( 'keydown', upArrowDomEvtDataStub );
+
+ sinon.assert.calledOnce( upArrowDomEvtDataStub.preventDefault );
+ sinon.assert.calledOnce( upArrowDomEvtDataStub.stopPropagation );
+
+ assertEqualMarkup( getModelData( model ), modelTable( [
+ [ '00', '01[]', '02' ],
+ [ '10', '11', '12' ],
+ [ '20', '21', '22' ]
+ ] ) );
+ } );
+
+ it( 'should move to the cell below the selection', () => {
+ editor.editing.view.document.fire( 'keydown', downArrowDomEvtDataStub );
+
+ sinon.assert.calledOnce( downArrowDomEvtDataStub.preventDefault );
+ sinon.assert.calledOnce( downArrowDomEvtDataStub.stopPropagation );
+
+ assertEqualMarkup( getModelData( model ), modelTable( [
+ [ '00', '01', '02' ],
+ [ '10', '11', '12' ],
+ [ '20', '[]21', '22' ]
+ ] ) );
+ } );
+ } );
+ } );
+ } );
+} );