Skip to content

Latest commit

 

History

History
471 lines (316 loc) · 18.7 KB

usage.md

File metadata and controls

471 lines (316 loc) · 18.7 KB

Usage

LRUD handles the registering of new navigation nodes, the handling of key events, and the emitting of events based on internal state changes.

These nodes represent 2D space in an "abstract" manner - there is no visual component to LRUD, but its data represents spatial grids, lists, and items.

For some examples of common desired behaviour and how to achieve it, see our common recipes.

Creating a new instance

const { Lrud } = require('Lrud')

const navigation = new Lrud()

Registering a node

A node can be added to the navigation tree by calling navigation.registerNode with the id of the node and any node options

navigation.registerNode('root', {})

The very first node that is registered will become the root node. From here, any node that is registered without a parent will automatically become a child of the root node.

navigation
    .registerNode('root', { orientation: 'horizontal' })
    .registerNode('alpha', { isFocusable: true })

// ...is the same as

navigation
    .registerNode('root', { orientation: 'horizontal' })
    .registerNode('alpha', { isFocusable: true, parent: 'root' })

Nodes must be registered with unique IDs. If registerNode() is called with an ID that has already been registered against the navigation instance, an exception will be thrown.

Registration options

Most options affect behaviour for when LRUD is handling key events and assigning focus.

orientation

"vertical" | "horizontal"

Any node that has children should be registered with an orientation. A node of vertical orientation will handle up/down key events while horizontal will handle left/right key events.

navigation
    .registerNode('row', { orientation: 'horizontal' })
    .registerNode('item-1', { parent: 'row', isFocusable: true })
    .registerNode('item-2', { parent: 'row', isFocusable: true })
    .registerNode('item-3', { parent: 'row', isFocusable: true })

If focus was on item-2, a left keypress will put focus to item-1, and right keypress will put focus to item-3. This is because the row is set to orientation: horizontal. If the row was set to orientation: vertical, it would respond to key presses of up and down, respectively.

If orientation is not defined than focus can not be moved within that node. This is a very powerful feature, that allows creating closed boxes, from which focus can not "jump out". It's the best to think about modal popups with semi transparent overlay here. It may contain Ok/Cancel buttons and focus must be moved only around those buttons. The rest of the page is still visible in the background and LRUD navigation tree may stay untouched. See the Recipe 5 - Error Modal Popup.

isWrapping

boolean

Used in conjunction with orientation to make a list wrap at the top/bottom or left/right depending on orientation.

navigation
    .registerNode('row', { orientation: 'horizontal', isWrapping: true })
    .registerNode('item-1', { parent: 'row', isFocusable: true })
    .registerNode('item-2', { parent: 'row', isFocusable: true })
    .registerNode('item-3', { parent: 'row', isFocusable: true })

In the above example, if the user was focused on item-3, and LRUD handled an event with a direction of right, usually focus would remain on item-3. However, because the row is set to isWrapping: true, focus will actually reset to item-1.

index

number

The zero-based index of the node, relative to its siblings. Index is kept coherent and compact. If index value is provided, then the child is inserted at that position and all other children indices, that are greater or equal to provided index, are shifted up by one. If index of the registering child is greater that current size of children list or is not given at all, it will be set as the next index under its parent.

navigation
    .registerNode('root', { isIndexCoherent: true })
    .registerNode('A', { parent: 'root' }) // order: |A|
    .registerNode('B', { parent: 'root' }) // order: A, |B|
    .registerNode('C', { parent: 'root' }) // order: A, B, |C|
    .registerNode('D', { parent: 'root' }) // order: A, B, C, |D|

// ...gives the same result as

navigation
    .registerNode('A', { parent: 'root', index: 0 }) // order: |A|
    .registerNode('C', { parent: 'root', index: 1 }) // order: A, |C|
    .registerNode('B', { parent: 'root', index: 1 }) // order: A, |B|, C
    .registerNode('D', { parent: 'root', index: 9 }) // order: A, B, C, |D|

