@placeholderco/placeholder-ui 1.0.3 → 1.0.6

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 (136) hide show
  1. package/LICENSE +26 -26
  2. package/README.md +179 -179
  3. package/dist/display/Alert.svelte +179 -179
  4. package/dist/display/Avatar.svelte +166 -166
  5. package/dist/display/LinkCollection.svelte +161 -161
  6. package/dist/display/Paper.svelte +118 -118
  7. package/dist/form/Autocomplete.svelte +223 -191
  8. package/dist/form/Autocomplete.svelte.d.ts +3 -1
  9. package/dist/form/AutocompleteMulti.svelte +356 -0
  10. package/dist/form/AutocompleteMulti.svelte.d.ts +28 -0
  11. package/dist/form/Checkbox.svelte +201 -201
  12. package/dist/form/Chips.svelte +128 -128
  13. package/dist/form/ComboBox.svelte +158 -158
  14. package/dist/form/ComboBox.svelte.d.ts +1 -1
  15. package/dist/form/ComboBoxItemBuilder.svelte +460 -460
  16. package/dist/form/ComboBoxMulti.svelte +197 -197
  17. package/dist/form/ComboBoxMulti.svelte.d.ts +1 -1
  18. package/dist/form/CronBuilder.svelte +693 -693
  19. package/dist/form/DatePicker.svelte +672 -672
  20. package/dist/form/DateTimePicker.svelte +712 -712
  21. package/dist/form/FileInput.svelte +235 -235
  22. package/dist/form/FormGroup.svelte +68 -68
  23. package/dist/form/Number.svelte +238 -238
  24. package/dist/form/PasswordInput.svelte +252 -252
  25. package/dist/form/RadioGroup.svelte +210 -210
  26. package/dist/form/Rating.svelte +235 -235
  27. package/dist/form/SegmentedControl.svelte +149 -149
  28. package/dist/form/Select.svelte +590 -590
  29. package/dist/form/Select.svelte.d.ts +1 -1
  30. package/dist/form/SelectMulti.svelte +613 -613
  31. package/dist/form/SelectMulti.svelte.d.ts +1 -1
  32. package/dist/form/Slider.svelte +358 -358
  33. package/dist/form/Switch.svelte +147 -147
  34. package/dist/form/TextArea.svelte +148 -148
  35. package/dist/form/Textbox.svelte +228 -228
  36. package/dist/form/TimePicker.svelte +267 -267
  37. package/dist/icon/Icon.svelte +52 -52
  38. package/dist/icon/alert-octagon.svg +5 -5
  39. package/dist/icon/alert-triangle.svg +5 -5
  40. package/dist/icon/archive.svg +1 -1
  41. package/dist/icon/arrow-down.svg +1 -1
  42. package/dist/icon/arrow-left.svg +1 -1
  43. package/dist/icon/arrow-right.svg +1 -1
  44. package/dist/icon/arrow-up.svg +1 -1
  45. package/dist/icon/at.svg +1 -1
  46. package/dist/icon/bell.svg +1 -1
  47. package/dist/icon/bookmark.svg +1 -1
  48. package/dist/icon/calendar.svg +1 -1
  49. package/dist/icon/camera.svg +1 -1
  50. package/dist/icon/chart-bar.svg +1 -1
  51. package/dist/icon/chart-line.svg +1 -1
  52. package/dist/icon/chart-pie.svg +1 -1
  53. package/dist/icon/checkbox.svg +1 -1
  54. package/dist/icon/checklist.svg +1 -1
  55. package/dist/icon/circle-check.svg +1 -1
  56. package/dist/icon/circle-x.svg +1 -1
  57. package/dist/icon/clock.svg +1 -1
  58. package/dist/icon/credit-card.svg +1 -1
  59. package/dist/icon/dots-vertical.svg +1 -1
  60. package/dist/icon/dots.svg +1 -1
  61. package/dist/icon/external-link.svg +1 -1
  62. package/dist/icon/eye-off.svg +1 -1
  63. package/dist/icon/eye.svg +1 -1
  64. package/dist/icon/filter.svg +1 -1
  65. package/dist/icon/fingerprint.svg +1 -1
  66. package/dist/icon/flag.svg +1 -1
  67. package/dist/icon/heart.svg +1 -1
  68. package/dist/icon/home.svg +1 -1
  69. package/dist/icon/key.svg +1 -1
  70. package/dist/icon/list-check.svg +1 -1
  71. package/dist/icon/login.svg +1 -1
  72. package/dist/icon/logout.svg +1 -1
  73. package/dist/icon/map-pin.svg +1 -1
  74. package/dist/icon/maximize.svg +1 -1
  75. package/dist/icon/microphone.svg +1 -1
  76. package/dist/icon/minimize.svg +1 -1
  77. package/dist/icon/note.svg +1 -1
  78. package/dist/icon/player-pause.svg +1 -1
  79. package/dist/icon/printer.svg +1 -1
  80. package/dist/icon/qrcode.svg +1 -1
  81. package/dist/icon/send.svg +1 -1
  82. package/dist/icon/settings.svg +1 -1
  83. package/dist/icon/share.svg +1 -1
  84. package/dist/icon/shopping-cart.svg +1 -1
  85. package/dist/icon/sort-ascending.svg +1 -1
  86. package/dist/icon/sort-descending.svg +1 -1
  87. package/dist/icon/star.svg +1 -1
  88. package/dist/icon/tag.svg +1 -1
  89. package/dist/icon/trending-down.svg +1 -1
  90. package/dist/icon/trending-up.svg +1 -1
  91. package/dist/icon/upload.svg +1 -1
  92. package/dist/icon/volume-off.svg +1 -1
  93. package/dist/icon/volume.svg +1 -1
  94. package/dist/icon/world.svg +1 -1
  95. package/dist/icon/zoom-in.svg +1 -1
  96. package/dist/icon/zoom-out.svg +1 -1
  97. package/dist/index.d.ts +1 -0
  98. package/dist/index.js +1 -0
  99. package/dist/layout/AppShell.svelte +169 -169
  100. package/dist/layout/CustomNavbar.svelte +61 -61
  101. package/dist/layout/Navbar.svelte +206 -206
  102. package/dist/layout/NavbarItemDisplay.svelte +29 -29
  103. package/dist/layout/Sidenav.svelte +712 -712
  104. package/dist/styles/components.css +199 -199
  105. package/dist/styles/dark.css +146 -146
  106. package/dist/styles/index.css +116 -116
  107. package/dist/styles/reset.css +110 -110
  108. package/dist/styles/semantic.css +86 -86
  109. package/dist/styles/tokens.css +203 -197
  110. package/dist/styles/utilities.css +523 -523
  111. package/dist/ui/Accordion.svelte +289 -289
  112. package/dist/ui/ActionIcon.svelte +76 -76
  113. package/dist/ui/Badge.svelte +329 -279
  114. package/dist/ui/Breadcrumbs.svelte +131 -131
  115. package/dist/ui/Button.svelte +432 -370
  116. package/dist/ui/ButtonVariant.d.ts +1 -1
  117. package/dist/ui/Dialog.svelte +307 -307
  118. package/dist/ui/Drawer.svelte +524 -524
  119. package/dist/ui/Dropdown.svelte +97 -97
  120. package/dist/ui/Dropzone.svelte +122 -122
  121. package/dist/ui/Link.svelte +32 -32
  122. package/dist/ui/Loader.svelte +70 -70
  123. package/dist/ui/LoadingOverlay.svelte +53 -53
  124. package/dist/ui/Pagination.svelte +135 -135
  125. package/dist/ui/Popover.svelte +225 -225
  126. package/dist/ui/Progress.svelte +191 -191
  127. package/dist/ui/RingProgress.svelte +141 -141
  128. package/dist/ui/Skeleton.svelte +85 -85
  129. package/dist/ui/Stepper.svelte +355 -355
  130. package/dist/ui/Table.svelte +345 -345
  131. package/dist/ui/Tabs.svelte +146 -146
  132. package/dist/ui/ThemeSwitcher.svelte +39 -39
  133. package/dist/ui/Timeline.svelte +225 -225
  134. package/dist/ui/Toaster.svelte +6 -6
  135. package/dist/ui/Tooltip.svelte +434 -434
  136. package/package.json +14 -14
