@stainless-api/ui-primitives 0.1.0-beta.4 → 0.1.0-beta.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.
package/CHANGELOG.md CHANGED
@@ -1,5 +1,21 @@
1
1
  # @stainless-api/ui-primitives
2
2
 
3
+ ## 0.1.0-beta.6
4
+
5
+ ### Minor Changes
6
+
7
+ - f664b4d: Creating design system css variables, markdown styles, and typography improvements
8
+
9
+ ### Patch Changes
10
+
11
+ - 34cbd12: verify publishing
12
+
13
+ ## 0.1.0-beta.5
14
+
15
+ ### Patch Changes
16
+
17
+ - e7a2a96: verifying dependent packages update
18
+
3
19
  ## 0.1.0-beta.4
4
20
 
5
21
  ### Patch Changes
package/package.json CHANGED
@@ -1,6 +1,6 @@
1
1
  {
2
2
  "name": "@stainless-api/ui-primitives",
3
- "version": "0.1.0-beta.4",
3
+ "version": "0.1.0-beta.6",
4
4
  "publishConfig": {
5
5
  "access": "public"
6
6
  },
@@ -10,11 +10,6 @@
10
10
  "import": "./src/index.ts",
11
11
  "require": "./src/index.ts"
12
12
  },
13
- "./scripts": {
14
- "types": "./src/scripts/index.ts",
15
- "import": "./src/scripts/index.ts",
16
- "require": "./src/scripts/index.ts"
17
- },
18
13
  "./styles.css": "./src/styles.css"
19
14
  },
20
15
  "types": "./src/index.ts",