index is used when calculating the "next" or "previous" node in a list.

isIndexAlign

boolean

To be used in conjunction with orientation to give a node index alignment functionality.

Index alignment ensures that when focus would move from one of this nodes descendants to another, LRUD will attempt to ensure that the focused node matches the index of the node that was left, e.g to make 2 lists behave as a "grid".

For further details, see the docs on index alignment.

indexRange

number[]

An array with 2 elements. Value [0] is the lower bound of matching indexes, [1] is the upper bound.

Used in conjunction with isIndexAlign behaviour, allows a node to replicate the effects of a "column span" by assuming the role of multiple indexes relative to its siblings.

For further details, see the docs on index alignment.

isStopPropagate

boolean

Allows dealing with a situation when focusable parent contains focusable children.

By default, the leaf is picked to be focused, "the deepest" focusable candidate. In such case parent node is simply not a hindrance. Setting isStopPropagate to true inverts this behaviour. Parent node grabs focus instead of passing it to the children.

navigation
    .registerNode('row-1', { orientation: 'horizontal', isFocusable: true })
    .registerNode('row-2', { orientation: 'horizontal', isFocusable: true })
    .registerNode('row-2-item-1', { parent: 'row-2', isFocusable: true })
    .registerNode('row-3', { orientation: 'horizontal', isFocusable: true, isStopPropagate: true })
    .registerNode('row-3-item-2', { parent: 'row-3', isFocusable: true })

In the above example, if the user was focused on row-1, and LRUD handled an event with a direction of down, than row-2-item-1 will be focused, because by default row-2 passes focus to the children. Next event with a direction of down will move focus to row-3. This is because, row-3 has isStopPropagate set to true which allowed him to grab the focus instead of passing it to the row-3-item-2, even if it's a focusable leaf.


Several functions can also be given as registration options to a node. These functions will be called at specific state change points for the node. See our Process Lifecycles doc for further details on the "lifecycle" of a move event in LRUD.

onFocus

function

If given, the onFocus function will be called when the node gets focussed on.

onBlur

function

If given, the onBlur function will be called when the node if the node had focus and a new node gains focus.

onSelect

function

If given, the onSelect function will be called when the node is focused and a key event of "ENTER" is handled.

onActive

function

If given, the onActive function will be called when the node is made active by either itself or one of its descendants gaining focus.

onInactive

function

If given, the onInactive function will be called when the node was active and due to an updated focus, is no longer active.

onLeave

function

If given, the onLeave function will be called when the node was focused and the handling of a key event means the node is no longer focused.

onEnter

function

If given, the onEnter function will be called when the node was not focused and the handling of a key event means the node is now focused.

shouldCancelLeave

function

If given, the shouldCancelLeave function will be called when a move is being processed, and the node is being left. If shouldCancelLeave returns true, the move will be cancelled.

shouldCancelEnter

function

If given, the shouldCancelEnter function will be called when a move is being processed, and the node is being entered. If shouldCancelEnter returns true, the move will be cancelled.

onLeaveCancelled

function

If given, the onLeaveCancelled function will be called if this node has a matching shouldCancelLeave, and when that function returns true.

onEnterCancelled

function

If given, the onEnterCancelled function will be called if this node has a matching shouldCancelEnter, and when that function returns true.

Unregistering a node

A node can be removed from the navigation tree by calling navigation.unregisterNode() with the id of the node.

Unregistering a node will also remove all of its children and trigger events correctly. Overrides pointing to the removed node (and any of its direct and indirect children) will be removed as well.

If an unregister causes the current focused node to be removed, focus will be moved to the last node that could be focused. This also works when unregistering a branch.

Unregistering the root node will cause the tree to become empty and also remove all overrides that have been set (see Overrides, below).

Unregistering Options

A config object can be given to unregisterNode(<nodeId>, <unregisterOptions>) to force specific behaviour.

  • forceRefocus:boolean When true, the default behaviour of finding a new node to focus on if unregistering the current focused node will continue to work as normal. This value also defaults to true. Pass as false to stop the auto-refocus behaviour. Remember, if you are unregistering the current focused node, and passing forceRefocus as false, you need to manually call assignFocus() afterwards or the user will be left in limbo!

