@hyvnt/hyvui 0.2.0 → 0.4.0

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 (181) hide show
  1. package/README.md +294 -253
  2. package/dist/components/ambient/ArcaneVein.svelte +151 -0
  3. package/dist/components/ambient/ArcaneVein.svelte.d.ts +31 -0
  4. package/dist/components/ambient/BrassFiligree.svelte +109 -0
  5. package/dist/components/ambient/BrassFiligree.svelte.d.ts +20 -0
  6. package/dist/components/ambient/CornerBrackets.svelte +91 -87
  7. package/dist/components/ambient/CornerBrackets.svelte.d.ts +8 -0
  8. package/dist/components/ambient/CrystalShard.svelte +151 -0
  9. package/dist/components/ambient/CrystalShard.svelte.d.ts +19 -0
  10. package/dist/components/ambient/DataStream.svelte +117 -94
  11. package/dist/components/ambient/DataStream.svelte.d.ts +6 -0
  12. package/dist/components/ambient/EnergyArc.svelte +189 -0
  13. package/dist/components/ambient/EnergyArc.svelte.d.ts +32 -0
  14. package/dist/components/ambient/GlyphMark.svelte +75 -69
  15. package/dist/components/ambient/GlyphMark.svelte.d.ts +6 -0
  16. package/dist/components/ambient/GridOverlay.svelte +34 -28
  17. package/dist/components/ambient/GridOverlay.svelte.d.ts +8 -0
  18. package/dist/components/ambient/HexGrid.svelte +119 -0
  19. package/dist/components/ambient/HexGrid.svelte.d.ts +21 -0
  20. package/dist/components/ambient/ParallaxLayer.svelte +45 -41
  21. package/dist/components/ambient/ParallaxLayer.svelte.d.ts +7 -0
  22. package/dist/components/ambient/ScanBand.svelte +103 -91
  23. package/dist/components/ambient/ScanBand.svelte.d.ts +8 -0
  24. package/dist/components/ambient/ShimmerCloud.svelte +180 -0
  25. package/dist/components/ambient/ShimmerCloud.svelte.d.ts +21 -0
  26. package/dist/components/ambient/SignalRing.svelte +106 -100
  27. package/dist/components/ambient/SignalRing.svelte.d.ts +6 -0
  28. package/dist/components/ambient/ThreadLine.svelte +78 -78
  29. package/dist/components/ambient/ThreadLine.svelte.d.ts +7 -0
  30. package/dist/components/ambient/Vignette.svelte +30 -26
  31. package/dist/components/ambient/Vignette.svelte.d.ts +6 -0
  32. package/dist/components/depth/DepthLayer.svelte +30 -27
  33. package/dist/components/depth/DepthLayer.svelte.d.ts +8 -0
  34. package/dist/components/depth/DepthStage.svelte +67 -62
  35. package/dist/components/depth/DepthStage.svelte.d.ts +8 -0
  36. package/dist/components/depth/FloatCard.svelte +129 -104
  37. package/dist/components/depth/FloatCard.svelte.d.ts +8 -0
  38. package/dist/components/depth/HorizonGrid.svelte +241 -160
  39. package/dist/components/depth/HorizonGrid.svelte.d.ts +9 -0
  40. package/dist/components/depth/Plinth.svelte +62 -57
  41. package/dist/components/depth/Plinth.svelte.d.ts +10 -0
  42. package/dist/components/display/Avatar.svelte +69 -69
  43. package/dist/components/display/Avatar.svelte.d.ts +5 -0
  44. package/dist/components/display/Badge.svelte +75 -63
  45. package/dist/components/display/Badge.svelte.d.ts +6 -0
  46. package/dist/components/display/Blockquote.svelte +35 -34
  47. package/dist/components/display/Blockquote.svelte.d.ts +4 -0
  48. package/dist/components/display/CodeBlock.svelte +76 -76
  49. package/dist/components/display/CodeBlock.svelte.d.ts +5 -0
  50. package/dist/components/display/MetricCard.svelte +100 -83
  51. package/dist/components/display/MetricCard.svelte.d.ts +6 -0
  52. package/dist/components/display/Table.svelte +106 -104
  53. package/dist/components/display/Table.svelte.d.ts +7 -0
  54. package/dist/components/feedback/Alert.svelte +95 -76
  55. package/dist/components/feedback/Alert.svelte.d.ts +6 -0
  56. package/dist/components/feedback/EmptyState.svelte +75 -68
  57. package/dist/components/feedback/EmptyState.svelte.d.ts +7 -0
  58. package/dist/components/feedback/ErrorState.svelte +78 -73
  59. package/dist/components/feedback/ErrorState.svelte.d.ts +5 -0
  60. package/dist/components/feedback/Skeleton.svelte +58 -52
  61. package/dist/components/feedback/Skeleton.svelte.d.ts +6 -0
  62. package/dist/components/feedback/StatusDot.svelte +84 -54
  63. package/dist/components/feedback/StatusDot.svelte.d.ts +6 -0
  64. package/dist/components/feedback/StatusLine.svelte +128 -122
  65. package/dist/components/feedback/StatusLine.svelte.d.ts +6 -0
  66. package/dist/components/feedback/Toast.svelte +144 -136
  67. package/dist/components/feedback/Toast.svelte.d.ts +10 -0
  68. package/dist/components/inputs/Button.svelte +310 -237
  69. package/dist/components/inputs/Button.svelte.d.ts +8 -0
  70. package/dist/components/inputs/Checkbox.svelte +109 -105
  71. package/dist/components/inputs/Checkbox.svelte.d.ts +5 -0
  72. package/dist/components/inputs/FileUpload.svelte +170 -163
  73. package/dist/components/inputs/FileUpload.svelte.d.ts +5 -0
  74. package/dist/components/inputs/Input.svelte +153 -147
  75. package/dist/components/inputs/Input.svelte.d.ts +7 -0
  76. package/dist/components/inputs/Select.svelte +164 -150
  77. package/dist/components/inputs/Select.svelte.d.ts +8 -0
  78. package/dist/components/inputs/Textarea.svelte +160 -154
  79. package/dist/components/inputs/Textarea.svelte.d.ts +6 -0
  80. package/dist/components/inputs/Toggle.svelte +125 -120
  81. package/dist/components/inputs/Toggle.svelte.d.ts +5 -0
  82. package/dist/components/layout/Card.svelte +81 -76
  83. package/dist/components/layout/Card.svelte.d.ts +11 -0
  84. package/dist/components/layout/Drawer.svelte +140 -109
  85. package/dist/components/layout/Drawer.svelte.d.ts +6 -0
  86. package/dist/components/layout/Grid.svelte +128 -43
  87. package/dist/components/layout/Grid.svelte.d.ts +18 -2
  88. package/dist/components/layout/Modal.svelte +191 -159
  89. package/dist/components/layout/Modal.svelte.d.ts +10 -0
  90. package/dist/components/layout/Panel.svelte +58 -54
  91. package/dist/components/layout/Panel.svelte.d.ts +9 -0
  92. package/dist/components/layout/Popover.svelte +188 -67
  93. package/dist/components/layout/Popover.svelte.d.ts +19 -1
  94. package/dist/components/layout/Stack.svelte +65 -53
  95. package/dist/components/layout/Stack.svelte.d.ts +12 -0
  96. package/dist/components/navigation/Breadcrumb.svelte +78 -73
  97. package/dist/components/navigation/Breadcrumb.svelte.d.ts +8 -0
  98. package/dist/components/navigation/DropdownMenu.svelte +179 -124
  99. package/dist/components/navigation/DropdownMenu.svelte.d.ts +24 -2
  100. package/dist/components/navigation/SidebarNav.svelte +96 -90
  101. package/dist/components/navigation/SidebarNav.svelte.d.ts +9 -0
  102. package/dist/components/navigation/Tabs.svelte +106 -86
  103. package/dist/components/navigation/Tabs.svelte.d.ts +8 -0
  104. package/dist/components/navigation/Topbar.svelte +94 -85
  105. package/dist/components/navigation/Topbar.svelte.d.ts +9 -0
  106. package/dist/components/patterns/ActionBar.svelte +82 -76
  107. package/dist/components/patterns/ActionBar.svelte.d.ts +10 -0
  108. package/dist/components/patterns/ChapterMark.svelte +152 -0
  109. package/dist/components/patterns/ChapterMark.svelte.d.ts +19 -0
  110. package/dist/components/patterns/ConfirmDialog.svelte +75 -64
  111. package/dist/components/patterns/ConfirmDialog.svelte.d.ts +12 -0
  112. package/dist/components/patterns/DepthPortal.svelte +123 -0
  113. package/dist/components/patterns/DepthPortal.svelte.d.ts +24 -0
  114. package/dist/components/patterns/Manifesto.svelte +171 -0
  115. package/dist/components/patterns/Manifesto.svelte.d.ts +25 -0
  116. package/dist/components/patterns/PageHeader.svelte +117 -114
  117. package/dist/components/patterns/PageHeader.svelte.d.ts +8 -0
  118. package/dist/components/patterns/PullQuote.svelte +145 -0
  119. package/dist/components/patterns/PullQuote.svelte.d.ts +23 -0
  120. package/dist/components/patterns/RegisterSwitcher.svelte +132 -0
  121. package/dist/components/patterns/RegisterSwitcher.svelte.d.ts +21 -0
  122. package/dist/components/patterns/SearchBar.svelte +59 -59
  123. package/dist/components/patterns/SearchBar.svelte.d.ts +5 -0
  124. package/dist/components/patterns/ShowcaseFrame.svelte +117 -0
  125. package/dist/components/patterns/ShowcaseFrame.svelte.d.ts +28 -0
  126. package/dist/components/patterns/TerminalBoot.svelte +118 -104
  127. package/dist/components/patterns/TerminalBoot.svelte.d.ts +12 -0
  128. package/dist/components/primitives/Divider.svelte +56 -29
  129. package/dist/components/primitives/Divider.svelte.d.ts +5 -0
  130. package/dist/components/primitives/Icon.svelte +53 -49
  131. package/dist/components/primitives/Icon.svelte.d.ts +9 -0
  132. package/dist/components/primitives/Label.svelte +45 -44
  133. package/dist/components/primitives/Label.svelte.d.ts +6 -0
  134. package/dist/components/primitives/Surface.svelte +154 -87
  135. package/dist/components/primitives/Surface.svelte.d.ts +7 -0
  136. package/dist/components/primitives/Text.svelte +130 -98
  137. package/dist/components/primitives/Text.svelte.d.ts +7 -0
  138. package/dist/components/scenes/ArchiveScene.svelte +102 -95
  139. package/dist/components/scenes/ArchiveScene.svelte.d.ts +17 -1
  140. package/dist/components/scenes/DepthScene.svelte +128 -0
  141. package/dist/components/scenes/DepthScene.svelte.d.ts +36 -0
  142. package/dist/components/scenes/LogScene.svelte +86 -77
  143. package/dist/components/scenes/LogScene.svelte.d.ts +14 -0
  144. package/dist/components/scenes/NarrativeScene.svelte +100 -92
  145. package/dist/components/scenes/NarrativeScene.svelte.d.ts +9 -0
  146. package/dist/components/scenes/ReadoutScene.svelte +131 -107
  147. package/dist/components/scenes/ReadoutScene.svelte.d.ts +14 -1
  148. package/dist/components/scenes/StageScene.svelte +111 -104
  149. package/dist/components/scenes/StageScene.svelte.d.ts +14 -0
  150. package/dist/components/system/AppShell.svelte +62 -0
  151. package/dist/components/system/AppShell.svelte.d.ts +32 -0
  152. package/dist/examples/ArcaneShard.svelte +364 -0
  153. package/dist/examples/ArcaneShard.svelte.d.ts +3 -0
  154. package/dist/examples/FieldReport.svelte +226 -223
  155. package/dist/examples/HextechForge.svelte +324 -0
  156. package/dist/examples/HextechForge.svelte.d.ts +3 -0
  157. package/dist/examples/ObservationDeck.svelte +333 -317
  158. package/dist/examples/SignalLost.svelte +191 -191
  159. package/dist/index.d.ts +15 -1
  160. package/dist/index.js +16 -1
  161. package/dist/styles.css +115 -0
  162. package/dist/system/actions/echo.js +21 -12
  163. package/dist/system/actions/resolve.js +28 -14
  164. package/dist/system/actions/reveal.js +2 -2
  165. package/dist/system/actions/surface.js +12 -2
  166. package/dist/system/depth/depth.css +49 -49
  167. package/dist/system/depth/depth.js +1 -1
  168. package/dist/system/expressions.css +80 -80
  169. package/dist/system/override-template.css +72 -72
  170. package/dist/system/register.css +74 -74
  171. package/dist/system/register.d.ts +1 -1
  172. package/dist/system/register.js +5 -1
  173. package/dist/system/scroll-lock.d.ts +6 -0
  174. package/dist/system/scroll-lock.js +23 -0
  175. package/dist/tokens/arcane.css +96 -0
  176. package/dist/tokens/hextech.css +96 -0
  177. package/dist/tokens/tokens.css +102 -86
  178. package/dist/tokens/tokens.d.ts +41 -0
  179. package/dist/tokens/tokens.js +44 -3
  180. package/dist/utils/motion.js +1 -1
  181. package/package.json +71 -60