@@ -1,4 +1,4 @@
1
- @layer stl-ui {
1
+ @layer stl-ui.components {
2
2
  .stl-ui-callout {
3
3
  --stl-ui-callout-background-color: var(--stl-ui-muted-background);
4
4
  --stl-ui-callout-border-color: var(--stl-ui-border);
@@ -11,115 +11,113 @@
11
11
  }
12
12
  }
13
13
 
14
- @layer stl-ui {
15
- @layer typography {
16
- .stl-ui-prose {
17
- details {
18
- --stl-ui-details-padding: 12px;
19
- --stl-ui-details-content-padding-bottom: 4px;
20
- --stl-ui-details-marker-size: 1em;
21
- --stl-ui-details-marker-margin: calc((1lh - var(--stl-ui-details-marker-size)) / 2);
22
- --stl-ui-details-row-gap: 8px;
23
- --stl-ui-details-summary-column-gap: 8px;
24
- --stl-ui-details-border-radius: var(--stl-ui-layout-border-radius);
25
- /* left inset to make content line up with summary content (after chevron) */
26
- --stl-ui--internal--details-padding-start: calc(
27
- var(--stl-ui-details-padding)
28
- + var(--stl-ui-details-marker-size)
29
- + var(--stl-ui-details-marker-margin) * 2
30
- + var(--stl-ui-details-summary-column-gap)
31
- );
32
-
33
- background-color: var(--stl-ui-card-background);
34
- border: 1px solid var(--stl-ui-border);
35
- border-radius: var(--stl-ui-details-border-radius);
36
- font-size: var(--stl-ui-typography-text-body);
37
-
14
+ @layer stl-ui.components {
15
+ .stl-ui-prose {
16
+ details {
17
+ --stl-ui-details-padding: 12px;
18
+ --stl-ui-details-content-padding-bottom: 4px;
19
+ --stl-ui-details-marker-size: 1em;
20
+ --stl-ui-details-marker-margin: calc((1lh - var(--stl-ui-details-marker-size)) / 2);
21
+ --stl-ui-details-row-gap: 8px;
22
+ --stl-ui-details-summary-column-gap: 8px;
23
+ --stl-ui-details-border-radius: var(--stl-ui-layout-border-radius);
24
+ /* left inset to make content line up with summary content (after chevron) */
25
+ --stl-ui--internal--details-padding-start: calc(
26
+ var(--stl-ui-details-padding)
27
+ + var(--stl-ui-details-marker-size)
28
+ + var(--stl-ui-details-marker-margin) * 2
29
+ + var(--stl-ui-details-summary-column-gap)
30
+ );
31
+
32
+ background-color: var(--stl-ui-card-background);
33
+ border: 1px solid var(--stl-ui-border);
34
+ border-radius: var(--stl-ui-details-border-radius);
35
+ font-size: var(--stl-ui-typography-text-body);
36
+
37
+ padding: var(--stl-ui-details-padding);
38
+ /* indent content to line up with left edge of summary content */
39
+ padding-inline-start: var(--stl-ui--internal--details-padding-start);
40
+
41
+ summary {
42
+ display: block;
43
+ list-style: none;
44
+
45
+ /* summary extends to the edges of the element in order to increase click target */
38
46
  padding: var(--stl-ui-details-padding);
39
- /* indent content to line up with left edge of summary content */
40
47
  padding-inline-start: var(--stl-ui--internal--details-padding-start);
48
+ margin: calc(-1 * var(--stl-ui-details-padding));
49
+ margin-inline-start: calc(-1 * var(--stl-ui--internal--details-padding-start));
41
50
 
42
- summary {
43
- display: block;
44
- list-style: none;
51
+ cursor: pointer;
52
+ font-weight: 500;
45
53
 
46
- /* summary extends to the edges of the element in order to increase click target */
47
- padding: var(--stl-ui-details-padding);
48
- padding-inline-start: var(--stl-ui--internal--details-padding-start);
49
- margin: calc(-1 * var(--stl-ui-details-padding));
50
- margin-inline-start: calc(-1 * var(--stl-ui--internal--details-padding-start));
54
+ border-radius: var(--stl-ui-details-border-radius);
51
55
 
52
- cursor: pointer;
53
- font-weight: 500;
56
+ &::before {
57
+ content: '';
58
+ width: var(--stl-ui-details-marker-size);
59
+ height: var(--stl-ui-details-marker-size);
60
+ margin: var(--stl-ui-details-marker-margin);
61
+ background-color: currentColor;
62
+ display: block;
63
+ position: absolute;
64
+ --stl-ui-details--internal--marker-width: calc(var(--stl-ui-details-marker-size) + var(--stl-ui-details-marker-margin) * 2);
65
+ --stl-ui-details--internal--summary-marker-transform: translateX(calc(-1 * var(--stl-ui-details--internal--marker-width) - var(--stl-ui-details-summary-column-gap)));
66
+ transform: var(--stl-ui-details--internal--summary-marker-transform);
54
67
 
55
- border-radius: var(--stl-ui-details-border-radius);
68
+ --lucide-chevron-right: url('');
69
+ mask-image: var(--lucide-chevron-right);
70
+ mask-size: contain;
71
+ mask-repeat: no-repeat;
56
72
 
57
- &::before {
58
- content: '';
59
- width: var(--stl-ui-details-marker-size);
60
- height: var(--stl-ui-details-marker-size);
61
- margin: var(--stl-ui-details-marker-margin);
62
- background-color: currentColor;
63
- display: block;
64
- position: absolute;
65
- --stl-ui-details--internal--marker-width: calc(var(--stl-ui-details-marker-size) + var(--stl-ui-details-marker-margin) * 2);
66
- --stl-ui-details--internal--summary-marker-transform: translateX(calc(-1 * var(--stl-ui-details--internal--marker-width) - var(--stl-ui-details-summary-column-gap)));
67
- transform: var(--stl-ui-details--internal--summary-marker-transform);
68
-
69
- --lucide-chevron-right: url('');
70
- mask-image: var(--lucide-chevron-right);
71
- mask-size: contain;
72
- mask-repeat: no-repeat;
73
-
74
- transition: transform 0.1s ease-out;
75
- }
73
+ transition: transform 0.1s ease-out;
74
+ }
76
75
 
77
- & > :first-child {
78
- margin-top: 0;
79
- }
76
+ & > :first-child {
77
+ margin-top: 0;
80
78
  }
79
+ }
81
80
 
82
- &[open] {
83
- padding-bottom: calc(var(--stl-ui-details-padding) + var(--stl-ui-details-content-padding-bottom));
81
+ &[open] {
82
+ padding-bottom: calc(var(--stl-ui-details-padding) + var(--stl-ui-details-content-padding-bottom));
84
83
 
85
- summary {
86
- /* padding-bottom shouldn’t extend beyond the row gap while open, i.e. we shouldn’t be negative-margin-placing content overlapping summary */
87
- --stl-ui--internal--summary-padding-bottom: min(var(--stl-ui-details-padding), var(--stl-ui-details-row-gap));
88
- padding-bottom: var(--stl-ui--internal--summary-padding-bottom);
89
- margin-bottom: calc(var(--stl-ui-details-row-gap) - var(--stl-ui--internal--summary-padding-bottom));
90
- border-bottom-left-radius: 0;
91
- border-bottom-right-radius: 0;
84
+ summary {
85
+ /* padding-bottom shouldn’t extend beyond the row gap while open, i.e. we shouldn’t be negative-margin-placing content overlapping summary */
86
+ --stl-ui--internal--summary-padding-bottom: min(var(--stl-ui-details-padding), var(--stl-ui-details-row-gap));
87
+ padding-bottom: var(--stl-ui--internal--summary-padding-bottom);
88
+ margin-bottom: calc(var(--stl-ui-details-row-gap) - var(--stl-ui--internal--summary-padding-bottom));
89
+ border-bottom-left-radius: 0;
90
+ border-bottom-right-radius: 0;
92
91
 
93
- &::before {
94
- transform: var(--stl-ui-details--internal--summary-marker-transform) rotate(90deg);
95
- }
92
+ &::before {
93
+ transform: var(--stl-ui-details--internal--summary-marker-transform) rotate(90deg);
96
94
  }
97
95
  }
96
+ }
98
97
 
99
- summary + * {
100
- margin-top: 0;
101
- }
98
+ summary + * {
99
+ margin-top: 0;
102
100
  }
103
101
  }
102
+ }
104
103
 
105
- .stl-ui-details-group {
106
- & > details:has(+ details) {
104
+ .stl-ui-details-group {
105
+ & > details:has(+ details) {
106
+ border-bottom-left-radius: 0;
107
+ border-bottom-right-radius: 0;
108
+ summary {
107
109
  border-bottom-left-radius: 0;
108
110
  border-bottom-right-radius: 0;
109
- summary {
110
- border-bottom-left-radius: 0;
111
- border-bottom-right-radius: 0;
112
- }
113
111
  }
114
- & > details + details {
115
- margin-top: 0;
116
- border-top: 0;
112
+ }
113
+ & > details + details {
114
+ margin-top: 0;
115
+ border-top: 0;
116
+ border-top-left-radius: 0;
117
+ border-top-right-radius: 0;
118
+ summary {
117
119
  border-top-left-radius: 0;
118
120
  border-top-right-radius: 0;
119
- summary {
120
- border-top-left-radius: 0;
121
- border-top-right-radius: 0;
122
- }
123
121
  }
124
122
  }
125
123
  }
package/src/index.ts CHANGED
@@ -1,4 +1,3 @@
1
1
  export * from './components/Button';
2
- export * from './components/DropdownButton';
3
2
  export * from './components/Callout';
4
3
  export * from './components/DetailsGroup';
@@ -1,8 +1,7 @@
1
- @layer stl-ui {
1
+ @layer stl-ui.starlight-compat {
2
2
  :root {
3
3
  --stl-ui-button-border-radius: 8px;
4
4
  --stl-ui-text-body: 14px;
5
- --stl-ui-font-family: 'Geist', system-ui, sans-serif;
6
5
  --stl-ui-button-size: 2rem;
7
6
 
8
7
  /* Brand colors */
@@ -11,6 +10,9 @@
11
10
  }
12
11
 
13
12
  :root {
13
+ --sl-font: var(--stl-ui-typography-font);
14
+ --sl-font-mono: var(--stl-ui-typography-font-mono);
15
+
14
16
  /* Layout and Typography */
15
17
  --sl-text-h1: var(--stl-ui-typography-text-h1);
16
18
  --sl-text-h2: var(--stl-ui-typography-text-h2);
@@ -71,18 +73,43 @@
71
73
 
72
74
  /* Starlight Compatibility */
73
75
  .stl-ui-prose {
74
- .sl-anchor-link svg {
75
- margin-top: 4px;
76
+ :is(h1, h2, h3, h4, h5) code {
77
+ font-size: 0.9em;
78
+ font-weight: inherit;
76
79
  }
77
-
78
- h2 code {
80
+ /* TODO: Disable starlight headingLinks and replace with our own */
81
+ /* Duplicate styles from h1–5 in typography; TODO: move to mixins after adopting preprocessor */
82
+ .sl-heading-wrapper.level-h1 {
83
+ font-size: var(--stl-ui-typography-text-h1);
84
+ letter-spacing: -0.03em;
85
+ margin-top: 64px;
86
+ }
87
+ .sl-heading-wrapper.level-h2 {
88
+ font-size: var(--stl-ui-typography-text-h2);
89
+ letter-spacing: -0.03em;
90
+ margin-top: 48px;
91
+ }
92
+ .sl-heading-wrapper.level-h3 {
93
+ font-size: var(--stl-ui-typography-text-h3);
94
+ letter-spacing: -0.02em;
95
+ margin-top: 40px;
96
+ }
97
+ .sl-heading-wrapper.level-h4 {
79
98
  font-size: var(--stl-ui-typography-text-h4);
80
- line-height: 120%;
99
+ letter-spacing: -0.02em;
100
+ margin-top: 32px;
81
101
  }
82
-
83
- h3 code {
102
+ .sl-heading-wrapper.level-h5 {
84
103
  font-size: var(--stl-ui-typography-text-h5);
85
- line-height: 120%;
104
+ letter-spacing: -0.02em;
105
+ margin-top: 24px;
106
+ }
107
+ .sl-heading-wrapper {
108
+ --sl-anchor-icon-size: 0.8em;
109
+ }
110
+ /* TODO: replace with an icon that matches Stainless branding */
111
+ .sl-anchor-link svg {
112
+ margin-top: 0;
86
113
  }
87
114
 
88
115
  /* Tab component */
@@ -53,10 +53,6 @@
53
53
  line-height: var(--stl-ui-typography-line-height);
54
54
  letter-spacing: -0.01em;
55
55
 
56
- * {
57
- margin: 0;
58
- }
59
-
60
56
  p {
61
57
  margin-top: 16px;
62
58
  }
package/src/styles.css CHANGED
@@ -6,6 +6,5 @@
6
6
  @import './styles/starlight-compat.css';
7
7
 
8
8
  @import './components/button.css';
9
- @import './components/dropdown-button.css';
10
9
  @import './components/callout.css';
11
10
  @import './components/details.css';
@@ -1,98 +0,0 @@
1
- import { ChevronsUpDown, ExternalLink } from 'lucide-react';
2
-
3
- export function DropdownButtonPrimaryActionText({ children }: { children?: React.ReactNode }) {
4
- return <span data-part="primary-action-text">{children}</span>;
5
- }
6
-
7
- export function DropdownButtonPrimaryAction({ children }: { children?: React.ReactNode }) {
8
- return (
9
- <button
10
- type="button"
11
- data-part="primary-action"
12
- className="stl-ui-dropdown-button__button stl-ui-dropdown-button--action"
13
- >
14
- {children}
15
- </button>
16
- );
17
- }
18
-
19
- export function DropdownButtonTrigger() {
20
- return (
21
- <button
22
- className="stl-ui-dropdown-button__button stl-ui-dropdown-button__trigger"
23
- aria-haspopup="listbox"
24
- aria-expanded="false"
25
- data-part="trigger"
26
- >
27
- <ChevronsUpDown size={16} />
28
- </button>
29
- );
30
- }
31
-
32
- export function DropdownButtonMenu({ children }: { children?: React.ReactNode }) {
33
- return (
34
- <div className="stl-ui-dropdown-button__menu" data-state="closed" data-part="menu">
35
- {children}
36
- </div>
37
- );
38
- }
39
-
40
- export function DropdownButtonItemIcon({ children }: { children?: React.ReactNode }) {
41
- return (
42
- <div className="stl-ui-dropdown-button__menu-item-icon" data-part="item-icon">
43
- {children}
44
- </div>
45
- );
46
- }
47
-
48
- export function DropdownButtonItemText({ children }: { children?: React.ReactNode }) {
49
- return (
50
- <span className="stl-ui-dropdown-button__menu-item-text" data-part="item-text">
51
- {children}
52
- </span>
53
- );
54
- }
55
-
56
- export function DropdownButtonItemTextSubtle({ children }: { children?: React.ReactNode }) {
57
- return (
58
- <span className="stl-ui-dropdown-button__menu-item-text-subtle" data-part="item-text-subtle">
59
- {children}
60
- </span>
61
- );
62
- }
63
-
64
- export function DropdownButtonItem({
65
- children,
66
- value,
67
- isExternalLink,
68
- }: {
69
- children?: React.ReactNode;
70
- value: string;
71
- isExternalLink?: boolean;
72
- }) {
73
- return (
74
- <div className="stl-ui-dropdown-button__menu-item" data-part="item" data-value={value}>
75
- <div className="stl-ui-dropdown-button__menu-item-content">{children}</div>
76
- {isExternalLink && (
77
- <div
78
- className="stl-ui-dropdown-button__menu-item-external-link-icon"
79
- data-part="item-external-link-icon"
80
- >
81
- <ExternalLink size={16} />
82
- </div>
83
- )}
84
- </div>
85
- );
86
- }
87
-
88
- export function DropdownButtonDivider() {
89
- return <div className="stl-ui-dropdown-button__divider" data-part="divider" />;
90
- }
91
-
92
- export function DropdownButton({ id, children }: { id: string; children?: React.ReactNode }) {
93
- return (
94
- <div className="stl-ui-dropdown-button stl-ui-not-prose not-content" id={id}>
95
- {children}
96
- </div>
97
- );
98
- }
@@ -1,162 +0,0 @@
1
- @layer stl-ui {
2
- .stl-ui-dropdown-button {
3
- --stl-ui-dropdown-button-color: var(--stl-ui-foreground);
4
- --stl-ui-dropdown-button-background-color: var(--stl-ui-card-background);
5
- --stl-ui-dropdown-button-border-color: var(--stl-ui-border);
6
- --stl-ui-dropdown-button-border-radius: var(--stl-ui-layout-border-radius-sml);
7
- --stl-ui-dropdown-button-font-size: var(--stl-ui-type-scale-text-sm);
8
-
9
- --stl-ui-dropdown-button-height: 32px;
10
- --stl-ui-dropdown-button-padding: 8px 10px;
11
- --stl-ui-dropdown-button-line-height: 100%;
12
- --stl-ui-dropdown-button-font-weight: 500;
13
-
14
- position: relative;
15
- display: inline-flex;
16
- align-items: center;
17
-
18
- background-color: var(--stl-ui-dropdown-button-background-color);
19
- border: 1px solid var(--stl-ui-dropdown-button-border-color);
20
- border-radius: var(--stl-ui-dropdown-button-border-radius);
21
- color: var(--stl-ui-dropdown-button-color);
22
- gap: 0;
23
- font-size: var(--stl-ui-dropdown-button-font-size);
24
- }
25
-
26
- .stl-ui-dropdown-button__button {
27
- border: none;
28
- background: none;
29
- height: var(--stl-ui-dropdown-button-height);
30
- padding: var(--stl-ui-dropdown-button-padding);
31
- line-height: var(--stl-ui-dropdown-button-line-height);
32
- font-weight: var(--stl-ui-dropdown-button-font-weight);
33
- cursor: pointer;
34
- display: flex;
35
- align-items: center;
36
- justify-content: center;
37
-
38
- &:hover {
39
- background-color: oklch(from var(--stl-ui-foreground) l c h / 0.05);
40
- border-color: var(--stl-ui-border-emphasis);
41
- }
42
- }
43
-
44
- .stl-ui-dropdown-button--action {
45
- display: flex;
46
- align-items: center;
47
- gap: 8px;
48
-
49
- &:hover {
50
- background-color: oklch(from var(--stl-ui-foreground) l c h / 0.05);
51
- border-color: var(--stl-ui-border-emphasis);
52
- }
53
-
54
- &.disabled {
55
- cursor: not-allowed;
56
- background-color: oklch(from var(--stl-ui-foreground) l c h / 0.05);
57
- }
58
- }
59
-
60
- .stl-ui-dropdown-button__trigger {
61
- border-left: 1px solid var(--stl-ui-border);
62
- border-radius: 0;
63
-
64
- &:hover {
65
- background-color: oklch(from var(--stl-ui-foreground) l c h / 0.05);
66
- border-color: var(--stl-ui-border-emphasis);
67
- }
68
- }
69
-
70
- .stl-ui-dropdown-button__menu {
71
- --stl-ui-dropdown-button__menu-background-color: var(--stl-ui-card-background);
72
- --stl-ui-dropdown-button__menu-border-color: var(--stl-ui-border);
73
- --stl-ui-dropdown-button__menu-box-shadow: var(--stl-ui-shadow-md);
74
- --stl-ui-dropdown-button__menu-border-radius: var(--stl-ui-layout-border-radius-sml);
75
-
76
- background-color: var(--stl-ui-dropdown-button__menu-background-color);
77
- border: 1px solid var(--stl-ui-dropdown-button__menu-border-color);
78
- box-shadow: var(--stl-ui-dropdown-button__menu-box-shadow);
79
- border-radius: var(--stl-ui-dropdown-button__menu-border-radius);
80
-
81
- position: absolute;
82
- top: 100%;
83
- right: 0;
84
- margin-top: 4px;
85
- z-index: 1000;
86
- min-width: 100%;
87
- padding: 4px;
88
- display: none;
89
-
90
- &[data-state='open'] {
91
- display: block;
92
- }
93
- }
94
-
95
- .stl-ui-dropdown-button__menu-item {
96
- /* border-radius, height, and lineheight */
97
- --stl-ui-dropdown-button__menu-item-border-radius: var(--stl-ui-dropdown-button-border-radius);
98
- --stl-ui-dropdown-button__menu-item-height: var(--stl-ui-dropdown-button-height);
99
- --stl-ui-dropdown-button__menu-item-line-height: var(--stl-ui-dropdown-button-line-height);
100
- --stl-ui-dropdown-button__menu-item-hover-background-color: oklch(
101
- from var(--stl-ui-foreground) l c h / 0.05
102
- );
103
-
104
- border-radius: var(--stl-ui-dropdown-button__menu-item-border-radius);
105
- height: var(--stl-ui-dropdown-button__menu-item-height);
106
- line-height: var(--stl-ui-dropdown-button__menu-item-line-height);
107
-
108
- padding: 8px;
109
- cursor: pointer;
110
- display: flex;
111
- align-items: center;
112
- justify-content: space-between;
113
- gap: 8px;
114
-
115
- &:hover {
116
- background-color: var(--stl-ui-dropdown-button__menu-item-hover-background-color);
117
- }
118
-
119
- .stl-ui-dropdown-button__menu-item-content {
120
- display: flex;
121
- align-items: center;
122
- gap: 8px;
123
- }
124
-
125
- .stl-ui-dropdown-button__menu-item-text {
126
- white-space: nowrap;
127
- }
128
-
129
- .stl-ui-dropdown-button__menu-item-text-subtle {
130
- --stl-ui-dropdown-button__menu-item-text-subtle-color: var(--stl-ui-foreground-muted);
131
- color: var(--stl-ui-dropdown-button__menu-item-text-subtle-color);
132
- }
133
-
134
- .stl-ui-dropdown-button__menu-item-icon {
135
- display: flex;
136
- align-items: center;
137
- justify-content: center;
138
- }
139
-
140
- .stl-ui-dropdown-button__menu-item-external-link-icon {
141
- --stl-ui-dropdown-button__menu-item-external-link-icon-color: oklch(
142
- from var(--stl-ui-foreground) l c h / 0.25
143
- );
144
- margin-left: 8px;
145
- svg {
146
- color: var(--stl-ui-dropdown-button__menu-item-external-link-icon-color);
147
- }
148
- }
149
- }
150
-
151
- .stl-ui-dropdown-button__divider {
152
- --stl-ui-dropdown-button__divider-height: 1px;
153
- --stl-ui-dropdown-button__divider-color: var(--stl-ui-border);
154
-
155
- height: var(--stl-ui-dropdown-button__divider-height);
156
- background-color: var(--stl-ui-dropdown-button__divider-color);
157
-
158
- margin: 4px 0;
159
- width: calc(100% + 8px);
160
- transform: translateX(-4px);
161
- }
162
- }
@@ -1,55 +0,0 @@
1
- export function initDropdownButton({
2
- dropdownId,
3
- onSelect,
4
- onPrimaryAction,
5
- }: {
6
- dropdownId: string;
7
- onSelect: (value: string) => void;
8
- onPrimaryAction: (primaryActionElement: Element) => void;
9
- }) {
10
- const dropdown = document.getElementById(dropdownId);
11
- if (!dropdown) return;
12
-
13
- const trigger = dropdown.querySelector('[data-part="trigger"]') as HTMLButtonElement | null;
14
- const menu = dropdown.querySelector('[data-part="menu"]') as HTMLElement | null;
15
- const primaryAction = dropdown.querySelector('[data-part="primary-action"]');
16
-
17
- if (!trigger || !menu || !primaryAction) return;
18
-
19
- let isOpen = false;
20
-
21
- function toggleDropdown() {
22
- if (!trigger || !menu) return;
23
- isOpen = !isOpen;
24
- menu.dataset.state = isOpen ? 'open' : 'closed';
25
- trigger.setAttribute('aria-expanded', String(isOpen));
26
- }
27
-
28
- trigger.addEventListener('click', toggleDropdown);
29
-
30
- document.addEventListener('click', (event) => {
31
- if (!isOpen) return;
32
- if (!dropdown.contains(event.target as Node)) {
33
- toggleDropdown();
34
- }
35
-
36
- if (primaryAction && primaryAction.contains(event.target as Node)) {
37
- toggleDropdown();
38
- }
39
- });
40
-
41
- const items = dropdown.querySelectorAll('[data-part="item"]');
42
- items.forEach((item) => {
43
- item.addEventListener('click', () => {
44
- const value = item.getAttribute('data-value');
45
- if (value) {
46
- onSelect(value);
47
- }
48
- toggleDropdown();
49
- });
50
- });
51
-
52
- primaryAction.addEventListener('click', () => {
53
- onPrimaryAction(primaryAction);
54
- });
55
- }
@@ -1 +0,0 @@
1
- export * from './dropdown-button';