Skip to content

Commit

Permalink
Expand selection on consecutive Meta+A presses
Browse files Browse the repository at this point in the history
  • Loading branch information
ellatrix committed May 15, 2018
1 parent 0f74b95 commit a2c5559
Show file tree
Hide file tree
Showing 4 changed files with 76 additions and 19 deletions.
15 changes: 0 additions & 15 deletions editor/components/editor-global-keyboard-shortcuts/index.js
Original file line number Diff line number Diff line change
@@ -1,8 +1,3 @@
/**
* External dependencies
*/
import { first, last } from 'lodash';

/**
* WordPress dependencies
*/
Expand All @@ -17,19 +12,12 @@ class EditorGlobalKeyboardShortcuts extends Component {
constructor() {
super( ...arguments );

this.selectAll = this.selectAll.bind( this );
this.undoOrRedo = this.undoOrRedo.bind( this );
this.save = this.save.bind( this );
this.deleteSelectedBlocks = this.deleteSelectedBlocks.bind( this );
this.clearMultiSelection = this.clearMultiSelection.bind( this );
}

selectAll( event ) {
const { uids, onMultiSelect } = this.props;
event.preventDefault();
onMultiSelect( first( uids ), last( uids ) );
}

undoOrRedo( event ) {
const { onRedo, onUndo } = this.props;

Expand Down Expand Up @@ -72,7 +60,6 @@ class EditorGlobalKeyboardShortcuts extends Component {
<Fragment>
<KeyboardShortcuts
shortcuts={ {
[ rawShortcut.primary( 'a' ) ]: this.selectAll,
[ rawShortcut.primary( 'z' ) ]: this.undoOrRedo,
[ rawShortcut.primaryShift( 'z' ) ]: this.undoOrRedo,
backspace: this.deleteSelectedBlocks,
Expand Down Expand Up @@ -111,7 +98,6 @@ export default compose( [
withDispatch( ( dispatch ) => {
const {
clearSelectedBlock,
multiSelect,
redo,
undo,
removeBlocks,
Expand All @@ -120,7 +106,6 @@ export default compose( [

return {
clearSelectedBlock,
onMultiSelect: multiSelect,
onRedo: redo,
onUndo: undo,
onRemove: removeBlocks,
Expand Down
28 changes: 25 additions & 3 deletions editor/components/writing-flow/index.js
Original file line number Diff line number Diff line change
@@ -1,7 +1,7 @@
/**
* External dependencies
*/
import { overEvery, find, findLast, reverse } from 'lodash';
import { overEvery, find, findLast, reverse, first, last } from 'lodash';

/**
* WordPress dependencies
Expand All @@ -16,6 +16,7 @@ import {
isVerticalEdge,
placeCaretAtHorizontalEdge,
placeCaretAtVerticalEdge,
isFullySelected,
} from '@wordpress/utils';
import { withSelect, withDispatch } from '@wordpress/data';

Expand All @@ -32,7 +33,7 @@ import {
* Module Constants
*/

const { UP, DOWN, LEFT, RIGHT } = keycodes;
const { UP, DOWN, LEFT, RIGHT, is } = keycodes;

/**
* Given an element, returns true if the element is a tabbable text field, or
Expand Down Expand Up @@ -179,7 +180,7 @@ class WritingFlow extends Component {
}

onKeyDown( event ) {
const { hasMultiSelection } = this.props;
const { hasMultiSelection, onMultiSelect, blocks } = this.props;

const { keyCode, target } = event;
const isUp = keyCode === UP;
Expand All @@ -201,6 +202,25 @@ class WritingFlow extends Component {
}

if ( ! isNav ) {
const activeElement = document.activeElement;

// Set right before the meta+a combination can be pressed.
if ( is.primary( event ) ) {
this.isFullySelected = isFullySelected( activeElement );
}

if ( is.primary( event, 'a' ) ) {
// In the case of contentEditable, we want to know the earlier value
// because the selection will have already been set by TinyMCE.
if ( activeElement.isContentEditable ? this.isFullySelected : isFullySelected( activeElement ) ) {
onMultiSelect( first( blocks ), last( blocks ) );
event.preventDefault();
}

// Set in case the meta key doesn't get released.
this.isFullySelected = isFullySelected( activeElement );
}

return;
}

Expand Down Expand Up @@ -278,6 +298,7 @@ export default compose( [
getFirstMultiSelectedBlockUid,
getLastMultiSelectedBlockUid,
hasMultiSelection,
getBlockOrder,
} = select( 'core/editor' );

const selectedBlockUID = getSelectedBlockUID();
Expand All @@ -292,6 +313,7 @@ export default compose( [
selectedFirstUid: getFirstMultiSelectedBlockUid(),
selectedLastUid: getLastMultiSelectedBlockUid(),
hasMultiSelection: hasMultiSelection(),
blocks: getBlockOrder(),
};
} ),
withDispatch( ( dispatch ) => {
Expand Down
34 changes: 34 additions & 0 deletions utils/dom.js
Original file line number Diff line number Diff line change
Expand Up @@ -395,6 +395,40 @@ export function documentHasSelection() {
return range && ! range.collapsed;
}

/**
* Check wether the contents of the element have been fully selected.
* Returns true if there is no possibility of full selection.
*
* @param {Element} element The element to check.
*
* @return {boolean} True if fully selected, false if not.
*/
export function isFullySelected( element ) {
if ( includes( [ 'INPUT', 'TEXTAREA' ], element.nodeName ) ) {
return element.selectionStart === 0 && element.value.length === element.selectionEnd;
}

if ( ! element.isContentEditable ) {
return true;
}

const selection = window.getSelection();
const range = selection.rangeCount ? selection.getRangeAt( 0 ) : null;

if ( ! range ) {
return true;
}

const { startContainer, endContainer, startOffset, endOffset } = range;

return (
startContainer === element &&
endContainer === element &&
startOffset === 0 &&
endOffset === element.childNodes.length
);
}

/**
* Given a DOM node, finds the closest scrollable container node.
*
Expand Down
18 changes: 17 additions & 1 deletion utils/keycodes.js
Original file line number Diff line number Diff line change
Expand Up @@ -12,7 +12,7 @@
/**
* External dependencies
*/
import { get, mapValues } from 'lodash';
import { get, mapValues, includes } from 'lodash';

export const BACKSPACE = 8;
export const TAB = 9;
Expand Down Expand Up @@ -90,3 +90,19 @@ export const displayShortcut = mapValues( modifiers, ( modifier ) => {
return shortcut.replace( /⌘\+([A-Z0-9])$/g, '⌘$1' );
};
} );

export const is = mapValues( modifiers, ( getModifiers ) => {
return ( event, character, _isMac = isMacOS ) => {
const mods = getModifiers( _isMac );

if ( ! mods.every( ( key ) => event[ `${ key }Key` ] ) ) {
return false;
}

if ( ! character ) {
return includes( mods, event.key.toLowerCase() );
}

return event.key === character;
};
} );

0 comments on commit a2c5559

Please sign in to comment.