@geoffcox/sterling-svelte 0.0.16 → 0.0.17

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 (111) hide show
  1. package/{buttons/Button.svelte → Button.svelte} +27 -27
  2. package/{inputs/Checkbox.svelte → Checkbox.svelte} +15 -15
  3. package/{surfaces/Dialog.svelte → Dialog.svelte} +34 -21
  4. package/{inputs/Input.svelte → Input.svelte} +22 -22
  5. package/Label.svelte +52 -0
  6. package/Label.svelte.d.ts +42 -0
  7. package/List.constants.d.ts +1 -0
  8. package/List.constants.js +1 -0
  9. package/List.svelte +324 -0
  10. package/List.svelte.d.ts +69 -0
  11. package/List.types.d.ts +6 -0
  12. package/ListItem.svelte +90 -0
  13. package/ListItem.svelte.d.ts +51 -0
  14. package/{containers/Menu.svelte → Menu.svelte} +42 -27
  15. package/{containers/MenuBar.svelte → MenuBar.svelte} +13 -13
  16. package/{buttons/MenuButton.svelte → MenuButton.svelte} +17 -17
  17. package/{buttons/MenuButton.svelte.d.ts → MenuButton.svelte.d.ts} +1 -1
  18. package/{containers/MenuItem.svelte → MenuItem.svelte} +42 -54
  19. package/{containers/MenuItem.svelte.d.ts → MenuItem.svelte.d.ts} +2 -2
  20. package/{containers/MenuSeparator.svelte → MenuSeparator.svelte} +2 -2
  21. package/Menus.types.d.ts +22 -0
  22. package/{containers/Menus.utils.d.ts → Menus.utils.d.ts} +2 -2
  23. package/{containers/Menus.utils.js → Menus.utils.js} +6 -6
  24. package/{surfaces/Portal.svelte → Portal.svelte} +1 -1
  25. package/{display/Progress.svelte → Progress.svelte} +14 -14
  26. package/{inputs/Radio.svelte → Radio.svelte} +15 -15
  27. package/{inputs/Select.svelte → Select.svelte} +92 -98
  28. package/{inputs/Select.svelte.d.ts → Select.svelte.d.ts} +20 -25
  29. package/{inputs/Slider.svelte → Slider.svelte} +49 -24
  30. package/Slider.svelte.d.ts +53 -0
  31. package/{inputs/Switch.svelte → Switch.svelte} +38 -38
  32. package/Tab.svelte +181 -0
  33. package/{containers/Tab.svelte.d.ts → Tab.svelte.d.ts} +12 -15
  34. package/TabList.svelte +247 -0
  35. package/{containers/TabList.svelte.d.ts → TabList.svelte.d.ts} +21 -21
  36. package/TabList.types.d.ts +7 -0
  37. package/{inputs/TextArea.svelte → TextArea.svelte} +22 -22
  38. package/Tooltip.svelte +182 -0
  39. package/Tooltip.svelte.d.ts +24 -0
  40. package/Tooltip.types.d.ts +3 -0
  41. package/Tooltip.types.js +1 -0
  42. package/Tree.constants.d.ts +2 -0
  43. package/Tree.constants.js +2 -0
  44. package/Tree.svelte +142 -0
  45. package/Tree.svelte.d.ts +25 -0
  46. package/Tree.types.d.ts +28 -0
  47. package/Tree.types.js +1 -0
  48. package/{containers/TreeChevron.svelte → TreeChevron.svelte} +3 -3
  49. package/TreeItem.svelte +276 -0
  50. package/TreeItem.svelte.d.ts +65 -0
  51. package/{containers/TreeItem.svelte → TreeItemDisplay.svelte} +18 -18
  52. package/{containers/TreeItem.svelte.d.ts → TreeItemDisplay.svelte.d.ts} +11 -14
  53. package/{forwardEvents.js → actions/forwardEvents.js} +0 -2
  54. package/index.d.ts +42 -31
  55. package/index.js +39 -25
  56. package/package.json +44 -41
  57. package/theme/darkTheme.js +66 -74
  58. package/theme/lightTheme.js +66 -74
  59. package/containers/List.svelte +0 -249
  60. package/containers/List.svelte.d.ts +0 -68
  61. package/containers/ListItem.svelte +0 -48
  62. package/containers/ListItem.svelte.d.ts +0 -26
  63. package/containers/Menus.types.d.ts +0 -22
  64. package/containers/Tab.svelte +0 -327
  65. package/containers/TabList.svelte +0 -126
  66. package/containers/Tabs.types.d.ts +0 -12
  67. package/containers/Tree.constants.d.ts +0 -2
  68. package/containers/Tree.constants.js +0 -2
  69. package/containers/Tree.svelte +0 -222
  70. package/containers/Tree.svelte.d.ts +0 -50
  71. package/containers/Tree.types.d.ts +0 -47
  72. package/containers/TreeNode.svelte +0 -266
  73. package/containers/TreeNode.svelte.d.ts +0 -43
  74. package/display/Label.svelte +0 -27
  75. package/display/Label.svelte.d.ts +0 -20
  76. package/inputs/Slider.svelte.d.ts +0 -28
  77. package/surfaces/CloseX.svelte +0 -5
  78. package/surfaces/CloseX.svelte.d.ts +0 -23
  79. /package/{buttons/Button.svelte.d.ts → Button.svelte.d.ts} +0 -0
  80. /package/{buttons/Button.types.d.ts → Button.types.d.ts} +0 -0
  81. /package/{buttons/Button.types.js → Button.types.js} +0 -0
  82. /package/{inputs/Checkbox.svelte.d.ts → Checkbox.svelte.d.ts} +0 -0
  83. /package/{surfaces/Dialog.svelte.d.ts → Dialog.svelte.d.ts} +0 -0
  84. /package/{inputs/Input.svelte.d.ts → Input.svelte.d.ts} +0 -0
  85. /package/{containers/Menus.types.js → List.types.js} +0 -0
  86. /package/{containers/Menu.svelte.d.ts → Menu.svelte.d.ts} +0 -0
  87. /package/{containers/MenuBar.svelte.d.ts → MenuBar.svelte.d.ts} +0 -0
  88. /package/{containers/MenuItemDisplay.svelte → MenuItemDisplay.svelte} +0 -0
  89. /package/{containers/MenuItemDisplay.svelte.d.ts → MenuItemDisplay.svelte.d.ts} +0 -0
  90. /package/{containers/MenuSeparator.svelte.d.ts → MenuSeparator.svelte.d.ts} +0 -0
  91. /package/{containers/Menus.constants.d.ts → Menus.constants.d.ts} +0 -0
  92. /package/{containers/Menus.constants.js → Menus.constants.js} +0 -0
  93. /package/{containers/Tabs.types.js → Menus.types.js} +0 -0
  94. /package/{surfaces/Portal.svelte.d.ts → Portal.svelte.d.ts} +0 -0
  95. /package/{display/Progress.svelte.d.ts → Progress.svelte.d.ts} +0 -0
  96. /package/{display/Progress.types.d.ts → Progress.types.d.ts} +0 -0
  97. /package/{display/Progress.types.js → Progress.types.js} +0 -0
  98. /package/{inputs/Radio.svelte.d.ts → Radio.svelte.d.ts} +0 -0
  99. /package/{inputs/Switch.svelte.d.ts → Switch.svelte.d.ts} +0 -0
  100. /package/{containers/Tabs.constants.d.ts → TabList.constants.d.ts} +0 -0
  101. /package/{containers/Tabs.constants.js → TabList.constants.js} +0 -0
  102. /package/{containers/Tree.types.js → TabList.types.js} +0 -0
  103. /package/{inputs/TextArea.svelte.d.ts → TextArea.svelte.d.ts} +0 -0
  104. /package/{inputs/TextArea.types.d.ts → TextArea.types.d.ts} +0 -0
  105. /package/{inputs/TextArea.types.js → TextArea.types.js} +0 -0
  106. /package/{containers/TreeChevron.svelte.d.ts → TreeChevron.svelte.d.ts} +0 -0
  107. /package/{clickOutside.d.ts → actions/clickOutside.d.ts} +0 -0
  108. /package/{clickOutside.js → actions/clickOutside.js} +0 -0
  109. /package/{forwardEvents.d.ts → actions/forwardEvents.d.ts} +0 -0
  110. /package/{portal.d.ts → actions/portal.d.ts} +0 -0
  111. /package/{portal.js → actions/portal.js} +0 -0