Moving a node

A node can be moved from its current parent to the new parent by calling navigation.moveNode() with the id of the node to be moved and the id of the new parent node.

This method has few advantages over regular unregisterNode/registerNode operation:

  • It maintains the currently focused node. If moved node (or its direct or indirect child) is a currently focused node, then it will stay focused.
  • It maintains overrides pointing to the moved node or its direct and indirect children.

Moving Options

A config object can be given to moveNode(<nodeId>, <newParentId>, <moveNodeOptions>) to force specific behaviour.

  • index:number Defines position at which node should be inserted into new parent's children list. Despite the given value, index of new parent's children is kept coherent and compact.
  • maintainIndex:boolean When true, the node will be inserted into new parent's children list at the same position as it was under old parent, if possible. Otherwise, the node will be appended to new parent's children list. Despite the node's current index value, index of new parent's children is kept coherent and compact. If index value is specified in options, the maintainIndex takes no effect, it's ignored.

Assigning Focus

You can give focus to a particular node by calling navigation.assignFocus() with the node id

navigation.assignFocus('list')

If the node that has been assigned focus is not focusable, LRUD will attempt to find the first active child of the node that is focusable, and focus on that instead.

Handling Key Events

Once focus has been assigned against the LRUD instance, LRUD can begin handling key events.

Every key event represents a user moving the "cursor"/"focus" in a given direction. The direction is based on the event.keyCode value - LRUD maintains an internal mapping of keyCode values to semantic directions.

You can pass key events into LRUD using the navigation.handleKeyEvent function:

document.onkeydown = function (event) {
  navigation.handleKeyEvent(event)
}

Handling Key Events Options

A config object can be given to handleKeyEvent(<event>, <handleKeyEventOptions>) to force specific behaviour.

  • forceFocus:boolean When true, if there's no focused node, then LRUD will attempt to find a first focusable candidate (the node that is focusable or contains at least one child that is a focusable candidate). Such node, when found, will be automatically focused and focus state of navigation tree is auto-initialized. The default value is false.

Events

LRUD emits events in response to key events. See the TAL docs for an explanation of 'focused' and 'active' nodes. Each of these callbacks is called with the node that changed state.

  • navigation.on('focus', function) - Focus was given to a node.
  • navigation.on('blur', function) - Focus was taken from a node.
  • navigation.on('active', function) - The node has become active.
  • navigation.on('inactive', function) - The node has become inactive.
  • navigation.on('select', function) - The current focused node was selected.
  • navigation.on('cancelled', function) - A movement has been cancelled.

A special event of move is emitted after handling a key event.

  • navigation.on('move', function) - Triggered when focus is changed within a list

The move event callback is called with a move event in the following shape:

{
    leave: <node>       // the node that was focused that we're now leaving
    enter: <node>       // the node that is now focused that we're entering
    offset: -1 : 1      // 1 if direction was RIGHT or DOWN, -1 if direction was LEFT or UP
}

Common usages for handling this move event include changing the style of a given DOM node to match a "focus" style, or to handle a DOM animation between the leave and enter nodes.

navigation.on('move', moveEvent => {
  const focusedDomNode = document.getElementById(moveEvent.enter.id);
  focusedDomNode.classList.add('focused');
})

To unregister an event callback, simply call the .off() method

navigation.off('focus', function);

Overrides

LRUD supports an override system, for times when correct product/UX behaviour requires focus to change in a way that is not strictly in accordance with the structure of the navigation tree.

New overrides can be registered with navigation.registerOverride(<sourceNodeId>, <targetNodeId>, <direction>, <registerOverrideOptions>), where options may contain parameters:

  • forceOverride:boolean When true, the existing override from source node in given direction will be overwritten.

To unregister override its enough to call navigation.unregisterOverride(<sourceNodeId>, <direction>).

