@llui/components 0.0.1

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 (213) hide show
  1. package/LICENSE +21 -0
  2. package/README.md +143 -0
  3. package/dist/components/accordion.d.ts +115 -0
  4. package/dist/components/accordion.d.ts.map +1 -0
  5. package/dist/components/accordion.js +138 -0
  6. package/dist/components/alert-dialog.d.ts +45 -0
  7. package/dist/components/alert-dialog.d.ts.map +1 -0
  8. package/dist/components/alert-dialog.js +12 -0
  9. package/dist/components/angle-slider.d.ts +121 -0
  10. package/dist/components/angle-slider.d.ts.map +1 -0
  11. package/dist/components/angle-slider.js +145 -0
  12. package/dist/components/async-list.d.ts +104 -0
  13. package/dist/components/async-list.d.ts.map +1 -0
  14. package/dist/components/async-list.js +117 -0
  15. package/dist/components/avatar.d.ts +58 -0
  16. package/dist/components/avatar.d.ts.map +1 -0
  17. package/dist/components/avatar.js +43 -0
  18. package/dist/components/carousel.d.ts +128 -0
  19. package/dist/components/carousel.d.ts.map +1 -0
  20. package/dist/components/carousel.js +131 -0
  21. package/dist/components/cascade-select.d.ts +95 -0
  22. package/dist/components/cascade-select.d.ts.map +1 -0
  23. package/dist/components/cascade-select.js +100 -0
  24. package/dist/components/checkbox.d.ts +74 -0
  25. package/dist/components/checkbox.d.ts.map +1 -0
  26. package/dist/components/checkbox.js +73 -0
  27. package/dist/components/clipboard.d.ts +72 -0
  28. package/dist/components/clipboard.d.ts.map +1 -0
  29. package/dist/components/clipboard.js +73 -0
  30. package/dist/components/collapsible.d.ts +64 -0
  31. package/dist/components/collapsible.d.ts.map +1 -0
  32. package/dist/components/collapsible.js +51 -0
  33. package/dist/components/color-picker.d.ts +125 -0
  34. package/dist/components/color-picker.d.ts.map +1 -0
  35. package/dist/components/color-picker.js +169 -0
  36. package/dist/components/combobox.d.ts +163 -0
  37. package/dist/components/combobox.d.ts.map +1 -0
  38. package/dist/components/combobox.js +345 -0
  39. package/dist/components/context-menu.d.ts +105 -0
  40. package/dist/components/context-menu.d.ts.map +1 -0
  41. package/dist/components/context-menu.js +177 -0
  42. package/dist/components/date-input.d.ts +117 -0
  43. package/dist/components/date-input.d.ts.map +1 -0
  44. package/dist/components/date-input.js +149 -0
  45. package/dist/components/date-picker.d.ts +142 -0
  46. package/dist/components/date-picker.d.ts.map +1 -0
  47. package/dist/components/date-picker.js +294 -0
  48. package/dist/components/dialog.d.ts +152 -0
  49. package/dist/components/dialog.d.ts.map +1 -0
  50. package/dist/components/dialog.js +140 -0
  51. package/dist/components/drawer.d.ts +106 -0
  52. package/dist/components/drawer.d.ts.map +1 -0
  53. package/dist/components/drawer.js +136 -0
  54. package/dist/components/editable.d.ts +92 -0
  55. package/dist/components/editable.d.ts.map +1 -0
  56. package/dist/components/editable.js +112 -0
  57. package/dist/components/file-upload.d.ts +251 -0
  58. package/dist/components/file-upload.d.ts.map +1 -0
  59. package/dist/components/file-upload.js +324 -0
  60. package/dist/components/floating-panel.d.ts +171 -0
  61. package/dist/components/floating-panel.d.ts.map +1 -0
  62. package/dist/components/floating-panel.js +198 -0
  63. package/dist/components/hover-card.d.ts +85 -0
  64. package/dist/components/hover-card.d.ts.map +1 -0
  65. package/dist/components/hover-card.js +128 -0
  66. package/dist/components/image-cropper.d.ts +129 -0
  67. package/dist/components/image-cropper.d.ts.map +1 -0
  68. package/dist/components/image-cropper.js +208 -0
  69. package/dist/components/index.d.ts +109 -0
  70. package/dist/components/index.d.ts.map +1 -0
  71. package/dist/components/index.js +54 -0
  72. package/dist/components/listbox.d.ts +98 -0
  73. package/dist/components/listbox.d.ts.map +1 -0
  74. package/dist/components/listbox.js +174 -0
  75. package/dist/components/marquee.d.ts +84 -0
  76. package/dist/components/marquee.d.ts.map +1 -0
  77. package/dist/components/marquee.js +73 -0
  78. package/dist/components/menu.d.ts +131 -0
  79. package/dist/components/menu.d.ts.map +1 -0
  80. package/dist/components/menu.js +262 -0
  81. package/dist/components/navigation-menu.d.ts +111 -0
  82. package/dist/components/navigation-menu.d.ts.map +1 -0
  83. package/dist/components/navigation-menu.js +102 -0
  84. package/dist/components/number-input.d.ts +106 -0
  85. package/dist/components/number-input.d.ts.map +1 -0
  86. package/dist/components/number-input.js +178 -0
  87. package/dist/components/pagination.d.ts +113 -0
  88. package/dist/components/pagination.d.ts.map +1 -0
  89. package/dist/components/pagination.js +135 -0
  90. package/dist/components/password-input.d.ts +64 -0
  91. package/dist/components/password-input.d.ts.map +1 -0
  92. package/dist/components/password-input.js +52 -0
  93. package/dist/components/pin-input.d.ts +89 -0
  94. package/dist/components/pin-input.d.ts.map +1 -0
  95. package/dist/components/pin-input.js +139 -0
  96. package/dist/components/popover.d.ts +116 -0
  97. package/dist/components/popover.d.ts.map +1 -0
  98. package/dist/components/popover.js +146 -0
  99. package/dist/components/presence.d.ts +71 -0
  100. package/dist/components/presence.d.ts.map +1 -0
  101. package/dist/components/presence.js +57 -0
  102. package/dist/components/progress.d.ts +74 -0
  103. package/dist/components/progress.d.ts.map +1 -0
  104. package/dist/components/progress.js +80 -0
  105. package/dist/components/qr-code.d.ts +114 -0
  106. package/dist/components/qr-code.d.ts.map +1 -0
  107. package/dist/components/qr-code.js +108 -0
  108. package/dist/components/radio-group.d.ts +89 -0
  109. package/dist/components/radio-group.d.ts.map +1 -0
  110. package/dist/components/radio-group.js +161 -0
  111. package/dist/components/rating-group.d.ts +88 -0
  112. package/dist/components/rating-group.d.ts.map +1 -0
  113. package/dist/components/rating-group.js +122 -0
  114. package/dist/components/scroll-area.d.ts +124 -0
  115. package/dist/components/scroll-area.d.ts.map +1 -0
  116. package/dist/components/scroll-area.js +152 -0
  117. package/dist/components/select.d.ts +161 -0
  118. package/dist/components/select.d.ts.map +1 -0
  119. package/dist/components/select.js +333 -0
  120. package/dist/components/signature-pad.d.ts +138 -0
  121. package/dist/components/signature-pad.d.ts.map +1 -0
  122. package/dist/components/signature-pad.js +142 -0
  123. package/dist/components/slider.d.ts +117 -0
  124. package/dist/components/slider.d.ts.map +1 -0
  125. package/dist/components/slider.js +210 -0
  126. package/dist/components/splitter.d.ts +87 -0
  127. package/dist/components/splitter.d.ts.map +1 -0
  128. package/dist/components/splitter.js +119 -0
  129. package/dist/components/steps.d.ts +104 -0
  130. package/dist/components/steps.d.ts.map +1 -0
  131. package/dist/components/steps.js +133 -0
  132. package/dist/components/switch.d.ts +66 -0
  133. package/dist/components/switch.d.ts.map +1 -0
  134. package/dist/components/switch.js +59 -0
  135. package/dist/components/tabs.d.ts +146 -0
  136. package/dist/components/tabs.d.ts.map +1 -0
  137. package/dist/components/tabs.js +244 -0
  138. package/dist/components/tags-input.d.ts +118 -0
  139. package/dist/components/tags-input.d.ts.map +1 -0
  140. package/dist/components/tags-input.js +168 -0
  141. package/dist/components/time-picker.d.ts +121 -0
  142. package/dist/components/time-picker.d.ts.map +1 -0
  143. package/dist/components/time-picker.js +147 -0
  144. package/dist/components/timer.d.ts +131 -0
  145. package/dist/components/timer.d.ts.map +1 -0
  146. package/dist/components/timer.js +117 -0
  147. package/dist/components/toast.d.ts +119 -0
  148. package/dist/components/toast.d.ts.map +1 -0
  149. package/dist/components/toast.js +102 -0
  150. package/dist/components/toc.d.ts +119 -0
  151. package/dist/components/toc.d.ts.map +1 -0
  152. package/dist/components/toc.js +107 -0
  153. package/dist/components/toggle-group.d.ts +80 -0
  154. package/dist/components/toggle-group.d.ts.map +1 -0
  155. package/dist/components/toggle-group.js +93 -0
  156. package/dist/components/toggle.d.ts +47 -0
  157. package/dist/components/toggle.d.ts.map +1 -0
  158. package/dist/components/toggle.js +41 -0
  159. package/dist/components/tooltip.d.ts +92 -0
  160. package/dist/components/tooltip.d.ts.map +1 -0
  161. package/dist/components/tooltip.js +147 -0
  162. package/dist/components/tour.d.ts +145 -0
  163. package/dist/components/tour.d.ts.map +1 -0
  164. package/dist/components/tour.js +133 -0
  165. package/dist/components/tree-view.d.ts +216 -0
  166. package/dist/components/tree-view.d.ts.map +1 -0
  167. package/dist/components/tree-view.js +293 -0
  168. package/dist/index.d.ts +3 -0
  169. package/dist/index.d.ts.map +1 -0
  170. package/dist/index.js +4 -0
  171. package/dist/patterns/confirm-dialog.d.ts +92 -0
  172. package/dist/patterns/confirm-dialog.d.ts.map +1 -0
  173. package/dist/patterns/confirm-dialog.js +92 -0
  174. package/dist/patterns/index.d.ts +3 -0
  175. package/dist/patterns/index.d.ts.map +1 -0
  176. package/dist/patterns/index.js +1 -0
  177. package/dist/utils/anatomy.d.ts +40 -0
  178. package/dist/utils/anatomy.d.ts.map +1 -0
  179. package/dist/utils/anatomy.js +41 -0
  180. package/dist/utils/aria-hidden.d.ts +12 -0
  181. package/dist/utils/aria-hidden.d.ts.map +1 -0
  182. package/dist/utils/aria-hidden.js +72 -0
  183. package/dist/utils/dismissable.d.ts +25 -0
  184. package/dist/utils/dismissable.d.ts.map +1 -0
  185. package/dist/utils/dismissable.js +65 -0
  186. package/dist/utils/dom.d.ts +8 -0
  187. package/dist/utils/dom.d.ts.map +1 -0
  188. package/dist/utils/dom.js +21 -0
  189. package/dist/utils/floating.d.ts +44 -0
  190. package/dist/utils/floating.d.ts.map +1 -0
  191. package/dist/utils/floating.js +44 -0
  192. package/dist/utils/focus-trap.d.ts +18 -0
  193. package/dist/utils/focus-trap.d.ts.map +1 -0
  194. package/dist/utils/focus-trap.js +85 -0
  195. package/dist/utils/focusables.d.ts +6 -0
  196. package/dist/utils/focusables.d.ts.map +1 -0
  197. package/dist/utils/focusables.js +65 -0
  198. package/dist/utils/index.d.ts +18 -0
  199. package/dist/utils/index.d.ts.map +1 -0
  200. package/dist/utils/index.js +10 -0
  201. package/dist/utils/interact-outside.d.ts +26 -0
  202. package/dist/utils/interact-outside.d.ts.map +1 -0
  203. package/dist/utils/interact-outside.js +46 -0
  204. package/dist/utils/remove-scroll.d.ts +8 -0
  205. package/dist/utils/remove-scroll.d.ts.map +1 -0
  206. package/dist/utils/remove-scroll.js +37 -0
  207. package/dist/utils/tree-collection.d.ts +61 -0
  208. package/dist/utils/tree-collection.d.ts.map +1 -0
  209. package/dist/utils/tree-collection.js +137 -0
  210. package/dist/utils/typeahead.d.ts +49 -0
  211. package/dist/utils/typeahead.d.ts.map +1 -0
  212. package/dist/utils/typeahead.js +81 -0
  213. package/package.json +282 -0