@@ -1,327 +0,0 @@
1
- <script>import { getContext } from "svelte";
2
- import { tabListContextKey } from "./Tabs.constants";
3
- export let data = void 0;
4
- export let disabled = false;
5
- export let tabId = void 0;
6
- export let text = void 0;
7
- export let selected = false;
8
- let tabRef;
9
- const {
10
- disabled: tabListDisabled,
11
- selectedTabId,
12
- selectionFollowsFocus,
13
- vertical
14
- } = getContext(tabListContextKey);
15
- $:
16
- _tabId = tabId || data?.tabId;
17
- $:
18
- _text = text || data?.text || _tabId;
19
- $:
20
- _disabled = $tabListDisabled || disabled || data?.disabled === true;
21
- $: {
22
- if (!_tabId) {
23
- throw new Error("Both tabId and data.tabId are missing");
24
- }
25
- }
26
- $: {
27
- selected = $selectedTabId === _tabId;
28
- }
29
- $: {
30
- if (selected) {
31
- selectedTabId.set(_tabId);
32
- tabRef?.focus();
33
- }
34
- }
35
- const getFirstActiveTab = () => {
36
- let foundTab = tabRef.parentElement?.firstElementChild;
37
- while (foundTab) {
38
- if (!foundTab.disabled && foundTab.getAttribute("data-tab-id")) {
39
- break;
40
- }
41
- foundTab = foundTab.nextElementSibling;
42
- }
43
- return foundTab;
44
- };
45
- const getLastActiveTab = () => {
46
- let foundTab = tabRef.parentElement?.lastElementChild;
47
- while (foundTab) {
48
- if (!foundTab.disabled && foundTab.getAttribute("data-tab-id")) {
49
- break;
50
- }
51
- foundTab = foundTab.previousElementSibling;
52
- }
53
- return foundTab;
54
- };
55
- const getPreviousActiveTab = () => {
56
- let foundTab = tabRef?.previousElementSibling;
57
- while (foundTab) {
58
- if (!foundTab.disabled && foundTab.getAttribute("data-tab-id")) {
59
- break;
60
- }
61
- foundTab = foundTab.previousElementSibling;
62
- }
63
- return foundTab;
64
- };
65
- const getNextActiveTab = () => {
66
- let foundTab = tabRef?.nextElementSibling;
67
- while (foundTab) {
68
- if (!foundTab.disabled && foundTab.getAttribute("data-tab-id")) {
69
- break;
70
- }
71
- foundTab = foundTab.nextElementSibling;
72
- }
73
- return foundTab;
74
- };
75
- const focusFirstTab = () => {
76
- if (!_disabled) {
77
- let foundTab = getFirstActiveTab();
78
- if (foundTab) {
79
- foundTab.focus();
80
- return true;
81
- }
82
- }
83
- return false;
84
- };
85
- const focusLastTab = () => {
86
- if (!_disabled) {
87
- let foundTab = getLastActiveTab();
88
- if (foundTab) {
89
- foundTab.focus();
90
- return true;
91
- }
92
- }
93
- return false;
94
- };
95
- const focusPreviousTab = () => {
96
- if (!_disabled) {
97
- let foundTab = getPreviousActiveTab();
98
- if (foundTab) {
99
- foundTab.focus();
100
- return true;
101
- }
102
- }
103
- return false;
104
- };
105
- const focusNextTab = () => {
106
- if (!_disabled) {
107
- let foundTab = getNextActiveTab();
108
- if (foundTab) {
109
- foundTab.focus();
110
- return true;
111
- }
112
- }
113
- return false;
114
- };
115
- const onClick = (event) => {
116
- if (!_disabled) {
117
- selectedTabId.set(_tabId);
118
- }
119
- };
120
- const onFocus = (event) => {
121
- if (!_disabled && $selectionFollowsFocus) {
122
- selectedTabId.set(_tabId);
123
- }
124
- };
125
- const onKeydown = (event) => {
126
- if (!_disabled && !event.ctrlKey && !event.shiftKey && !event.altKey && !event.metaKey) {
127
- switch (event.key) {
128
- case "Home":
129
- if (focusFirstTab()) {
130
- event.preventDefault();
131
- return false;
132
- }
133
- break;
134
- case "End":
135
- if (focusLastTab()) {
136
- event.preventDefault();
137
- return false;
138
- }
139
- break;
140
- case "ArrowLeft":
141
- if (!$vertical && focusPreviousTab()) {
142
- event.preventDefault();
143
- return false;
144
- }
145
- break;
146
- case "ArrowRight":
147
- if (!$vertical && focusNextTab()) {
148
- event.preventDefault();
149
- return false;
150
- }
151
- break;
152
- case "ArrowUp":
153
- if ($vertical && focusPreviousTab()) {
154
- event.preventDefault();
155
- return false;
156
- }
157
- break;
158
- case "ArrowDown":
159
- if ($vertical && focusNextTab()) {
160
- event.preventDefault();
161
- return false;
162
- }
163
- break;
164
- default:
165
- break;
166
- }
167
- }
168
- };
169
- </script>
170
-
171
- <button
172
- bind:this={tabRef}
173
- class="sterling-tab"
174
- disabled={_disabled}
175
- class:selected
176
- class:vertical={$vertical}
177
- data-tab-id={_tabId}
178
- role="tab"
179
- tabindex={selected ? 0 : -1}
180
- type="button"
181
- on:blur
182
- on:click
183
- on:dblclick
184
- on:focus
185
- on:focusin
186
- on:focusout
187
- on:keydown
188
- on:keypress
189
- on:keyup
190
- on:mousedown
191
- on:mouseenter
192
- on:mouseleave
193
- on:mousemove
194
- on:mouseover
195
- on:mouseout
196
- on:mouseup
197
- on:pointercancel
198
- on:pointerdown
199
- on:pointerenter
200
- on:pointerleave
201
- on:pointermove
202
- on:pointerover
203
- on:pointerout
204
- on:pointerup
205
- on:wheel
206
- on:click={onClick}
207
- on:focus={onFocus}
208
- on:keydown={onKeydown}
209
- >
210
- <div class="content">
211
- <slot {data} disabled={_disabled} {selected} tabId={_tabId} text={_text}>
212
- <div class="text">
213
- {_text || _tabId}
214
- </div>
215
- </slot>
216
- </div>
217
- <div class="indicator" />
218
- </button>
219
-
220
- <style>
221
- .sterling-tab {
222
- align-content: center;
223
- align-items: center;
224
- background-color: var(--Common__background-color);
225
- border-color: transparent;
226
- border-radius: var(--Button__border-radius);
227
- border-style: var(--Common__border-style);
228
- border-width: 0;
229
- box-sizing: border-box;
230
- color: var(--Common__color);
231
- cursor: pointer;
232
- display: grid;
233
- font: inherit;
234
- overflow: hidden;
235
- padding: 0.5em 1em 0.25em 1em;
236
- text-decoration: none;
237
- text-overflow: ellipsis;
238
- transition: background-color 250ms, color 250ms, border-color 250ms;
239
- white-space: nowrap;
240
- }
241
-
242
- .sterling-tab:not(.vertical) {
243
- grid-template-columns: 1fr;
244
- grid-template-rows: 1fr 1em;
245
- justify-content: center;
246
- justify-items: center;
247
- row-gap: 0.15em;
248
- }
249
-
250
- .sterling-tab.vertical {
251
- grid-template-columns: auto 1em;
252
- grid-template-rows: 1fr;
253
- align-content: center;
254
- align-items: center;
255
- justify-content: flex-end;
256
- justify-items: flex-end;
257
- column-gap: 0.15em;
258
- }
259
-
260
- .sterling-tab:hover {
261
- background-color: var(--Common__background-color--hover);
262
- color: var(--Common__color--hover);
263
- }
264
-
265
- .sterling-tab:active {
266
- background-color: var(--Common__background-color--active);
267
- color: var(--Common__color--active);
268
- }
269
-
270
- .sterling-tab:focus-visible {
271
- outline-color: var(--Common__outline-color);
272
- outline-offset: var(--Common__outline-offset);
273
- outline-style: var(--Common__outline-style);
274
- outline-width: var(--Common__outline-width);
275
- }
276
-
277
- .sterling-tab:disabled {
278
- color: var(--Common__color--disabled);
279
- cursor: not-allowed;
280
- }
281
-
282
- .content {
283
- padding: 0 0.2em;
284
- }
285
-
286
- .sterling-tab.vertical .content .text {
287
- padding-top: 0.25em;
288
- }
289
-
290
- .indicator {
291
- background-color: transparent;
292
- border-radius: 10000px;
293
- transition: background-color 250ms;
294
- }
295
-
296
- .sterling-tab:not(.vertical) .indicator {
297
- justify-self: stretch;
298
- height: 0.4em;
299
- }
300
-
301
- .sterling-tab.vertical .indicator {
302
- align-self: stretch;
303
- width: 0.4em;
304
- }
305
-
306
- .sterling-tab:hover .indicator {
307
- background-color: var(--Display__color--faint);
308
- }
309
-
310
- .sterling-tab:active .indicator {
311
- background-color: var(--Button__border-color--hover);
312
- }
313
-
314
- .sterling-tab:disabled .indicator,
315
- .sterling-tab:disabled:hover .indicator,
316
- .sterling-tab:disabled:active .indicator {
317
- background-color: transparent;
318
- }
319
-
320
- .sterling-tab.selected .indicator {
321
- background-color: var(--Input__color);
322
- }
323
-
324
- .sterling-tab.selected:disabled .indicator {
325
- background-color: var(--Input__color--disabled);
326
- }
327
- </style>
@@ -1,126 +0,0 @@
1
- <script>import { createEventDispatcher, setContext } from "svelte";
2
- import Tab from "./Tab.svelte";
3
- import { writable } from "svelte/store";
4
- import { tabListContextKey } from "./Tabs.constants";
5
- export let disabled = false;
6
- export let tabs = [];
7
- export let vertical = false;
8
- export let selectedTabId = void 0;
9
- export let selectionFollowsFocus = false;
10
- export let selectedTab = void 0;
11
- const selectedTabIdStore = writable(selectedTabId);
12
- const selectionFollowsFocusStore = writable(selectionFollowsFocus);
13
- const verticalStore = writable(vertical);
14
- const disabledStore = writable(disabled);
15
- setContext(tabListContextKey, {
16
- disabled: disabledStore,
17
- selectedTabId: selectedTabIdStore,
18
- selectionFollowsFocus: selectionFollowsFocusStore,
19
- vertical: verticalStore
20
- });
21
- const dispatch = createEventDispatcher();
22
- const raiseSelected = (tabId, tab) => {
23
- dispatch("select", { tabId, tab });
24
- };
25
- $:
26
- disabledStore.set(disabled);
27
- $:
28
- selectedTabIdStore.set(selectedTabId);
29
- $: {
30
- selectedTabId = $selectedTabIdStore;
31
- selectedTab = tabs.find((tab) => tab.tabId === selectedTabId);
32
- raiseSelected(selectedTabId, selectedTab);
33
- }
34
- $:
35
- selectionFollowsFocusStore.set(selectionFollowsFocus);
36
- $:
37
- verticalStore.set(vertical);
38
- </script>
39
-
40
- <!--
41
- @component
42
- A list of items where a single item can be selected.
43
- -->
44
- <!-- svelte-ignore a11y-no-noninteractive-tabindex -->
45
- <div
46
- class="sterling-tab-list"
47
- role="tablist"
48
- class:vertical
49
- class:disabled
50
- on:blur
51
- on:click
52
- on:copy
53
- on:cut
54
- on:dblclick
55
- on:focus
56
- on:focusin
57
- on:focusout
58
- on:keydown
59
- on:keypress
60
- on:keyup
61
- on:mousedown
62
- on:mouseenter
63
- on:mouseleave
64
- on:mousemove
65
- on:mouseover
66
- on:mouseout
67
- on:mouseup
68
- on:scroll
69
- on:wheel
70
- on:paste
71
- {...$$restProps}
72
- >
73
- {#each tabs as tab (tab.tabId)}
74
- <Tab data={tab}>
75
- <svelte:fragment let:data let:disabled let:selected let:tabId let:text>
76
- <slot name="tabContent" {data} {disabled} {selected} {tabId} {text}>
77
- <div class="text">
78
- {text}
79
- </div>
80
- </slot>
81
- </svelte:fragment>
82
- </Tab>
83
- {/each}
84
- <slot />
85
- </div>
86
-
87
- <style>
88
- .sterling-tab-list {
89
- box-sizing: border-box;
90
- display: grid;
91
- margin: 0;
92
- padding: calc(2 * var(--Common__outline-width));
93
- }
94
-
95
- .sterling-tab-list:not(.vertical) {
96
- column-gap: 0.5em;
97
- grid-auto-flow: column;
98
- grid-template-columns: repeat(auto-fill, auto);
99
- grid-template-rows: 1fr;
100
- overflow-x: scroll;
101
- overflow-y: hidden;
102
- }
103
-
104
- .sterling-tab-list.vertical {
105
- grid-auto-flow: row;
106
- grid-template-rows: auto;
107
- grid-template-columns: 1fr;
108
- overflow-x: hidden;
109
- overflow-y: scroll;
110
- row-gap: 0.5em;
111
- }
112
-
113
- .sterling-tab-list:hover {
114
- color: var(--Common__color--hover);
115
- }
116
-
117
- .sterling-tab-list.vertical .text {
118
- padding-top: 0.25em;
119
- }
120
-
121
- @media (prefers-reduced-motion) {
122
- .sterling-tab-list {
123
- transition: none;
124
- }
125
- }
126
- </style>
@@ -1,12 +0,0 @@
1
- import type { Readable, Writable } from 'svelte/store';
2
- export type TabData<T> = T & {
3
- tabId: string;
4
- text?: string;
5
- disabled?: boolean;
6
- };
7
- export type TabListContext<T> = {
8
- disabled: Readable<boolean>;
9
- selectedTabId: Writable<string | undefined>;
10
- selectionFollowsFocus: Readable<boolean>;
11
- vertical: Readable<boolean>;
12
- };
@@ -1,2 +0,0 @@
1
- export declare const treeContextKey = "sterlingTree";
2
- export declare const treeNodeContextKey = "sterlingTreeNode";
@@ -1,2 +0,0 @@
1
- export const treeContextKey = 'sterlingTree';
2
- export const treeNodeContextKey = 'sterlingTreeNode';
@@ -1,222 +0,0 @@
1
- <script>import { v4 as uuid } from "uuid";
2
- import { createEventDispatcher, setContext } from "svelte";
3
- import { writable } from "svelte/store";
4
- import { treeContextKey, treeNodeContextKey } from "./Tree.constants";
5
- import Label from "../display/Label.svelte";
6
- import TreeNode from "./TreeNode.svelte";
7
- import TreeItem from "./TreeItem.svelte";
8
- const inputId = uuid();
9
- let nodeIdCounter = 0;
10
- const getDefaultNodeId = (node) => {
11
- if (node.nodeId) {
12
- return node.nodeId;
13
- } else {
14
- nodeIdCounter++;
15
- return `node-${nodeIdCounter}`;
16
- }
17
- };
18
- export let composed = false;
19
- export let disabled = false;
20
- export let getNodeId = getDefaultNodeId;
21
- export let nodes = void 0;
22
- export let selectedNodeId = void 0;
23
- export let expandedNodeIds = [];
24
- const selectedNodeIdStore = writable(selectedNodeId);
25
- const expandedNodeIdStore = writable(expandedNodeIds);
26
- const selectedNodeEventHandlersStore = writable();
27
- setContext(treeContextKey, {
28
- expandedNodeIds: expandedNodeIdStore,
29
- getNodeId,
30
- selectedNodeId: selectedNodeIdStore,
31
- selectedNodeEventHandlers: selectedNodeEventHandlersStore
32
- });
33
- setContext(treeNodeContextKey, { parentNodeId: void 0, depth: 0 });
34
- const dispatch = createEventDispatcher();
35
- const raiseExpandCollapse = (expandedNodeIds2) => {
36
- dispatch("expandCollapse", { expandedNodeIds: expandedNodeIds2 });
37
- };
38
- const raiseSelect = (selectedNodeId2) => {
39
- dispatch("select", { selectedNodeId: selectedNodeId2 });
40
- };
41
- $: {
42
- selectedNodeIdStore.set(selectedNodeId);
43
- }
44
- $: {
45
- selectedNodeId = $selectedNodeIdStore;
46
- raiseSelect($selectedNodeIdStore);
47
- }
48
- $: {
49
- expandedNodeIdStore.set(expandedNodeIds);
50
- }
51
- $: {
52
- expandedNodeIds = $expandedNodeIdStore;
53
- raiseExpandCollapse($expandedNodeIdStore);
54
- }
55
- const getAllNodeIds = (node, nodeIds) => {
56
- nodeIds.push(getNodeId(node));
57
- node.children?.forEach((child) => getAllNodeIds(child, nodeIds));
58
- };
59
- const setToggleNodeIdsToAll = () => {
60
- const nodeIds = [];
61
- nodes?.forEach((node) => getAllNodeIds(node, nodeIds));
62
- expandedNodeIds = nodeIds;
63
- };
64
- export const collapseAll = () => {
65
- expandedNodeIds = [];
66
- };
67
- export const expandAll = () => {
68
- setToggleNodeIdsToAll();
69
- };
70
- const onKeydown = (event) => {
71
- return $selectedNodeEventHandlersStore?.onKeydown?.(event);
72
- };
73
- </script>
74
-
75
- <!-- svelte-ignore a11y-no-noninteractive-tabindex -->
76
- <div
77
- class="sterling-tree"
78
- class:disabled
79
- class:composed
80
- tabindex={!disabled ? 0 : undefined}
81
- on:keydown={onKeydown}
82
- >
83
- {#if $$slots.label}
84
- <Label {disabled} for={inputId}>
85
- <slot name="label" />
86
- </Label>
87
- {/if}
88
- <div class="tree" role="tree">
89
- {#if nodes}
90
- {#each nodes as node}
91
- <TreeNode {disabled} {node} nodeId={getNodeId(node)}>
92
- <!--
93
- Forward the item slot into each tree node.
94
- It is cleanest to have a fragment for the TreeNode item slot to
95
- capture the let params, then apply them to the Tree item slot.
96
- -->
97
- <svelte:fragment
98
- slot="item"
99
- let:disabled
100
- let:expanded
101
- let:hasChildren
102
- let:depth
103
- let:node
104
- let:nodeId
105
- let:selected
106
- >
107
- <slot
108
- name="item"
109
- {disabled}
110
- {expanded}
111
- {hasChildren}
112
- {depth}
113
- {node}
114
- {nodeId}
115
- {selected}
116
- >
117
- <!--
118
- Svelte prevents conditionally applying slots.
119
- This repeats exact same item slot default for this node
120
- so the item slot is passed down the tree.
121
- -->
122
- <TreeItem {disabled} {expanded} {hasChildren} {depth} {node} {nodeId} {selected}>
123
- <svelte:fragment
124
- let:disabled
125
- let:expanded
126
- let:hasChildren
127
- let:depth
128
- let:node
129
- let:nodeId
130
- let:selected
131
- >
132
- <!-- This uses the nodeLabel slot for the TreeItem default slot. -->
133
- <slot
134
- name="nodeLabel"
135
- {disabled}
136
- {expanded}
137
- {hasChildren}
138
- {depth}
139
- {node}
140
- {nodeId}
141
- {selected}>{nodeId}</slot
142
- >
143
- </svelte:fragment>
144
- </TreeItem>
145
- </slot>
146
- </svelte:fragment>
147
- </TreeNode>
148
- {/each}
149
- {/if}
150
- <slot />
151
- </div>
152
- </div>
153
-
154
- <style>
155
- .sterling-tree {
156
- background-color: var(--Common__background-color);
157
- border-color: var(--Common__border-color);
158
- border-radius: var(--Common__border-radius);
159
- border-style: var(--Common__border-style);
160
- border-width: var(--Common__border-width);
161
- box-sizing: border-box;
162
- color: var(--Common__color);
163
- display: grid;
164
- grid-template-columns: 1fr;
165
- grid-template-rows: auto 1fr;
166
- height: 100%;
167
- margin: 0;
168
- overflow: hidden;
169
- padding: 0;
170
- transition: background-color 250ms, color 250ms, border-color 250ms;
171
- }
172
-
173
- .sterling-tree:hover {
174
- border-color: var(--Common__border-color--hover);
175
- color: var(--Common__color--hover);
176
- }
177
-
178
- .sterling-tree:focus-visible {
179
- border-color: var(--Common__border-color--focus);
180
- color: var(--Common__color--focus);
181
- outline-color: var(--Common__outline-color);
182
- outline-offset: var(--Common__outline-offset);
183
- outline-style: var(--Common__outline-style);
184
- outline-width: var(--Common__outline-width);
185
- }
186
-
187
- .sterling-tree.disabled {
188
- background-color: var(--Common__background-color--disabled);
189
- border-color: var(---Common__border-color--disabled);
190
- color: var(--Common__color--disabled);
191
- cursor: not-allowed;
192
- }
193
-
194
- .sterling-tree.composed,
195
- .sterling-tree:hover.composed,
196
- .sterling-tree:focus-visible.composed,
197
- .sterling-tree.disabled.composed {
198
- background: none;
199
- border: none;
200
- outline: none;
201
- }
202
-
203
- .sterling-tree > :global(label) {
204
- font-size: 0.7em;
205
- margin: 0.5em 0.7em;
206
- }
207
-
208
- .sterling-tree > :global(label):empty {
209
- margin: 0;
210
- }
211
-
212
- .tree {
213
- display: flex;
214
- flex-direction: column;
215
- flex-wrap: nowrap;
216
- grid-row: 2 / span 1;
217
- overflow-x: hidden;
218
- overflow-y: scroll;
219
- outline: none;
220
- position: relative;
221
- }
222
- </style>