sh3-core 0.13.0 → 0.13.2

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 (120) hide show
  1. package/dist/BrandSlot.svelte +62 -13
  2. package/dist/__test__/setup-dom.js +5 -0
  3. package/dist/api.d.ts +3 -0
  4. package/dist/api.js +3 -0
  5. package/dist/apps/lifecycle.js +10 -2
  6. package/dist/apps/types.d.ts +11 -4
  7. package/dist/apps/workspace-rekey.d.ts +1 -0
  8. package/dist/apps/workspace-rekey.js +35 -0
  9. package/dist/apps/workspace-rekey.test.js +23 -0
  10. package/dist/auth/admin-users.svelte.d.ts +9 -0
  11. package/dist/auth/admin-users.svelte.js +42 -0
  12. package/dist/auth/admin-users.test.d.ts +1 -0
  13. package/dist/auth/admin-users.test.js +52 -0
  14. package/dist/createShell.js +5 -5
  15. package/dist/documents/config.d.ts +5 -1
  16. package/dist/documents/config.js +16 -8
  17. package/dist/documents/index.d.ts +1 -1
  18. package/dist/documents/index.js +1 -1
  19. package/dist/host-entry.d.ts +1 -1
  20. package/dist/host-entry.js +1 -1
  21. package/dist/host.d.ts +1 -1
  22. package/dist/host.js +8 -2
  23. package/dist/primitives/Button.svelte +50 -4
  24. package/dist/primitives/Button.svelte.d.ts +3 -1
  25. package/dist/primitives/Collapsible.svelte +110 -0
  26. package/dist/primitives/Collapsible.svelte.d.ts +14 -0
  27. package/dist/primitives/widgets/AppPicker.svelte +41 -0
  28. package/dist/primitives/widgets/AppPicker.svelte.d.ts +9 -0
  29. package/dist/primitives/widgets/AppPicker.svelte.test.d.ts +1 -0
  30. package/dist/primitives/widgets/AppPicker.svelte.test.js +26 -0
  31. package/dist/primitives/widgets/AppPicker.test.d.ts +1 -0
  32. package/dist/primitives/widgets/AppPicker.test.js +74 -0
  33. package/dist/primitives/widgets/ColorSwatch.svelte +7 -2
  34. package/dist/primitives/widgets/ColorSwatch.svelte.d.ts +2 -1
  35. package/dist/primitives/widgets/ColorSwatch.svelte.test.d.ts +1 -0
  36. package/dist/primitives/widgets/ColorSwatch.svelte.test.js +31 -0
  37. package/dist/primitives/widgets/Field.svelte +6 -1
  38. package/dist/primitives/widgets/Field.svelte.d.ts +2 -1
  39. package/dist/primitives/widgets/Field.svelte.test.d.ts +1 -0
  40. package/dist/primitives/widgets/Field.svelte.test.js +33 -0
  41. package/dist/primitives/widgets/FilePicker.svelte +6 -3
  42. package/dist/primitives/widgets/FilePicker.svelte.d.ts +2 -1
  43. package/dist/primitives/widgets/FilePicker.svelte.test.d.ts +1 -0
  44. package/dist/primitives/widgets/FilePicker.svelte.test.js +31 -0
  45. package/dist/primitives/widgets/IconToggleGroup.svelte +6 -3
  46. package/dist/primitives/widgets/IconToggleGroup.svelte.d.ts +3 -2
  47. package/dist/primitives/widgets/IconToggleGroup.svelte.test.d.ts +1 -0
  48. package/dist/primitives/widgets/IconToggleGroup.svelte.test.js +40 -0
  49. package/dist/primitives/widgets/NumberInput.svelte +26 -7
  50. package/dist/primitives/widgets/NumberInput.svelte.d.ts +2 -1
  51. package/dist/primitives/widgets/NumberInput.svelte.test.d.ts +1 -0
  52. package/dist/primitives/widgets/NumberInput.svelte.test.js +48 -0
  53. package/dist/primitives/widgets/PickerList.d.ts +24 -0
  54. package/dist/primitives/widgets/PickerList.js +21 -0
  55. package/dist/primitives/widgets/PickerList.svelte +150 -0
  56. package/dist/primitives/widgets/PickerList.svelte.d.ts +16 -0
  57. package/dist/primitives/widgets/PickerList.svelte.test.d.ts +1 -0
  58. package/dist/primitives/widgets/PickerList.svelte.test.js +31 -0
  59. package/dist/primitives/widgets/PickerList.test.d.ts +1 -0
  60. package/dist/primitives/widgets/PickerList.test.js +218 -0
  61. package/dist/primitives/widgets/RangeSlider.svelte +16 -5
  62. package/dist/primitives/widgets/RangeSlider.svelte.d.ts +2 -1
  63. package/dist/primitives/widgets/RangeSlider.svelte.test.d.ts +1 -0
  64. package/dist/primitives/widgets/RangeSlider.svelte.test.js +38 -0
  65. package/dist/primitives/widgets/Segmented.svelte +12 -5
  66. package/dist/primitives/widgets/Segmented.svelte.d.ts +3 -2
  67. package/dist/primitives/widgets/Segmented.svelte.test.d.ts +1 -0
  68. package/dist/primitives/widgets/Segmented.svelte.test.js +25 -0
  69. package/dist/primitives/widgets/Select.svelte +7 -3
  70. package/dist/primitives/widgets/Select.svelte.d.ts +3 -2
  71. package/dist/primitives/widgets/Select.svelte.test.d.ts +1 -0
  72. package/dist/primitives/widgets/Select.svelte.test.js +37 -0
  73. package/dist/primitives/widgets/Slider.svelte +6 -1
  74. package/dist/primitives/widgets/Slider.svelte.d.ts +2 -1
  75. package/dist/primitives/widgets/Slider.svelte.test.d.ts +1 -0
  76. package/dist/primitives/widgets/Slider.svelte.test.js +22 -0
  77. package/dist/primitives/widgets/SliderGroup.svelte +6 -1
  78. package/dist/primitives/widgets/SliderGroup.svelte.d.ts +2 -1
  79. package/dist/primitives/widgets/SliderGroup.svelte.test.d.ts +1 -0
  80. package/dist/primitives/widgets/SliderGroup.svelte.test.js +34 -0
  81. package/dist/primitives/widgets/Textarea.svelte +7 -1
  82. package/dist/primitives/widgets/Textarea.svelte.d.ts +2 -1
  83. package/dist/primitives/widgets/Textarea.svelte.test.d.ts +1 -0
  84. package/dist/primitives/widgets/Textarea.svelte.test.js +29 -0
  85. package/dist/primitives/widgets/UserPicker.svelte +53 -0
  86. package/dist/primitives/widgets/UserPicker.svelte.d.ts +9 -0
  87. package/dist/primitives/widgets/UserPicker.svelte.test.d.ts +1 -0
  88. package/dist/primitives/widgets/UserPicker.svelte.test.js +30 -0
  89. package/dist/primitives/widgets/UserPicker.test.d.ts +1 -0
  90. package/dist/primitives/widgets/UserPicker.test.js +115 -0
  91. package/dist/primitives/widgets/_contract.d.ts +27 -0
  92. package/dist/primitives/widgets/_contract.js +10 -0
  93. package/dist/projects/session-state.svelte.d.ts +17 -0
  94. package/dist/projects/session-state.svelte.js +39 -0
  95. package/dist/projects/session-state.test.d.ts +1 -0
  96. package/dist/projects/session-state.test.js +55 -0
  97. package/dist/projects-shard/DeleteProjectDialog.svelte +150 -0
  98. package/dist/projects-shard/DeleteProjectDialog.svelte.d.ts +12 -0
  99. package/dist/projects-shard/DeleteProjectDialog.test.d.ts +1 -0
  100. package/dist/projects-shard/DeleteProjectDialog.test.js +120 -0
  101. package/dist/projects-shard/ProjectManage.svelte +209 -0
  102. package/dist/projects-shard/ProjectManage.svelte.d.ts +8 -0
  103. package/dist/projects-shard/ProjectsSection.svelte +120 -0
  104. package/dist/projects-shard/ProjectsSection.svelte.d.ts +3 -0
  105. package/dist/projects-shard/index.d.ts +4 -0
  106. package/dist/projects-shard/index.js +4 -0
  107. package/dist/projects-shard/projectsApi.d.ts +20 -0
  108. package/dist/projects-shard/projectsApi.js +44 -0
  109. package/dist/projects-shard/projectsApi.test.d.ts +1 -0
  110. package/dist/projects-shard/projectsApi.test.js +71 -0
  111. package/dist/projects-shard/projectsShard.svelte.d.ts +10 -0
  112. package/dist/projects-shard/projectsShard.svelte.js +148 -0
  113. package/dist/sh3core-shard/ShellHome.svelte +19 -1
  114. package/dist/shards/activate-scopeid.test.d.ts +1 -0
  115. package/dist/shards/{activate-tenantid.test.js → activate-scopeid.test.js} +6 -6
  116. package/dist/tokens.css +4 -2
  117. package/dist/version.d.ts +1 -1
  118. package/dist/version.js +1 -1
  119. package/package.json +1 -1
  120. /package/dist/{shards/activate-tenantid.test.d.ts → apps/workspace-rekey.test.d.ts} +0 -0
