Skip to main content

Keyboard Drag and Drop

Keyboard-controlled Drag and Drop for assistive technologies

SourceView Source
TypesView Types
Importimport { keyboard-drag-and-dropFeature } from "@headless-tree/core
Type DocumentationConfigurationStateTree InstanceItem Instance

The Keyboard Drag and Drop feature provides a way to use the drag-and-drop functionality of the tree using only the keyboard. This allows you to implement drag-based functionality while staying compliant with accessibility standards, by providing the default drag capability with keyboard-controlled hotkeys and a visually hidden assistive text indicating the drag-process to screen readers.

Dragging items with Keyboard controls Demo

Setup

The Keyboard Drag and Drop feature mostly builds upon the normal Drag and Drop feature, integrating it mostly just means to add it as feature to the tree configuration and it will provide the necessary functionality for keyboard-controlled drag-and-drop.

Note that this feature needs both the dragAndDropFeature and the hotkeysCoreFeature to work properly.

features: [
syncDataLoaderFeature,
selectionFeature,
hotkeysCoreFeature,
dragAndDropFeature,
keyboardDragAndDropFeature,
],

One important aspect of the Keyboard Drag and Drop feature is the assistive text, which is used to inform screen readers about the drag-and-drop process.

The @headless-tree/react package exposes a AssistiveTreeDescription react component, that will automatically render the english assistive text for you in a visually hidden way. It is recommended to put this component inside the tree root container, or otherwise reference it with aria-tags in the tree.

<div {...tree.getContainerProps()} className="tree">
<AssistiveTreeDescription tree={tree} />
{/* ...tree items... */}
</div>

Alternatively, you can build this text yourself by using the data available in the tree state that you can retrieve with tree.getState(). You can use treeState.assistiveDndState of type AssistiveDndState to determine in which state the drag-and-drop process is, and treeState.dnd for details about what is being dragged and where the user is dropping it.

enum AssistiveDndState {
None,
Started,
Dragging,
Completed,
Aborted,
}

Dragging items out of the tree with keyboard

Selecting items inside the tree, and then dragging them out of the tree into a third-party component is a core-functionality of the normal DnD Feature. You can find out more about that in the Guide on dragging in and out of trees.

Since there is no actual drag-event happening in keyboard-bound situations, you need to implement custom logic to accept a keyboard-controlled drag initiated in a Headless Tree instance.

You can detect whether the user is dragging items via keyboard by inspecting tree.getState().assistiveDndState, if it is either AssistiveDndState.Started or AssistiveDndState.Dragging. In the logic that accepts such a drag, you can get information about the items that are being dragged by inspecting tree.getState().dnd?.draggedItems. You can then handle the dragged items (e.g. by removing them from the tree with removeItemsFromParents()) and complete the keyboard drag with tree.stopKeyboardDrag().

<button
onClick={async () => {
const draggedItems = tree.getState().dnd.draggedItems;
if (draggedItems) {
await onCompleteForeignDrop(draggedItems);
alert(draggedItems.map((item) => item.getItemName()));
}
tree.stopKeyboardDrag();
}}
>
Accept dragged items from tree
</button>

Keyboard Drag tree items out of tree demo

Dragging foreign objects inside the tree with keyboard

Similarly, moving data from outside the tree into it via a drag event is also a core-functionality of the normal DnD Feature. You can find out more about that in the Guide on dragging in and out of trees.

Again, this functionality works mostly out-of-the-box with headless tree, but the initiation of a keyboard-bound drag event that can be accepted by Headless Tree needs to be done externally. If the user triggers something that you consider a keyboard-bound drag event within the boundaries of your application, you can inform Headless Tree that this is happening by calling tree.startKeyboardDragOnForeignObject(dataTransfer); with a custom dataTranfer object, that you handle the same way as with normal foreign drag events. You can create this object manually.


<button
onClick={() => {
const dataTransfer = new DataTransfer();
dataTransfer.setData("text/plain", "hello world");
tree.startKeyboardDragOnForeignObject(dataTransfer);
}}
>
Initiate keyboard drag on a foreign object
</button>

Keyboard Drag foreign object inside of tree demo