useListData
Manages state for an immutable list data structure, and provides convenience methods to update the data over time.
| install | yarn add @react-stately/data | 
|---|---|
| version | 3.1.2 | 
| usage | import {useListData} from '@react-stately/data' | 
Introduction#
React requires all data structures passed as props to be immutable. This enables them to be diffed correctly to determine what has changed since the last render. This can be challenging to accomplish from scratch in a performant way in JavaScript.
useListData helps manage an immutable list data structure, with helper methods to update the data in an efficient way.
Since the data is stored in React state, calling these methods to update the data automatically causes the component
to re-render accordingly.
In addition, useListData stores selection state for the list, based on unique item keys. This can be updated programmatically,
and is automatically updated when items are removed from the list.
API#
useListData<T>(
  (options: ListOptions<T>
)): ListData<T>Options#
| Name | Type | Description | 
| initialItems | T[] | Initial items in the list. | 
| initialSelectedKeys | 'all' | Iterable<Key> | The keys for the initially selected items. | 
| initialFilterText | string | The initial text to filter the list by. | 
| getKey | (
  (item: T
)) => Key | A function that returns a unique key for an item object. | 
| filter | (
  (item: T,
  , filterText: string
)) => boolean | A function that returns whether a item matches the current filter text. | 
Interface#
Properties
| Name | Type | Description | 
| items | T[] | The items in the list. | 
| selectedKeys | Selection | The keys of the currently selected items in the list. | 
| filterText | string | The current filter text. | 
Methods
| Method | Description | 
| setSelectedKeys(
  (keys: Selection
)): void | Sets the selected keys. | 
| setFilterText(
  (filterText: string
)): void | Sets the filter text. | 
| getItem(
  (key: Key
)): T | Gets an item from the list by key. | 
| insert(
  (index: number,
  , ...values: T[]
)): void | Inserts items into the list at the given index. | 
| insertBefore(
  (key: Key,
  , ...values: T[]
)): void | Inserts items into the list before the item at the given key. | 
| insertAfter(
  (key: Key,
  , ...values: T[]
)): void | Inserts items into the list after the item at the given key. | 
| append(
  (...values: T[]
)): void | Appends items to the list. | 
| prepend(
  (...values: T[]
)): void | Prepends items to the list. | 
| remove(
  (...keys: Key[]
)): void | Removes items from the list by their keys. | 
| removeSelectedItems(): void | Removes all items from the list that are currently in the set of selected items. | 
| move(
  (key: Key,
  , toIndex: number
)): void | Moves an item within the list. | 
| update(
  (key: Key,
  , newValue: T
)): void | Updates an item in the list. | 
Example#
To construct a list, pass an initial set of items along with a function to get a key for each item.
You can use the state returned by useListData to render a collection component.
This example renders a ListBox using the items managed by useListData. It uses the name property of each item
as the unique key for that item, and the items property as the children. In addition, it manages the selection state
for the listbox, which will automatically be updated when items are removed from the tree.
let list = useListData({
  initialItems: [{name: 'Aardvark'} {name: 'Kangaroo'} {name: 'Snake'}]
  initialSelectedKeys: ['Kangaroo']
  getKey: (item) => itemname
});
<ListBox
  items=listitems
  selectedKeys=listselectedKeys
  onSelectionChange=listsetSelectedKeys>
  (item) => <Item key=itemname>itemname</Item>
</ListBox>let list = useListData({
  initialItems: [
    {name: 'Aardvark'}
    {name: 'Kangaroo'}
    {name: 'Snake'}
  ]
  initialSelectedKeys: ['Kangaroo']
  getKey: (item) => itemname
});
<ListBox
  items=listitems
  selectedKeys=listselectedKeys
  onSelectionChange=listsetSelectedKeys>
  (item) => <Item key=itemname>itemname</Item>
</ListBox>let list = useListData({
  initialItems: [
    {name: 'Aardvark'}
    {name: 'Kangaroo'}
    {name: 'Snake'}
  ]
  initialSelectedKeys: [
    'Kangaroo'
  ]
  getKey: (item) =>
    itemname
});
<ListBox
  items=listitems
  selectedKeys=
    listselectedKeys
  
  onSelectionChange=
    listsetSelectedKeys
  >
  (item) => (
    <Item
      key=itemname>
      itemname
    </Item>
  )