The override below represents that, when LRUD is bubbling its key event and when it hits the box-1 node, and direction of travel is down, STOP the propagation of the bubble event and focus directly on box-2.

navigation.registerOverride(
  'box-1',   // the ID to trigger the override on
  'box-2',   // the ID of the node we want to focus on
  'down'     // the direction of travel in order for the override to trigger
)

Modifying Node Focusability

Ability of a node to be focused can be modified using navigation.setNodeFocusable

navigation.registerNode('root', { isFocusable: true })
navigation.setNodeFocusable('root', false)

Tree and Partial Tree Insertion & Registering

LRUD supports the ability to register an entire tree at once.

const instance = new Lrud();
const tree = {
  id: 'root',
  orientation: 'horizontal',
  children: [
    { id: 'alpha', isFocusable: true },
    { id: 'beta', isFocusable: true },
    { id: 'charlie', isFocusable: true },
  ]
}

instance.registerTree(tree);
// `instance` now has the above tree registered, and has correctly setup active children, indexes, etc. 

insertTree() and nested tree registration

LRUD also supports the ability to register a tree/insert a tree into an already existing node branch.

If no parent is given on the top level node of the passed tree, the tree will be inserted under the root node, as per standard registerNode() behaviour.

Otherwise, if a parent is given, the tree will be inserted under that parent.

Inserting a tree under the root node

const instance = new Lrud();
instance
  .registerNode('root', { orientation: 'horizontal' })
  .registerNode('alpha', { isFocusable: true })
  .registerNode('beta', { isFocusable: true })

const tree = {
  id: 'charlie',
  orientation: 'vertical',
  children: [
    { id: 'charlie_1', isFocusable: true },
    { id: 'charlie_2', isFocusable: true },
  ]
}
instance.registerTree(tree);
/*
the full tree of `instance` now looks like:
{
  id: 'root',
  orientation: 'horizontal',
  children: [
    { id: 'alpha', isFocusable: true }
    { id: 'beta', isFocusable: true }
    {
      id: 'charlie',
      orientation: 'vertical'
      children: [
        { id: 'charlie_1', isFocusable: true }
        { id: 'charlie_2', isFocusable: true }
      ]
    }
  ]
}
*/

Inserting a tree under a specified branch

const instance = new Lrud();
instance
  .registerNode('root', { orientation: 'horizontal' })
  .registerNode('alpha', { isFocusable: true })
  .registerNode('beta', { orientation: 'vertical' })

const tree = {
  id: 'charlie',
  orientation: 'vertical',
  parent: 'beta',
  children: [
    { id: 'charlie_1', isFocusable: true },
    { id: 'charlie_2', isFocusable: true },
  ]
}
instance.registerTree(tree);
/*
the full tree of `instance` now looks like:
{
  id: 'root',
  orientation: 'horizontal',
  children: [
    { id: 'alpha', isFocusable: true }
    {
      id: 'beta',
      orientation: 'vertical',
      children: [
        {
          id: 'charlie',
          orientation: 'vertical'
          children: [
            { id: 'charlie_1', isFocusable: true }
            { id: 'charlie_2', isFocusable: true }
          ]
        }
      ]
    }
  ]
}
*/

F.A.Q

Q: A node that should be focusabled is never receiving focus - whats happening?

A: Ensure that the parent nodes, etc. have the correct orientation in order to be able to jump inbetween nodes.

Q: All my parents have orientations, everything is setup in the navigation tree, and I STILL can't focus on the node I expect.

A: Ensure the node has either isFocusable: true or a selectAction registered against it. A node needs either one of these in order to be considered "focusable". isFocusable is prioritised over selectAction so a node with isFocusable: false will never be focusable.

Q: What is the different between onBlur/onFocus and onLeave/onEnter?

A: onBlur and onFocus can/will be called at any point in time for when a node loses focus. This includes a direct call to assignFocus() from user land. On the other hand, onLeave and onEnter will only be called via the result of a handleKeyEvent()