@@ -1,22 +1,24 @@
1
1
  <script lang="ts">
2
2
  import type { Snippet } from 'svelte';
3
+ import type { CommitOnlyEvents } from './_contract';
3
4
  import { toggleSingle, toggleMultiple } from './IconToggleGroup';
4
5
 
5
6
  type Option = { value: string; icon: Snippet; tooltip?: string };
6
7
 
7
8
  let {
8
9
  options,
9
- value = $bindable(),
10
10
  multiple = false,
11
+ value = $bindable<string | string[]>(multiple ? [] : (options[0]?.value ?? '')),
11
12
  disabled = false,
12
13
  size = 'md',
14
+ onchange,
13
15
  }: {
14
16
  options: Option[];
15
- value: string | string[];
16
17
  multiple?: boolean;
18
+ value?: string | string[];
17
19
  disabled?: boolean;
18
20
  size?: 'sm' | 'md';
19
- } = $props();
21
+ } & CommitOnlyEvents<string | string[]> = $props();
20
22
 
21
23
  function isActive(v: string): boolean {
22
24
  if (multiple && Array.isArray(value)) return value.includes(v);
@@ -31,6 +33,7 @@
31
33
  } else {
32
34
  value = toggleSingle(typeof value === 'string' ? value : '', v);
33
35
  }
