native-document 1.0.166 → 1.0.168
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/.vitepress/config.js +166 -0
- package/CHANGELOG.md +153 -0
- package/components.js +2 -1
- package/dist/native-document.components.min.js +495 -228
- package/dist/native-document.dev.js +7 -0
- package/dist/native-document.dev.js.map +1 -1
- package/dist/native-document.min.js +1 -1
- package/docs/advanced-components.md +213 -608
- package/docs/anchor.md +173 -312
- package/docs/cache.md +95 -803
- package/docs/cli.md +179 -0
- package/docs/components/accordion.md +172 -0
- package/docs/components/alert.md +99 -0
- package/docs/components/avatar.md +160 -0
- package/docs/components/badge.md +102 -0
- package/docs/components/breadcrumb.md +89 -0
- package/docs/components/button.md +183 -0
- package/docs/components/card.md +69 -0
- package/docs/components/context-menu.md +118 -0
- package/docs/components/data-table.md +345 -0
- package/docs/components/dropdown.md +214 -0
- package/docs/components/form/autocomplete-field.md +81 -0
- package/docs/components/form/checkbox-field.md +41 -0
- package/docs/components/form/checkbox-group-field.md +54 -0
- package/docs/components/form/color-field.md +64 -0
- package/docs/components/form/date-field.md +92 -0
- package/docs/components/form/field-collection.md +63 -0
- package/docs/components/form/file-field.md +203 -0
- package/docs/components/form/form-control.md +87 -0
- package/docs/components/form/image-field.md +90 -0
- package/docs/components/form/index.md +115 -0
- package/docs/components/form/number-field.md +65 -0
- package/docs/components/form/radio-field.md +51 -0
- package/docs/components/form/select-field.md +123 -0
- package/docs/components/form/slider.md +136 -0
- package/docs/components/form/string-field.md +134 -0
- package/docs/components/form/textarea-field.md +65 -0
- package/docs/components/form-fields.md +372 -0
- package/docs/components/getting-started.md +264 -0
- package/docs/components/index.md +337 -0
- package/docs/components/layout.md +279 -0
- package/docs/components/list.md +73 -0
- package/docs/components/menu.md +215 -0
- package/docs/components/modal.md +156 -0
- package/docs/components/pagination.md +95 -0
- package/docs/components/popover.md +131 -0
- package/docs/components/progress.md +111 -0
- package/docs/components/shortcut-manager.md +221 -0
- package/docs/components/simple-table.md +107 -0
- package/docs/components/skeleton.md +155 -0
- package/docs/components/spinner.md +100 -0
- package/docs/components/splitter.md +133 -0
- package/docs/components/stepper.md +163 -0
- package/docs/components/switch.md +113 -0
- package/docs/components/tabs.md +153 -0
- package/docs/components/toast.md +119 -0
- package/docs/components/tooltip.md +151 -0
- package/docs/components/traits.md +261 -0
- package/docs/conditional-rendering.md +170 -588
- package/docs/contributing.md +300 -25
- package/docs/core-concepts.md +205 -374
- package/docs/elements.md +251 -367
- package/docs/extending-native-document-element.md +192 -207
- package/docs/filters.md +153 -1122
- package/docs/getting-started.md +193 -267
- package/docs/i18n.md +241 -0
- package/docs/index.md +76 -0
- package/docs/lifecycle-events.md +143 -75
- package/docs/list-rendering.md +227 -852
- package/docs/memory-management.md +134 -47
- package/docs/native-document-element.md +337 -186
- package/docs/native-fetch.md +99 -630
- package/docs/observable-resource.md +364 -0
- package/docs/observables.md +592 -526
- package/docs/routing.md +244 -653
- package/docs/state-management.md +134 -241
- package/docs/svg-elements.md +231 -0
- package/docs/theming.md +409 -0
- package/docs/tutorials/.gitkeep +0 -0
- package/docs/validation.md +95 -97
- package/docs/vitepress-conventions.md +219 -0
- package/package.json +34 -13
- package/readme.md +269 -89
- package/src/components/card/Card.js +93 -39
- package/src/components/card/index.js +1 -1
- package/src/components/list/HasListItem.js +171 -0
- package/src/components/list/List.js +41 -107
- package/src/components/list/ListDivider.js +39 -0
- package/src/components/list/ListGroup.js +76 -59
- package/src/components/list/ListItem.js +117 -69
- package/src/components/list/index.js +3 -1
- package/src/components/list/types/ListItem.d.ts +45 -34
- package/src/components/spacer/Spacer.js +1 -1
- package/src/core/data/ObservableResource.js +5 -0
- package/src/core/data/observable-helpers/observable.prototypes.js +2 -0
- package/src/ui/components/card/CardRender.js +133 -0
- package/src/ui/components/card/card.css +169 -0
- package/src/ui/components/contextmenu/ContextmenuRender.js +1 -1
- package/src/ui/components/list/ListRender.js +18 -0
- package/src/ui/components/list/divider/ListDividerRender.js +10 -0
- package/src/ui/components/list/divider/list-divider.css +12 -0
- package/src/ui/components/list/group/ListGroupRender.js +61 -0
- package/src/ui/components/list/group/list-group.css +62 -0
- package/src/ui/components/list/item/ListItemRender.js +238 -0
- package/src/ui/components/list/item/list-item.css +191 -0
- package/src/ui/components/list/list.css +24 -0
- package/src/ui/components/spacer/SpacerRender.js +10 -0
- package/src/ui/index.js +8 -0
|
@@ -0,0 +1,261 @@
|
|
|
1
|
+
---
|
|
2
|
+
title: Traits
|
|
3
|
+
description: Reusable behavior mixins for NativeDocument components - HasEventEmitter, HasDraggable, HasResizable, HasItems, HasPosition
|
|
4
|
+
---
|
|
5
|
+
|
|
6
|
+
# Traits
|
|
7
|
+
|
|
8
|
+
Traits are behavior mixins that add capabilities to components. They are applied via `BaseComponent.use()` and follow the same composition pattern as the rest of the framework.
|
|
9
|
+
|
|
10
|
+
```javascript
|
|
11
|
+
import { BaseComponent, HasEventEmitter, HasDraggable, HasResizable } from '@native-document/components';
|
|
12
|
+
|
|
13
|
+
BaseComponent.use(MyComponent, HasEventEmitter, HasDraggable, HasResizable);
|
|
14
|
+
```
|
|
15
|
+
|
|
16
|
+
> `HasDraggable` and `HasResizable` both **require** `HasEventEmitter` to be applied first.
|
|
17
|
+
|
|
18
|
+
---
|
|
19
|
+
|
|
20
|
+
## `HasEventEmitter`
|
|
21
|
+
|
|
22
|
+
Adds a custom event system to a component. Components use this to communicate state changes to their parent.
|
|
23
|
+
|
|
24
|
+
### Methods
|
|
25
|
+
|
|
26
|
+
| Method | Parameters | Description |
|
|
27
|
+
|---|---|---|
|
|
28
|
+
| `.on(eventName, callback)` | `eventName: string`, `callback: function` | Register a listener. Multiple listeners allowed per event. |
|
|
29
|
+
| `.emit(eventName, ...args)` | `eventName: string`, `...args` | Emit an event. Async - all listeners awaited in order. |
|
|
30
|
+
| `.trigger(eventName, ...args)` | same as `emit` | Alias for `emit`. |
|
|
31
|
+
| `.hasListeners(eventName)` | `eventName: string` | Returns `true` if any listeners are registered. |
|
|
32
|
+
|
|
33
|
+
### Built-in component events
|
|
34
|
+
|
|
35
|
+
| Event | When |
|
|
36
|
+
|---|---|
|
|
37
|
+
| `open` | Component becomes visible |
|
|
38
|
+
| `close` | Component is dismissed |
|
|
39
|
+
| `change` | Value or selection changes |
|
|
40
|
+
| `select` | An item is selected |
|
|
41
|
+
|
|
42
|
+
### Example
|
|
43
|
+
|
|
44
|
+
```javascript
|
|
45
|
+
import { Modal } from '@native-document/components';
|
|
46
|
+
|
|
47
|
+
const confirmModal = Modal(Div('Are you sure?'))
|
|
48
|
+
.title('Confirm')
|
|
49
|
+
.on('close', (confirmed) => {
|
|
50
|
+
if (confirmed) deleteItem();
|
|
51
|
+
});
|
|
52
|
+
|
|
53
|
+
Button('Delete')
|
|
54
|
+
.danger()
|
|
55
|
+
.nd.onClick(() => confirmModal.open())
|
|
56
|
+
```
|
|
57
|
+
|
|
58
|
+
---
|
|
59
|
+
|
|
60
|
+
## `HasDraggable`
|
|
61
|
+
|
|
62
|
+
Adds mouse-driven drag behavior to a component. Requires `HasEventEmitter`.
|
|
63
|
+
|
|
64
|
+
### Methods
|
|
65
|
+
|
|
66
|
+
| Method | Parameters | Description |
|
|
67
|
+
|---|---|---|
|
|
68
|
+
| `.makeDraggable(element, grip?)` | `element: HTMLElement`, `grip?: HTMLElement` | Makes the element draggable. Returns a cleanup function. `grip` defaults to `element`. |
|
|
69
|
+
| `.move(x, y)` | `x: number`, `y: number` | Programmatically move to an absolute position (px). |
|
|
70
|
+
|
|
71
|
+
### CSS classes
|
|
72
|
+
|
|
73
|
+
| Class | When applied |
|
|
74
|
+
|---|---|
|
|
75
|
+
| `is-draggable` | On the grip element |
|
|
76
|
+
| `is-dragging` | While drag is active |
|
|
77
|
+
|
|
78
|
+
### Events
|
|
79
|
+
|
|
80
|
+
| Event | Args | When |
|
|
81
|
+
|---|---|---|
|
|
82
|
+
| `onDragStart` | `(event, initialX, initialY)` | Mouse down on grip |
|
|
83
|
+
| `onDrag` | `(event, x, y)` | Mouse move while dragging |
|
|
84
|
+
| `onDragEnd` | - | Mouse up |
|
|
85
|
+
|
|
86
|
+
```javascript
|
|
87
|
+
MyModal
|
|
88
|
+
.on('onDragStart', (e, x, y) => console.log('Started at', x, y))
|
|
89
|
+
.on('onDrag', (e, x, y) => console.log('Moving to', x, y))
|
|
90
|
+
.on('onDragEnd', () => console.log('Done'))
|
|
91
|
+
```
|
|
92
|
+
|
|
93
|
+
---
|
|
94
|
+
|
|
95
|
+
## `HasResizable`
|
|
96
|
+
|
|
97
|
+
Adds mouse-driven resize behavior to a component. Requires `HasEventEmitter`.
|
|
98
|
+
|
|
99
|
+
### Methods
|
|
100
|
+
|
|
101
|
+
| Method | Parameters | Description |
|
|
102
|
+
|---|---|---|
|
|
103
|
+
| `.makeResizable(element, options?)` | `element: HTMLElement`, `options?: object` | Makes the element resizable. Returns a cleanup function. |
|
|
104
|
+
|
|
105
|
+
Options:
|
|
106
|
+
|
|
107
|
+
| Option | Type | Default | Description |
|
|
108
|
+
|---|---|---|---|
|
|
109
|
+
| `directions` | `string[]` | `['right', 'bottom', 'bottom-right']` | Which resize handles to show. Available: `right`, `left`, `top`, `bottom`, `top-right`, `top-left`, `bottom-right`, `bottom-left` |
|
|
110
|
+
| `size.minWidth` | `number` | `200` | Minimum width in px |
|
|
111
|
+
| `size.minHeight` | `number` | `200` | Minimum height in px |
|
|
112
|
+
| `size.maxWidth` | `number` | - | Maximum width in px |
|
|
113
|
+
| `size.maxHeight` | `number` | - | Maximum height in px |
|
|
114
|
+
|
|
115
|
+
### CSS classes
|
|
116
|
+
|
|
117
|
+
| Class | Applied to | When |
|
|
118
|
+
|---|---|---|
|
|
119
|
+
| `is-resizable` | Parent element | Always (when resizable) |
|
|
120
|
+
| `resize-handle is-{direction}` | Handle divs | Always |
|
|
121
|
+
| `is-resizing` | `document.body` | While resize is active |
|
|
122
|
+
|
|
123
|
+
### Events
|
|
124
|
+
|
|
125
|
+
| Event | Args | When |
|
|
126
|
+
|---|---|---|
|
|
127
|
+
| `onResizeStart` | `(event, width, height)` | Mouse down on handle |
|
|
128
|
+
| `onResize` | `(event, width, height)` | Mouse move while resizing |
|
|
129
|
+
| `onResizeEnd` | `(width, height)` | Mouse up |
|
|
130
|
+
|
|
131
|
+
```javascript
|
|
132
|
+
MyPanel
|
|
133
|
+
.on('onResizeStart', (e, w, h) => console.log('Started at', w, h))
|
|
134
|
+
.on('onResize', (e, w, h) => saveSize(w, h))
|
|
135
|
+
.on('onResizeEnd', (w, h) => persist({ width: w, height: h }))
|
|
136
|
+
```
|
|
137
|
+
|
|
138
|
+
---
|
|
139
|
+
|
|
140
|
+
## `HasItems`
|
|
141
|
+
|
|
142
|
+
Manages a reactive collection of items. Used internally by `Accordion`, `Menu`, `Tabs`, `Dropdown`.
|
|
143
|
+
|
|
144
|
+
### Methods
|
|
145
|
+
|
|
146
|
+
| Method | Parameters | Description |
|
|
147
|
+
|---|---|---|
|
|
148
|
+
| `.dynamic(observableArray?)` | `observableArray?: ObservableArray` | Binds to an existing array or creates a new `Observable.array()`. |
|
|
149
|
+
| `.bind(observableArray?)` | same as `dynamic` | Alias for `dynamic`. |
|
|
150
|
+
| `.items(items)` | `items: array` | Replaces all current items. |
|
|
151
|
+
| `.clear()` | - | Removes all items. |
|
|
152
|
+
| `.removeItem(item)` | `item: *` | Removes a specific item by reference. |
|
|
153
|
+
|
|
154
|
+
### Example
|
|
155
|
+
|
|
156
|
+
```javascript
|
|
157
|
+
import { BaseComponent, HasEventEmitter, HasItems } from '@native-document/components';
|
|
158
|
+
import { Div, Span } from 'native-document/elements';
|
|
159
|
+
import { ForEach } from 'native-document/elements';
|
|
160
|
+
|
|
161
|
+
function TagList(props = {}) {
|
|
162
|
+
if (!(this instanceof TagList)) return new TagList(props);
|
|
163
|
+
BaseComponent.call(this);
|
|
164
|
+
this.$description = { items: [], props };
|
|
165
|
+
}
|
|
166
|
+
|
|
167
|
+
BaseComponent.extends(TagList);
|
|
168
|
+
BaseComponent.use(TagList, HasEventEmitter, HasItems);
|
|
169
|
+
|
|
170
|
+
TagList.use(($d) => {
|
|
171
|
+
return Div({ class: 'tag-list' },
|
|
172
|
+
ForEach($d.items, tag => Span({ class: 'tag' }, tag))
|
|
173
|
+
);
|
|
174
|
+
});
|
|
175
|
+
|
|
176
|
+
TagList()
|
|
177
|
+
.dynamic()
|
|
178
|
+
.items(['JavaScript', 'TypeScript', 'Python'])
|
|
179
|
+
```
|
|
180
|
+
|
|
181
|
+
---
|
|
182
|
+
|
|
183
|
+
## `HasPosition` / `HasFullPosition`
|
|
184
|
+
|
|
185
|
+
Position shorthand methods for components that accept a `position` prop (`Tooltip`, `Popover`, `Toast`). Applied internally - you do not need to apply them manually.
|
|
186
|
+
|
|
187
|
+
### `HasPosition` - 4 directions + center
|
|
188
|
+
|
|
189
|
+
| Method | Sets `position` to |
|
|
190
|
+
|---|---|
|
|
191
|
+
| `.atTop()` | `'top'` |
|
|
192
|
+
| `.atBottom()` | `'bottom'` |
|
|
193
|
+
| `.atLeft()` | `'left'` |
|
|
194
|
+
| `.atRight()` | `'right'` |
|
|
195
|
+
| `.atCenter()` | `'center'` |
|
|
196
|
+
|
|
197
|
+
### `HasFullPosition` - all positions
|
|
198
|
+
|
|
199
|
+
| Method | Sets `position` to |
|
|
200
|
+
|---|---|
|
|
201
|
+
| `.atTop()` | `'top'` |
|
|
202
|
+
| `.atBottom()` | `'bottom'` |
|
|
203
|
+
| `.atLeft()` | `'left'` |
|
|
204
|
+
| `.atRight()` | `'right'` |
|
|
205
|
+
| `.atTopLeading()` | `'top-leading'` |
|
|
206
|
+
| `.atTopTrailing()` | `'top-trailing'` |
|
|
207
|
+
| `.atTopCenter()` | `'top-center'` |
|
|
208
|
+
| `.atBottomLeading()` | `'bottom-leading'` |
|
|
209
|
+
| `.atBottomTrailing()` | `'bottom-trailing'` |
|
|
210
|
+
| `.atBottomCenter()` | `'bottom-center'` |
|
|
211
|
+
| `.atLeadingCenter()` | `'leading-center'` |
|
|
212
|
+
| `.atTrailingCenter()` | `'trailing-center'` |
|
|
213
|
+
|
|
214
|
+
---
|
|
215
|
+
|
|
216
|
+
## Building Custom Components with Traits
|
|
217
|
+
|
|
218
|
+
```javascript
|
|
219
|
+
import { BaseComponent, HasEventEmitter, HasDraggable } from '@native-document/components';
|
|
220
|
+
import { Div, H2 } from 'native-document/elements';
|
|
221
|
+
|
|
222
|
+
function FloatingPanel(title, content, props = {}) {
|
|
223
|
+
if (!(this instanceof FloatingPanel)) {
|
|
224
|
+
return new FloatingPanel(title, content, props);
|
|
225
|
+
}
|
|
226
|
+
BaseComponent.call(this);
|
|
227
|
+
this.$description = { title, content, props };
|
|
228
|
+
}
|
|
229
|
+
|
|
230
|
+
FloatingPanel.defaultTemplate = null;
|
|
231
|
+
FloatingPanel.use = function(template) {
|
|
232
|
+
FloatingPanel.defaultTemplate = template;
|
|
233
|
+
};
|
|
234
|
+
|
|
235
|
+
BaseComponent.extends(FloatingPanel);
|
|
236
|
+
BaseComponent.use(FloatingPanel, HasEventEmitter, HasDraggable);
|
|
237
|
+
|
|
238
|
+
FloatingPanel.use(($d, component) => {
|
|
239
|
+
const header = Div({ class: 'panel-header' }, H2($d.title));
|
|
240
|
+
const body = Div({ class: 'panel-body' }, $d.content);
|
|
241
|
+
const panel = Div({ class: 'panel', style: 'position: absolute', ...$d.props }, [header, body]);
|
|
242
|
+
|
|
243
|
+
component.postBuild(() => {
|
|
244
|
+
component.makeDraggable(panel, header);
|
|
245
|
+
});
|
|
246
|
+
|
|
247
|
+
return panel;
|
|
248
|
+
});
|
|
249
|
+
|
|
250
|
+
FloatingPanel('Settings', Div('Panel content'))
|
|
251
|
+
.on('onDragEnd', () => console.log('Moved'))
|
|
252
|
+
.nd
|
|
253
|
+
```
|
|
254
|
+
|
|
255
|
+
---
|
|
256
|
+
|
|
257
|
+
## Next Steps
|
|
258
|
+
|
|
259
|
+
- **[Components Overview](./index.md)** - BaseComponent philosophy
|
|
260
|
+
- **[Modal](./modal.md)** - Uses HasEventEmitter, HasDraggable, HasResizable
|
|
261
|
+
- **[Getting Started](./getting-started.md)** - Register renderers
|