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

t/ckeditor5/416d: Implemented automatic items grouping in the ToolbarView component #523

Merged
merged 45 commits into from
Oct 15, 2019
Merged
Changes from 1 commit
Commits
Show all changes
45 commits
Select commit Hold shift + click to select a range
6fb0736
Implemented ToolbarView#enableWrappedItemsGroupping method to improve…
oleq Oct 16, 2018
4107b70
Merge branch 'master' into t/ckeditor5/416b
oleq Aug 2, 2019
ebdc70c
Some improvements to the PoC.
oleq Aug 2, 2019
b203e12
Another PoC.
oleq Aug 5, 2019
26fcfea
Stabilized the toolbar grouping PoC.
oleq Aug 6, 2019
6708e71
Changed the template of the toolbar for better grouping feature. Code…
oleq Aug 6, 2019
9ac351f
Further improvements and code refactoring.
oleq Aug 7, 2019
68cc9c5
Fixed the broken keyboard navigation in the toolbar.
oleq Aug 7, 2019
2125464
Re-enabled toolbar keyboard navigation on arrowup and down.
oleq Aug 8, 2019
a762b19
Merge branch 'master' into t/ckeditor5/416d
oleq Sep 9, 2019
823c550
Added a RTL UI support in the automatic toolbar grouping feature.
oleq Sep 9, 2019
f6eed57
Aligned API to changes in the "resizeobserver" module.
oleq Sep 9, 2019
86fb29e
Code refactoring and docs in the ToolbarView class.
oleq Sep 11, 2019
f4c45d9
Docs: Improved ToolbarView docs.
oleq Sep 11, 2019
ab1884b
Code refac: Renamd ToolbarView method. Updated docs.
oleq Sep 11, 2019
bc55b5b
Code refactoring and tests in the ToolbarView class.
oleq Sep 12, 2019
51562ed
Tests and code refactoring in ToolbarView.
oleq Sep 13, 2019
95b325f
More tests in ToolbarView.
oleq Sep 13, 2019
b382915
Code refactoring.
oleq Sep 16, 2019
a197586
Tests: Added manual test for automatic ToolbarView items grouping.
oleq Sep 16, 2019
a15d3df
Made sure the toolbar does not cause console.warns when detached from…
oleq Sep 16, 2019
9ede6a4
Tests: Added a test to check if #updateGroupedItems is called after t…
oleq Sep 16, 2019
75e5106
Docs: Improvements in ToolbarView docs.
oleq Sep 16, 2019
197de63
Merge branch 'master' into t/ckeditor5/416d
Reinmar Oct 3, 2019
6ec572d
Merge branch 'master' into t/ckeditor5/416d
oleq Oct 7, 2019
0bbaebd
Code refactoring.
oleq Oct 7, 2019
b8b9f59
Code refactoring.
oleq Oct 8, 2019
64e58bc
Docs: Updated the documentation after the refactoring.
oleq Oct 8, 2019
c69262b
Docs: Updated the toolbar.js file documentation after the refactoring.
oleq Oct 8, 2019
53150ae
Code refactoring - used composition in the ToolbarView.
oleq Oct 10, 2019
7411974
Code refactoring.
oleq Oct 10, 2019
f0b258d
Merge branch 'master' into t/ckeditor5/416e
oleq Oct 10, 2019
fafcbb2
Reverted changes made to ToolbarView#fillFromConfig.
oleq Oct 10, 2019
5423147
Code refactoring.
oleq Oct 10, 2019
647fc1f
Docs.
oleq Oct 14, 2019
bfb3487
Removed obsolete lock property.
oleq Oct 14, 2019
f53d979
Docs.
oleq Oct 14, 2019
7c7cb17
Tests: Stabilized ToolbarView tests after refactoring.
oleq Oct 14, 2019
436464b
Code refactoring.
oleq Oct 14, 2019
384c2e7
Code refactoring.
oleq Oct 14, 2019
2a2fe4a
Merge branch 't/ckeditor5/416e' into t/ckeditor5/416d
oleq Oct 14, 2019
f87eaca
Update src/toolbar/toolbarview.js
oleq Oct 15, 2019
ed780e5
Code refactoring: Removed obsolete ToolbarView docs.
oleq Oct 15, 2019
d900dd7
Docs: Improvements in ToolbarView docs.
oleq Oct 15, 2019
8c47c13
Code refactoring: Improved docs, re-organized some methods, fixed min…
oleq Oct 15, 2019
File filter