36
+ onchange?.(value);
34
37
  }
35
38
  </script>
36
39
 
@@ -1,4 +1,5 @@
1
1
  import type { Snippet } from 'svelte';
2
+ import type { CommitOnlyEvents } from './_contract';
2
3
  type Option = {
3
4
  value: string;
4
5
  icon: Snippet;
@@ -6,11 +7,11 @@ type Option = {
6
7
  };
7
8
  type $$ComponentProps = {
8
9
  options: Option[];
9
- value: string | string[];
10
10
  multiple?: boolean;
11
+ value?: string | string[];
11
12
  disabled?: boolean;
12
13
  size?: 'sm' | 'md';
13
- };
14
+ } & CommitOnlyEvents<string | string[]>;
14
15
  declare const IconToggleGroup: import("svelte").Component<$$ComponentProps, {}, "value">;
15
16
  type IconToggleGroup = ReturnType<typeof IconToggleGroup>;
16
17
  export default IconToggleGroup;
@@ -0,0 +1,40 @@
1
+ import { describe, it, expect, vi } from 'vitest';
2
+ import { render, fireEvent } from '@testing-library/svelte';
3
+ import { createRawSnippet } from 'svelte';
4
+ import IconToggleGroup from './IconToggleGroup.svelte';
5
+ const blank = createRawSnippet(() => ({ render: () => '<span/>' }));
6
+ const options = [
7
+ { value: 'b', icon: blank, tooltip: 'Bold' },
8
+ { value: 'i', icon: blank, tooltip: 'Italic' },
9
+ { value: 'u', icon: blank, tooltip: 'Under' },
10
+ ];
11
+ describe('IconToggleGroup event contract', () => {
12
+ it('fires onchange on click in single mode', async () => {
13
+ const onchange = vi.fn();
14
+ const { getByLabelText } = render(IconToggleGroup, {
15
+ props: { options, value: 'b', onchange },
16
+ });
17
+ await fireEvent.click(getByLabelText('Italic'));
18
+ expect(onchange).toHaveBeenCalledTimes(1);
19
+ expect(onchange).toHaveBeenCalledWith('i');
20
+ });
21
+ it('defaults to first option in single mode', () => {
22
+ const { getByLabelText } = render(IconToggleGroup, { props: { options } });
23
+ const bold = getByLabelText('Bold');
24
+ expect(bold.getAttribute('aria-pressed')).toBe('true');
25
+ });
26
+ it('defaults to empty array in multi mode', () => {
27
+ const { getByLabelText } = render(IconToggleGroup, { props: { options, multiple: true } });
28
+ expect(getByLabelText('Bold').getAttribute('aria-pressed')).toBe('false');
29
+ expect(getByLabelText('Italic').getAttribute('aria-pressed')).toBe('false');
30
+ });
31
+ it('toggles individual items in multi mode', async () => {
32
+ const onchange = vi.fn();
33
+ const { getByLabelText } = render(IconToggleGroup, {
34
+ props: { options, multiple: true, onchange },
35
+ });
36
+ await fireEvent.click(getByLabelText('Bold'));
37
+ await fireEvent.click(getByLabelText('Italic'));
38
+ expect(onchange).toHaveBeenLastCalledWith(['b', 'i']);
39
+ });
40
+ });
@@ -1,6 +1,7 @@
1
1
  <script lang="ts">