</ListBox>Inserting items#
To insert a new item into the list, use the insert method or one of the other convenience methods.
Each of these methods also accepts multiple items, so you can insert multiple items at once.
// Insert an item after the first one
listinsert(1 {name: 'Horse'});
// Insert multiple items
listinsert(1 {name: 'Horse'} {name: 'Giraffe'});
// Insert an item after the first one
listinsert(1 {name: 'Horse'});
// Insert multiple items
listinsert(1 {name: 'Horse'} {name: 'Giraffe'});
// Insert an item after the first one
listinsert(1 {
  name: 'Horse'
});
// Insert multiple items
listinsert(
  1
  {name: 'Horse'}
  {name: 'Giraffe'}
);
// Insert an item before another item
listinsertAfter('Kangaroo' {name: 'Horse'});
// Insert multiple items before another item
listinsertAfter('Kangaroo' {name: 'Horse'} {name: 'Giraffe'});
// Insert an item before another item
listinsertAfter('Kangaroo' {name: 'Horse'});
// Insert multiple items before another item
listinsertAfter(
  'Kangaroo'
  {name: 'Horse'}
  {name: 'Giraffe'}
);
// Insert an item before another item
listinsertAfter(
  'Kangaroo'
  {name: 'Horse'}
);
// Insert multiple items before another item
listinsertAfter(
  'Kangaroo'
  {name: 'Horse'}
  {name: 'Giraffe'}
);
// Insert an item after another item
listinsertAfter('Kangaroo' {name: 'Horse'});
// Insert multiple items after another item
listinsertAfter('Kangaroo' {name: 'Horse'} {name: 'Giraffe'});
// Insert an item after another item
listinsertAfter('Kangaroo' {name: 'Horse'});
// Insert multiple items after another item
listinsertAfter(
  'Kangaroo'
  {name: 'Horse'}
  {name: 'Giraffe'}
);
// Insert an item after another item
listinsertAfter(
  'Kangaroo'
  {name: 'Horse'}
);
// Insert multiple items after another item
listinsertAfter(
  'Kangaroo'
  {name: 'Horse'}
  {name: 'Giraffe'}
);
// Append an item
listappend({name: 'Horse'});
// Append multiple items
listappend({name: 'Horse'} {name: 'Giraffe'});
// Append an item
listappend({name: 'Horse'});
// Append multiple items
listappend({name: 'Horse'} {name: 'Giraffe'});
// Append an item
listappend({
  name: 'Horse'
});
// Append multiple items
listappend(
  {name: 'Horse'}
  {name: 'Giraffe'}
);
// Prepend an item
listprepend({name: 'Horse'});
// Prepend multiple items
listprepend({name: 'Horse'} {name: 'Giraffe'});
// Prepend an item
listprepend({name: 'Horse'});
// Prepend multiple items
listprepend({name: 'Horse'} {name: 'Giraffe'});
// Prepend an item
listprepend({
  name: 'Horse'
});
// Prepend multiple items
listprepend(
  {name: 'Horse'}
  {name: 'Giraffe'}
);
Removing items#
// Remove an item
listremove('Kangaroo');
// Remove multiple items
listremove('Kangaroo' 'Snake');
// Remove an item
listremove('Kangaroo');
// Remove multiple items
listremove('Kangaroo' 'Snake');
// Remove an item
listremove('Kangaroo');
// Remove multiple items
listremove(
  'Kangaroo'
  'Snake'
);
// Remove all selected items
listremoveSelectedItems();
// Remove all selected items
listremoveSelectedItems();
// Remove all selected items
listremoveSelectedItems();
Moving items#
listmove('Snake' 0);
listmove('Snake' 0);
listmove('Snake' 0);
Updating items#
listupdate('Snake' {name: 'Rattle Snake'});
listupdate('Snake' {name: 'Rattle Snake'});
listupdate('Snake' {
  name: 'Rattle Snake'
});