@@ -1,252 +1,252 @@
1
- <script lang="ts">
2
- import { type Snippet } from 'svelte';
3
- import FormGroup from './FormGroup.svelte';
4
- import Loader from '../ui/Loader.svelte';
5
- import Icon from '../icon/Icon.svelte';
6
- import { iconEye, iconEyeOff } from '../icon/index.js';
7
-
8
- interface Props {
9
- label?: string;
10
- inputId?: string;
11
- placeholder?: string;
12
- required?: boolean;
13
- autofocus?: boolean;
14
- noAutocomplete?: boolean;
15
- loading?: boolean;
16
- groupClass?: string;
17
- leftIconSvg?: string;
18
- class?: string;
19
- containerClass?: string;
20
- disabled?: boolean;
21
- value?: string;
22
- visible?: boolean;
23
- isFocused?: boolean;
24
- showError?: boolean;
25
- errorText?: string;
26
- tooltipLocation?: 'top' | 'bottom' | 'left' | 'right';
27
- tooltipText?: string;
28
- onenter?: (v: string) => void;
29
- onchange?: (e: Event) => void;
30
- oninput?: (e: Event) => void;
31
- onkeydown?: (e: KeyboardEvent) => void;
32
- onfocus?: (e: FocusEvent) => void;
33
- onblur?: (e: FocusEvent) => void;
34
- onVisibilityChange?: (visible: boolean) => void;
35
- children?: Snippet;
36
- }
37
-
38
- let {
39
- label = '',
40
- inputId = undefined,
41
- placeholder = undefined,
42
- required = false,
43
- autofocus = false,
44
- noAutocomplete = false,
45
- loading = false,
46
- class: classes = '',
47
- containerClass = '',
48
- groupClass = '',
49
- leftIconSvg = undefined,
50
- disabled = false,
51
- value = $bindable(''),
52
- visible = $bindable(false),
53
- isFocused = false,
54
- showError = false,
55
- errorText = '',
56
- tooltipLocation = 'top',
57
- tooltipText = undefined,
58
- onenter = undefined,
59
- onchange = undefined,
60
- oninput = undefined,
61
- onkeydown = undefined,
62
- onfocus = undefined,
63
- onblur = undefined,
64
- onVisibilityChange = undefined,
65
- children
66
- }: Props = $props();
67
-
68
- // Generate ID on client-side only to avoid hydration mismatches
69
- let autoId = $state('');
70
-
71
- $effect(() => {
72
- if (!inputId && !label) {
73
- autoId = Math.random().toString(36).substring(2, 8);
74
- }
75
- });
76
-
77
- let id = $derived.by(() => {
78
- if (inputId) return inputId;
79
- if (label) return `input-${label.toLowerCase().replace(/ /g, '-')}`;
80
- return autoId;
81
- });
82
-
83
- const inputType = $derived(visible ? 'text' : 'password');
84
-
85
- const extraClasses: string[] = $derived.by(() => {
86
- const classes = [];
87
- if (leftIconSvg) classes.push('with-left-icon');
88
- if (disabled) classes.push('disabled');
89
- if (showError) classes.push('show-error');
90
- return classes;
91
- });
92
-
93
- function toggleVisibility() {
94
- visible = !visible;
95
- onVisibilityChange?.(visible);
96
- }
97
- </script>
98
-
99
- <div class="password-container {containerClass}">
100
- <FormGroup {label} {required} {id} class={groupClass} {tooltipLocation} {tooltipText}>
101
- <div class="password-input">
102
- <!-- svelte-ignore a11y_autofocus -->
103
- <input
104
- class="password {classes} {extraClasses.join(' ')}"
105
- type={inputType}
106
- {id}
107
- {placeholder}
108
- {disabled}
109
- {autofocus}
110
- {required}
111
- autocomplete={noAutocomplete ? 'off' : 'current-password'}
112
- bind:value
113
- onblur={() => {
114
- isFocused = false;
115
- onblur?.(new FocusEvent('blur'));
116
- }}
117
- onfocus={() => {
118
- isFocused = true;
119
- onfocus?.(new FocusEvent('focus'));
120
- }}
121
- {onchange}
122
- {oninput}
123
- onkeydown={(e) => {
124
- if (e.key === 'Enter') {
125
- onenter?.(value);
126
- }
127
- onkeydown?.(e);
128
- }}
129
- />
130
- {#if leftIconSvg && !loading}
131
- <div class="left-icon">
132
- <Icon svg={leftIconSvg} size="1.125em" />
133
- </div>
134
- {/if}
135
- {#if loading}
136
- <div class="loader-container">
137
- <Loader sizeOverride="1.1rem" />
138
- </div>
139
- {:else}
140
- <button
141
- type="button"
142
- class="visibility-toggle"
143
- onclick={toggleVisibility}
144
- tabindex={-1}
145
- aria-label={visible ? 'Hide password' : 'Show password'}
146
- {disabled}
147
- >
148
- <Icon svg={visible ? iconEyeOff : iconEye} size="1.125em" />
149
- </button>
150
- {/if}
151
- </div>
152
- {#if showError && errorText}
153
- <div class="text-error">{errorText}</div>
154
- {/if}
155
- {@render children?.()}
156
- </FormGroup>
157
- </div>
158
-
159
- <style>
160
- .password-input {
161
- position: relative;
162
- display: flex;
163
- }
164
-
165
- .password {
166
- width: 100%;
167
- font-size: var(--pui-font-size-md);
168
- line-height: var(--pui-line-height-normal);
169
- padding: var(--pui-spacing-1) var(--pui-spacing-10) var(--pui-spacing-1) var(--pui-spacing-2);
170
- border: 1px solid var(--pui-input-border);
171
- background-color: var(--pui-input-bg);
172
- color: var(--pui-input-text);
173
- border-radius: var(--pui-radius-base);
174
- box-sizing: border-box;
175
- }
176
-
177
- .password.disabled {
178
- opacity: 0.5;
179
- cursor: not-allowed;
180
- background: var(--pui-input-bg-disabled);
181
- }
182
-
183
- .password:focus {
184
- border-color: var(--pui-input-border-focus);
185
- outline: 2px solid transparent;
186
- outline-offset: 2px;
187
- }
188
-
189
- .password:focus ~ .left-icon {
190
- color: var(--pui-input-border-focus);
191
- }
192
-
193
- .with-left-icon {
194
- padding-left: 2rem;
195
- }
196
-
197
- .left-icon {
198
- position: absolute;
199
- pointer-events: none;
200
- top: 50%;
201
- transform: translateY(-50%);
202
- left: 0.5rem;
203
- color: var(--pui-text-muted);
204
- }
205
-
206
- .visibility-toggle {
207
- position: absolute;
208
- right: 0.5rem;
209
- top: 50%;
210
- transform: translateY(-50%);
211
- background: none;
212
- border: none;
213
- cursor: pointer;
214
- padding: 0.25rem;
215
- color: var(--pui-text-muted);
216
- display: flex;
217
- align-items: center;
218
- justify-content: center;
219
- border-radius: var(--pui-radius-sm);
220
- transition: color var(--pui-transition-fast) var(--pui-ease-in-out);
221
- }
222
-
223
- .visibility-toggle:hover:not(:disabled) {
224
- color: var(--pui-color-gray-700);
225
- }
226
-
227
- :global(.dark) .visibility-toggle:hover:not(:disabled) {
228
- color: var(--pui-color-gray-300);
229
- }
230
-
231
- .visibility-toggle:disabled {
232
- cursor: not-allowed;
233
- opacity: 0.5;
234
- }
235
-
236
- .loader-container {
237
- position: absolute;
238
- right: 0.5rem;
239
- top: 50%;
240
- transform: translateY(-50%);
241
- }
242
-
243
- .show-error {
244
- border-color: var(--pui-text-danger);
245
- }
246
-
247
- .text-error {
248
- color: var(--pui-text-danger);
249
- font-size: var(--pui-font-size-sm);
250
- margin-top: var(--pui-spacing-1);
251
- }
252
- </style>
1
+ <script lang="ts">
2
+ import { type Snippet } from 'svelte';
3
+ import FormGroup from './FormGroup.svelte';
4
+ import Loader from '../ui/Loader.svelte';
5
+ import Icon from '../icon/Icon.svelte';
6
+ import { iconEye, iconEyeOff } from '../icon/index.js';
7
+
8
+ interface Props {
9
+ label?: string;
10
+ inputId?: string;
11
+ placeholder?: string;
12
+ required?: boolean;
13
+ autofocus?: boolean;
14
+ noAutocomplete?: boolean;
15
+ loading?: boolean;
16
+ groupClass?: string;
17
+ leftIconSvg?: string;
18
+ class?: string;
19
+ containerClass?: string;
20
+ disabled?: boolean;
21
+ value?: string;
22
+ visible?: boolean;
23
+ isFocused?: boolean;
24
+ showError?: boolean;
25
+ errorText?: string;
26
+ tooltipLocation?: 'top' | 'bottom' | 'left' | 'right';
27
+ tooltipText?: string;
28
+ onenter?: (v: string) => void;
29
+ onchange?: (e: Event) => void;
30
+ oninput?: (e: Event) => void;
31
+ onkeydown?: (e: KeyboardEvent) => void;
32
+ onfocus?: (e: FocusEvent) => void;
33
+ onblur?: (e: FocusEvent) => void;
34
+ onVisibilityChange?: (visible: boolean) => void;
35
+ children?: Snippet;
36
+ }
37
+
38
+ let {
39
+ label = '',
40
+ inputId = undefined,
41
+ placeholder = undefined,
42
+ required = false,
43
+ autofocus = false,
44
+ noAutocomplete = false,
45
+ loading = false,
46
+ class: classes = '',
47
+ containerClass = '',
48
+ groupClass = '',
49
+ leftIconSvg = undefined,
50
+ disabled = false,
51
+ value = $bindable(''),
52
+ visible = $bindable(false),
53
+ isFocused = false,
54
+ showError = false,
55
+ errorText = '',
56
+ tooltipLocation = 'top',
57
+ tooltipText = undefined,
58
+ onenter = undefined,
59
+ onchange = undefined,
60
+ oninput = undefined,
61
+ onkeydown = undefined,
62
+ onfocus = undefined,
63
+ onblur = undefined,
64
+ onVisibilityChange = undefined,
65
+ children
66
+ }: Props = $props();
67
+
68
+ // Generate ID on client-side only to avoid hydration mismatches
69
+ let autoId = $state('');
70
+
71
+ $effect(() => {
72
+ if (!inputId && !label) {
73
+ autoId = Math.random().toString(36).substring(2, 8);
74
+ }
75
+ });
76
+
77
+ let id = $derived.by(() => {
78
+ if (inputId) return inputId;
79
+ if (label) return `input-${label.toLowerCase().replace(/ /g, '-')}`;
80
+ return autoId;
81
+ });
82
+
83
+ const inputType = $derived(visible ? 'text' : 'password');
84
+
85
+ const extraClasses: string[] = $derived.by(() => {
86
+ const classes = [];
87
+ if (leftIconSvg) classes.push('with-left-icon');
88
+ if (disabled) classes.push('disabled');
89
+ if (showError) classes.push('show-error');
90
+ return classes;
91
+ });
92
+
93
+ function toggleVisibility() {
94
+ visible = !visible;
95
+ onVisibilityChange?.(visible);
96
+ }
97
+ </script>
98
+
99
+ <div class="password-container {containerClass}">
100
+ <FormGroup {label} {required} {id} class={groupClass} {tooltipLocation} {tooltipText}>
101
+ <div class="password-input">
102
+ <!-- svelte-ignore a11y_autofocus -->
103
+ <input
104
+ class="password {classes} {extraClasses.join(' ')}"
105
+ type={inputType}
106
+ {id}
107
+ {placeholder}
108
+ {disabled}
109
+ {autofocus}
110
+ {required}
111
+ autocomplete={noAutocomplete ? 'off' : 'current-password'}
112
+ bind:value
113
+ onblur={() => {
114
+ isFocused = false;
115
+ onblur?.(new FocusEvent('blur'));
116
+ }}
117
+ onfocus={() => {
118
+ isFocused = true;
119
+ onfocus?.(new FocusEvent('focus'));
120
+ }}
121
+ {onchange}
122
+ {oninput}
123
+ onkeydown={(e) => {
124
+ if (e.key === 'Enter') {
125
+ onenter?.(value);
126
+ }
127
+ onkeydown?.(e);
128
+ }}
129
+ />
130
+ {#if leftIconSvg && !loading}
131
+ <div class="left-icon">
132
+ <Icon svg={leftIconSvg} size="1.125em" />
133
+ </div>
134
+ {/if}
135
+ {#if loading}
136
+ <div class="loader-container">
137
+ <Loader sizeOverride="1.1rem" />
138
+ </div>
139
+ {:else}
140
+ <button
141
+ type="button"
142
+ class="visibility-toggle"
143
+ onclick={toggleVisibility}
144
+ tabindex={-1}
145
+ aria-label={visible ? 'Hide password' : 'Show password'}
146
+ {disabled}
147
+ >
148
+ <Icon svg={visible ? iconEyeOff : iconEye} size="1.125em" />
149
+ </button>
150
+ {/if}
151
+ </div>
152
+ {#if showError && errorText}
153
+ <div class="text-error">{errorText}</div>
154
+ {/if}
155
+ {@render children?.()}
156
+ </FormGroup>
157
+ </div>
158
+
159
+ <style>
160
+ .password-input {
161
+ position: relative;
162
+ display: flex;
163
+ }
164
+
165
+ .password {
166
+ width: 100%;
167
+ font-size: var(--pui-font-size-md);
168
+ line-height: var(--pui-line-height-normal);
169
+ padding: var(--pui-spacing-1) var(--pui-spacing-10) var(--pui-spacing-1) var(--pui-spacing-2);
170
+ border: 1px solid var(--pui-input-border);
171
+ background-color: var(--pui-input-bg);
172
+ color: var(--pui-input-text);
173
+ border-radius: var(--pui-radius-base);
174
+ box-sizing: border-box;
175
+ }
176
+
177
+ .password.disabled {
178
+ opacity: 0.5;
179
+ cursor: not-allowed;
180
+ background: var(--pui-input-bg-disabled);
181
+ }
182
+
183
+ .password:focus {
184
+ border-color: var(--pui-input-border-focus);
185
+ outline: 2px solid transparent;
186
+ outline-offset: 2px;
187
+ }
188
+
189
+ .password:focus ~ .left-icon {
190
+ color: var(--pui-input-border-focus);
191
+ }
192
+
193
+ .with-left-icon {
194
+ padding-left: 2rem;
195
+ }
196
+
197
+ .left-icon {
198
+ position: absolute;
199
+ pointer-events: none;
200
+ top: 50%;
201
+ transform: translateY(-50%);
202
+ left: 0.5rem;
203
+ color: var(--pui-text-muted);
204
+ }
205
+
206
+ .visibility-toggle {
207
+ position: absolute;
208
+ right: 0.5rem;
209
+ top: 50%;
210
+ transform: translateY(-50%);
211
+ background: none;
212
+ border: none;
213
+ cursor: pointer;
214
+ padding: 0.25rem;
215
+ color: var(--pui-text-muted);
216
+ display: flex;
217
+ align-items: center;
218
+ justify-content: center;
219
+ border-radius: var(--pui-radius-sm);
220
+ transition: color var(--pui-transition-fast) var(--pui-ease-in-out);
221
+ }
222
+
223
+ .visibility-toggle:hover:not(:disabled) {
224
+ color: var(--pui-color-gray-700);
225
+ }
226
+
227
+ :global(.dark) .visibility-toggle:hover:not(:disabled) {
228
+ color: var(--pui-color-gray-300);
229
+ }
230
+
231
+ .visibility-toggle:disabled {
232
+ cursor: not-allowed;
233
+ opacity: 0.5;
234
+ }
235
+
236
+ .loader-container {
237
+ position: absolute;
238
+ right: 0.5rem;
239
+ top: 50%;
240
+ transform: translateY(-50%);
241
+ }
242
+
243
+ .show-error {
244
+ border-color: var(--pui-text-danger);
245
+ }
246
+
247
+ .text-error {
248
+ color: var(--pui-text-danger);
249
+ font-size: var(--pui-font-size-sm);
250
+ margin-top: var(--pui-spacing-1);
251
+ }
252
+ </style>