zdp-design-system 0.43.8
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.
- package/CHANGELOG.md +449 -0
- package/LICENSE +21 -0
- package/README.md +568 -0
- package/THIRD_PARTY_NOTICES.md +34 -0
- package/dist/code.ts +2 -0
- package/dist/combobox.ts +9 -0
- package/dist/command.ts +1 -0
- package/dist/components/Accordion.svelte +97 -0
- package/dist/components/Avatar.svelte +90 -0
- package/dist/components/Badge.svelte +61 -0
- package/dist/components/Breadcrumb.svelte +97 -0
- package/dist/components/Button.svelte +163 -0
- package/dist/components/Callout.svelte +81 -0
- package/dist/components/Card.svelte +151 -0
- package/dist/components/CardHeader.svelte +58 -0
- package/dist/components/Checkbox.svelte +135 -0
- package/dist/components/CodeBlock.svelte +247 -0
- package/dist/components/Combobox.svelte +552 -0
- package/dist/components/CommandField.svelte +230 -0
- package/dist/components/ConfirmAction.svelte +307 -0
- package/dist/components/Container.svelte +63 -0
- package/dist/components/Dialog.svelte +303 -0
- package/dist/components/Disclosure.svelte +176 -0
- package/dist/components/Divider.svelte +41 -0
- package/dist/components/EmptyState.svelte +79 -0
- package/dist/components/ErrorText.svelte +18 -0
- package/dist/components/Field.svelte +38 -0
- package/dist/components/Grid.svelte +76 -0
- package/dist/components/HelpText.svelte +17 -0
- package/dist/components/Icon.svelte +45 -0
- package/dist/components/IconButton.svelte +162 -0
- package/dist/components/IdentityChip.svelte +130 -0
- package/dist/components/Inline.svelte +85 -0
- package/dist/components/InlineCode.svelte +27 -0
- package/dist/components/Input.svelte +109 -0
- package/dist/components/Kbd.svelte +63 -0
- package/dist/components/KeyValue.svelte +73 -0
- package/dist/components/Label.svelte +43 -0
- package/dist/components/Link.svelte +70 -0
- package/dist/components/LocaleSwitcher.svelte +209 -0
- package/dist/components/Menu.svelte +491 -0
- package/dist/components/Page.svelte +36 -0
- package/dist/components/PageHeader.svelte +93 -0
- package/dist/components/Pagination.svelte +297 -0
- package/dist/components/Popover.svelte +208 -0
- package/dist/components/Progress.svelte +111 -0
- package/dist/components/Radio.svelte +132 -0
- package/dist/components/Section.svelte +52 -0
- package/dist/components/SegmentedControl.svelte +190 -0
- package/dist/components/Select.svelte +88 -0
- package/dist/components/ShareDock.svelte +304 -0
- package/dist/components/Sheet.svelte +332 -0
- package/dist/components/ShortcutHint.svelte +52 -0
- package/dist/components/Skeleton.svelte +82 -0
- package/dist/components/SkipLink.svelte +40 -0
- package/dist/components/SortHeader.svelte +138 -0
- package/dist/components/Spinner.svelte +82 -0
- package/dist/components/Stack.svelte +62 -0
- package/dist/components/StatusToast.svelte +133 -0
- package/dist/components/Surface.svelte +53 -0
- package/dist/components/Switch.svelte +152 -0
- package/dist/components/Table.svelte +94 -0
- package/dist/components/TableToolbar.svelte +195 -0
- package/dist/components/Tabs.svelte +205 -0
- package/dist/components/TermSheet.svelte +392 -0
- package/dist/components/TermTrigger.svelte +70 -0
- package/dist/components/TextScaleControl.svelte +219 -0
- package/dist/components/Textarea.svelte +106 -0
- package/dist/components/ThemeToggle.svelte +148 -0
- package/dist/components/Toast.svelte +180 -0
- package/dist/components/Toolbar.svelte +83 -0
- package/dist/components/Tooltip.svelte +199 -0
- package/dist/components/VisuallyHidden.svelte +18 -0
- package/dist/disclosure.ts +11 -0
- package/dist/focusable.ts +36 -0
- package/dist/identity.ts +5 -0
- package/dist/index.d.ts +106 -0
- package/dist/index.js +76 -0
- package/dist/index.ts +106 -0
- package/dist/menu.ts +12 -0
- package/dist/modal-layer.ts +108 -0
- package/dist/pagination.ts +10 -0
- package/dist/preferences.js +14 -0
- package/dist/preferences.ts +36 -0
- package/dist/progress.ts +4 -0
- package/dist/schemas/design-tokens.schema.json +119 -0
- package/dist/segmented.ts +8 -0
- package/dist/share.d.ts +48 -0
- package/dist/share.js +115 -0
- package/dist/share.ts +99 -0
- package/dist/sheet.ts +3 -0
- package/dist/shortcuts.js +125 -0
- package/dist/shortcuts.ts +153 -0
- package/dist/styles/brand-fonts.css +10 -0
- package/dist/styles/components.css +4686 -0
- package/dist/styles/expressive-fonts.css +2 -0
- package/dist/styles/index.css +2 -0
- package/dist/styles/locale-fonts.css +4 -0
- package/dist/styles/tokens.css +413 -0
- package/dist/table-tools.ts +10 -0
- package/dist/term.ts +16 -0
- package/dist/theme.ts +2 -0
- package/dist/toast.ts +14 -0
- package/dist/tokens/zdp.tokens.json +241 -0
- package/dist/tokens.js +122 -0
- package/dist/tokens.ts +123 -0
- package/docs/CONSUMER_CONTRACT.md +482 -0
- package/docs/EXTERNAL_UI_ADOPTION.md +141 -0
- package/docs/INTERACTIVE_PRIMITIVE_AUDIT.md +127 -0
- package/package.json +78 -0
|
@@ -0,0 +1,82 @@
|
|
|
1
|
+
<script lang="ts">
|
|
2
|
+
import type { ZdpSkeletonVariant } from '../progress';
|
|
3
|
+
|
|
4
|
+
export let variant: ZdpSkeletonVariant = 'block';
|
|
5
|
+
export let lines = 1;
|
|
6
|
+
export let animated = true;
|
|
7
|
+
export let decorative = true;
|
|
8
|
+
export let ariaLabel = 'Loading content';
|
|
9
|
+
export let labelledBy: string | null = null;
|
|
10
|
+
|
|
11
|
+
$: lineCount = Math.max(1, Math.min(8, Math.floor(Number.isFinite(lines) ? lines : 1)));
|
|
12
|
+
$: placeholders = Array.from({ length: variant === 'text' ? lineCount : 1 });
|
|
13
|
+
</script>
|
|
14
|
+
|
|
15
|
+
<div
|
|
16
|
+
class={`zdp-skeleton zdp-skeleton--${variant}`}
|
|
17
|
+
aria-hidden={decorative ? 'true' : undefined}
|
|
18
|
+
aria-label={!decorative && !labelledBy ? ariaLabel : undefined}
|
|
19
|
+
aria-labelledby={!decorative ? labelledBy ?? undefined : undefined}
|
|
20
|
+
data-animated={animated ? 'true' : 'false'}
|
|
21
|
+
role={decorative ? undefined : 'status'}
|
|
22
|
+
>
|
|
23
|
+
{#each placeholders as _, index}
|
|
24
|
+
<span
|
|
25
|
+
class={`zdp-skeleton__line ${index === placeholders.length - 1 ? 'zdp-skeleton__line--last' : ''}`}
|
|
26
|
+
aria-hidden="true"
|
|
27
|
+
></span>
|
|
28
|
+
{/each}
|
|
29
|
+
</div>
|
|
30
|
+
|
|
31
|
+
<style>
|
|
32
|
+
.zdp-skeleton {
|
|
33
|
+
--zdp-skeleton-fill: var(--zdp-color-surface-raised);
|
|
34
|
+
--zdp-skeleton-line: var(--zdp-color-line-subtle);
|
|
35
|
+
|
|
36
|
+
box-sizing: border-box;
|
|
37
|
+
display: grid;
|
|
38
|
+
gap: var(--zdp-space-2);
|
|
39
|
+
inline-size: 100%;
|
|
40
|
+
max-inline-size: 100%;
|
|
41
|
+
min-width: 0;
|
|
42
|
+
-webkit-user-select: none;
|
|
43
|
+
user-select: none;
|
|
44
|
+
}
|
|
45
|
+
|
|
46
|
+
.zdp-skeleton__line {
|
|
47
|
+
background: var(--zdp-skeleton-fill);
|
|
48
|
+
border: var(--zdp-control-border-width) solid var(--zdp-skeleton-line);
|
|
49
|
+
border-radius: var(--zdp-control-radius);
|
|
50
|
+
box-sizing: border-box;
|
|
51
|
+
display: block;
|
|
52
|
+
inline-size: 100%;
|
|
53
|
+
min-width: 0;
|
|
54
|
+
-webkit-user-select: none;
|
|
55
|
+
user-select: none;
|
|
56
|
+
}
|
|
57
|
+
|
|
58
|
+
.zdp-skeleton--text .zdp-skeleton__line {
|
|
59
|
+
block-size: calc(var(--zdp-type-body-small-size) * var(--zdp-type-body-small-line-height));
|
|
60
|
+
}
|
|
61
|
+
|
|
62
|
+
.zdp-skeleton--text .zdp-skeleton__line--last {
|
|
63
|
+
inline-size: 72%;
|
|
64
|
+
}
|
|
65
|
+
|
|
66
|
+
.zdp-skeleton--block .zdp-skeleton__line {
|
|
67
|
+
block-size: min(9rem, 28vw);
|
|
68
|
+
}
|
|
69
|
+
|
|
70
|
+
.zdp-skeleton--avatar {
|
|
71
|
+
inline-size: var(--zdp-control-height-md);
|
|
72
|
+
}
|
|
73
|
+
|
|
74
|
+
.zdp-skeleton--avatar .zdp-skeleton__line {
|
|
75
|
+
aspect-ratio: 1;
|
|
76
|
+
border-radius: 50%;
|
|
77
|
+
}
|
|
78
|
+
|
|
79
|
+
.zdp-skeleton[data-animated="false"] .zdp-skeleton__line {
|
|
80
|
+
opacity: 0.72;
|
|
81
|
+
}
|
|
82
|
+
</style>
|
|
@@ -0,0 +1,40 @@
|
|
|
1
|
+
<script lang="ts">
|
|
2
|
+
export let href = '#content';
|
|
3
|
+
</script>
|
|
4
|
+
|
|
5
|
+
<a class="zdp-skip-link" {href}>
|
|
6
|
+
<slot />
|
|
7
|
+
</a>
|
|
8
|
+
|
|
9
|
+
<style>
|
|
10
|
+
.zdp-skip-link {
|
|
11
|
+
background: var(--zdp-color-focus-surface);
|
|
12
|
+
border: var(--zdp-control-border-width) solid var(--zdp-color-focus-line);
|
|
13
|
+
border-radius: var(--zdp-control-radius);
|
|
14
|
+
color: var(--zdp-color-focus-text);
|
|
15
|
+
font-family: var(--zdp-font-family-sans);
|
|
16
|
+
font-size: var(--zdp-type-control-size);
|
|
17
|
+
font-weight: var(--zdp-font-weight-medium);
|
|
18
|
+
left: var(--zdp-space-3);
|
|
19
|
+
line-height: var(--zdp-type-control-line-height);
|
|
20
|
+
opacity: 0;
|
|
21
|
+
padding: var(--zdp-space-2) var(--zdp-space-3);
|
|
22
|
+
pointer-events: none;
|
|
23
|
+
position: fixed;
|
|
24
|
+
text-decoration-line: none;
|
|
25
|
+
top: var(--zdp-space-3);
|
|
26
|
+
transition:
|
|
27
|
+
background-color var(--zdp-motion-fast) ease,
|
|
28
|
+
border-color var(--zdp-motion-fast) ease,
|
|
29
|
+
color var(--zdp-motion-fast) ease,
|
|
30
|
+
opacity var(--zdp-motion-fast) ease;
|
|
31
|
+
z-index: 10000;
|
|
32
|
+
}
|
|
33
|
+
|
|
34
|
+
.zdp-skip-link:focus-visible {
|
|
35
|
+
opacity: 1;
|
|
36
|
+
outline: var(--zdp-control-focus-outline-width) solid var(--zdp-color-focus-surface);
|
|
37
|
+
outline-offset: var(--zdp-control-focus-outline-offset);
|
|
38
|
+
pointer-events: auto;
|
|
39
|
+
}
|
|
40
|
+
</style>
|
|
@@ -0,0 +1,138 @@
|
|
|
1
|
+
<script lang="ts">
|
|
2
|
+
import type { ZdpSortDirection } from '../table-tools';
|
|
3
|
+
|
|
4
|
+
export let label = 'Sort';
|
|
5
|
+
export let direction: ZdpSortDirection = 'none';
|
|
6
|
+
export let disabled = false;
|
|
7
|
+
export let ariaLabel: string | null = null;
|
|
8
|
+
export let onSort:
|
|
9
|
+
| ((event: MouseEvent, nextDirection: Exclude<ZdpSortDirection, 'none'>) => void)
|
|
10
|
+
| null = null;
|
|
11
|
+
|
|
12
|
+
$: normalizedDirection = normalizeDirection(direction);
|
|
13
|
+
$: nextDirection = getNextDirection(normalizedDirection);
|
|
14
|
+
$: directionLabel =
|
|
15
|
+
normalizedDirection === 'ascending'
|
|
16
|
+
? 'Ascending'
|
|
17
|
+
: normalizedDirection === 'descending'
|
|
18
|
+
? 'Descending'
|
|
19
|
+
: 'Not sorted';
|
|
20
|
+
$: resolvedAriaLabel = ariaLabel ?? `${label} ${directionLabel}`;
|
|
21
|
+
|
|
22
|
+
function handleClick(event: MouseEvent): void {
|
|
23
|
+
if (disabled) {
|
|
24
|
+
return;
|
|
25
|
+
}
|
|
26
|
+
|
|
27
|
+
onSort?.(event, nextDirection);
|
|
28
|
+
}
|
|
29
|
+
|
|
30
|
+
function normalizeDirection(value: ZdpSortDirection): ZdpSortDirection {
|
|
31
|
+
if (value === 'ascending' || value === 'descending') {
|
|
32
|
+
return value;
|
|
33
|
+
}
|
|
34
|
+
|
|
35
|
+
return 'none';
|
|
36
|
+
}
|
|
37
|
+
|
|
38
|
+
function getNextDirection(value: ZdpSortDirection): Exclude<ZdpSortDirection, 'none'> {
|
|
39
|
+
return value === 'ascending' ? 'descending' : 'ascending';
|
|
40
|
+
}
|
|
41
|
+
</script>
|
|
42
|
+
|
|
43
|
+
<button
|
|
44
|
+
class={`zdp-sort-header zdp-sort-header--${normalizedDirection}`}
|
|
45
|
+
type="button"
|
|
46
|
+
aria-label={resolvedAriaLabel}
|
|
47
|
+
aria-disabled={disabled ? 'true' : undefined}
|
|
48
|
+
data-sort-direction={normalizedDirection}
|
|
49
|
+
disabled={disabled}
|
|
50
|
+
onclick={handleClick}
|
|
51
|
+
>
|
|
52
|
+
<span class="zdp-sort-header__label"><slot>{label}</slot></span>
|
|
53
|
+
<span class="zdp-sort-header__mark" aria-hidden="true"></span>
|
|
54
|
+
</button>
|
|
55
|
+
|
|
56
|
+
<style>
|
|
57
|
+
.zdp-sort-header {
|
|
58
|
+
align-items: center;
|
|
59
|
+
background: transparent;
|
|
60
|
+
border: var(--zdp-control-border-width) solid transparent;
|
|
61
|
+
border-radius: var(--zdp-control-radius);
|
|
62
|
+
box-sizing: border-box;
|
|
63
|
+
color: inherit;
|
|
64
|
+
cursor: pointer;
|
|
65
|
+
display: inline-flex;
|
|
66
|
+
font-family: var(--zdp-font-family-sans);
|
|
67
|
+
font-size: inherit;
|
|
68
|
+
font-weight: inherit;
|
|
69
|
+
gap: var(--zdp-space-2);
|
|
70
|
+
justify-content: space-between;
|
|
71
|
+
line-height: inherit;
|
|
72
|
+
margin: calc(var(--zdp-space-1) * -1);
|
|
73
|
+
max-width: 100%;
|
|
74
|
+
min-block-size: var(--zdp-control-height-sm);
|
|
75
|
+
min-width: 0;
|
|
76
|
+
padding: var(--zdp-space-1) var(--zdp-space-2);
|
|
77
|
+
text-align: start;
|
|
78
|
+
transition:
|
|
79
|
+
background-color var(--zdp-motion-fast) ease,
|
|
80
|
+
border-color var(--zdp-motion-fast) ease,
|
|
81
|
+
color var(--zdp-motion-fast) ease;
|
|
82
|
+
-webkit-user-select: none;
|
|
83
|
+
user-select: none;
|
|
84
|
+
vertical-align: middle;
|
|
85
|
+
}
|
|
86
|
+
|
|
87
|
+
.zdp-sort-header:hover:not(:disabled) {
|
|
88
|
+
background: var(--zdp-color-surface-panel);
|
|
89
|
+
border-color: var(--zdp-color-line-strong);
|
|
90
|
+
color: var(--zdp-color-ink-strong);
|
|
91
|
+
}
|
|
92
|
+
|
|
93
|
+
.zdp-sort-header:focus-visible {
|
|
94
|
+
border-color: var(--zdp-color-focus-line);
|
|
95
|
+
outline: var(--zdp-control-focus-outline-width) solid var(--zdp-color-focus-surface);
|
|
96
|
+
outline-offset: var(--zdp-control-focus-outline-offset);
|
|
97
|
+
}
|
|
98
|
+
|
|
99
|
+
.zdp-sort-header:disabled {
|
|
100
|
+
cursor: not-allowed;
|
|
101
|
+
opacity: 0.56;
|
|
102
|
+
}
|
|
103
|
+
|
|
104
|
+
.zdp-sort-header__label {
|
|
105
|
+
min-width: 0;
|
|
106
|
+
overflow-wrap: var(--zdp-i18n-overflow-wrap);
|
|
107
|
+
}
|
|
108
|
+
|
|
109
|
+
.zdp-sort-header__mark {
|
|
110
|
+
align-items: center;
|
|
111
|
+
color: var(--zdp-color-ink-muted);
|
|
112
|
+
display: inline-flex;
|
|
113
|
+
flex: 0 0 auto;
|
|
114
|
+
font-size: var(--zdp-type-caption-size);
|
|
115
|
+
justify-content: center;
|
|
116
|
+
line-height: 1;
|
|
117
|
+
min-inline-size: 1em;
|
|
118
|
+
-webkit-user-select: none;
|
|
119
|
+
user-select: none;
|
|
120
|
+
}
|
|
121
|
+
|
|
122
|
+
.zdp-sort-header__mark::before {
|
|
123
|
+
content: '↕';
|
|
124
|
+
}
|
|
125
|
+
|
|
126
|
+
.zdp-sort-header--ascending .zdp-sort-header__mark::before {
|
|
127
|
+
content: '↑';
|
|
128
|
+
}
|
|
129
|
+
|
|
130
|
+
.zdp-sort-header--descending .zdp-sort-header__mark::before {
|
|
131
|
+
content: '↓';
|
|
132
|
+
}
|
|
133
|
+
|
|
134
|
+
.zdp-sort-header--ascending .zdp-sort-header__mark,
|
|
135
|
+
.zdp-sort-header--descending .zdp-sort-header__mark {
|
|
136
|
+
color: var(--zdp-color-accent-primary-strong);
|
|
137
|
+
}
|
|
138
|
+
</style>
|
|
@@ -0,0 +1,82 @@
|
|
|
1
|
+
<script lang="ts">
|
|
2
|
+
import type { ZdpProgressTone, ZdpSpinnerSize } from '../progress';
|
|
3
|
+
|
|
4
|
+
export let size: ZdpSpinnerSize = 'md';
|
|
5
|
+
export let tone: ZdpProgressTone = 'primary';
|
|
6
|
+
export let label = 'Loading';
|
|
7
|
+
export let decorative = false;
|
|
8
|
+
export let semanticRole: 'status' | 'img' | null = 'status';
|
|
9
|
+
</script>
|
|
10
|
+
|
|
11
|
+
<span
|
|
12
|
+
class={`zdp-spinner zdp-spinner--${tone} zdp-spinner--${size}`}
|
|
13
|
+
aria-hidden={decorative ? 'true' : undefined}
|
|
14
|
+
aria-label={decorative ? undefined : label}
|
|
15
|
+
role={decorative ? undefined : semanticRole ?? undefined}
|
|
16
|
+
>
|
|
17
|
+
<span class="zdp-spinner__mark" aria-hidden="true"></span>
|
|
18
|
+
</span>
|
|
19
|
+
|
|
20
|
+
<style>
|
|
21
|
+
.zdp-spinner {
|
|
22
|
+
--zdp-spinner-accent: var(--zdp-color-accent-primary);
|
|
23
|
+
|
|
24
|
+
align-items: center;
|
|
25
|
+
box-sizing: border-box;
|
|
26
|
+
color: var(--zdp-spinner-accent);
|
|
27
|
+
display: inline-flex;
|
|
28
|
+
flex: 0 0 auto;
|
|
29
|
+
justify-content: center;
|
|
30
|
+
line-height: 1;
|
|
31
|
+
-webkit-user-select: none;
|
|
32
|
+
user-select: none;
|
|
33
|
+
vertical-align: middle;
|
|
34
|
+
}
|
|
35
|
+
|
|
36
|
+
.zdp-spinner__mark {
|
|
37
|
+
border: var(--zdp-control-border-width) solid var(--zdp-color-line-subtle);
|
|
38
|
+
border-block-start-color: currentColor;
|
|
39
|
+
border-inline-end-color: currentColor;
|
|
40
|
+
border-radius: 50%;
|
|
41
|
+
box-sizing: border-box;
|
|
42
|
+
display: block;
|
|
43
|
+
-webkit-user-select: none;
|
|
44
|
+
user-select: none;
|
|
45
|
+
}
|
|
46
|
+
|
|
47
|
+
.zdp-spinner--sm .zdp-spinner__mark {
|
|
48
|
+
block-size: var(--zdp-control-glyph-sm);
|
|
49
|
+
inline-size: var(--zdp-control-glyph-sm);
|
|
50
|
+
}
|
|
51
|
+
|
|
52
|
+
.zdp-spinner--md .zdp-spinner__mark {
|
|
53
|
+
block-size: var(--zdp-control-glyph-md);
|
|
54
|
+
inline-size: var(--zdp-control-glyph-md);
|
|
55
|
+
}
|
|
56
|
+
|
|
57
|
+
.zdp-spinner--lg .zdp-spinner__mark {
|
|
58
|
+
block-size: var(--zdp-control-height-sm);
|
|
59
|
+
inline-size: var(--zdp-control-height-sm);
|
|
60
|
+
}
|
|
61
|
+
|
|
62
|
+
.zdp-spinner--neutral {
|
|
63
|
+
--zdp-spinner-accent: var(--zdp-color-line-strong);
|
|
64
|
+
}
|
|
65
|
+
|
|
66
|
+
.zdp-spinner--primary {
|
|
67
|
+
--zdp-spinner-accent: var(--zdp-color-accent-primary-strong);
|
|
68
|
+
}
|
|
69
|
+
|
|
70
|
+
.zdp-spinner--success {
|
|
71
|
+
--zdp-spinner-accent: var(--zdp-color-accent-success);
|
|
72
|
+
}
|
|
73
|
+
|
|
74
|
+
.zdp-spinner--warning {
|
|
75
|
+
--zdp-spinner-accent: var(--zdp-color-accent-warning);
|
|
76
|
+
}
|
|
77
|
+
|
|
78
|
+
.zdp-spinner--danger {
|
|
79
|
+
--zdp-spinner-accent: var(--zdp-color-accent-danger);
|
|
80
|
+
}
|
|
81
|
+
|
|
82
|
+
</style>
|
|
@@ -0,0 +1,62 @@
|
|
|
1
|
+
<script lang="ts">
|
|
2
|
+
export let as: 'div' | 'section' | 'article' | 'nav' | 'header' | 'footer' = 'div';
|
|
3
|
+
export let gap: 'xs' | 'sm' | 'md' | 'lg' | 'xl' = 'md';
|
|
4
|
+
export let align: 'stretch' | 'start' | 'center' | 'end' = 'stretch';
|
|
5
|
+
export let labelledBy: string | null = null;
|
|
6
|
+
|
|
7
|
+
$: labelledGroupRole = as === 'div' && labelledBy ? 'group' : undefined;
|
|
8
|
+
</script>
|
|
9
|
+
|
|
10
|
+
<svelte:element
|
|
11
|
+
this={as}
|
|
12
|
+
class={`zdp-stack zdp-stack--gap-${gap} zdp-stack--align-${align}`}
|
|
13
|
+
role={labelledGroupRole}
|
|
14
|
+
aria-labelledby={labelledBy ?? undefined}
|
|
15
|
+
>
|
|
16
|
+
<slot />
|
|
17
|
+
</svelte:element>
|
|
18
|
+
|
|
19
|
+
<style>
|
|
20
|
+
.zdp-stack {
|
|
21
|
+
align-items: stretch;
|
|
22
|
+
display: grid;
|
|
23
|
+
gap: var(--zdp-space-4);
|
|
24
|
+
min-width: 0;
|
|
25
|
+
}
|
|
26
|
+
|
|
27
|
+
.zdp-stack--gap-xs {
|
|
28
|
+
gap: var(--zdp-space-1);
|
|
29
|
+
}
|
|
30
|
+
|
|
31
|
+
.zdp-stack--gap-sm {
|
|
32
|
+
gap: var(--zdp-space-2);
|
|
33
|
+
}
|
|
34
|
+
|
|
35
|
+
.zdp-stack--gap-md {
|
|
36
|
+
gap: var(--zdp-space-4);
|
|
37
|
+
}
|
|
38
|
+
|
|
39
|
+
.zdp-stack--gap-lg {
|
|
40
|
+
gap: var(--zdp-space-6);
|
|
41
|
+
}
|
|
42
|
+
|
|
43
|
+
.zdp-stack--gap-xl {
|
|
44
|
+
gap: var(--zdp-space-8);
|
|
45
|
+
}
|
|
46
|
+
|
|
47
|
+
.zdp-stack--align-stretch {
|
|
48
|
+
align-items: stretch;
|
|
49
|
+
}
|
|
50
|
+
|
|
51
|
+
.zdp-stack--align-start {
|
|
52
|
+
align-items: start;
|
|
53
|
+
}
|
|
54
|
+
|
|
55
|
+
.zdp-stack--align-center {
|
|
56
|
+
align-items: center;
|
|
57
|
+
}
|
|
58
|
+
|
|
59
|
+
.zdp-stack--align-end {
|
|
60
|
+
align-items: end;
|
|
61
|
+
}
|
|
62
|
+
</style>
|
|
@@ -0,0 +1,133 @@
|
|
|
1
|
+
<script lang="ts">
|
|
2
|
+
import Toast from './Toast.svelte';
|
|
3
|
+
import type { ZdpStatusToastItem } from '../toast';
|
|
4
|
+
|
|
5
|
+
export let items: readonly ZdpStatusToastItem[] = [];
|
|
6
|
+
export let placement:
|
|
7
|
+
| 'inline'
|
|
8
|
+
| 'top-start'
|
|
9
|
+
| 'top-end'
|
|
10
|
+
| 'bottom-start'
|
|
11
|
+
| 'bottom-end' = 'bottom-end';
|
|
12
|
+
export let idPrefix = 'zdp-status-toast';
|
|
13
|
+
export let ariaLabel = 'Status notifications';
|
|
14
|
+
export let labelledBy: string | null = null;
|
|
15
|
+
export let onDismiss: ((event: MouseEvent, item: ZdpStatusToastItem) => void) | null = null;
|
|
16
|
+
|
|
17
|
+
function titleId(item: ZdpStatusToastItem): string | null {
|
|
18
|
+
return item.title ? `${idPrefix}-${item.id}-title` : null;
|
|
19
|
+
}
|
|
20
|
+
|
|
21
|
+
function messageId(item: ZdpStatusToastItem): string {
|
|
22
|
+
return `${idPrefix}-${item.id}-message`;
|
|
23
|
+
}
|
|
24
|
+
|
|
25
|
+
function handleDismiss(event: MouseEvent, item: ZdpStatusToastItem): void {
|
|
26
|
+
onDismiss?.(event, item);
|
|
27
|
+
}
|
|
28
|
+
|
|
29
|
+
function handleAction(event: MouseEvent, item: ZdpStatusToastItem): void {
|
|
30
|
+
item.onclick?.(event, item);
|
|
31
|
+
}
|
|
32
|
+
|
|
33
|
+
function resolvedRel(item: ZdpStatusToastItem): string | undefined {
|
|
34
|
+
return item.target === '_blank' ? item.rel ?? 'noopener noreferrer' : item.rel;
|
|
35
|
+
}
|
|
36
|
+
</script>
|
|
37
|
+
|
|
38
|
+
<div
|
|
39
|
+
class={`zdp-status-toast zdp-status-toast--${placement}`}
|
|
40
|
+
aria-label={labelledBy ? undefined : ariaLabel}
|
|
41
|
+
aria-labelledby={labelledBy ?? undefined}
|
|
42
|
+
role="group"
|
|
43
|
+
>
|
|
44
|
+
{#each items as item (item.id)}
|
|
45
|
+
<Toast
|
|
46
|
+
tone={item.tone ?? 'neutral'}
|
|
47
|
+
labelledBy={titleId(item)}
|
|
48
|
+
describedBy={messageId(item)}
|
|
49
|
+
dismissLabel={item.dismissLabel ?? 'Dismiss notification'}
|
|
50
|
+
onClose={onDismiss ? (event) => handleDismiss(event, item) : null}
|
|
51
|
+
>
|
|
52
|
+
{#if item.title}
|
|
53
|
+
<strong class="zdp-toast__title" id={titleId(item) ?? undefined}>{item.title}</strong>
|
|
54
|
+
{/if}
|
|
55
|
+
<p class="zdp-toast__message" id={messageId(item)}>{item.message}</p>
|
|
56
|
+
{#if item.actionLabel}
|
|
57
|
+
{#if item.href}
|
|
58
|
+
<a
|
|
59
|
+
class="zdp-toast__action"
|
|
60
|
+
href={item.href}
|
|
61
|
+
target={item.target}
|
|
62
|
+
rel={resolvedRel(item)}
|
|
63
|
+
onclick={item.onclick ? (event) => handleAction(event, item) : undefined}
|
|
64
|
+
>
|
|
65
|
+
{item.actionLabel}
|
|
66
|
+
</a>
|
|
67
|
+
{:else if item.onclick}
|
|
68
|
+
<button class="zdp-toast__action" type="button" onclick={(event) => handleAction(event, item)}>
|
|
69
|
+
{item.actionLabel}
|
|
70
|
+
</button>
|
|
71
|
+
{/if}
|
|
72
|
+
{/if}
|
|
73
|
+
</Toast>
|
|
74
|
+
{/each}
|
|
75
|
+
</div>
|
|
76
|
+
|
|
77
|
+
<style>
|
|
78
|
+
.zdp-status-toast {
|
|
79
|
+
box-sizing: border-box;
|
|
80
|
+
display: grid;
|
|
81
|
+
gap: var(--zdp-space-3);
|
|
82
|
+
inline-size: min(28rem, calc(100vw - var(--zdp-space-6)));
|
|
83
|
+
max-inline-size: 100%;
|
|
84
|
+
pointer-events: none;
|
|
85
|
+
z-index: 50;
|
|
86
|
+
}
|
|
87
|
+
|
|
88
|
+
.zdp-status-toast :global(.zdp-toast) {
|
|
89
|
+
pointer-events: auto;
|
|
90
|
+
}
|
|
91
|
+
|
|
92
|
+
.zdp-status-toast--inline {
|
|
93
|
+
inline-size: min(100%, 28rem);
|
|
94
|
+
position: static;
|
|
95
|
+
}
|
|
96
|
+
|
|
97
|
+
.zdp-status-toast--top-start,
|
|
98
|
+
.zdp-status-toast--top-end,
|
|
99
|
+
.zdp-status-toast--bottom-start,
|
|
100
|
+
.zdp-status-toast--bottom-end {
|
|
101
|
+
position: fixed;
|
|
102
|
+
}
|
|
103
|
+
|
|
104
|
+
.zdp-status-toast--top-start {
|
|
105
|
+
left: var(--zdp-space-4);
|
|
106
|
+
top: var(--zdp-space-4);
|
|
107
|
+
}
|
|
108
|
+
|
|
109
|
+
.zdp-status-toast--top-end {
|
|
110
|
+
right: var(--zdp-space-4);
|
|
111
|
+
top: var(--zdp-space-4);
|
|
112
|
+
}
|
|
113
|
+
|
|
114
|
+
.zdp-status-toast--bottom-start {
|
|
115
|
+
bottom: var(--zdp-space-4);
|
|
116
|
+
left: var(--zdp-space-4);
|
|
117
|
+
}
|
|
118
|
+
|
|
119
|
+
.zdp-status-toast--bottom-end {
|
|
120
|
+
bottom: var(--zdp-space-4);
|
|
121
|
+
right: var(--zdp-space-4);
|
|
122
|
+
}
|
|
123
|
+
|
|
124
|
+
@media (max-width: 42rem) {
|
|
125
|
+
.zdp-status-toast--top-start,
|
|
126
|
+
.zdp-status-toast--top-end,
|
|
127
|
+
.zdp-status-toast--bottom-start,
|
|
128
|
+
.zdp-status-toast--bottom-end {
|
|
129
|
+
left: var(--zdp-space-3);
|
|
130
|
+
right: var(--zdp-space-3);
|
|
131
|
+
}
|
|
132
|
+
}
|
|
133
|
+
</style>
|
|
@@ -0,0 +1,53 @@
|
|
|
1
|
+
<script lang="ts">
|
|
2
|
+
export let tone: 'panel' | 'raised' | 'inverse' = 'panel';
|
|
3
|
+
export let padding: 'none' | 'sm' | 'md' | 'lg' = 'md';
|
|
4
|
+
export let labelledBy: string | null = null;
|
|
5
|
+
</script>
|
|
6
|
+
|
|
7
|
+
<section
|
|
8
|
+
class={`zdp-surface zdp-surface--${tone} zdp-surface--padding-${padding}`}
|
|
9
|
+
aria-labelledby={labelledBy ?? undefined}
|
|
10
|
+
>
|
|
11
|
+
<slot />
|
|
12
|
+
</section>
|
|
13
|
+
|
|
14
|
+
<style>
|
|
15
|
+
.zdp-surface {
|
|
16
|
+
border: 1px solid var(--zdp-color-line-subtle);
|
|
17
|
+
border-radius: var(--zdp-radius-lg);
|
|
18
|
+
color: var(--zdp-color-ink-normal);
|
|
19
|
+
}
|
|
20
|
+
|
|
21
|
+
.zdp-surface--panel {
|
|
22
|
+
background: var(--zdp-color-surface-panel);
|
|
23
|
+
}
|
|
24
|
+
|
|
25
|
+
:global([data-zdp-theme="dark"]) .zdp-surface--panel {
|
|
26
|
+
background: var(--zdp-color-surface-panel);
|
|
27
|
+
}
|
|
28
|
+
|
|
29
|
+
.zdp-surface--raised {
|
|
30
|
+
background: var(--zdp-color-surface-raised);
|
|
31
|
+
}
|
|
32
|
+
|
|
33
|
+
.zdp-surface--inverse {
|
|
34
|
+
background: var(--zdp-color-surface-inverse);
|
|
35
|
+
color: var(--zdp-color-ink-inverse);
|
|
36
|
+
}
|
|
37
|
+
|
|
38
|
+
.zdp-surface--padding-none {
|
|
39
|
+
padding: 0;
|
|
40
|
+
}
|
|
41
|
+
|
|
42
|
+
.zdp-surface--padding-sm {
|
|
43
|
+
padding: var(--zdp-space-3);
|
|
44
|
+
}
|
|
45
|
+
|
|
46
|
+
.zdp-surface--padding-md {
|
|
47
|
+
padding: var(--zdp-space-4);
|
|
48
|
+
}
|
|
49
|
+
|
|
50
|
+
.zdp-surface--padding-lg {
|
|
51
|
+
padding: var(--zdp-space-6);
|
|
52
|
+
}
|
|
53
|
+
</style>
|