2
2
  import type { Snippet } from 'svelte';
3
- import { clamp, applyStep, formatNumber } from './NumberInput';
3
+ import type { LiveInputEvents } from './_contract';
4
+ import { clamp, applyStep } from './NumberInput';
4
5
 
5
6
  let {
6
7
  value = $bindable(0),
@@ -14,6 +15,8 @@
14
15
  disabled = false,
15
16
  invalid = false,
16
17
  size = 'md',
18
+ oninput,
19
+ onchange,
17
20
  }: {
18
21
  value?: number;
19
22
  min?: number;
@@ -26,23 +29,26 @@
26
29
  disabled?: boolean;
27
30
  invalid?: boolean;
28
31
  size?: 'sm' | 'md';
29
- } = $props();
30
-
31
- const display = $derived(formatNumber(value, precision));
32
+ } & LiveInputEvents<number> = $props();
32
33
 
33
34
  function commit(next: number) {
34
35
  if (Number.isNaN(next)) return;
35
- value = clamp(applyStep(next, { min, step }), min, max);
36
+ let v = clamp(applyStep(next, { min, step }), min, max);
37
+ if (precision !== undefined) v = Number(v.toFixed(precision));
38
+ value = v;
36
39
  }
37
40
 
38
41
  function bump(direction: 1 | -1) {
39
42
  commit(value + direction * step);
43
+ oninput?.(value);
40
44
  }
41
45
 
42
46
  let holdTimer: ReturnType<typeof setTimeout> | null = null;
43
47
  let holdInterval: ReturnType<typeof setInterval> | null = null;
