UNPKG

react-treeselectjs

Version:
220 lines (190 loc) 11.3 kB
# Treeselect JS React component It is a wrapper for https://www.npmjs.com/package/treeselectjs A multi-select js component with nested options. - Full key support (ArrowUp, ArrowDown, Space, ArrowLeft, ArrowRight, Enter) - Screen sensitive direction - Typescript support Build data: - react-treeselectjs.mjs 2.14 kB │ gzip: 0.83 kB - react-treeselectjs.umd.js 1.91 kB │ gzip: 0.80 kB - react-treeselectjs.css 7.08 kB │ gzip: 1.41 kB **Live Demo:** https://dipson88.github.io/treeselectjs/ ![Example img](https://github.com/dipson88/react-treeselectjs/blob/main/treeselectjs.png?raw=true) ### Getting Started ```bash npm install --save react-treeselectjs ``` Import react-treeselectjs (ES) ``` import Treeselect from 'react-treeselectjs' import 'react-treeselectjs/dist/react-treeselectjs.css' ``` Import treeselectjs (UMD) ``` <!DOCTYPE html> <html> <head> <script src="https://unpkg.com/react@18/umd/react.development.js" crossorigin></script> <script src="https://unpkg.com/react-dom@18/umd/react-dom.development.js" crossorigin></script> <script src="https://unpkg.com/@babel/standalone/babel.min.js"></script> <script src="https://cdn.jsdelivr.net/npm/treeselectjs@0.13.0/dist/treeselectjs.umd.js"></script> <script src="https://cdn.jsdelivr.net/npm/react-treeselectjs@0.7.0/dist/react-treeselectjs.umd.js"></script> <link rel="stylesheet" href="https://cdn.jsdelivr.net/npm/react-treeselectjs@0.7.0/dist/react-treeselectjs.css" /> </head> <body> <div id="root"></div> <script type="text/babel"> const root = ReactDOM.createRoot(document.getElementById('root')); root.render(<ReactTreeselect />); </script> </body> </html> ``` All Available Imports ``` import Treeselect, { DirectionType, IconsType, OptionType, TreeselectProps, TreeselectValue } from 'react-treeselectjs' ``` Example ``` import React, { useCallback } from 'react' import Treeselect, { TreeselectValue } from 'react-treeselectjs' import 'react-treeselectjs/dist/react-treeselectjs.css' const options = [ { name: 'England', value: 1, children: [ { name: 'London', value: 2, children: [ { name: 'Chelsea', value: 3, children: [] }, { name: 'West End', value: 4, children: [] } ] }, { name: 'Brighton', value: 5, children: [] } ] }, { name: 'France', value: 6, children: [ { name: 'Paris', value: 7, children: [] }, { name: 'Lyon', value: 8, children: [] } ] } ] function App() { // Also for value type, you can use value: string | number | (string | number)[] | null | undefined const onInput = useCallback((value: TreeselectValue) => { console.log('onInput', value) }, []) return ( <div> <Treeselect options={options} value={[4, 7, 8]} onInput={onInput} > {/* Use slot if you need */} <a className="treeselect-demo__slot" href="">Click!</a>' </Treeselect> </div> ) } export default App ``` --- ### Props #### Core props Name | Type (default) | Description ------------- | ------------- | ------------- **value** | Array[String \| Number] ([]) | An array of `value` from `options` prop. This value will be selected on load of the treeselect. The `value` changes if you check/uncheck checkboxes or remove tags from the input. **options** | Array[Object] ([]) | It is an array of objects ```{name: String, value: String \| Number, disabled?: Boolean, htmlAttr?: object, isGroupSelectable?: boolean, children: [] }```, where children are the same array of objects. Do not use duplicated `value` field. But you can use duplicated names. [Read more](#option-description). **disabled** | Boolean (false) | List will be disabled. **id** | String ('') | id attribute for the accessibility. **ariaLabel** | String ('') | ariaLabel attribute for the accessibility. **isSingleSelect** | Boolean (false) | Converts multi-select to the single value select. Checkboxes will be removed. You should pass only one id instead of array of values. Also you can set **showTags** to false. It helps to show treeselect as a dropdown. **isGroupedValue** | Boolean (false) | Return groups if they selected instead of separate ids. Treeselect returns only leaves ids by default. **isIndependentNodes** | Boolean (false) | All nodes in treeselect work as an independent entity. Check/uncheck action ignore children/parent updates workflow. Disabled nodes ignore children/parent workflow as well. **rtl** | Boolean (false) | RTL mode. **isBoostedRendering** | Boolean (false) | ***Experimental*** - Improves list rendering performance by using visibility-based optimizations and IntersectionObserver. Useful for efficiently rendering large lists. #### List settings props Name | Type (default) | Description ------------- | ------------- | ------------- **disabledBranchNode** | Boolean (false) | It is impossible to select groups. You can select only leaves. **openLevel** | Number (0) | All groups will be opened to this level. **appendToBody** | Boolean (false) | List will be appended to the body instead of the input container. **alwaysOpen** | Boolean (false) | List will be always opened. You can use it for comfortable style changing. If you what to use it as an opened list, turn `staticList` to `true`. **staticList** | Boolean (false) | Add the list as a static DOM element. List doesn't overlap content. This prop will be ignored if you use `appendToBody`. **emptyText** | String ('No results found...') | A empty list text. **direction** | String (auto) | A force direction for the list. Supported values: `auto`, `top`, `bottom`. **expandSelected** | Boolean (false) | All groups which have checked values will be expanded on the init. **saveScrollPosition** | Boolean (true) | The list saves the last scroll position before close. If you open the list your scroll will be on the previous position. If you set the value to `false` - the scroll will have position 0 and the first item will be focused every time. **listClassName** | String ('') | A class name for list. Useful to change styles for `appendToBody` mode. #### Input settings props Name | Type (default) | Description ------------- | ------------- | ------------- **showTags** | Boolean (true) | Selected values look like tags. The false value shows results as '{count} elements selected'. You can change text if you use `tagsCountText` prop. For one selected element, you will see a name of this element. **tagsCountText** | String ('elements selected') | This text will be shown if you use 'showTags'. This text will be inserted after the count of the selected elements - ```'{count} {tagsCountText}'```. **showCount** | Boolean (false) | Shows count of children near the group's name. **tagsSortFn** | `(a: TagsSortItem, b: TagsSortItem) => number` \| `null` (null) | Defines the sorting order for tags in the input field.<br>`TagsSortItem` - `{ value: ValueOptionType, name: string }`. **clearable** | Boolean (true) | Clear icon is available. **searchable** | Boolean (true) | Search is available. **placeholder** | String ('Search...') | Placeholder text. **grouped** | Boolean (true) | Show groups in the input and group leafs if all group selected. #### Callback props Name | Type (default) | Description ------------- | ------------- | ------------- **onInput** | (value) => void (undefined) | Returns selected values, action is triggered on change the list value. **onOpen** | (value) => void (undefined) | Returns selected values, action is triggered on opening the list. **onClose** | (value) => void (undefined) | Returns selected values, action is triggered on closing the list. **onNameChange** | (name) => void (undefined) | Returns selected name inside the input, action is triggered on on change the list. **onSearch** | String | Returns entered search value, action is triggered on change search value during the typing. You can try create something like autocomplete with help of this event. **onOpenCloseGroup** | (groupId: ValueOptionType, isClosed: boolean) => void (undefined) | Returns groupId and closed/open status of this group, action is triggered on open/close group in the list. #### Additional props Name | Type (default) | Description ------------- | ------------- | ------------- **iconElements** | Object({ arrowUp, ... }) | Object contains all svg icons. You can use HTMLElement or a String to reset values from the default Object. Object: ```iconElements: { arrowUp, arrowDown, arrowRight, attention, clear, cross, check, partialCheck }```. After reset of icon you have to update styles if it is necessary, use `alwaysOpen` prop for more comfortable work with styles changes. --- ### Option description Is is description of the one option of the [`options`](#core-props) prop: Name | Type | Description ------------- | ------------- | ------------- **value** | String \| Number (required!) | It is a value of the node. **It should be unique!** **name** | String (required!) | It is the name of the node. **Can be duplicated.** **disabled** | Boolean (optional) | The node will be disabled. It is an optional field, you can skip it if no need to work with disabled values. **htmlAttr** | Object (optional) | The object of the HTML attributes, the value of the object should be a String type. These attributes will be merged into the node HTML tag. **isGroupSelectable** | Boolean (optional - true) | Determines whether groups are selectable. This behavior is similar to the disabledBranchNode prop but applies specifically to groups. It does not affect regular (non-group) items. **children** | {name: String, value: String, disabled?: Boolean, htmlAttr?: object, children: [] }[] | Children are the same array of objects. --- ### Notes 1) If you want to change the padding of the element you can use CSS selector. I've added **'group'** and **'level'** attributes, but you have to use **!important**. 2) Do not use **duplicated** values for the options. You will see a error with duplicated values. But you can use duplicated names. 3) **Value** prop inside the **options** prop should be a **String** or **Number**. 4) If you use **isSingleSelect** prop, you should pass only a single **value** without an array. 5) If you use **isSingleSelect** prop, you can set **showTags** to false. It helps to show treeselect as a dropdown. Also you can disable selecting of group's nodes with help of **disabledBranchNode**. 6) If you use a large list of options and see a problem with performance, try to use **isBoostedRendering** prop. ### Support You can buy me a coffee if you want to support my work. Thank you! <a href="https://www.buymeacoffee.com/dipson88" target="_blank"><img src="https://cdn.buymeacoffee.com/buttons/v2/default-yellow.png" alt="Buy Me A Coffee" style="height: 60px !important;width: 217px !important;" ></a>