@@ -1,105 +1,109 @@
1
- <script lang="ts">
2
- import { cn } from '../../utils/cn.js';
3
-
4
- interface Props {
5
- /** Whether the checkbox is checked (bindable). */
6
- checked?: boolean;
7
- /** Label text. */
8
- label?: string;
9
- /** Disables the checkbox. */
10
- disabled?: boolean;
11
- /** Additional CSS classes. */
12
- class?: string;
13
- /** Change handler. */
14
- onchange?: (e: Event) => void;
15
- }
16
-
17
- let {
18
- checked = $bindable(false),
19
- label = '',
20
- disabled = false,
21
- class: className = '',
22
- onchange,
23
- }: Props = $props();
24
- </script>
25
-
26
- <label class={cn('hyvui-checkbox', disabled && 'hyvui-checkbox-disabled', className)}>
27
- <input
28
- type="checkbox"
29
- bind:checked
30
- {disabled}
31
- class="hyvui-checkbox-input"
32
- {onchange}
33
- />
34
- <span class="hyvui-checkbox-box" class:hyvui-checkbox-checked={checked}>
35
- {#if checked}
36
- <svg width="10" height="10" viewBox="0 0 10 10" aria-hidden="true">
37
- <path d="M2 5.5L4 7.5L8 3" stroke="var(--bg)" stroke-width="1.5" fill="none" stroke-linecap="round" stroke-linejoin="round"/>
38
- </svg>
39
- {/if}
40
- </span>
41
- {#if label}
42
- <span class="hyvui-checkbox-label">{label}</span>
43
- {/if}
44
- </label>
45
-
46
- <style>
47
- .hyvui-checkbox {
48
- display: inline-flex;
49
- align-items: flex-start;
50
- gap: var(--space-sm);
51
- cursor: pointer;
52
- min-width: 0;
53
- }
54
-
55
- .hyvui-checkbox-disabled {
56
- opacity: 0.4;
57
- cursor: not-allowed;
58
- }
59
-
60
- .hyvui-checkbox-input {
61
- position: absolute;
62
- opacity: 0;
63
- width: 0;
64
- height: 0;
65
- pointer-events: none;
66
- }
67
-
68
- .hyvui-checkbox-box {
69
- width: 16px;
70
- height: 16px;
71
- margin-top: 0.1rem;
72
- border: 1px solid var(--line-strong);
73
- border-radius: var(--radius-sm);
74
- background:
75
- linear-gradient(180deg, rgba(240, 232, 218, 0.02), transparent 48%),
76
- var(--bg-elev);
77
- display: inline-flex;
78
- align-items: center;
79
- justify-content: center;
80
- flex-shrink: 0;
81
- transition:
82
- background-color var(--transition-fast),
83
- border-color var(--transition-fast),
84
- box-shadow var(--transition-fast);
85
- box-shadow: inset 0 1px 0 rgba(240, 232, 218, 0.03);
86
- }
87
-
88
- .hyvui-checkbox-checked {
89
- background-color: var(--accent);
90
- border-color: var(--accent);
91
- }
92
-
93
- .hyvui-checkbox-label {
94
- font-family: var(--font-body);
95
- font-size: 0.98rem;
96
- color: var(--text-soft);
97
- line-height: 1.5;
98
- }
99
-
100
- @media (prefers-reduced-motion: reduce) {
101
- .hyvui-checkbox-box {
102
- transition: none;
103
- }
104
- }
105
- </style>
1
+ <script lang="ts">
2
+ import { cn } from '../../utils/cn.js';
3
+
4
+ /**
5
+ * @example
6
+ * <Checkbox label="accept terms" bind:checked={accepted} />
7
+ * <Checkbox label="notify me" bind:checked={notify} onchange={handleChange} />
8
+ */
9
+ interface Props {
10
+ /** Whether the checkbox is checked (bindable). */
11
+ checked?: boolean;
12
+ /** Label text. */
13
+ label?: string;
14
+ /** Disables the checkbox. */
15
+ disabled?: boolean;
16
+ /** Additional CSS classes. */
17
+ class?: string;
18
+ /** Change handler. */
19
+ onchange?: (e: Event) => void;
20
+ }
21
+
22
+ let {
23
+ checked = $bindable(false),
24
+ label = '',
25
+ disabled = false,
26
+ class: className = '',
27
+ onchange
28
+ }: Props = $props();
29
+ </script>
30
+
31
+ <label class={cn('hyvui-checkbox', disabled && 'hyvui-checkbox-disabled', className)}>
32
+ <input type="checkbox" bind:checked {disabled} class="hyvui-checkbox-input" {onchange} />
33
+ <span class="hyvui-checkbox-box" class:hyvui-checkbox-checked={checked}>
34
+ {#if checked}
35
+ <svg width="10" height="10" viewBox="0 0 10 10" aria-hidden="true">
36
+ <path
37
+ d="M2 5.5L4 7.5L8 3"
38
+ stroke="var(--bg)"
39
+ stroke-width="1.5"
40
+ fill="none"
41
+ stroke-linecap="round"
42
+ stroke-linejoin="round"
43
+ />
44
+ </svg>
45
+ {/if}
46
+ </span>
47
+ {#if label}
48
+ <span class="hyvui-checkbox-label">{label}</span>
49
+ {/if}
50
+ </label>
51
+
52
+ <style>
53
+ .hyvui-checkbox {
54
+ display: inline-flex;
55
+ align-items: flex-start;
56
+ gap: var(--space-sm);
57
+ cursor: pointer;
58
+ min-width: 0;
59
+ }
60
+
61
+ .hyvui-checkbox-disabled {
62
+ opacity: 0.4;
63
+ cursor: not-allowed;
64
+ }
65
+
66
+ .hyvui-checkbox-input {
67
+ position: absolute;
68
+ opacity: 0;
69
+ width: 0;
70
+ height: 0;
71
+ pointer-events: none;
72
+ }
73
+
74
+ .hyvui-checkbox-box {
75
+ width: 16px;
76
+ height: 16px;
77
+ margin-top: 0.1rem;
78
+ border: 1px solid var(--line-strong);
79
+ border-radius: var(--radius-sm);
80
+ background: linear-gradient(180deg, rgba(240, 232, 218, 0.02), transparent 48%), var(--bg-elev);
81
+ display: inline-flex;
82
+ align-items: center;
83
+ justify-content: center;
84
+ flex-shrink: 0;
85
+ transition:
86
+ background-color var(--transition-fast),
87
+ border-color var(--transition-fast),
88
+ box-shadow var(--transition-fast);
89
+ box-shadow: inset 0 1px 0 rgba(240, 232, 218, 0.03);
90
+ }
91
+
92
+ .hyvui-checkbox-checked {
93
+ background-color: var(--accent);
94
+ border-color: var(--accent);
95
+ }
96
+
97
+ .hyvui-checkbox-label {
98
+ font-family: var(--font-body);
99
+ font-size: 0.98rem;
100
+ color: var(--text-soft);
101
+ line-height: 1.5;
102
+ }
103
+
104
+ @media (prefers-reduced-motion: reduce) {
105
+ .hyvui-checkbox-box {
106
+ transition: none;
107
+ }
108
+ }
109
+ </style>
@@ -1,3 +1,8 @@
1
+ /**
2
+ * @example
3
+ * <Checkbox label="accept terms" bind:checked={accepted} />
4
+ * <Checkbox label="notify me" bind:checked={notify} onchange={handleChange} />
5
+ */
1
6
  interface Props {
2
7
  /** Whether the checkbox is checked (bindable). */
3
8
  checked?: boolean;
@@ -1,163 +1,170 @@
1
- <script lang="ts">
2
- import { cn } from '../../utils/cn.js';
3
-
4
- interface Props {
5
- /** Accepted file types (e.g. "image/*"). */
6
- accept?: string;
7
- /** Allow multiple file selection. */
8
- multiple?: boolean;
9
- /** Disables the upload zone. */
10
- disabled?: boolean;
11
- /** Label text displayed in the zone. */
12
- label?: string;
13
- /** Additional CSS classes. */
14
- class?: string;
15
- /** Fires with the selected files. */
16
- onfiles?: (files: File[]) => void;
17
- }
18
-
19
- let {
20
- accept = '',
21
- multiple = false,
22
- disabled = false,
23
- label = 'drop files here or click to browse',
24
- class: className = '',
25
- onfiles,
26
- }: Props = $props();
27
-
28
- let dragging = $state(false);
29
- let fileNames = $state<string[]>([]);
30
- let inputEl: HTMLInputElement | undefined = $state();
31
-
32
- function handleFiles(fileList: FileList | null) {
33
- if (!fileList) return;
34
- const files = Array.from(fileList);
35
- fileNames = files.map((f) => f.name);
36
- onfiles?.(files);
37
- }
38
-
39
- function onDrop(e: DragEvent) {
40
- e.preventDefault();
41
- dragging = false;
42
- if (disabled) return;
43
- handleFiles(e.dataTransfer?.files ?? null);
44
- }
45
-
46
- function onDragOver(e: DragEvent) {
47
- e.preventDefault();
48
- if (!disabled) dragging = true;
49
- }
50
-
51
- function onDragLeave() {
52
- dragging = false;
53
- }
54
-
55
- function onClick() {
56
- if (!disabled) inputEl?.click();
57
- }
58
-
59
- function onInputChange(e: Event) {
60
- const target = e.target as HTMLInputElement;
61
- handleFiles(target.files);
62
- }
63
- </script>
64
-
65
- <div class={cn('hyvui-file-upload', className)}>
66
- <button
67
- type="button"
68
- class={cn(
69
- 'hyvui-file-zone',
70
- dragging && 'hyvui-file-zone-active',
71
- disabled && 'hyvui-file-zone-disabled'
72
- )}
73
- ondrop={onDrop}
74
- ondragover={onDragOver}
75
- ondragleave={onDragLeave}
76
- onclick={onClick}
77
- {disabled}
78
- >
79
- <span class="hyvui-file-label">{label}</span>
80
- </button>
81
- <input
82
- bind:this={inputEl}
83
- type="file"
84
- {accept}
85
- {multiple}
86
- class="hyvui-file-input"
87
- onchange={onInputChange}
88
- tabindex="-1"
89
- />
90
- {#if fileNames.length > 0}
91
- <div class="hyvui-file-names">
92
- {#each fileNames as name}
93
- <span class="hyvui-file-name">{name}</span>
94
- {/each}
95
- </div>
96
- {/if}
97
- </div>
98
-
99
- <style>
100
- .hyvui-file-upload {
101
- display: flex;
102
- flex-direction: column;
103
- gap: 0.5rem;
104
- }
105
-
106
- .hyvui-file-zone {
107
- border: 1px dashed var(--line);
108
- background: transparent;
109
- padding: 2rem;
110
- display: flex;
111
- align-items: center;
112
- justify-content: center;
113
- cursor: pointer;
114
- transition: border-color var(--transition-fast), background-color var(--transition-fast);
115
- width: 100%;
116
- }
117
-
118
- .hyvui-file-zone:hover:not(:disabled),
119
- .hyvui-file-zone-active {
120
- border-color: var(--line-strong);
121
- background-color: rgba(199, 156, 87, 0.04);
122
- }
123
-
124
- .hyvui-file-zone-disabled {
125
- opacity: 0.4;
126
- cursor: not-allowed;
127
- }
128
-
129
- .hyvui-file-label {
130
- font-family: var(--font-mono);
131
- font-size: 0.75rem;
132
- color: var(--muted);
133
- letter-spacing: 0.08em;
134
- }
135
-
136
- .hyvui-file-input {
137
- position: absolute;
138
- width: 0;
139
- height: 0;
140
- opacity: 0;
141
- pointer-events: none;
142
- }
143
-
144
- .hyvui-file-names {
145
- display: flex;
146
- flex-wrap: wrap;
147
- gap: 0.375rem;
148
- }
149
-
150
- .hyvui-file-name {
151
- font-family: var(--font-mono);
152
- font-size: 0.68rem;
153
- letter-spacing: 0.14em;
154
- text-transform: uppercase;
155
- color: var(--muted-strong);
156
- }
157
-
158
- @media (prefers-reduced-motion: reduce) {
159
- .hyvui-file-zone {
160
- transition: none;
161
- }
162
- }
163
- </style>
1
+ <script lang="ts">
2
+ import { cn } from '../../utils/cn.js';
3
+
4
+ /**
5
+ * @example
6
+ * <FileUpload accept="image/*" onfiles={handleImages} />
7
+ * <FileUpload multiple accept=".csv,.json" onfiles={handleData} label="drop data files here" />
8
+ */
9
+ interface Props {
10
+ /** Accepted file types (e.g. "image/*"). */
11
+ accept?: string;
12
+ /** Allow multiple file selection. */
13
+ multiple?: boolean;
14
+ /** Disables the upload zone. */
15
+ disabled?: boolean;
16
+ /** Label text displayed in the zone. */
17
+ label?: string;
18
+ /** Additional CSS classes. */
19
+ class?: string;
20
+ /** Fires with the selected files. */
21
+ onfiles?: (files: File[]) => void;
22
+ }
23
+
24
+ let {
25
+ accept = '',
26
+ multiple = false,
27
+ disabled = false,
28
+ label = 'drop files here or click to browse',
29
+ class: className = '',
30
+ onfiles
31
+ }: Props = $props();
32
+
33
+ let dragging = $state(false);
34
+ let fileNames = $state<string[]>([]);
35
+ let inputEl: HTMLInputElement | undefined = $state();
36
+
37
+ function handleFiles(fileList: FileList | null) {
38
+ if (!fileList) return;
39
+ const files = Array.from(fileList);
40
+ fileNames = files.map((f) => f.name);
41
+ onfiles?.(files);
42
+ }
43
+
44
+ function onDrop(e: DragEvent) {
45
+ e.preventDefault();
46
+ dragging = false;
47
+ if (disabled) return;
48
+ handleFiles(e.dataTransfer?.files ?? null);
49
+ }
50
+
51
+ function onDragOver(e: DragEvent) {
52
+ e.preventDefault();
53
+ if (!disabled) dragging = true;
54
+ }
55
+
56
+ function onDragLeave() {
57
+ dragging = false;
58
+ }
59
+
60
+ function onClick() {
61
+ if (!disabled) inputEl?.click();
62
+ }
63
+
64
+ function onInputChange(e: Event) {
65
+ const target = e.target as HTMLInputElement;
66
+ handleFiles(target.files);
67
+ }
68
+ </script>
69
+
70
+ <div class={cn('hyvui-file-upload', className)}>
71
+ <button
72
+ type="button"
73
+ class={cn(
74
+ 'hyvui-file-zone',
75
+ dragging && 'hyvui-file-zone-active',
76
+ disabled && 'hyvui-file-zone-disabled'
77
+ )}
78
+ ondrop={onDrop}
79
+ ondragover={onDragOver}
80
+ ondragleave={onDragLeave}
81
+ onclick={onClick}
82
+ {disabled}
83
+ >
84
+ <span class="hyvui-file-label">{label}</span>
85
+ </button>
86
+ <input
87
+ bind:this={inputEl}
88
+ type="file"
89
+ {accept}
90
+ {multiple}
91
+ class="hyvui-file-input"
92
+ onchange={onInputChange}
93
+ tabindex="-1"
94
+ />
95
+ {#if fileNames.length > 0}
96
+ <div class="hyvui-file-names">
97
+ {#each fileNames as name}
98
+ <span class="hyvui-file-name">{name}</span>
99
+ {/each}
100
+ </div>
101
+ {/if}
102
+ </div>
103
+
104
+ <style>
105
+ .hyvui-file-upload {
106
+ display: flex;
107
+ flex-direction: column;
108
+ gap: 0.5rem;
109
+ }
110
+
111
+ .hyvui-file-zone {
112
+ border: 1px dashed var(--line);
113
+ background: transparent;
114
+ padding: 2rem;
115
+ display: flex;
116
+ align-items: center;
117
+ justify-content: center;
118
+ cursor: pointer;
119
+ transition:
120
+ border-color var(--transition-fast),
121
+ background-color var(--transition-fast);
122
+ width: 100%;
123
+ }
124
+
125
+ .hyvui-file-zone:hover:not(:disabled),
126
+ .hyvui-file-zone-active {
127
+ border-color: var(--line-strong);
128
+ background-color: rgba(199, 156, 87, 0.04);
129
+ }
130
+
131
+ .hyvui-file-zone-disabled {
132
+ opacity: 0.4;
133
+ cursor: not-allowed;
134
+ }
135
+
136
+ .hyvui-file-label {
137
+ font-family: var(--font-mono);
138
+ font-size: 0.75rem;
139
+ color: var(--muted);
140
+ letter-spacing: 0.08em;
141
+ }
142
+
143
+ .hyvui-file-input {
144
+ position: absolute;
145
+ width: 0;
146
+ height: 0;
147
+ opacity: 0;
148
+ pointer-events: none;
149
+ }
150
+
151
+ .hyvui-file-names {
152
+ display: flex;
153
+ flex-wrap: wrap;
154
+ gap: 0.375rem;
155
+ }
156
+
157
+ .hyvui-file-name {
158
+ font-family: var(--font-mono);
159
+ font-size: 0.68rem;
160
+ letter-spacing: 0.14em;
161
+ text-transform: uppercase;
162
+ color: var(--muted-strong);
163
+ }
164
+
165
+ @media (prefers-reduced-motion: reduce) {
166
+ .hyvui-file-zone {
167
+ transition: none;
168
+ }
169
+ }
170
+ </style>
@@ -1,3 +1,8 @@
1
+ /**
2
+ * @example
3
+ * <FileUpload accept="image/*" onfiles={handleImages} />
4
+ * <FileUpload multiple accept=".csv,.json" onfiles={handleData} label="drop data files here" />
5
+ */
1
6
  interface Props {
2
7
  /** Accepted file types (e.g. "image/*"). */
3
8
  accept?: string;