48
+ let bumping = false;
44
49
  function startHold(direction: 1 | -1) {
45
50
  if (disabled) return;
51
+ bumping = true;
46
52
  bump(direction);
47
53
  holdTimer = setTimeout(() => {
48
54
  let elapsed = 0;
@@ -59,6 +65,10 @@
59
65
  function stopHold() {
60
66
  if (holdTimer) { clearTimeout(holdTimer); holdTimer = null; }
61
67
  if (holdInterval) { clearInterval(holdInterval); holdInterval = null; }
68
+ if (bumping) {
69
+ bumping = false;
70
+ onchange?.(value);
71
+ }
62
72
  }
63
73
  </script>
64
74
 
@@ -74,8 +84,17 @@
74
84
  {step}
75
85
  {disabled}
76
86
  aria-invalid={invalid || undefined}
77
- value={display}
78
- oninput={(e) => commit(parseFloat((e.currentTarget as HTMLInputElement).value))}
87
+ value={value}
88
+ oninput={(e) => {
89
+ const raw = (e.currentTarget as HTMLInputElement).valueAsNumber;
90
+ if (Number.isNaN(raw)) return;
91
+ value = raw;
92
+ oninput?.(raw);
93
+ }}
94
+ onblur={() => {
95
+ commit(value);
96
+ onchange?.(value);
97
+ }}
79
98
  />
80
99
  {#if suffix}<span class="sh3-num__affix">{@render suffix()}</span>{/if}
81
100
  <span class="sh3-num__steppers">
@@ -1,4 +1,5 @@
1
1
  import type { Snippet } from 'svelte';
2
+ import type { LiveInputEvents } from './_contract';
2
3
  type $$ComponentProps = {
3
4
  value?: number;
4
5
  min?: number;
@@ -11,7 +12,7 @@ type $$ComponentProps = {
11
12
  disabled?: boolean;
12
13
  invalid?: boolean;
13
14
  size?: 'sm' | 'md';
14
- };
15
+ } & LiveInputEvents<number>;
15
16
  declare const NumberInput: import("svelte").Component<$$ComponentProps, {}, "value">;
16
17
  type NumberInput = ReturnType<typeof NumberInput>;
17
18
  export default NumberInput;
@@ -0,0 +1,48 @@
1
+ import { describe, it, expect, vi } from 'vitest';
2
+ import { render, fireEvent } from '@testing-library/svelte';
3
+ import NumberInput from './NumberInput.svelte';
4
+ describe('NumberInput event contract', () => {
5
+ it('does not snap mid-typing when step=1', async () => {
6
+ const oninput = vi.fn();
7
+ const { container } = render(NumberInput, { props: { value: 0, step: 1, oninput } });
8
+ const input = container.querySelector('input');
9
+ await fireEvent.input(input, { target: { value: '5' } });
10
+ await fireEvent.input(input, { target: { value: '5.5' } });
11
+ expect(oninput).toHaveBeenCalledTimes(2);
12
+ expect(oninput).toHaveBeenNthCalledWith(1, 5);
13
+ expect(oninput).toHaveBeenNthCalledWith(2, 5.5);
14
+ });
15
+ it('swallows partial parses (NaN does not call oninput)', async () => {
16
+ const oninput = vi.fn();
17
+ const { container } = render(NumberInput, { props: { value: 1, oninput } });
18
+ const input = container.querySelector('input');
19
+ await fireEvent.input(input, { target: { value: '' } });
20
+ expect(oninput).not.toHaveBeenCalled();
21
+ });
22
+ it('commits applyStep on blur and fires onchange once', async () => {
23
+ const onchange = vi.fn();
24
+ const { container } = render(NumberInput, { props: { value: 0, step: 1, onchange } });
25
+ const input = container.querySelector('input');
26
+ await fireEvent.input(input, { target: { value: '5.5' } });
27
+ await fireEvent.blur(input);
28
+ expect(onchange).toHaveBeenCalledTimes(1);
29
+ expect(onchange).toHaveBeenCalledWith(6);
30
+ });
31
+ it('clamps to max on blur', async () => {
32
+ const onchange = vi.fn();
33
+ const { container } = render(NumberInput, { props: { value: 0, max: 10, onchange } });
34
+ const input = container.querySelector('input');
35
+ await fireEvent.input(input, { target: { value: '999' } });
36
+ await fireEvent.blur(input);
37
+ expect(onchange).toHaveBeenCalledWith(10);
38
+ });
39
+ it('rounds to precision on blur', async () => {
40
+ const onchange = vi.fn();
41
+ const { container } = render(NumberInput, { props: { value: 0, step: 0.01, precision: 2, onchange } });
42
+ const input = container.querySelector('input');
43
+ await fireEvent.input(input, { target: { value: '1.23456' } });
44
+ await fireEvent.blur(input);
45
+ expect(onchange).toHaveBeenCalledTimes(1);
46
+ expect(onchange.mock.calls[0][0]).toBeCloseTo(1.23, 5);
47
+ });
48
+ });
@@ -0,0 +1,24 @@
1
+ export interface PickerItem {
2
+ id: string;
3
+ label: string;
4
+ /** Optional secondary line shown beneath the label in muted style. */
5
+ sublabel?: string;
6
+ }
7
+ export interface PickerListProps {
8
+ items: PickerItem[];
9
+ value: string[];
10
+ onchange?: (next: string[]) => void;
11
+ loading?: boolean;
12
+ error?: string | null;
13
+ onRetry?: () => void;
14
+ /** Empty-state message when items.length === 0 and not loading/error. */
15
+ emptyText?: string;
16
+ /** Search input is rendered when items.length >= this value. Default: 8. */
17
+ searchThreshold?: number;
18
+ disabled?: boolean;
19
+ size?: 'sm' | 'md';
20
+ }
21
+ /** Case-insensitive substring match over label, id, and sublabel. */
22
+ export declare function filterItems(items: PickerItem[], query: string): PickerItem[];
23
+ /** Toggle the given id in or out of value, returning a NEW array. */
24
+ export declare function toggle(value: string[], id: string): string[];
@@ -0,0 +1,21 @@
1
+ /** Case-insensitive substring match over label, id, and sublabel. */
2
+ export function filterItems(items, query) {
3
+ if (!query)
4
+ return items;
5
+ const q = query.toLowerCase();
6
+ return items.filter((it) => {
7
+ if (it.label.toLowerCase().includes(q))
8
+ return true;
9
+ if (it.id.toLowerCase().includes(q))
10
+ return true;
11
+ if (it.sublabel && it.sublabel.toLowerCase().includes(q))
12
+ return true;
13
+ return false;
14
+ });
15
+ }
16
+ /** Toggle the given id in or out of value, returning a NEW array. */
17
+ export function toggle(value, id) {
18
+ if (value.includes(id))
19
+ return value.filter((v) => v !== id);
20
+ return [...value, id];
21
+ }
@@ -0,0 +1,150 @@
1
+ <script lang="ts">
2
+ import type { CommitOnlyEvents } from './_contract';
3
+ import { filterItems, toggle, type PickerItem } from './PickerList';
4
+
5
+ let {
6
+ items,
7
+ value = $bindable<string[]>([]),
8
+ loading = false,
9
+ error = null,
10
+ onRetry,
11
+ emptyText = 'No items.',
12
+ searchThreshold = 8,
13
+ disabled = false,
14
+ size = 'md',
15
+ onchange,
16
+ }: {
17
+ items: PickerItem[];
18
+ value?: string[];
19
+ loading?: boolean;
20
+ error?: string | null;
21
+ onRetry?: () => void;
22
+ emptyText?: string;
23
+ searchThreshold?: number;
24
+ disabled?: boolean;
25
+ size?: 'sm' | 'md';
26
+ } & CommitOnlyEvents<string[]> = $props();
27
+
28
+ let query = $state('');
29
+ const showSearch = $derived(items.length >= searchThreshold);
30
+ const filtered = $derived(filterItems(items, query));
31
+ const totalCount = $derived(items.length);
32
+ const selectedCount = $derived(value.length);
33
+
34
+ function toggleId(id: string) {
35
+ if (disabled) return;
36
+ const next = toggle(value, id);
37
+ value = next;
38
+ onchange?.(next);
39
+ }
40
+ </script>
41
+
42
+ <div class="sh3-picker" class:sh3-picker--sm={size === 'sm'} class:sh3-picker--disabled={disabled}>
43
+ {#if showSearch}
44
+ <input
45
+ type="search"
46
+ class="sh3-picker__search"
47
+ placeholder="Filter…"
48
+ bind:value={query}
49
+ {disabled}
50
+ />
51
+ {/if}
52
+
53
+ <div class="sh3-picker__list">
54
+ {#if loading}
55
+ <p class="sh3-picker__status">Loading…</p>
56
+ {:else if error}
57
+ <p class="sh3-picker__status sh3-picker__status--error">{error}</p>
58
+ {#if onRetry}
59
+ <button type="button" class="sh3-picker__retry" onclick={() => onRetry?.()}>Retry</button>
60
+ {/if}
61
+ {:else if items.length === 0}
62
+ <p class="sh3-picker__status">{emptyText}</p>
63
+ {:else}
64
+ {#each filtered as item (item.id)}
65
+ <label class="sh3-picker__row">
66
+ <input
67
+ type="checkbox"
68
+ checked={value.includes(item.id)}
69
+ {disabled}
70
+ onchange={() => toggleId(item.id)}
71
+ />
72
+ <span class="sh3-picker__row-text">
73
+ <span class="sh3-picker__row-label">{item.label}</span>
74
+ {#if item.sublabel}
75
+ <span class="sh3-picker__row-sub">{item.sublabel}</span>
76
+ {/if}
77
+ </span>
78
+ </label>
79
+ {/each}
80
+ {/if}
81
+ </div>
82
+
83
+ <p class="sh3-picker__footer">{selectedCount} of {totalCount} selected</p>
84
+ </div>
85
+
86
+ <style>
87
+ .sh3-picker {
88
+ display: flex;
89
+ flex-direction: column;
90
+ gap: 6px;
91
+ border: 1px solid var(--shell-border);
92
+ border-radius: var(--shell-radius-sm, 3px);
93
+ padding: 6px;
94
+ background: var(--shell-bg-elevated);
95
+ }
96
+ .sh3-picker__search {
97
+ width: 100%;
98
+ padding: 4px 6px;
99
+ border: 1px solid var(--shell-border);
100
+ border-radius: var(--shell-radius-sm, 3px);
101
+ background: var(--shell-bg);
102
+ color: var(--shell-fg);
103
+ font: inherit;
104
+ font-size: 12px;
105
+ }
106
+ .sh3-picker__list {
107
+ display: flex;
108
+ flex-direction: column;
109
+ max-height: 200px;
110
+ overflow-y: auto;
111
+ }
112
+ .sh3-picker__row {
113
+ display: flex;
114
+ align-items: center;
115
+ gap: 8px;
116
+ padding: 4px 6px;
117
+ cursor: pointer;
118
+ border-radius: var(--shell-radius-sm, 3px);
119
+ }
120
+ .sh3-picker__row:hover { background: var(--shell-bg); }
121
+ .sh3-picker__row-text { display: flex; flex-direction: column; gap: 0; }
122
+ .sh3-picker__row-label { font-size: 13px; }
123
+ .sh3-picker__row-sub { font-size: 11px; color: var(--shell-fg-muted); }
124
+ .sh3-picker__status {
125
+ margin: 0;
126
+ padding: 8px 6px;
127
+ color: var(--shell-fg-muted);
128
+ font-size: 12px;
129
+ }
130
+ .sh3-picker__status--error { color: var(--shell-error, #c33); }
131
+ .sh3-picker__retry {
132
+ align-self: flex-start;
133
+ margin: 0 6px 8px;
134
+ background: var(--shell-bg);
135
+ color: var(--shell-fg);
136
+ border: 1px solid var(--shell-border);
137
+ border-radius: var(--shell-radius-sm, 3px);
138
+ padding: 2px 8px;
139
+ font: inherit;
140
+ font-size: 12px;
141
+ cursor: pointer;
142
+ }
143
+ .sh3-picker__footer {
144
+ margin: 0;
145
+ font-size: 11px;
146
+ color: var(--shell-fg-muted);
147
+ text-align: right;
148
+ }
149
+ .sh3-picker--disabled .sh3-picker__row { cursor: not-allowed; opacity: 0.6; }
150
+ </style>
@@ -0,0 +1,16 @@
1
+ import type { CommitOnlyEvents } from './_contract';
2
+ import { type PickerItem } from './PickerList';
3
+ type $$ComponentProps = {
4
+ items: PickerItem[];
5
+ value?: string[];
6
+ loading?: boolean;
7
+ error?: string | null;
8
+ onRetry?: () => void;
9
+ emptyText?: string;
10
+ searchThreshold?: number;
11
+ disabled?: boolean;
12
+ size?: 'sm' | 'md';
13
+ } & CommitOnlyEvents<string[]>;
14
+ declare const PickerList: import("svelte").Component<$$ComponentProps, {}, "value">;
15
+ type PickerList = ReturnType<typeof PickerList>;
16
+ export default PickerList;
@@ -0,0 +1,31 @@
1
+ import { describe, it, expect, vi } from 'vitest';
2
+ import { render, fireEvent } from '@testing-library/svelte';
3
+ import PickerList from './PickerList.svelte';
4
+ const items = [
5
+ { id: '1', label: 'One' },
6
+ { id: '2', label: 'Two' },
7
+ { id: '3', label: 'Three' },
8
+ ];
9
+ describe('PickerList event contract', () => {
10
+ it('fires onchange with updated array when row toggled', async () => {
11
+ const onchange = vi.fn();
12
+ const { container } = render(PickerList, { props: { items, value: [], onchange } });
13
+ const checkboxes = container.querySelectorAll('input[type=checkbox]');
14
+ await fireEvent.click(checkboxes[1]);
15
+ expect(onchange).toHaveBeenCalledTimes(1);
16
+ expect(onchange).toHaveBeenCalledWith(['2']);
17
+ });
18
+ it('renders loading state', () => {
19
+ const { getByText } = render(PickerList, { props: { items: [], loading: true } });
20
+ expect(getByText('Loading…')).toBeInTheDocument();
21
+ });
22
+ it('renders error state with retry button when provided', async () => {
23
+ const onRetry = vi.fn();
24
+ const { getByText } = render(PickerList, {
25
+ props: { items: [], error: 'Boom', onRetry },
26
+ });
27
+ expect(getByText('Boom')).toBeInTheDocument();
28
+ await fireEvent.click(getByText('Retry'));
29
+ expect(onRetry).toHaveBeenCalled();
30
+ });
31
+ });
@@ -0,0 +1 @@
1
+ export {};