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.
Files changed (108) hide show
  1. package/.vitepress/config.js +166 -0
  2. package/CHANGELOG.md +153 -0
  3. package/components.js +2 -1
  4. package/dist/native-document.components.min.js +495 -228
  5. package/dist/native-document.dev.js +7 -0
  6. package/dist/native-document.dev.js.map +1 -1
  7. package/dist/native-document.min.js +1 -1
  8. package/docs/advanced-components.md +213 -608
  9. package/docs/anchor.md +173 -312
  10. package/docs/cache.md +95 -803
  11. package/docs/cli.md +179 -0
  12. package/docs/components/accordion.md +172 -0
  13. package/docs/components/alert.md +99 -0
  14. package/docs/components/avatar.md +160 -0
  15. package/docs/components/badge.md +102 -0
  16. package/docs/components/breadcrumb.md +89 -0
  17. package/docs/components/button.md +183 -0
  18. package/docs/components/card.md +69 -0
  19. package/docs/components/context-menu.md +118 -0
  20. package/docs/components/data-table.md +345 -0
  21. package/docs/components/dropdown.md +214 -0
  22. package/docs/components/form/autocomplete-field.md +81 -0
  23. package/docs/components/form/checkbox-field.md +41 -0
  24. package/docs/components/form/checkbox-group-field.md +54 -0
  25. package/docs/components/form/color-field.md +64 -0
  26. package/docs/components/form/date-field.md +92 -0
  27. package/docs/components/form/field-collection.md +63 -0
  28. package/docs/components/form/file-field.md +203 -0
  29. package/docs/components/form/form-control.md +87 -0
  30. package/docs/components/form/image-field.md +90 -0
  31. package/docs/components/form/index.md +115 -0
  32. package/docs/components/form/number-field.md +65 -0
  33. package/docs/components/form/radio-field.md +51 -0
  34. package/docs/components/form/select-field.md +123 -0
  35. package/docs/components/form/slider.md +136 -0
  36. package/docs/components/form/string-field.md +134 -0
  37. package/docs/components/form/textarea-field.md +65 -0
  38. package/docs/components/form-fields.md +372 -0
  39. package/docs/components/getting-started.md +264 -0
  40. package/docs/components/index.md +337 -0
  41. package/docs/components/layout.md +279 -0
  42. package/docs/components/list.md +73 -0
  43. package/docs/components/menu.md +215 -0
  44. package/docs/components/modal.md +156 -0
  45. package/docs/components/pagination.md +95 -0
  46. package/docs/components/popover.md +131 -0
  47. package/docs/components/progress.md +111 -0
  48. package/docs/components/shortcut-manager.md +221 -0
  49. package/docs/components/simple-table.md +107 -0
  50. package/docs/components/skeleton.md +155 -0
  51. package/docs/components/spinner.md +100 -0
  52. package/docs/components/splitter.md +133 -0
  53. package/docs/components/stepper.md +163 -0
  54. package/docs/components/switch.md +113 -0
  55. package/docs/components/tabs.md +153 -0
  56. package/docs/components/toast.md +119 -0
  57. package/docs/components/tooltip.md +151 -0
  58. package/docs/components/traits.md +261 -0
  59. package/docs/conditional-rendering.md +170 -588
  60. package/docs/contributing.md +300 -25
  61. package/docs/core-concepts.md +205 -374
  62. package/docs/elements.md +251 -367
  63. package/docs/extending-native-document-element.md +192 -207
  64. package/docs/filters.md +153 -1122
  65. package/docs/getting-started.md +193 -267
  66. package/docs/i18n.md +241 -0
  67. package/docs/index.md +76 -0
  68. package/docs/lifecycle-events.md +143 -75
  69. package/docs/list-rendering.md +227 -852
  70. package/docs/memory-management.md +134 -47
  71. package/docs/native-document-element.md +337 -186
  72. package/docs/native-fetch.md +99 -630
  73. package/docs/observable-resource.md +364 -0
  74. package/docs/observables.md +592 -526
  75. package/docs/routing.md +244 -653
  76. package/docs/state-management.md +134 -241
  77. package/docs/svg-elements.md +231 -0
  78. package/docs/theming.md +409 -0
  79. package/docs/tutorials/.gitkeep +0 -0
  80. package/docs/validation.md +95 -97
  81. package/docs/vitepress-conventions.md +219 -0
  82. package/package.json +34 -13
  83. package/readme.md +269 -89
  84. package/src/components/card/Card.js +93 -39
  85. package/src/components/card/index.js +1 -1
  86. package/src/components/list/HasListItem.js +171 -0
  87. package/src/components/list/List.js +41 -107
  88. package/src/components/list/ListDivider.js +39 -0
  89. package/src/components/list/ListGroup.js +76 -59
  90. package/src/components/list/ListItem.js +117 -69
  91. package/src/components/list/index.js +3 -1
  92. package/src/components/list/types/ListItem.d.ts +45 -34
  93. package/src/components/spacer/Spacer.js +1 -1
  94. package/src/core/data/ObservableResource.js +5 -0
  95. package/src/core/data/observable-helpers/observable.prototypes.js +2 -0
  96. package/src/ui/components/card/CardRender.js +133 -0
  97. package/src/ui/components/card/card.css +169 -0
  98. package/src/ui/components/contextmenu/ContextmenuRender.js +1 -1
  99. package/src/ui/components/list/ListRender.js +18 -0
  100. package/src/ui/components/list/divider/ListDividerRender.js +10 -0
  101. package/src/ui/components/list/divider/list-divider.css +12 -0
  102. package/src/ui/components/list/group/ListGroupRender.js +61 -0
  103. package/src/ui/components/list/group/list-group.css +62 -0
  104. package/src/ui/components/list/item/ListItemRender.js +238 -0
  105. package/src/ui/components/list/item/list-item.css +191 -0
  106. package/src/ui/components/list/list.css +24 -0
  107. package/src/ui/components/spacer/SpacerRender.js +10 -0
  108. 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