Filter by extension

Filter by extension

Conversations
Failed to load comments.
Loading
Jump to
Jump to file
Failed to load files.
Loading
Diff view
Diff view
Prev Previous commit
Next Next commit
More tests in ToolbarView.
  • Loading branch information
oleq committed Sep 13, 2019
commit 95b325f7c364c4d5f5df81d48b040c2280f6ca36
55 changes: 26 additions & 29 deletions src/toolbar/toolbarview.js
Original file line number Diff line number Diff line change
@@ -333,14 +333,18 @@ export default class ToolbarView extends View {
}

/**
* Focuses the first focusable in {@link #items}.
* Focuses the first focusable in element in the toolbar.
*/
focus() {
this._componentsFocusCycler.focusFirst();
if ( this._itemsFocusCycler.first ) {
this._itemsFocusCycler.focusFirst();
} else if ( this.groupedItems && this.groupedItems.length ) {
this.groupedItemsDropdown.focus();
}
}

/**
* Focuses the last focusable in {@link #items}.
* Focuses the last focusable element in the toolbar.
*/
focusLast() {
const last = this._componentsFocusCycler.last;
@@ -554,16 +558,19 @@ export default class ToolbarView extends View {
* +────────────────────────────────────────────────────────────────────────────────────┘
*/
_focusNext( keyEvtData, cancel ) {
const itemsFocusCycler = this._itemsFocusCycler;
const componentsFocusCycler = this._componentsFocusCycler;

if ( this.itemsView.focusTracker.isFocused ) {
if ( !this._itemsFocusCycler.next || this._itemsFocusCycler.next === this._itemsFocusCycler.first ) {
this._componentsFocusCycler.focusNext();
if ( !itemsFocusCycler.next || itemsFocusCycler.next === itemsFocusCycler.first ) {
componentsFocusCycler.focusNext();
} else {
this._itemsFocusCycler.focusNext();
itemsFocusCycler.focusNext();
}

cancel();
} else {
this._componentsFocusCycler.focusNext();
componentsFocusCycler.focusNext();

cancel();
}
@@ -588,24 +595,21 @@ export default class ToolbarView extends View {
* +────────────────────────────────────────────────────────────────────────────────────┘
*/
_focusPrevious( keyEvtData, cancel ) {
const itemsFocusCycler = this._itemsFocusCycler;
const componentsFocusCycler = this._componentsFocusCycler;

if ( this.itemsView.focusTracker.isFocused ) {
if ( !this._itemsFocusCycler.next || this._itemsFocusCycler.previous === this._itemsFocusCycler.last ) {
if ( this.groupedItems && this.groupedItems.length ) {
this._componentsFocusCycler.focusLast();
} else {
this._itemsFocusCycler.focusPrevious();
}
const hasGroupedItems = this.groupedItems && this.groupedItems.length;

if ( hasGroupedItems && ( !itemsFocusCycler.previous || itemsFocusCycler.previous === itemsFocusCycler.last ) ) {
componentsFocusCycler.focusLast();
} else {
this._itemsFocusCycler.focusPrevious();
itemsFocusCycler.focusPrevious();
}

cancel();
} else {
if ( this._componentsFocusCycler.previous === this.itemsView ) {
this._itemsFocusCycler.focusLast();
} else {
this._componentsFocusCycler.focusPrevious();
}
itemsFocusCycler.focusLast();

cancel();
}
@@ -631,11 +635,11 @@ export default class ToolbarView extends View {

// TODO: Consider debounce.
this._resizeObserver = getResizeObserver( ( [ entry ] ) => {
if ( !previousWidth || previousWidth.width !== entry.contentRect.width ) {
if ( !previousWidth || previousWidth !== entry.contentRect.width ) {
this.updateGroupedItems();
}

previousWidth = entry.contentRect.width;
previousWidth = entry.contentRect.width;
}
} );

this._resizeObserver.observe( this.element );
@@ -759,11 +763,4 @@ class ToolbarItemsView extends View {
focus() {
this._focusCycler.focusFirst();
}

/**
* Focuses the last focusable in {@link #items}.
*/
focusLast() {
this._focusCycler.focusLast();
}
}
101 changes: 93 additions & 8 deletions tests/toolbar/toolbarview.js
Original file line number Diff line number Diff line change
@@ -3,7 +3,7 @@
* For licensing, see LICENSE.md or https://ckeditor.com/legal/ckeditor-oss-license
*/

/* global document, Event, console */
/* global document, Event, console, setTimeout */

import ToolbarView from '../../src/toolbar/toolbarview';
import ToolbarSeparatorView from '../../src/toolbar/toolbarseparatorview';
@@ -470,10 +470,8 @@ describe( 'ToolbarView', () => {
view.items.add( itemC );
view.items.add( itemD );

view.shouldGroupWhenFull = true;

// The dropdown shows up.
view.updateGroupedItems();
view.shouldGroupWhenFull = true;
sinon.spy( view.groupedItemsDropdown, 'destroy' );

view.element.style.width = '500px';
@@ -524,6 +522,20 @@ describe( 'ToolbarView', () => {

sinon.assert.calledOnce( view.items.get( 1 ).focus );
} );

it( 'if no items the first focusable of #items in DOM', () => {
document.body.appendChild( view.element );
view.element.style.width = '10px';

view.items.add( focusable() );
view.items.add( focusable() );

view.shouldGroupWhenFull = true;
sinon.spy( view.groupedItemsDropdown, 'focus' );

view.focus();
sinon.assert.calledOnce( view.groupedItemsDropdown.focus );
} );
} );

describe( 'focusLast()', () => {
@@ -790,6 +802,15 @@ describe( 'ToolbarView', () => {
} );

describe( 'automatic toolbar grouping (#shouldGroupWhenFull = true)', () => {
beforeEach( () => {
document.body.appendChild( view.element );
view.element.style.width = '200px';
} );

afterEach( () => {
view.element.remove();
} );

it( 'updates the UI as new #items are added', () => {
sinon.spy( view, 'updateGroupedItems' );
sinon.assert.notCalled( view.updateGroupedItems );
@@ -810,12 +831,76 @@ describe( 'ToolbarView', () => {
sinon.assert.calledTwice( view.updateGroupedItems );
} );

it( 'updates the UI when the toolbar is being resized (expanding)', () => {
// TODO
it( 'updates the UI when the toolbar is being resized (expanding)', done => {
view.items.add( focusable() );
view.items.add( focusable() );
view.items.add( focusable() );
view.items.add( focusable() );
view.items.add( focusable() );

view.element.style.width = '200px';
view.shouldGroupWhenFull = true;

expect( view.items ).to.have.length( 1 );
expect( view.groupedItems ).to.have.length( 4 );

view.element.style.width = '500px';

setTimeout( () => {
expect( view.items ).to.have.length( 5 );
expect( view.groupedItems ).to.have.length( 0 );

done();
}, 100 );
} );

it( 'updates the UI when the toolbar is being resized (narrowing)', done => {
view.items.add( focusable() );
view.items.add( focusable() );
view.items.add( focusable() );
view.items.add( focusable() );
view.items.add( focusable() );

view.element.style.width = '500px';
view.shouldGroupWhenFull = true;

expect( view.items ).to.have.length( 5 );
expect( view.groupedItems ).to.be.null;

view.element.style.width = '200px';

setTimeout( () => {
expect( view.items ).to.have.length( 1 );
expect( view.groupedItems ).to.have.length( 4 );

done();
}, 100 );
} );

it( 'updates the UI when the toolbar is being resized (narrowing)', () => {
// TODO
it( 'does not react to changes in height', done => {
view.element.style.width = '500px';
view.element.style.height = '200px';

view.items.add( focusable() );
view.items.add( focusable() );
view.items.add( focusable() );
view.items.add( focusable() );
view.items.add( focusable() );

view.shouldGroupWhenFull = true;
sinon.spy( view, 'updateGroupedItems' );

expect( view.items ).to.have.length( 5 );
expect( view.groupedItems ).to.be.null;

setTimeout( () => {
view.element.style.height = '500px';

setTimeout( () => {
sinon.assert.calledOnce( view.updateGroupedItems );
done();
}, 100 );
}, 100 );
} );
} );
} );