Skip to content
This repository has been archived by the owner on Jun 26, 2020. It is now read-only.

Commit

Permalink
Changed: Add insertTable dropdown.
Browse files Browse the repository at this point in the history
  • Loading branch information
jodator committed May 28, 2018
1 parent c0d5b1c commit 88c5fcc
Show file tree
Hide file tree
Showing 5 changed files with 415 additions and 17 deletions.
28 changes: 21 additions & 7 deletions src/tableui.js
Original file line number Diff line number Diff line change
Expand Up @@ -8,11 +8,12 @@
*/

import Plugin from '@ckeditor/ckeditor5-core/src/plugin';
import ButtonView from '@ckeditor/ckeditor5-ui/src/button/buttonview';
import { addListToDropdown, createDropdown } from '@ckeditor/ckeditor5-ui/src/dropdown/utils';
import Model from '@ckeditor/ckeditor5-ui/src/model';
import Collection from '@ckeditor/ckeditor5-utils/src/collection';

import InsertTableView from './ui/inserttableview';

import tableIcon from './../theme/icons/table.svg';
import tableColumnIcon from './../theme/icons/table-column.svg';
import tableRowIcon from './../theme/icons/table-row.svg';
Expand All @@ -33,22 +34,35 @@ export default class TableUI extends Plugin {

editor.ui.componentFactory.add( 'insertTable', locale => {
const command = editor.commands.get( 'insertTable' );
const buttonView = new ButtonView( locale );
const dropdownView = createDropdown( locale );

buttonView.bind( 'isEnabled' ).to( command );
dropdownView.bind( 'isEnabled' ).to( command );

buttonView.set( {
dropdownView.buttonView.set( {
icon: tableIcon,
label: 'Insert table',
tooltip: true
} );

buttonView.on( 'execute', () => {
editor.execute( 'insertTable' );
const insertTableView = new InsertTableView( locale );

insertTableView.delegate( 'execute' ).to( dropdownView );

dropdownView.buttonView.on( 'open', () => {
// Reset the chooser before showing it to the user.
insertTableView.rows = 0;
insertTableView.columns = 0;
} );

dropdownView.on( 'execute', () => {
editor.execute( 'insertTable', { rows: insertTableView.rows, columns: insertTableView.columns } );

editor.editing.view.focus();
} );

return buttonView;
dropdownView.panelView.children.add( insertTableView );

return dropdownView;
} );

editor.ui.componentFactory.add( 'tableColumn', locale => {
Expand Down
193 changes: 193 additions & 0 deletions src/ui/inserttableview.js
Original file line number Diff line number Diff line change
@@ -0,0 +1,193 @@
/**
* @license Copyright (c) 2003-2018, CKSource - Frederico Knabben. All rights reserved.
* For licensing, see LICENSE.md.
*/

/**
* @module table/ui/inserttableview
*/

import View from '@ckeditor/ckeditor5-ui/src/view';
import FocusTracker from '@ckeditor/ckeditor5-utils/src/focustracker';
import FocusCycler from '@ckeditor/ckeditor5-ui/src/focuscycler';

import './../../theme/inserttable.css';

/**
* The table size view.
*
* It renders a 10x10 grid to choose inserted table size.
*
* @extends module:ui/view~View
*/
export default class InsertTableView extends View {
/**
* @inheritDoc
*/
constructor( locale ) {
super( locale );

const bind = this.bindTemplate;

/**
* Collection of the table size box items.
*
* @readonly
* @member {module:ui/viewcollection~ViewCollection}
*/
this.items = this.createCollection();

/**
* Tracks information about DOM focus in the list.
*
* @readonly
* @member {module:utils/focustracker~FocusTracker}
*/
this.focusTracker = new FocusTracker();

/**
* Helps cycling over focusable {@link #items} in the toolbar.
*
* @readonly
* @protected
* @member {module:ui/focuscycler~FocusCycler}
*/
this._focusCycler = new FocusCycler( {
focusables: this.items,
focusTracker: this.focusTracker,
keystrokeHandler: this.keystrokes,
actions: {
// Navigate toolbar items backwards using the arrow[left,up] keys.
focusPrevious: [ 'arrowleft', 'arrowup' ],

// Navigate toolbar items forwards using the arrow[right,down] keys.
focusNext: [ 'arrowright', 'arrowdown' ]
}
} );

/**
* Currently selected number of rows of a new table.
*
* @observable
* @member {Number} #rows
*/
this.set( 'rows', 0 );

/**
* Currently selected number of columns of a new table.
*
* @observable
* @member {Number} #columns
*/
this.set( 'columns', 0 );

/**
* The label text displayed under the boxes.
*
* @observable
* @member {String} #label
*/
this.bind( 'label' )
.to( this, 'columns', this, 'rows', ( columns, rows ) => `${ rows } x ${ columns }` );

this.setTemplate( {
tag: 'div',
attributes: {
class: [ 'ck' ]
},

children: [
{
tag: 'div',
attributes: {
class: [ 'ck-table-size-choose-box-container' ]
},
children: this.items
},
{
tag: 'div',
attributes: {
class: [ 'ck-table-size-label' ]
},
children: [
{
text: bind.to( 'label' )
}
]
}
],

on: {
click: bind.to( () => {
this.fire( 'execute' );
} )
}
} );

for ( let i = 0; i < 100; i++ ) {
const view = new TableSizeChooser();

view.on( 'over', () => {
const row = parseInt( i / 10 );
const column = i % 10;

this.set( 'rows', row + 1 );
this.set( 'columns', column + 1 );
} );

this.items.add( view );
}

this.on( 'change:columns', () => {
this._updateItems();
} );

this.on( 'change:rows', () => {
this._updateItems();
} );
}

/**
* Enables current table size selection depending on rows & columns set.
*
* @private
*/
_updateItems() {
const row = this.rows - 1;
const column = this.columns - 1;

this.items.map( ( item, index ) => {
const itemRow = parseInt( index / 10 );
const itemColumn = index % 10;

if ( itemRow <= row && itemColumn <= column ) {
item.set( 'isOn', true );
} else {
item.set( 'isOn', false );
}
} );
}
}

class TableSizeChooser extends View {
constructor( locale ) {
super( locale );

const bind = this.bindTemplate;

this.set( 'isOn', false );

this.setTemplate( {
tag: 'div',
attributes: {
class: [
'ck-table-size-choose-box',
bind.if( 'isOn', 'ck-on' )
]
},
on: {
mouseover: bind.to( 'over' )
}
} );
}
}
38 changes: 28 additions & 10 deletions tests/tableui.js
Original file line number Diff line number Diff line change
Expand Up @@ -7,7 +7,6 @@

import ClassicTestEditor from '@ckeditor/ckeditor5-core/tests/_utils/classictesteditor';
import { _clear as clearTranslations, add as addTranslations } from '@ckeditor/ckeditor5-utils/src/translation-service';
import ButtonView from '@ckeditor/ckeditor5-ui/src/button/buttonview';
import testUtils from '@ckeditor/ckeditor5-core/tests/_utils/utils';

import TableEditing from '../src/tableediting';
Expand Down Expand Up @@ -54,31 +53,50 @@ describe( 'TableUI', () => {
insertTable = editor.ui.componentFactory.create( 'insertTable' );
} );

it( 'should register insertTable buton', () => {
expect( insertTable ).to.be.instanceOf( ButtonView );
expect( insertTable.isOn ).to.be.false;
expect( insertTable.label ).to.equal( 'Insert table' );
expect( insertTable.icon ).to.match( /<svg / );
it( 'should register insertTable button', () => {
expect( insertTable ).to.be.instanceOf( DropdownView );
expect( insertTable.buttonView.label ).to.equal( 'Insert table' );
expect( insertTable.buttonView.icon ).to.match( /<svg / );
} );

it( 'should bind to insertTable command', () => {
const command = editor.commands.get( 'insertTable' );

command.isEnabled = true;
expect( insertTable.isOn ).to.be.false;
expect( insertTable.isEnabled ).to.be.true;
expect( insertTable.buttonView.isOn ).to.be.false;
expect( insertTable.buttonView.isEnabled ).to.be.true;

command.isEnabled = false;
expect( insertTable.isEnabled ).to.be.false;
expect( insertTable.buttonView.isEnabled ).to.be.false;
} );

it( 'should execute insertTable command on button execute event', () => {
const executeSpy = testUtils.sinon.spy( editor, 'execute' );

const tableSizeView = insertTable.panelView.children.get( 0 );

tableSizeView.rows = 2;
tableSizeView.columns = 7;

insertTable.fire( 'execute' );

sinon.assert.calledOnce( executeSpy );
sinon.assert.calledWithExactly( executeSpy, 'insertTable' );
sinon.assert.calledWithExactly( executeSpy, 'insertTable', { rows: 2, columns: 7 } );
} );

it( 'should reset rows & columns on dropdown open', () => {
const tableSizeView = insertTable.panelView.children.get( 0 );

expect( tableSizeView.rows ).to.equal( 0 );
expect( tableSizeView.columns ).to.equal( 0 );

tableSizeView.rows = 2;
tableSizeView.columns = 2;

insertTable.buttonView.fire( 'open' );

expect( tableSizeView.rows ).to.equal( 0 );
expect( tableSizeView.columns ).to.equal( 0 );
} );
} );

Expand Down
Loading

0 comments on commit 88c5fcc

Please sign in to comment.