@dxos/react-list 0.8.4-main.c4373fc → 0.8.4-main.cb12b3f963
This diff represents the content of publicly available package versions that have been released to one of the supported registries. The information contained in this diff is provided for informational purposes only and reflects changes between package versions as they appear in their respective public registries.
- package/README.md +134 -1
- package/dist/lib/browser/index.mjs +61 -76
- package/dist/lib/browser/index.mjs.map +3 -3
- package/dist/lib/browser/meta.json +1 -1
- package/dist/lib/node-esm/index.mjs +61 -76
- package/dist/lib/node-esm/index.mjs.map +3 -3
- package/dist/lib/node-esm/meta.json +1 -1
- package/dist/types/src/List.d.ts +12 -0
- package/dist/types/src/List.d.ts.map +1 -1
- package/dist/types/src/List.stories.d.ts +12 -0
- package/dist/types/src/List.stories.d.ts.map +1 -0
- package/dist/types/src/ListItem.d.ts +4 -4
- package/dist/types/src/ListItem.d.ts.map +1 -1
- package/dist/types/tsconfig.tsbuildinfo +1 -1
- package/package.json +14 -15
- package/src/List.stories.tsx +149 -0
- package/src/List.tsx +58 -4
- package/src/ListItem.tsx +5 -5
package/README.md
CHANGED
|
@@ -1 +1,134 @@
|
|
|
1
|
-
#
|
|
1
|
+
# `@dxos/react-list` — elemental list primitive
|
|
2
|
+
|
|
3
|
+
ARIA-only foundation for DXOS list components. Renders semantically-correct
|
|
4
|
+
`<ol>` / `<ul>` markup; when `selectable` is set, becomes a
|
|
5
|
+
`role="listbox"` with `role="option"` children that carry `aria-selected`.
|
|
6
|
+
|
|
7
|
+
This package applies **no** styling, **no** keyboard navigation, and **no**
|
|
8
|
+
`dx-*` utility classes. Those live above:
|
|
9
|
+
|
|
10
|
+
```text
|
|
11
|
+
┌────────────────────────────────────────────────────────────┐
|
|
12
|
+
│ @dxos/react-ui-mosaic virtualization, DnD, cards/board │
|
|
13
|
+
├────────────────────────────────────────────────────────────┤
|
|
14
|
+
│ @dxos/react-ui-list dx-* styling, keyboard nav, │
|
|
15
|
+
│ RowList / CardList containers │
|
|
16
|
+
├────────────────────────────────────────────────────────────┤
|
|
17
|
+
│ @dxos/react-list ARIA + structure only ← this pkg │
|
|
18
|
+
└────────────────────────────────────────────────────────────┘
|
|
19
|
+
```
|
|
20
|
+
|
|
21
|
+
Most app code should reach for `@dxos/react-ui-list`. Use this primitive
|
|
22
|
+
directly only when building a new selectable surface that needs full
|
|
23
|
+
control over styling and keyboard handling — e.g. a custom Combobox.
|
|
24
|
+
|
|
25
|
+
## Components
|
|
26
|
+
|
|
27
|
+
| Component | Renders |
|
|
28
|
+
| ------------------------------ | ---------------------------------------------------- |
|
|
29
|
+
| `List` | `<ol>` / `<ul>`; optionally `role="listbox"` |
|
|
30
|
+
| `ListItem` | `<li>`; optionally `role="option"` + `aria-selected` |
|
|
31
|
+
| `ListItemHeading` | `<div>` with `id` linked via `aria-labelledby` |
|
|
32
|
+
| `ListItemOpenTrigger` | Radix `Collapsible.Trigger` |
|
|
33
|
+
| `ListItemCollapsibleContent` | Radix `Collapsible.Content` |
|
|
34
|
+
|
|
35
|
+
`ListItem` accepts `selected`, `defaultSelected`, `open`, `defaultOpen`,
|
|
36
|
+
and `collapsible` props; selection / open state can be controlled or
|
|
37
|
+
uncontrolled (uses `useControllableState`).
|
|
38
|
+
|
|
39
|
+
## Quick start
|
|
40
|
+
|
|
41
|
+
### Static unordered list (no selection)
|
|
42
|
+
|
|
43
|
+
```tsx
|
|
44
|
+
import { List, ListItem, ListItemHeading } from '@dxos/react-list';
|
|
45
|
+
|
|
46
|
+
<List variant='unordered'>
|
|
47
|
+
<ListItem>
|
|
48
|
+
<ListItemHeading>Coffee</ListItemHeading>
|
|
49
|
+
</ListItem>
|
|
50
|
+
<ListItem>
|
|
51
|
+
<ListItemHeading>Tea</ListItemHeading>
|
|
52
|
+
</ListItem>
|
|
53
|
+
</List>;
|
|
54
|
+
```
|
|
55
|
+
|
|
56
|
+
### Single-select listbox
|
|
57
|
+
|
|
58
|
+
```tsx
|
|
59
|
+
import { useState } from 'react';
|
|
60
|
+
import { List, ListItem, ListItemHeading } from '@dxos/react-list';
|
|
61
|
+
|
|
62
|
+
const Picker = ({ items }: { items: { id: string; label: string }[] }) => {
|
|
63
|
+
const [selected, setSelected] = useState<string | undefined>(items[0]?.id);
|
|
64
|
+
return (
|
|
65
|
+
<List variant='unordered' selectable aria-label='Items'>
|
|
66
|
+
{items.map((item) => (
|
|
67
|
+
<ListItem
|
|
68
|
+
key={item.id}
|
|
69
|
+
selected={item.id === selected}
|
|
70
|
+
onClick={() => setSelected(item.id)}
|
|
71
|
+
// Pair `aria-selected` (set by the primitive when `selectable`)
|
|
72
|
+
// with `dx-selected` from `@dxos/ui-theme` to get the canonical
|
|
73
|
+
// selected-row visual treatment. See
|
|
74
|
+
// ui-theme/src/css/components/selected.md.
|
|
75
|
+
className='dx-hover dx-selected'
|
|
76
|
+
>
|
|
77
|
+
<ListItemHeading>{item.label}</ListItemHeading>
|
|
78
|
+
</ListItem>
|
|
79
|
+
))}
|
|
80
|
+
</List>
|
|
81
|
+
);
|
|
82
|
+
};
|
|
83
|
+
```
|
|
84
|
+
|
|
85
|
+
### Multi-select listbox
|
|
86
|
+
|
|
87
|
+
```tsx
|
|
88
|
+
<List variant='unordered' selectable multiSelectable aria-label='Tags'>
|
|
89
|
+
…
|
|
90
|
+
</List>;
|
|
91
|
+
```
|
|
92
|
+
|
|
93
|
+
`multiSelectable` adds `aria-multiselectable="true"`; without it the
|
|
94
|
+
listbox is single-select (the default; `aria-multiselectable` is
|
|
95
|
+
intentionally omitted to keep assistive-tech announcements concise).
|
|
96
|
+
|
|
97
|
+
### Collapsible item
|
|
98
|
+
|
|
99
|
+
```tsx
|
|
100
|
+
import {
|
|
101
|
+
List,
|
|
102
|
+
ListItem,
|
|
103
|
+
ListItemHeading,
|
|
104
|
+
ListItemOpenTrigger,
|
|
105
|
+
ListItemCollapsibleContent,
|
|
106
|
+
} from '@dxos/react-list';
|
|
107
|
+
|
|
108
|
+
<List variant='unordered'>
|
|
109
|
+
<ListItem collapsible defaultOpen>
|
|
110
|
+
<ListItemOpenTrigger asChild>
|
|
111
|
+
<ListItemHeading>Section A</ListItemHeading>
|
|
112
|
+
</ListItemOpenTrigger>
|
|
113
|
+
<ListItemCollapsibleContent>
|
|
114
|
+
…content…
|
|
115
|
+
</ListItemCollapsibleContent>
|
|
116
|
+
</ListItem>
|
|
117
|
+
</List>;
|
|
118
|
+
```
|
|
119
|
+
|
|
120
|
+
## What this primitive does NOT provide
|
|
121
|
+
|
|
122
|
+
- **Keyboard navigation** — wire your own (or use `@dxos/react-ui-list`'s
|
|
123
|
+
`RowList`, which integrates `@fluentui/react-tabster`).
|
|
124
|
+
- **Styling** — pair with `dx-hover` / `dx-selected` / `dx-current` from
|
|
125
|
+
`@dxos/ui-theme` per the grammar in
|
|
126
|
+
`ui-theme/src/css/components/selected.md`.
|
|
127
|
+
- **Virtualization or drag-and-drop** — see `@dxos/react-ui-mosaic`.
|
|
128
|
+
|
|
129
|
+
## Related
|
|
130
|
+
|
|
131
|
+
- [`@dxos/react-ui-list`](../../react-ui-list) — the styled / opinionated layer.
|
|
132
|
+
- [`@dxos/react-ui-mosaic`](../../react-ui-mosaic) — virtualized / draggable.
|
|
133
|
+
- [`packages/ui/react-ui-list/AUDIT.md`](../../react-ui-list/AUDIT.md) —
|
|
134
|
+
rationale for the layering.
|
|
@@ -1,5 +1,4 @@
|
|
|
1
1
|
// src/List.tsx
|
|
2
|
-
import { useSignals as _useSignals } from "@preact-signals/safe-react/tracking";
|
|
3
2
|
import { createContextScope } from "@radix-ui/react-context";
|
|
4
3
|
import { Primitive } from "@radix-ui/react-primitive";
|
|
5
4
|
import React, { forwardRef } from "react";
|
|
@@ -7,31 +6,27 @@ var LIST_NAME = "List";
|
|
|
7
6
|
var [createListContext, createListScope] = createContextScope(LIST_NAME, []);
|
|
8
7
|
var [ListProvider, useListContext] = createListContext(LIST_NAME);
|
|
9
8
|
var List = /* @__PURE__ */ forwardRef((props, forwardedRef) => {
|
|
10
|
-
|
|
11
|
-
|
|
12
|
-
|
|
13
|
-
|
|
14
|
-
|
|
15
|
-
...
|
|
16
|
-
role: "listbox",
|
|
9
|
+
const { __listScope, variant = "ordered", selectable = false, multiSelectable = false, itemSizes, children, ...rootProps } = props;
|
|
10
|
+
const ListRoot = variant === "ordered" ? Primitive.ol : Primitive.ul;
|
|
11
|
+
return /* @__PURE__ */ React.createElement(ListRoot, {
|
|
12
|
+
...selectable && {
|
|
13
|
+
role: "listbox",
|
|
14
|
+
...multiSelectable && {
|
|
17
15
|
"aria-multiselectable": true
|
|
18
|
-
}
|
|
19
|
-
|
|
20
|
-
|
|
21
|
-
|
|
22
|
-
|
|
23
|
-
|
|
24
|
-
|
|
25
|
-
|
|
26
|
-
|
|
27
|
-
}
|
|
28
|
-
_effect.f();
|
|
29
|
-
}
|
|
16
|
+
}
|
|
17
|
+
},
|
|
18
|
+
...rootProps,
|
|
19
|
+
ref: forwardedRef
|
|
20
|
+
}, /* @__PURE__ */ React.createElement(ListProvider, {
|
|
21
|
+
scope: __listScope,
|
|
22
|
+
variant,
|
|
23
|
+
selectable,
|
|
24
|
+
itemSizes
|
|
25
|
+
}, children));
|
|
30
26
|
});
|
|
31
27
|
List.displayName = LIST_NAME;
|
|
32
28
|
|
|
33
29
|
// src/ListItem.tsx
|
|
34
|
-
import { useSignals as _useSignals2 } from "@preact-signals/safe-react/tracking";
|
|
35
30
|
import * as Collapsible from "@radix-ui/react-collapsible";
|
|
36
31
|
import { createContextScope as createContextScope2 } from "@radix-ui/react-context";
|
|
37
32
|
import { Primitive as Primitive2 } from "@radix-ui/react-primitive";
|
|
@@ -43,65 +38,55 @@ var LIST_ITEM_NAME = "ListItem";
|
|
|
43
38
|
var [createListItemContext, createListItemScope] = createContextScope2(LIST_ITEM_NAME, []);
|
|
44
39
|
var [ListItemProvider, useListItemContext] = createListItemContext(LIST_ITEM_NAME);
|
|
45
40
|
var ListItemHeading = /* @__PURE__ */ forwardRef2(({ children, asChild, __listItemScope, ...props }, forwardedRef) => {
|
|
46
|
-
|
|
47
|
-
|
|
48
|
-
|
|
49
|
-
|
|
50
|
-
|
|
51
|
-
|
|
52
|
-
|
|
53
|
-
ref: forwardedRef
|
|
54
|
-
}, children);
|
|
55
|
-
} finally {
|
|
56
|
-
_effect.f();
|
|
57
|
-
}
|
|
41
|
+
const { headingId } = useListItemContext(LIST_ITEM_NAME, __listItemScope);
|
|
42
|
+
const Comp = asChild ? Slot : Primitive2.div;
|
|
43
|
+
return /* @__PURE__ */ React2.createElement(Comp, {
|
|
44
|
+
...props,
|
|
45
|
+
id: headingId,
|
|
46
|
+
ref: forwardedRef
|
|
47
|
+
}, children);
|
|
58
48
|
});
|
|
59
49
|
var ListItemOpenTrigger = Collapsible.Trigger;
|
|
60
50
|
var ListItemCollapsibleContent = Collapsible.Content;
|
|
61
51
|
var ListItem = /* @__PURE__ */ forwardRef2((props, forwardedRef) => {
|
|
62
|
-
|
|
63
|
-
|
|
64
|
-
|
|
65
|
-
|
|
66
|
-
|
|
67
|
-
|
|
68
|
-
|
|
69
|
-
|
|
70
|
-
|
|
71
|
-
|
|
72
|
-
|
|
73
|
-
|
|
74
|
-
|
|
75
|
-
|
|
76
|
-
|
|
77
|
-
|
|
78
|
-
|
|
79
|
-
|
|
80
|
-
|
|
81
|
-
|
|
82
|
-
"
|
|
83
|
-
|
|
84
|
-
|
|
85
|
-
|
|
86
|
-
|
|
87
|
-
|
|
88
|
-
|
|
89
|
-
|
|
90
|
-
|
|
91
|
-
|
|
92
|
-
|
|
93
|
-
|
|
94
|
-
|
|
95
|
-
|
|
96
|
-
|
|
97
|
-
|
|
98
|
-
|
|
99
|
-
|
|
100
|
-
onOpenChange: setOpen
|
|
101
|
-
}, listItem) : listItem);
|
|
102
|
-
} finally {
|
|
103
|
-
_effect.f();
|
|
104
|
-
}
|
|
52
|
+
const id = useId("listItem", props.id);
|
|
53
|
+
const { __listScope, __listItemScope, children, selected: propsSelected, defaultSelected, onSelectedChange, open: propsOpen, defaultOpen, onOpenChange, collapsible, labelId, ...listItemProps } = props;
|
|
54
|
+
const { selectable } = useListContext(LIST_NAME, __listScope);
|
|
55
|
+
const [selected = false, setSelected] = useControllableState({
|
|
56
|
+
prop: propsSelected,
|
|
57
|
+
defaultProp: defaultSelected,
|
|
58
|
+
onChange: onSelectedChange
|
|
59
|
+
});
|
|
60
|
+
const [open = false, setOpen] = useControllableState({
|
|
61
|
+
prop: propsOpen,
|
|
62
|
+
defaultProp: defaultOpen,
|
|
63
|
+
onChange: onOpenChange
|
|
64
|
+
});
|
|
65
|
+
const headingId = useId("listItem__heading", labelId);
|
|
66
|
+
const listItem = /* @__PURE__ */ React2.createElement(Primitive2.li, {
|
|
67
|
+
...listItemProps,
|
|
68
|
+
id,
|
|
69
|
+
ref: forwardedRef,
|
|
70
|
+
"aria-labelledby": headingId,
|
|
71
|
+
...selectable && {
|
|
72
|
+
role: "option",
|
|
73
|
+
"aria-selected": !!selected
|
|
74
|
+
},
|
|
75
|
+
...open && {
|
|
76
|
+
"aria-expanded": true
|
|
77
|
+
}
|
|
78
|
+
}, children);
|
|
79
|
+
return /* @__PURE__ */ React2.createElement(ListItemProvider, {
|
|
80
|
+
scope: __listItemScope,
|
|
81
|
+
headingId,
|
|
82
|
+
open,
|
|
83
|
+
selected,
|
|
84
|
+
setSelected
|
|
85
|
+
}, collapsible ? /* @__PURE__ */ React2.createElement(Collapsible.Root, {
|
|
86
|
+
asChild: true,
|
|
87
|
+
open,
|
|
88
|
+
onOpenChange: setOpen
|
|
89
|
+
}, listItem) : listItem);
|
|
105
90
|
});
|
|
106
91
|
ListItem.displayName = LIST_ITEM_NAME;
|
|
107
92
|
export {
|
|
@@ -1,7 +1,7 @@
|
|
|
1
1
|
{
|
|
2
2
|
"version": 3,
|
|
3
3
|
"sources": ["../../../src/List.tsx", "../../../src/ListItem.tsx"],
|
|
4
|
-
"sourcesContent": ["//\n// Copyright 2023 DXOS.org\n//\n\
|
|
5
|
-
"mappings": "
|
|
6
|
-
"names": ["createContextScope", "Primitive", "React", "forwardRef", "LIST_NAME", "createListContext", "createListScope", "
|
|
4
|
+
"sourcesContent": ["//\n// Copyright 2023 DXOS.org\n//\n\n// Elemental list / listbox primitive.\n//\n// This is the ARIA-only foundation of the DXOS list stack. It renders a\n// semantically-correct `<ol>` / `<ul>` (or, when `selectable={true}`, a\n// `role=\"listbox\"` element with `role=\"option\"` children carrying\n// `aria-selected`). It applies no styling, no keyboard navigation, and\n// no `dx-*` utility classes — those are layered above in\n// `@dxos/react-ui-list`.\n//\n// Layering:\n// - `@dxos/react-list` — this package; ARIA + structure only.\n// - `@dxos/react-ui-list` — adds `dx-*` styling, keyboard nav, and\n// opinionated `RowList`/`CardList` containers.\n// - `@dxos/react-ui-mosaic` — virtualized / draggable / card-board\n// layouts; composes the above where useful.\n//\n// Most app code should reach for `@dxos/react-ui-list`. Use this primitive\n// directly only when building a *new* selectable surface that needs full\n// control over styling and keyboard handling (e.g. a custom Combobox).\n//\n// See:\n// - `packages/ui/ui-theme/src/css/components/selected.md` for the\n// `aria-selected` ↔ `dx-selected` pairing rules.\n// - `packages/ui/react-ui-list/AUDIT.md` for why this layering exists.\n// - https://developer.mozilla.org/en-US/docs/Web/Accessibility/ARIA/Roles/listbox_role\n\nimport { type Scope, createContextScope } from '@radix-ui/react-context';\nimport { Primitive } from '@radix-ui/react-primitive';\nimport React, { type ComponentPropsWithRef, forwardRef } from 'react';\n\nconst LIST_NAME = 'List';\n\ntype ListScopedProps<P> = P & { __listScope?: Scope };\n\ntype ListVariant = 'ordered' | 'unordered';\n\ntype ListItemSizes = 'one' | 'many';\n\ntype ListProps = ComponentPropsWithRef<typeof Primitive.ol> & {\n /**\n * If true, render as `role=\"listbox\"` and let `ListItem` children become\n * `role=\"option\"` + `aria-selected`. If false (default) the list is a\n * plain `<ol>` / `<ul>` with no selection semantics — pick this for\n * static lists.\n */\n selectable?: boolean;\n /**\n * If true, the listbox advertises multi-select via\n * `aria-multiselectable=\"true\"`. Defaults to false (single-select).\n * Has no effect unless `selectable` is also true.\n */\n multiSelectable?: boolean;\n variant?: ListVariant;\n itemSizes?: ListItemSizes;\n};\n\nconst [createListContext, createListScope] = createContextScope(LIST_NAME, []);\n\ntype ListContextValue = {\n selectable: Exclude<ListProps['selectable'], undefined>;\n variant: Exclude<ListProps['variant'], undefined>;\n itemSizes?: ListItemSizes;\n};\n\nconst [ListProvider, useListContext] = createListContext<ListContextValue>(LIST_NAME);\n\nconst List = forwardRef<HTMLOListElement, ListProps>((props: ListScopedProps<ListProps>, forwardedRef) => {\n const {\n __listScope,\n variant = 'ordered',\n selectable = false,\n multiSelectable = false,\n itemSizes,\n children,\n ...rootProps\n } = props;\n const ListRoot = variant === 'ordered' ? Primitive.ol : Primitive.ul;\n return (\n <ListRoot\n // `aria-multiselectable` is only meaningful on `role=\"listbox\"`,\n // and even there is omitted in the single-select default to keep\n // assistive tech announcements concise.\n {...(selectable && {\n role: 'listbox',\n ...(multiSelectable && { 'aria-multiselectable': true as const }),\n })}\n {...rootProps}\n ref={forwardedRef}\n >\n <ListProvider\n {...{\n scope: __listScope,\n variant,\n selectable,\n itemSizes,\n }}\n >\n {children}\n </ListProvider>\n </ListRoot>\n );\n});\n\nList.displayName = LIST_NAME;\n\nexport { List, createListScope, useListContext, LIST_NAME };\n\nexport type { ListProps, ListVariant, ListScopedProps };\n", "//\n// Copyright 2023 DXOS.org\n//\n\nimport type { CheckboxProps } from '@radix-ui/react-checkbox';\nimport { type CollapsibleContentProps, type CollapsibleTriggerProps } from '@radix-ui/react-collapsible';\nimport * as Collapsible from '@radix-ui/react-collapsible';\nimport { type Scope, createContextScope } from '@radix-ui/react-context';\nimport { Primitive } from '@radix-ui/react-primitive';\nimport { Slot } from '@radix-ui/react-slot';\nimport { useControllableState } from '@radix-ui/react-use-controllable-state';\nimport React, {\n type ComponentProps,\n type ComponentPropsWithoutRef,\n type Dispatch,\n type ComponentRef,\n type ForwardRefExoticComponent,\n type RefAttributes,\n type SetStateAction,\n forwardRef,\n} from 'react';\n\nimport { useId } from '@dxos/react-hooks';\n\nimport { LIST_NAME, type ListScopedProps, useListContext } from './List';\n\nconst LIST_ITEM_NAME = 'ListItem';\n\ntype ListItemScopedProps<P> = P & { __listItemScope?: Scope };\n\ninterface ListItemData {\n id: string;\n labelId?: string;\n selected?: CheckboxProps['checked'];\n open?: boolean;\n}\n\ntype ListItemProps = Omit<ListItemData, 'id'> & { collapsible?: boolean } & RefAttributes<HTMLLIElement> &\n ComponentPropsWithoutRef<'li'> & {\n defaultOpen?: boolean;\n onOpenChange?: (nextOpen: boolean) => void;\n } & {\n onSelectedChange?: CheckboxProps['onCheckedChange'];\n defaultSelected?: CheckboxProps['defaultChecked'];\n };\n\ntype ListItemElement = ComponentRef<'li'>;\n\nconst [createListItemContext, createListItemScope] = createContextScope(LIST_ITEM_NAME, []);\n\ntype ListItemContextValue = {\n headingId: string;\n open: boolean;\n selected: CheckboxProps['checked'];\n setSelected: Dispatch<SetStateAction<CheckboxProps['checked']>>;\n};\n\nconst [ListItemProvider, useListItemContext] = createListItemContext<ListItemContextValue>(LIST_ITEM_NAME);\n\ntype ListItemHeadingProps = ListItemScopedProps<Omit<ComponentPropsWithoutRef<'p'>, 'id'>> &\n RefAttributes<HTMLParagraphElement> & {\n asChild?: boolean;\n };\n\nconst ListItemHeading = forwardRef<HTMLDivElement, ListItemHeadingProps>(\n ({ children, asChild, __listItemScope, ...props }, forwardedRef) => {\n const { headingId } = useListItemContext(LIST_ITEM_NAME, __listItemScope);\n const Comp = asChild ? Slot : Primitive.div;\n return (\n <Comp {...props} id={headingId} ref={forwardedRef}>\n {children}\n </Comp>\n );\n },\n);\n\ntype ListItemOpenTriggerProps = ListItemScopedProps<CollapsibleTriggerProps>;\n\nconst ListItemOpenTrigger = Collapsible.Trigger;\n\ntype ListItemCollapsibleContentProps = ComponentProps<typeof Collapsible.Content>;\n\nconst ListItemCollapsibleContent: ForwardRefExoticComponent<CollapsibleContentProps> = Collapsible.Content;\n\nconst ListItem = forwardRef<ListItemElement, ListItemProps>(\n (props: ListItemScopedProps<ListScopedProps<ListItemProps>>, forwardedRef) => {\n const id = useId('listItem', props.id);\n\n const {\n __listScope,\n __listItemScope,\n children,\n selected: propsSelected,\n defaultSelected,\n onSelectedChange,\n open: propsOpen,\n defaultOpen,\n onOpenChange,\n collapsible,\n labelId,\n ...listItemProps\n } = props;\n const { selectable } = useListContext(LIST_NAME, __listScope);\n\n const [selected = false, setSelected] = useControllableState({\n prop: propsSelected,\n defaultProp: defaultSelected,\n onChange: onSelectedChange,\n });\n\n const [open = false, setOpen] = useControllableState({\n prop: propsOpen,\n defaultProp: defaultOpen,\n onChange: onOpenChange,\n });\n\n const headingId = useId('listItem__heading', labelId);\n\n const listItem = (\n <Primitive.li\n {...listItemProps}\n id={id}\n ref={forwardedRef}\n aria-labelledby={headingId}\n {...(selectable && { role: 'option', 'aria-selected': !!selected })}\n {...(open && { 'aria-expanded': true })}\n >\n {children}\n </Primitive.li>\n );\n\n return (\n <ListItemProvider\n scope={__listItemScope}\n headingId={headingId}\n open={open}\n selected={selected}\n setSelected={setSelected}\n >\n {collapsible ? (\n <Collapsible.Root asChild open={open} onOpenChange={setOpen}>\n {listItem}\n </Collapsible.Root>\n ) : (\n listItem\n )}\n </ListItemProvider>\n );\n },\n);\n\nListItem.displayName = LIST_ITEM_NAME;\n\nexport {\n ListItem,\n ListItemHeading,\n ListItemCollapsibleContent,\n ListItemOpenTrigger,\n createListItemScope,\n useListItemContext,\n LIST_ITEM_NAME,\n};\n\nexport type {\n ListItemProps,\n ListItemHeadingProps,\n ListItemCollapsibleContentProps,\n ListItemOpenTriggerProps,\n ListItemScopedProps,\n};\n"],
|
|
5
|
+
"mappings": "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",
|
|
6
|
+
"names": ["createContextScope", "Primitive", "React", "forwardRef", "LIST_NAME", "createListContext", "createListScope", "ListProvider", "useListContext", "List", "props", "forwardedRef", "__listScope", "variant", "selectable", "multiSelectable", "itemSizes", "children", "rootProps", "ListRoot", "ol", "ul", "role", "ref", "scope", "displayName", "Collapsible", "createContextScope", "Primitive", "Slot", "useControllableState", "React", "forwardRef", "useId", "LIST_ITEM_NAME", "createListItemContext", "createListItemScope", "createContextScope", "ListItemProvider", "useListItemContext", "ListItemHeading", "forwardRef", "children", "asChild", "__listItemScope", "props", "forwardedRef", "headingId", "Comp", "Slot", "Primitive", "div", "React", "id", "ref", "ListItemOpenTrigger", "Trigger", "ListItemCollapsibleContent", "Content", "ListItem", "useId", "__listScope", "selected", "propsSelected", "defaultSelected", "onSelectedChange", "open", "propsOpen", "defaultOpen", "onOpenChange", "collapsible", "labelId", "listItemProps", "selectable", "useListContext", "LIST_NAME", "setSelected", "useControllableState", "prop", "defaultProp", "onChange", "setOpen", "listItem", "li", "aria-labelledby", "role", "scope", "Root", "displayName"]
|
|
7
7
|
}
|
|
@@ -1 +1 @@
|
|
|
1
|
-
{"inputs":{"src/List.tsx":{"bytes":
|
|
1
|
+
{"inputs":{"src/List.tsx":{"bytes":9674,"imports":[{"path":"@radix-ui/react-context","kind":"import-statement","external":true},{"path":"@radix-ui/react-primitive","kind":"import-statement","external":true},{"path":"react","kind":"import-statement","external":true}],"format":"esm"},"src/ListItem.tsx":{"bytes":12922,"imports":[{"path":"@radix-ui/react-collapsible","kind":"import-statement","external":true},{"path":"@radix-ui/react-context","kind":"import-statement","external":true},{"path":"@radix-ui/react-primitive","kind":"import-statement","external":true},{"path":"@radix-ui/react-slot","kind":"import-statement","external":true},{"path":"@radix-ui/react-use-controllable-state","kind":"import-statement","external":true},{"path":"react","kind":"import-statement","external":true},{"path":"@dxos/react-hooks","kind":"import-statement","external":true},{"path":"src/List.tsx","kind":"import-statement","original":"./List"}],"format":"esm"},"src/index.ts":{"bytes":456,"imports":[{"path":"src/List.tsx","kind":"import-statement","original":"./List"},{"path":"src/ListItem.tsx","kind":"import-statement","original":"./ListItem"}],"format":"esm"}},"outputs":{"dist/lib/browser/index.mjs.map":{"imports":[],"exports":[],"inputs":{},"bytes":12305},"dist/lib/browser/index.mjs":{"imports":[{"path":"@radix-ui/react-context","kind":"import-statement","external":true},{"path":"@radix-ui/react-primitive","kind":"import-statement","external":true},{"path":"react","kind":"import-statement","external":true},{"path":"@radix-ui/react-collapsible","kind":"import-statement","external":true},{"path":"@radix-ui/react-context","kind":"import-statement","external":true},{"path":"@radix-ui/react-primitive","kind":"import-statement","external":true},{"path":"@radix-ui/react-slot","kind":"import-statement","external":true},{"path":"@radix-ui/react-use-controllable-state","kind":"import-statement","external":true},{"path":"react","kind":"import-statement","external":true},{"path":"@dxos/react-hooks","kind":"import-statement","external":true}],"exports":["LIST_ITEM_NAME","LIST_NAME","List","ListItem","ListItemCollapsibleContent","ListItemHeading","ListItemOpenTrigger","createListItemScope","createListScope","useListContext","useListItemContext"],"entryPoint":"src/index.ts","inputs":{"src/List.tsx":{"bytesInOutput":1000},"src/index.ts":{"bytesInOutput":0},"src/ListItem.tsx":{"bytesInOutput":2528}},"bytes":3816}}}
|
|
@@ -1,7 +1,6 @@
|
|
|
1
1
|
import { createRequire } from 'node:module';const require = createRequire(import.meta.url);
|
|
2
2
|
|
|
3
3
|
// src/List.tsx
|
|
4
|
-
import { useSignals as _useSignals } from "@preact-signals/safe-react/tracking";
|
|
5
4
|
import { createContextScope } from "@radix-ui/react-context";
|
|
6
5
|
import { Primitive } from "@radix-ui/react-primitive";
|
|
7
6
|
import React, { forwardRef } from "react";
|
|
@@ -9,31 +8,27 @@ var LIST_NAME = "List";
|
|
|
9
8
|
var [createListContext, createListScope] = createContextScope(LIST_NAME, []);
|
|
10
9
|
var [ListProvider, useListContext] = createListContext(LIST_NAME);
|
|
11
10
|
var List = /* @__PURE__ */ forwardRef((props, forwardedRef) => {
|
|
12
|
-
|
|
13
|
-
|
|
14
|
-
|
|
15
|
-
|
|
16
|
-
|
|
17
|
-
...
|
|
18
|
-
role: "listbox",
|
|
11
|
+
const { __listScope, variant = "ordered", selectable = false, multiSelectable = false, itemSizes, children, ...rootProps } = props;
|
|
12
|
+
const ListRoot = variant === "ordered" ? Primitive.ol : Primitive.ul;
|
|
13
|
+
return /* @__PURE__ */ React.createElement(ListRoot, {
|
|
14
|
+
...selectable && {
|
|
15
|
+
role: "listbox",
|
|
16
|
+
...multiSelectable && {
|
|
19
17
|
"aria-multiselectable": true
|
|
20
|
-
}
|
|
21
|
-
|
|
22
|
-
|
|
23
|
-
|
|
24
|
-
|
|
25
|
-
|
|
26
|
-
|
|
27
|
-
|
|
28
|
-
|
|
29
|
-
}
|
|
30
|
-
_effect.f();
|
|
31
|
-
}
|
|
18
|
+
}
|
|
19
|
+
},
|
|
20
|
+
...rootProps,
|
|
21
|
+
ref: forwardedRef
|
|
22
|
+
}, /* @__PURE__ */ React.createElement(ListProvider, {
|
|
23
|
+
scope: __listScope,
|
|
24
|
+
variant,
|
|
25
|
+
selectable,
|
|
26
|
+
itemSizes
|
|
27
|
+
}, children));
|
|
32
28
|
});
|
|
33
29
|
List.displayName = LIST_NAME;
|
|
34
30
|
|
|
35
31
|
// src/ListItem.tsx
|
|
36
|
-
import { useSignals as _useSignals2 } from "@preact-signals/safe-react/tracking";
|
|
37
32
|
import * as Collapsible from "@radix-ui/react-collapsible";
|
|
38
33
|
import { createContextScope as createContextScope2 } from "@radix-ui/react-context";
|
|
39
34
|
import { Primitive as Primitive2 } from "@radix-ui/react-primitive";
|
|
@@ -45,65 +40,55 @@ var LIST_ITEM_NAME = "ListItem";
|
|
|
45
40
|
var [createListItemContext, createListItemScope] = createContextScope2(LIST_ITEM_NAME, []);
|
|
46
41
|
var [ListItemProvider, useListItemContext] = createListItemContext(LIST_ITEM_NAME);
|
|
47
42
|
var ListItemHeading = /* @__PURE__ */ forwardRef2(({ children, asChild, __listItemScope, ...props }, forwardedRef) => {
|
|
48
|
-
|
|
49
|
-
|
|
50
|
-
|
|
51
|
-
|
|
52
|
-
|
|
53
|
-
|
|
54
|
-
|
|
55
|
-
ref: forwardedRef
|
|
56
|
-
}, children);
|
|
57
|
-
} finally {
|
|
58
|
-
_effect.f();
|
|
59
|
-
}
|
|
43
|
+
const { headingId } = useListItemContext(LIST_ITEM_NAME, __listItemScope);
|
|
44
|
+
const Comp = asChild ? Slot : Primitive2.div;
|
|
45
|
+
return /* @__PURE__ */ React2.createElement(Comp, {
|
|
46
|
+
...props,
|
|
47
|
+
id: headingId,
|
|
48
|
+
ref: forwardedRef
|
|
49
|
+
}, children);
|
|
60
50
|
});
|
|
61
51
|
var ListItemOpenTrigger = Collapsible.Trigger;
|
|
62
52
|
var ListItemCollapsibleContent = Collapsible.Content;
|
|
63
53
|
var ListItem = /* @__PURE__ */ forwardRef2((props, forwardedRef) => {
|
|
64
|
-
|
|
65
|
-
|
|
66
|
-
|
|
67
|
-
|
|
68
|
-
|
|
69
|
-
|
|
70
|
-
|
|
71
|
-
|
|
72
|
-
|
|
73
|
-
|
|
74
|
-
|
|
75
|
-
|
|
76
|
-
|
|
77
|
-
|
|
78
|
-
|
|
79
|
-
|
|
80
|
-
|
|
81
|
-
|
|
82
|
-
|
|
83
|
-
|
|
84
|
-
"
|
|
85
|
-
|
|
86
|
-
|
|
87
|
-
|
|
88
|
-
|
|
89
|
-
|
|
90
|
-
|
|
91
|
-
|
|
92
|
-
|
|
93
|
-
|
|
94
|
-
|
|
95
|
-
|
|
96
|
-
|
|
97
|
-
|
|
98
|
-
|
|
99
|
-
|
|
100
|
-
|
|
101
|
-
|
|
102
|
-
onOpenChange: setOpen
|
|
103
|
-
}, listItem) : listItem);
|
|
104
|
-
} finally {
|
|
105
|
-
_effect.f();
|
|
106
|
-
}
|
|
54
|
+
const id = useId("listItem", props.id);
|
|
55
|
+
const { __listScope, __listItemScope, children, selected: propsSelected, defaultSelected, onSelectedChange, open: propsOpen, defaultOpen, onOpenChange, collapsible, labelId, ...listItemProps } = props;
|
|
56
|
+
const { selectable } = useListContext(LIST_NAME, __listScope);
|
|
57
|
+
const [selected = false, setSelected] = useControllableState({
|
|
58
|
+
prop: propsSelected,
|
|
59
|
+
defaultProp: defaultSelected,
|
|
60
|
+
onChange: onSelectedChange
|
|
61
|
+
});
|
|
62
|
+
const [open = false, setOpen] = useControllableState({
|
|
63
|
+
prop: propsOpen,
|
|
64
|
+
defaultProp: defaultOpen,
|
|
65
|
+
onChange: onOpenChange
|
|
66
|
+
});
|
|
67
|
+
const headingId = useId("listItem__heading", labelId);
|
|
68
|
+
const listItem = /* @__PURE__ */ React2.createElement(Primitive2.li, {
|
|
69
|
+
...listItemProps,
|
|
70
|
+
id,
|
|
71
|
+
ref: forwardedRef,
|
|
72
|
+
"aria-labelledby": headingId,
|
|
73
|
+
...selectable && {
|
|
74
|
+
role: "option",
|
|
75
|
+
"aria-selected": !!selected
|
|
76
|
+
},
|
|
77
|
+
...open && {
|
|
78
|
+
"aria-expanded": true
|
|
79
|
+
}
|
|
80
|
+
}, children);
|
|
81
|
+
return /* @__PURE__ */ React2.createElement(ListItemProvider, {
|
|
82
|
+
scope: __listItemScope,
|
|
83
|
+
headingId,
|
|
84
|
+
open,
|
|
85
|
+
selected,
|
|
86
|
+
setSelected
|
|
87
|
+
}, collapsible ? /* @__PURE__ */ React2.createElement(Collapsible.Root, {
|
|
88
|
+
asChild: true,
|
|
89
|
+
open,
|
|
90
|
+
onOpenChange: setOpen
|
|
91
|
+
}, listItem) : listItem);
|
|
107
92
|
});
|
|
108
93
|
ListItem.displayName = LIST_ITEM_NAME;
|
|
109
94
|
export {
|