@@ -0,0 +1,171 @@
1
+ import type { Send } from '@llui/dom';
2
+ /**
3
+ * Floating panel — a draggable + resizable window-like surface, useful
4
+ * for dev tools overlays, pop-out inspectors, preview panels, etc. The
5
+ * state machine tracks position and size; the view layer wires pointer
6
+ * events on the drag handle and resize grips and dispatches the
7
+ * corresponding messages.
8
+ *
9
+ * Coordinates are in pixels relative to the positioning container
10
+ * (typically `position: fixed` relative to the viewport).
11
+ */
12
+ export type ResizeHandle = 'n' | 's' | 'e' | 'w' | 'ne' | 'nw' | 'se' | 'sw';
13
+ export interface FloatingPanelState {
14
+ position: {
15
+ x: number;
16
+ y: number;
17
+ };
18
+ size: {
19
+ width: number;
20
+ height: number;
21
+ };
22
+ minSize: {
23
+ width: number;
24
+ height: number;
25
+ };
26
+ maxSize: {
27
+ width: number;
28
+ height: number;
29
+ } | null;
30
+ open: boolean;
31
+ minimized: boolean;
32
+ maximized: boolean;
33
+ dragging: boolean;
34
+ resizing: ResizeHandle | null;
35
+ /** Snapshot of the pre-maximize geometry (for restore). */
36
+ restoreBounds: {
37
+ x: number;
38
+ y: number;
39
+ width: number;
40
+ height: number;
41
+ } | null;
42
+ disabled: boolean;
43
+ }
44
+ export type FloatingPanelMsg = {
45
+ type: 'open';
46
+ } | {
47
+ type: 'close';
48
+ } | {
49
+ type: 'minimize';
50
+ } | {
51
+ type: 'restoreFromMinimized';
52
+ } | {
53
+ type: 'maximize';
54
+ } | {
55
+ type: 'restoreFromMaximized';
56
+ } | {
57
+ type: 'toggleMinimize';
58
+ } | {
59
+ type: 'toggleMaximize';
60
+ } | {
61
+ type: 'dragStart';
62
+ } | {
63
+ type: 'dragMove';
64
+ dx: number;
65
+ dy: number;
66
+ } | {
67
+ type: 'dragEnd';
68
+ } | {
69
+ type: 'resizeStart';
70
+ handle: ResizeHandle;
71
+ } | {
72
+ type: 'resizeMove';
73
+ dx: number;
74
+ dy: number;
75
+ } | {
76
+ type: 'resizeEnd';
77
+ } | {
78
+ type: 'setPosition';
79
+ x: number;
80
+ y: number;
81
+ } | {
82
+ type: 'setSize';
83
+ width: number;
84
+ height: number;
85
+ };
86
+ export interface FloatingPanelInit {
87
+ position?: {
88
+ x: number;
89
+ y: number;
90
+ };
91
+ size?: {
92
+ width: number;
93
+ height: number;
94
+ };
95
+ minSize?: {
96
+ width: number;
97
+ height: number;
98
+ };
99
+ maxSize?: {
100
+ width: number;
101
+ height: number;
102
+ } | null;
103
+ open?: boolean;
104
+ disabled?: boolean;
105
+ }
106
+ export declare function init(opts?: FloatingPanelInit): FloatingPanelState;
107
+ export declare function update(state: FloatingPanelState, msg: FloatingPanelMsg): [FloatingPanelState, never[]];
108
+ export interface FloatingPanelParts<S> {
109
+ root: {
110
+ role: 'dialog';
111
+ 'aria-label': string;
112
+ 'data-scope': 'floating-panel';
113
+ 'data-part': 'root';
114
+ 'data-dragging': (s: S) => '' | undefined;
115
+ 'data-resizing': (s: S) => '' | undefined;
116
+ 'data-minimized': (s: S) => '' | undefined;
117
+ 'data-maximized': (s: S) => '' | undefined;
118
+ hidden: (s: S) => boolean;
119
+ style: (s: S) => string;
120
+ };
121
+ dragHandle: {
122
+ 'data-scope': 'floating-panel';
123
+ 'data-part': 'drag-handle';
124
+ onPointerDown: (e: PointerEvent) => void;
125
+ };
126
+ content: {
127
+ 'data-scope': 'floating-panel';
128
+ 'data-part': 'content';
129
+ hidden: (s: S) => boolean;
130
+ };
131
+ minimizeTrigger: {
132
+ type: 'button';
133
+ 'aria-label': string;
134
+ 'data-scope': 'floating-panel';
135
+ 'data-part': 'minimize-trigger';
136
+ onClick: (e: MouseEvent) => void;
137
+ };
138
+ maximizeTrigger: {
139
+ type: 'button';
140
+ 'aria-label': string;
141
+ 'data-scope': 'floating-panel';
142
+ 'data-part': 'maximize-trigger';
143
+ onClick: (e: MouseEvent) => void;
144
+ };
145
+ closeTrigger: {
146
+ type: 'button';
147
+ 'aria-label': string;
148
+ 'data-scope': 'floating-panel';
149
+ 'data-part': 'close-trigger';
150
+ onClick: (e: MouseEvent) => void;
151
+ };
152
+ resizeHandle: (handle: ResizeHandle) => {
153
+ 'data-scope': 'floating-panel';
154
+ 'data-part': 'resize-handle';
155
+ 'data-handle': ResizeHandle;
156
+ onPointerDown: (e: PointerEvent) => void;
157
+ };
158
+ }
159
+ export interface ConnectOptions {
160
+ label?: string;
161
+ minimizeLabel?: string;
162
+ maximizeLabel?: string;
163
+ closeLabel?: string;
164
+ }
165
+ export declare function connect<S>(get: (s: S) => FloatingPanelState, send: Send<FloatingPanelMsg>, opts?: ConnectOptions): FloatingPanelParts<S>;
166
+ export declare const floatingPanel: {
167
+ init: typeof init;
168
+ update: typeof update;
169
+ connect: typeof connect;
170
+ };
171
+ //# sourceMappingURL=floating-panel.d.ts.map
@@ -0,0 +1 @@
1
+ {"version":3,"file":"floating-panel.d.ts","sourceRoot":"","sources":["../../src/components/floating-panel.ts"],"names":[],"mappings":"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"}
@@ -0,0 +1,198 @@
1
+ function clamp(v, min, max) {
2
+ return Math.max(min, Math.min(max, v));
3
+ }
4
+ export function init(opts = {}) {
5
+ return {
6
+ position: opts.position ?? { x: 100, y: 100 },
7
+ size: opts.size ?? { width: 400, height: 300 },
8
+ minSize: opts.minSize ?? { width: 200, height: 150 },
9
+ maxSize: opts.maxSize ?? null,
10
+ open: opts.open ?? true,
11
+ minimized: false,
12
+ maximized: false,
13
+ dragging: false,
14
+ resizing: null,
15
+ restoreBounds: null,
16
+ disabled: opts.disabled ?? false,
17
+ };
18
+ }
19
+ function clampSize(width, height, min, max) {
20
+ const maxW = max?.width ?? Infinity;
21
+ const maxH = max?.height ?? Infinity;
22
+ return {
23
+ width: clamp(width, min.width, maxW),
24
+ height: clamp(height, min.height, maxH),
25
+ };
26
+ }
27
+ function applyResize(state, dx, dy, handle) {
28
+ let { x, y } = state.position;
29
+ let { width, height } = state.size;
30
+ if (handle.includes('e'))
31
+ width += dx;
32
+ if (handle.includes('w')) {
33
+ width -= dx;
34
+ x += dx;
35
+ }
36
+ if (handle.includes('s'))
37
+ height += dy;
38
+ if (handle.includes('n')) {
39
+ height -= dy;
40
+ y += dy;
41
+ }
42
+ const size = clampSize(width, height, state.minSize, state.maxSize);
43
+ // If clamping changed width/height, undo the x/y shift by that delta.
44
+ if (handle.includes('w'))
45
+ x += width - size.width;
46
+ if (handle.includes('n'))
47
+ y += height - size.height;
48
+ return { ...state, position: { x, y }, size };
49
+ }
50
+ export function update(state, msg) {
51
+ if (state.disabled)
52
+ return [state, []];
53
+ switch (msg.type) {
54
+ case 'open':
55
+ return [{ ...state, open: true }, []];
56
+ case 'close':
57
+ return [{ ...state, open: false, dragging: false, resizing: null }, []];
58
+ case 'minimize':
59
+ return [{ ...state, minimized: true, dragging: false, resizing: null }, []];
60
+ case 'restoreFromMinimized':
61
+ return [{ ...state, minimized: false }, []];
62
+ case 'toggleMinimize':
63
+ return [{ ...state, minimized: !state.minimized }, []];
64
+ case 'maximize': {
65
+ if (state.maximized)
66
+ return [state, []];
67
+ return [
68
+ {
69
+ ...state,
70
+ maximized: true,
71
+ restoreBounds: {
72
+ x: state.position.x,
73
+ y: state.position.y,
74
+ width: state.size.width,
75
+ height: state.size.height,
76
+ },
77
+ },
78
+ [],
79
+ ];
80
+ }
81
+ case 'restoreFromMaximized': {
82
+ if (!state.maximized || !state.restoreBounds) {
83
+ return [{ ...state, maximized: false }, []];
84
+ }
85
+ const b = state.restoreBounds;
86
+ return [
87
+ {
88
+ ...state,
89
+ maximized: false,
90
+ position: { x: b.x, y: b.y },
91
+ size: { width: b.width, height: b.height },
92
+ restoreBounds: null,
93
+ },
94
+ [],
95
+ ];
96
+ }
97
+ case 'toggleMaximize':
98
+ return update(state, {
99
+ type: state.maximized ? 'restoreFromMaximized' : 'maximize',
100
+ });
101
+ case 'dragStart':
102
+ if (state.maximized)
103
+ return [state, []];
104
+ return [{ ...state, dragging: true }, []];
105
+ case 'dragMove':
106
+ if (!state.dragging)
107
+ return [state, []];
108
+ return [
109
+ {
110
+ ...state,
111
+ position: {
112
+ x: state.position.x + msg.dx,
113
+ y: state.position.y + msg.dy,
114
+ },
115
+ },
116
+ [],
117
+ ];
118
+ case 'dragEnd':
119
+ return [{ ...state, dragging: false }, []];
120
+ case 'resizeStart':
121
+ if (state.maximized)
122
+ return [state, []];
123
+ return [{ ...state, resizing: msg.handle }, []];
124
+ case 'resizeMove':
125
+ if (state.resizing === null)
126
+ return [state, []];
127
+ return [applyResize(state, msg.dx, msg.dy, state.resizing), []];
128
+ case 'resizeEnd':
129
+ return [{ ...state, resizing: null }, []];
130
+ case 'setPosition':
131
+ return [{ ...state, position: { x: msg.x, y: msg.y } }, []];
132
+ case 'setSize': {
133
+ const size = clampSize(msg.width, msg.height, state.minSize, state.maxSize);
134
+ return [{ ...state, size }, []];
135
+ }
136
+ }
137
+ }
138
+ export function connect(get, send, opts = {}) {
139
+ return {
140
+ root: {
141
+ role: 'dialog',
142
+ 'aria-label': opts.label ?? 'Floating panel',
143
+ 'data-scope': 'floating-panel',
144
+ 'data-part': 'root',
145
+ 'data-dragging': (s) => (get(s).dragging ? '' : undefined),
146
+ 'data-resizing': (s) => (get(s).resizing !== null ? '' : undefined),
147
+ 'data-minimized': (s) => (get(s).minimized ? '' : undefined),
148
+ 'data-maximized': (s) => (get(s).maximized ? '' : undefined),
149
+ hidden: (s) => !get(s).open,
150
+ style: (s) => {
151
+ const st = get(s);
152
+ if (st.maximized)
153
+ return 'position:fixed;inset:0;width:auto;height:auto;';
154
+ return (`position:fixed;` +
155
+ `left:${st.position.x}px;top:${st.position.y}px;` +
156
+ `width:${st.size.width}px;height:${st.size.height}px;`);
157
+ },
158
+ },
159
+ dragHandle: {
160
+ 'data-scope': 'floating-panel',
161
+ 'data-part': 'drag-handle',
162
+ onPointerDown: () => send({ type: 'dragStart' }),
163
+ },
164
+ content: {
165
+ 'data-scope': 'floating-panel',
166
+ 'data-part': 'content',
167
+ hidden: (s) => get(s).minimized,
168
+ },
169
+ minimizeTrigger: {
170
+ type: 'button',
171
+ 'aria-label': opts.minimizeLabel ?? 'Minimize',
172
+ 'data-scope': 'floating-panel',
173
+ 'data-part': 'minimize-trigger',
174
+ onClick: () => send({ type: 'toggleMinimize' }),
175
+ },
176
+ maximizeTrigger: {
177
+ type: 'button',
178
+ 'aria-label': opts.maximizeLabel ?? 'Maximize',
179
+ 'data-scope': 'floating-panel',
180
+ 'data-part': 'maximize-trigger',
181
+ onClick: () => send({ type: 'toggleMaximize' }),
182
+ },
183
+ closeTrigger: {
184
+ type: 'button',
185
+ 'aria-label': opts.closeLabel ?? 'Close',
186
+ 'data-scope': 'floating-panel',
187
+ 'data-part': 'close-trigger',
188
+ onClick: () => send({ type: 'close' }),
189
+ },
190
+ resizeHandle: (handle) => ({
191
+ 'data-scope': 'floating-panel',
192
+ 'data-part': 'resize-handle',
193
+ 'data-handle': handle,
194
+ onPointerDown: () => send({ type: 'resizeStart', handle }),
195
+ }),
196
+ };
197
+ }
198
+ export const floatingPanel = { init, update, connect };
@@ -0,0 +1,85 @@
1
+ import type { Send, TransitionOptions } from '@llui/dom';
2
+ import { type Placement } from '../utils/floating';
3
+ /**
4
+ * Hover card — richer tooltip-like popup triggered by hover or focus.
5
+ * Unlike tooltip, it uses `role="dialog"` (not `role="tooltip"`) and
6
+ * allows interactive content. Content can be hovered without closing.
7
+ */
8
+ export interface HoverCardState {
9
+ open: boolean;
10
+ }
11
+ export type HoverCardMsg = {
12
+ type: 'show';
13
+ } | {
14
+ type: 'hide';
15
+ } | {
16
+ type: 'setOpen';
17
+ open: boolean;
18
+ };
19
+ export interface HoverCardInit {
20
+ open?: boolean;
21
+ }
22
+ export declare function init(opts?: HoverCardInit): HoverCardState;
23
+ export declare function update(state: HoverCardState, msg: HoverCardMsg): [HoverCardState, never[]];
24
+ export interface HoverCardParts<S> {
25
+ trigger: {
26
+ id: string;
27
+ 'aria-haspopup': 'dialog';
28
+ 'aria-expanded': (s: S) => boolean;
29
+ 'aria-controls': string;
30
+ 'data-state': (s: S) => 'open' | 'closed';
31
+ 'data-scope': 'hover-card';
32
+ 'data-part': 'trigger';
33
+ onPointerEnter: (e: PointerEvent) => void;
34
+ onPointerLeave: (e: PointerEvent) => void;
35
+ onFocus: (e: FocusEvent) => void;
36
+ onBlur: (e: FocusEvent) => void;
37
+ };
38
+ positioner: {
39
+ 'data-scope': 'hover-card';
40
+ 'data-part': 'positioner';
41
+ style: string;
42
+ };
43
+ content: {
44
+ role: 'dialog';
45
+ id: string;
46
+ 'data-state': (s: S) => 'open' | 'closed';
47
+ 'data-scope': 'hover-card';
48
+ 'data-part': 'content';
49
+ onPointerEnter: (e: PointerEvent) => void;
50
+ onPointerLeave: (e: PointerEvent) => void;
51
+ };
52
+ arrow: {
53
+ 'data-scope': 'hover-card';
54
+ 'data-part': 'arrow';
55
+ };
56
+ }
57
+ export interface ConnectOptions {
58
+ id: string;
59
+ /** ms before showing on hover (default: 700). */
60
+ openDelay?: number;
61
+ /** ms before hiding after pointer leaves (default: 300). */
62
+ closeDelay?: number;
63
+ }
64
+ export declare function connect<S>(get: (s: S) => HoverCardState, send: Send<HoverCardMsg>, opts: ConnectOptions): HoverCardParts<S>;
65
+ export interface OverlayOptions<S> {
66
+ get: (s: S) => HoverCardState;
67
+ send: Send<HoverCardMsg>;
68
+ parts: HoverCardParts<S>;
69
+ content: () => Node[];
70
+ placement?: Placement;
71
+ offset?: number;
72
+ flip?: boolean;
73
+ shift?: boolean;
74
+ transition?: TransitionOptions;
75
+ target?: string | HTMLElement;
76
+ arrowSelector?: string;
77
+ }
78
+ export declare function overlay<S>(opts: OverlayOptions<S>): Node[];
79
+ export declare const hoverCard: {
80
+ init: typeof init;
81
+ update: typeof update;
82
+ connect: typeof connect;
83
+ overlay: typeof overlay;
84
+ };
85
+ //# sourceMappingURL=hover-card.d.ts.map
@@ -0,0 +1 @@
1
+ {"version":3,"file":"hover-card.d.ts","sourceRoot":"","sources":["../../src/components/hover-card.ts"],"names":[],"mappings":"AAAA,OAAO,KAAK,EAAE,IAAI,EAAE,iBAAiB,EAAE,MAAM,WAAW,CAAA;AAExD,OAAO,EAAkB,KAAK,SAAS,EAAE,MAAM,mBAAmB,CAAA;AAElE;;;;GAIG;AAEH,MAAM,WAAW,cAAc;IAC7B,IAAI,EAAE,OAAO,CAAA;CACd;AAED,MAAM,MAAM,YAAY,GAAG;IAAE,IAAI,EAAE,MAAM,CAAA;CAAE,GAAG;IAAE,IAAI,EAAE,MAAM,CAAA;CAAE,GAAG;IAAE,IAAI,EAAE,SAAS,CAAC;IAAC,IAAI,EAAE,OAAO,CAAA;CAAE,CAAA;AAEnG,MAAM,WAAW,aAAa;IAC5B,IAAI,CAAC,EAAE,OAAO,CAAA;CACf;AAED,wBAAgB,IAAI,CAAC,IAAI,GAAE,aAAkB,GAAG,cAAc,CAE7D;AAED,wBAAgB,MAAM,CAAC,KAAK,EAAE,cAAc,EAAE,GAAG,EAAE,YAAY,GAAG,CAAC,cAAc,EAAE,KAAK,EAAE,CAAC,CAS1F;AAED,MAAM,WAAW,cAAc,CAAC,CAAC;IAC/B,OAAO,EAAE;QACP,EAAE,EAAE,MAAM,CAAA;QACV,eAAe,EAAE,QAAQ,CAAA;QACzB,eAAe,EAAE,CAAC,CAAC,EAAE,CAAC,KAAK,OAAO,CAAA;QAClC,eAAe,EAAE,MAAM,CAAA;QACvB,YAAY,EAAE,CAAC,CAAC,EAAE,CAAC,KAAK,MAAM,GAAG,QAAQ,CAAA;QACzC,YAAY,EAAE,YAAY,CAAA;QAC1B,WAAW,EAAE,SAAS,CAAA;QACtB,cAAc,EAAE,CAAC,CAAC,EAAE,YAAY,KAAK,IAAI,CAAA;QACzC,cAAc,EAAE,CAAC,CAAC,EAAE,YAAY,KAAK,IAAI,CAAA;QACzC,OAAO,EAAE,CAAC,CAAC,EAAE,UAAU,KAAK,IAAI,CAAA;QAChC,MAAM,EAAE,CAAC,CAAC,EAAE,UAAU,KAAK,IAAI,CAAA;KAChC,CAAA;IACD,UAAU,EAAE;QACV,YAAY,EAAE,YAAY,CAAA;QAC1B,WAAW,EAAE,YAAY,CAAA;QACzB,KAAK,EAAE,MAAM,CAAA;KACd,CAAA;IACD,OAAO,EAAE;QACP,IAAI,EAAE,QAAQ,CAAA;QACd,EAAE,EAAE,MAAM,CAAA;QACV,YAAY,EAAE,CAAC,CAAC,EAAE,CAAC,KAAK,MAAM,GAAG,QAAQ,CAAA;QACzC,YAAY,EAAE,YAAY,CAAA;QAC1B,WAAW,EAAE,SAAS,CAAA;QACtB,cAAc,EAAE,CAAC,CAAC,EAAE,YAAY,KAAK,IAAI,CAAA;QACzC,cAAc,EAAE,CAAC,CAAC,EAAE,YAAY,KAAK,IAAI,CAAA;KAC1C,CAAA;IACD,KAAK,EAAE;QACL,YAAY,EAAE,YAAY,CAAA;QAC1B,WAAW,EAAE,OAAO,CAAA;KACrB,CAAA;CACF;AAED,MAAM,WAAW,cAAc;IAC7B,EAAE,EAAE,MAAM,CAAA;IACV,iDAAiD;IACjD,SAAS,CAAC,EAAE,MAAM,CAAA;IAClB,4DAA4D;IAC5D,UAAU,CAAC,EAAE,MAAM,CAAA;CACpB;AAED,wBAAgB,OAAO,CAAC,CAAC,EACvB,GAAG,EAAE,CAAC,CAAC,EAAE,CAAC,KAAK,cAAc,EAC7B,IAAI,EAAE,IAAI,CAAC,YAAY,CAAC,EACxB,IAAI,EAAE,cAAc,GACnB,cAAc,CAAC,CAAC,CAAC,CA2EnB;AAED,MAAM,WAAW,cAAc,CAAC,CAAC;IAC/B,GAAG,EAAE,CAAC,CAAC,EAAE,CAAC,KAAK,cAAc,CAAA;IAC7B,IAAI,EAAE,IAAI,CAAC,YAAY,CAAC,CAAA;IACxB,KAAK,EAAE,cAAc,CAAC,CAAC,CAAC,CAAA;IACxB,OAAO,EAAE,MAAM,IAAI,EAAE,CAAA;IACrB,SAAS,CAAC,EAAE,SAAS,CAAA;IACrB,MAAM,CAAC,EAAE,MAAM,CAAA;IACf,IAAI,CAAC,EAAE,OAAO,CAAA;IACd,KAAK,CAAC,EAAE,OAAO,CAAA;IACf,UAAU,CAAC,EAAE,iBAAiB,CAAA;IAC9B,MAAM,CAAC,EAAE,MAAM,GAAG,WAAW,CAAA;IAC7B,aAAa,CAAC,EAAE,MAAM,CAAA;CACvB;AAED,wBAAgB,OAAO,CAAC,CAAC,EAAE,IAAI,EAAE,cAAc,CAAC,CAAC,CAAC,GAAG,IAAI,EAAE,CAyC1D;AAED,eAAO,MAAM,SAAS;;;;;CAAqC,CAAA"}
@@ -0,0 +1,128 @@
1
+ import { show, portal, onMount, div } from '@llui/dom';
2
+ import { attachFloating } from '../utils/floating';
3
+ export function init(opts = {}) {
4
+ return { open: opts.open ?? false };
5
+ }
6
+ export function update(state, msg) {
7
+ switch (msg.type) {
8
+ case 'show':
9
+ return [{ ...state, open: true }, []];
10
+ case 'hide':
11
+ return [{ ...state, open: false }, []];
12
+ case 'setOpen':
13
+ return [{ ...state, open: msg.open }, []];
14
+ }
15
+ }
16
+ export function connect(get, send, opts) {
17
+ const base = opts.id;
18
+ const triggerId = `${base}:trigger`;
19
+ const contentId = `${base}:content`;
20
+ const openDelay = opts.openDelay ?? 700;
21
+ const closeDelay = opts.closeDelay ?? 300;
22
+ let openTimer = null;
23
+ let closeTimer = null;
24
+ const clearTimers = () => {
25
+ if (openTimer) {
26
+ clearTimeout(openTimer);
27
+ openTimer = null;
28
+ }
29
+ if (closeTimer) {
30
+ clearTimeout(closeTimer);
31
+ closeTimer = null;
32
+ }
33
+ };
34
+ const scheduleOpen = () => {
35
+ clearTimers();
36
+ openTimer = setTimeout(() => {
37
+ openTimer = null;
38
+ send({ type: 'show' });
39
+ }, openDelay);
40
+ };
41
+ const scheduleClose = () => {
42
+ clearTimers();
43
+ closeTimer = setTimeout(() => {
44
+ closeTimer = null;
45
+ send({ type: 'hide' });
46
+ }, closeDelay);
47
+ };
48
+ return {
49
+ trigger: {
50
+ id: triggerId,
51
+ 'aria-haspopup': 'dialog',
52
+ 'aria-expanded': (s) => get(s).open,
53
+ 'aria-controls': contentId,
54
+ 'data-state': (s) => (get(s).open ? 'open' : 'closed'),
55
+ 'data-scope': 'hover-card',
56
+ 'data-part': 'trigger',
57
+ onPointerEnter: scheduleOpen,
58
+ onPointerLeave: scheduleClose,
59
+ onFocus: scheduleOpen,
60
+ onBlur: scheduleClose,
61
+ },
62
+ positioner: {
63
+ 'data-scope': 'hover-card',
64
+ 'data-part': 'positioner',
65
+ style: 'position:absolute;top:0;left:0;',
66
+ },
67
+ content: {
68
+ role: 'dialog',
69
+ id: contentId,
70
+ 'data-state': (s) => (get(s).open ? 'open' : 'closed'),
71
+ 'data-scope': 'hover-card',
72
+ 'data-part': 'content',
73
+ onPointerEnter: () => {
74
+ if (closeTimer) {
75
+ clearTimeout(closeTimer);
76
+ closeTimer = null;
77
+ }
78
+ },
79
+ onPointerLeave: scheduleClose,
80
+ },
81
+ arrow: {
82
+ 'data-scope': 'hover-card',
83
+ 'data-part': 'arrow',
84
+ },
85
+ };
86
+ }
87
+ export function overlay(opts) {
88
+ const target = opts.target ?? 'body';
89
+ const placement = opts.placement ?? 'bottom';
90
+ const offset = opts.offset ?? 8;
91
+ const flip = opts.flip !== false;
92
+ const shift = opts.shift !== false;
93
+ const parts = opts.parts;
94
+ const contentId = parts.content.id;
95
+ const triggerId = parts.trigger.id;
96
+ return show({
97
+ when: (s) => opts.get(s).open,
98
+ render: () => portal({
99
+ target,
100
+ render: () => {
101
+ onMount(() => {
102
+ const contentEl = document.getElementById(contentId);
103
+ const triggerEl = document.getElementById(triggerId);
104
+ if (!contentEl || !triggerEl)
105
+ return;
106
+ const positioner = contentEl.closest('[data-part="positioner"]');
107
+ const floatingEl = positioner ?? contentEl;
108
+ const arrow = opts.arrowSelector
109
+ ? contentEl.querySelector(opts.arrowSelector)
110
+ : null;
111
+ return attachFloating({
112
+ anchor: triggerEl,
113
+ floating: floatingEl,
114
+ placement,
115
+ offset,
116
+ flip,
117
+ shift,
118
+ arrow: arrow ?? undefined,
119
+ });
120
+ });
121
+ return [div(parts.positioner, opts.content())];
122
+ },
123
+ }),
124
+ enter: opts.transition?.enter,
125
+ leave: opts.transition?.leave,
126
+ });
127
+ }
128
+ export const hoverCard = { init, update, connect, overlay };