@backstage/ui 0.15.0 → 0.16.0
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 +243 -0
- package/dist/components/Combobox/Combobox.esm.js +150 -52
- package/dist/components/Combobox/Combobox.esm.js.map +1 -1
- package/dist/components/Combobox/Combobox.module.css.esm.js +2 -2
- package/dist/components/Combobox/ComboboxItem.esm.js +76 -0
- package/dist/components/Combobox/ComboboxItem.esm.js.map +1 -0
- package/dist/components/Combobox/ComboboxListBox.esm.js +215 -17
- package/dist/components/Combobox/ComboboxListBox.esm.js.map +1 -1
- package/dist/components/Combobox/definition.esm.js +62 -3
- package/dist/components/Combobox/definition.esm.js.map +1 -1
- package/dist/components/Combobox/useAsyncComboboxState.esm.js +133 -0
- package/dist/components/Combobox/useAsyncComboboxState.esm.js.map +1 -0
- package/dist/components/Header/Header.module.css.esm.js +2 -2
- package/dist/components/Header/HeaderNav.esm.js +0 -1
- package/dist/components/Header/HeaderNav.esm.js.map +1 -1
- package/dist/components/NumberField/NumberField.esm.js +113 -0
- package/dist/components/NumberField/NumberField.esm.js.map +1 -0
- package/dist/components/NumberField/NumberField.module.css.esm.js +8 -0
- package/dist/components/NumberField/NumberField.module.css.esm.js.map +1 -0
- package/dist/components/NumberField/definition.esm.js +34 -0
- package/dist/components/NumberField/definition.esm.js.map +1 -0
- package/dist/components/Select/Select.esm.js +87 -19
- package/dist/components/Select/Select.esm.js.map +1 -1
- package/dist/components/Select/Select.module.css.esm.js +2 -2
- package/dist/components/Select/SelectContent.esm.js +70 -18
- package/dist/components/Select/SelectContent.esm.js.map +1 -1
- package/dist/components/Select/SelectItem.esm.js +76 -0
- package/dist/components/Select/SelectItem.esm.js.map +1 -0
- package/dist/components/Select/SelectListBox.esm.js +175 -19
- package/dist/components/Select/SelectListBox.esm.js.map +1 -1
- package/dist/components/Select/SelectTrigger.esm.js +1 -1
- package/dist/components/Select/SelectTrigger.esm.js.map +1 -1
- package/dist/components/Select/definition.esm.js +72 -9
- package/dist/components/Select/definition.esm.js.map +1 -1
- package/dist/components/Skeleton/Skeleton.module.css.esm.js +2 -2
- package/dist/components/Skeleton/definition.esm.js +1 -0
- package/dist/components/Skeleton/definition.esm.js.map +1 -1
- package/dist/components/Switch/Switch.esm.js +17 -5
- package/dist/components/Switch/Switch.esm.js.map +1 -1
- package/dist/components/Switch/Switch.module.css.esm.js +2 -2
- package/dist/components/Switch/definition.esm.js +1 -0
- package/dist/components/Switch/definition.esm.js.map +1 -1
- package/dist/components/Table/Table.module.css.esm.js +2 -2
- package/dist/components/Table/components/Table.esm.js +60 -57
- package/dist/components/Table/components/Table.esm.js.map +1 -1
- package/dist/components/Table/definition.esm.js +2 -1
- package/dist/components/Table/definition.esm.js.map +1 -1
- package/dist/components/TablePagination/TablePagination.esm.js +4 -1
- package/dist/components/TablePagination/TablePagination.esm.js.map +1 -1
- package/dist/components/Tabs/TabsIndicators.esm.js +155 -108
- package/dist/components/Tabs/TabsIndicators.esm.js.map +1 -1
- package/dist/css/styles.css +184 -68
- package/dist/hooks/useCollectionAdapter.esm.js +67 -0
- package/dist/hooks/useCollectionAdapter.esm.js.map +1 -0
- package/dist/hooks/useDelayedVisibility.esm.js +17 -0
- package/dist/hooks/useDelayedVisibility.esm.js.map +1 -0
- package/dist/hooks/useTrackedSelectionKeys.esm.js +23 -0
- package/dist/hooks/useTrackedSelectionKeys.esm.js.map +1 -0
- package/dist/index.d.ts +805 -77
- package/dist/index.esm.js +7 -2
- package/dist/index.esm.js.map +1 -1
- package/dist/utils/selectableCollection.esm.js +75 -0
- package/dist/utils/selectableCollection.esm.js.map +1 -0
- package/package.json +3 -3
|
@@ -9,133 +9,180 @@ const TabsIndicators = (props) => {
|
|
|
9
9
|
const { classes, tabRefs, tabsRef, hoveredKey, prevHoveredKey } = ownProps;
|
|
10
10
|
const state = useContext(TabListStateContext);
|
|
11
11
|
const prevSelectedKey = useRef(null);
|
|
12
|
-
const updateCSSVariables = useCallback(
|
|
13
|
-
|
|
14
|
-
|
|
15
|
-
|
|
16
|
-
|
|
17
|
-
|
|
18
|
-
|
|
19
|
-
|
|
20
|
-
|
|
21
|
-
|
|
22
|
-
|
|
23
|
-
|
|
12
|
+
const updateCSSVariables = useCallback(
|
|
13
|
+
(animate = true) => {
|
|
14
|
+
if (state == null) return;
|
|
15
|
+
if (!tabsRef.current) return;
|
|
16
|
+
const tabsRect = tabsRef.current.getBoundingClientRect();
|
|
17
|
+
if (state?.selectedKey != null && state.selectedKey !== "") {
|
|
18
|
+
const activeTab = tabRefs.current.get(state.selectedKey.toString());
|
|
19
|
+
if (activeTab) {
|
|
20
|
+
const activeRect = activeTab.getBoundingClientRect();
|
|
21
|
+
const relativeLeft = activeRect.left - tabsRect.left;
|
|
22
|
+
const relativeTop = activeRect.top - tabsRect.top;
|
|
23
|
+
const isFirstActiveTab = prevSelectedKey.current === null;
|
|
24
|
+
if (isFirstActiveTab || !animate) {
|
|
25
|
+
tabsRef.current.style.setProperty(
|
|
26
|
+
"--active-transition-duration",
|
|
27
|
+
"0s"
|
|
28
|
+
);
|
|
29
|
+
if (animate) {
|
|
30
|
+
requestAnimationFrame(() => {
|
|
31
|
+
if (tabsRef.current) {
|
|
32
|
+
tabsRef.current.style.setProperty(
|
|
33
|
+
"--active-transition-duration",
|
|
34
|
+
"0.25s"
|
|
35
|
+
);
|
|
36
|
+
}
|
|
37
|
+
});
|
|
38
|
+
}
|
|
39
|
+
} else {
|
|
40
|
+
tabsRef.current.style.setProperty(
|
|
41
|
+
"--active-transition-duration",
|
|
42
|
+
"0.25s"
|
|
43
|
+
);
|
|
44
|
+
}
|
|
45
|
+
prevSelectedKey.current = state.selectedKey.toString();
|
|
24
46
|
tabsRef.current.style.setProperty(
|
|
25
|
-
"--active-
|
|
26
|
-
|
|
47
|
+
"--active-tab-left",
|
|
48
|
+
`${relativeLeft}px`
|
|
49
|
+
);
|
|
50
|
+
tabsRef.current.style.setProperty(
|
|
51
|
+
"--active-tab-right",
|
|
52
|
+
`${relativeLeft + activeRect.width}px`
|
|
53
|
+
);
|
|
54
|
+
tabsRef.current.style.setProperty(
|
|
55
|
+
"--active-tab-top",
|
|
56
|
+
`${relativeTop}px`
|
|
57
|
+
);
|
|
58
|
+
tabsRef.current.style.setProperty(
|
|
59
|
+
"--active-tab-bottom",
|
|
60
|
+
`${relativeTop + activeRect.height}px`
|
|
61
|
+
);
|
|
62
|
+
tabsRef.current.style.setProperty(
|
|
63
|
+
"--active-tab-width",
|
|
64
|
+
`${activeRect.width}px`
|
|
27
65
|
);
|
|
28
|
-
requestAnimationFrame(() => {
|
|
29
|
-
if (tabsRef.current) {
|
|
30
|
-
tabsRef.current.style.setProperty(
|
|
31
|
-
"--active-transition-duration",
|
|
32
|
-
"0.25s"
|
|
33
|
-
);
|
|
34
|
-
}
|
|
35
|
-
});
|
|
36
|
-
} else {
|
|
37
66
|
tabsRef.current.style.setProperty(
|
|
38
|
-
"--active-
|
|
39
|
-
|
|
67
|
+
"--active-tab-height",
|
|
68
|
+
`${activeRect.height}px`
|
|
40
69
|
);
|
|
70
|
+
tabsRef.current.style.setProperty("--active-tab-opacity", "1");
|
|
41
71
|
}
|
|
42
|
-
|
|
43
|
-
tabsRef.current.style.setProperty(
|
|
44
|
-
|
|
45
|
-
`${relativeLeft}px`
|
|
46
|
-
);
|
|
47
|
-
tabsRef.current.style.setProperty(
|
|
48
|
-
"--active-tab-right",
|
|
49
|
-
`${relativeLeft + activeRect.width}px`
|
|
50
|
-
);
|
|
51
|
-
tabsRef.current.style.setProperty(
|
|
52
|
-
"--active-tab-top",
|
|
53
|
-
`${relativeTop}px`
|
|
54
|
-
);
|
|
55
|
-
tabsRef.current.style.setProperty(
|
|
56
|
-
"--active-tab-bottom",
|
|
57
|
-
`${relativeTop + activeRect.height}px`
|
|
58
|
-
);
|
|
59
|
-
tabsRef.current.style.setProperty(
|
|
60
|
-
"--active-tab-width",
|
|
61
|
-
`${activeRect.width}px`
|
|
62
|
-
);
|
|
63
|
-
tabsRef.current.style.setProperty(
|
|
64
|
-
"--active-tab-height",
|
|
65
|
-
`${activeRect.height}px`
|
|
66
|
-
);
|
|
67
|
-
tabsRef.current.style.setProperty("--active-tab-opacity", "1");
|
|
72
|
+
} else {
|
|
73
|
+
tabsRef.current.style.setProperty("--active-tab-opacity", "0");
|
|
74
|
+
prevSelectedKey.current = null;
|
|
68
75
|
}
|
|
69
|
-
|
|
70
|
-
|
|
71
|
-
|
|
72
|
-
|
|
73
|
-
|
|
74
|
-
|
|
75
|
-
if (hoveredTab) {
|
|
76
|
-
const hoveredRect = hoveredTab.getBoundingClientRect();
|
|
77
|
-
const relativeLeft = hoveredRect.left - tabsRect.left;
|
|
78
|
-
const relativeTop = hoveredRect.top - tabsRect.top;
|
|
79
|
-
tabsRef.current.style.setProperty(
|
|
80
|
-
"--hovered-tab-left",
|
|
81
|
-
`${relativeLeft}px`
|
|
82
|
-
);
|
|
83
|
-
tabsRef.current.style.setProperty(
|
|
84
|
-
"--hovered-tab-right",
|
|
85
|
-
`${relativeLeft + hoveredRect.width}px`
|
|
86
|
-
);
|
|
87
|
-
tabsRef.current.style.setProperty(
|
|
88
|
-
"--hovered-tab-top",
|
|
89
|
-
`${relativeTop}px`
|
|
90
|
-
);
|
|
91
|
-
tabsRef.current.style.setProperty(
|
|
92
|
-
"--hovered-tab-bottom",
|
|
93
|
-
`${relativeTop + hoveredRect.height}px`
|
|
94
|
-
);
|
|
95
|
-
tabsRef.current.style.setProperty(
|
|
96
|
-
"--hovered-tab-width",
|
|
97
|
-
`${hoveredRect.width}px`
|
|
98
|
-
);
|
|
99
|
-
tabsRef.current.style.setProperty(
|
|
100
|
-
"--hovered-tab-height",
|
|
101
|
-
`${hoveredRect.height}px`
|
|
102
|
-
);
|
|
103
|
-
const isNewHoverSession = prevHoveredKey.current === null;
|
|
104
|
-
if (isNewHoverSession) {
|
|
76
|
+
if (hoveredKey) {
|
|
77
|
+
const hoveredTab = tabRefs.current.get(hoveredKey);
|
|
78
|
+
if (hoveredTab) {
|
|
79
|
+
const hoveredRect = hoveredTab.getBoundingClientRect();
|
|
80
|
+
const relativeLeft = hoveredRect.left - tabsRect.left;
|
|
81
|
+
const relativeTop = hoveredRect.top - tabsRect.top;
|
|
105
82
|
tabsRef.current.style.setProperty(
|
|
106
|
-
"--hovered-
|
|
107
|
-
|
|
83
|
+
"--hovered-tab-left",
|
|
84
|
+
`${relativeLeft}px`
|
|
85
|
+
);
|
|
86
|
+
tabsRef.current.style.setProperty(
|
|
87
|
+
"--hovered-tab-right",
|
|
88
|
+
`${relativeLeft + hoveredRect.width}px`
|
|
108
89
|
);
|
|
109
|
-
requestAnimationFrame(() => {
|
|
110
|
-
if (tabsRef.current) {
|
|
111
|
-
tabsRef.current.style.setProperty(
|
|
112
|
-
"--hovered-transition-duration",
|
|
113
|
-
"0.2s"
|
|
114
|
-
);
|
|
115
|
-
}
|
|
116
|
-
});
|
|
117
|
-
} else {
|
|
118
90
|
tabsRef.current.style.setProperty(
|
|
119
|
-
"--hovered-
|
|
120
|
-
|
|
91
|
+
"--hovered-tab-top",
|
|
92
|
+
`${relativeTop}px`
|
|
121
93
|
);
|
|
94
|
+
tabsRef.current.style.setProperty(
|
|
95
|
+
"--hovered-tab-bottom",
|
|
96
|
+
`${relativeTop + hoveredRect.height}px`
|
|
97
|
+
);
|
|
98
|
+
tabsRef.current.style.setProperty(
|
|
99
|
+
"--hovered-tab-width",
|
|
100
|
+
`${hoveredRect.width}px`
|
|
101
|
+
);
|
|
102
|
+
tabsRef.current.style.setProperty(
|
|
103
|
+
"--hovered-tab-height",
|
|
104
|
+
`${hoveredRect.height}px`
|
|
105
|
+
);
|
|
106
|
+
const isNewHoverSession = prevHoveredKey.current === null;
|
|
107
|
+
if (isNewHoverSession || !animate) {
|
|
108
|
+
tabsRef.current.style.setProperty(
|
|
109
|
+
"--hovered-transition-duration",
|
|
110
|
+
"0s"
|
|
111
|
+
);
|
|
112
|
+
if (animate) {
|
|
113
|
+
requestAnimationFrame(() => {
|
|
114
|
+
if (tabsRef.current) {
|
|
115
|
+
tabsRef.current.style.setProperty(
|
|
116
|
+
"--hovered-transition-duration",
|
|
117
|
+
"0.2s"
|
|
118
|
+
);
|
|
119
|
+
}
|
|
120
|
+
});
|
|
121
|
+
}
|
|
122
|
+
} else {
|
|
123
|
+
tabsRef.current.style.setProperty(
|
|
124
|
+
"--hovered-transition-duration",
|
|
125
|
+
"0.2s"
|
|
126
|
+
);
|
|
127
|
+
}
|
|
128
|
+
prevHoveredKey.current = hoveredKey;
|
|
129
|
+
tabsRef.current.style.setProperty("--hovered-tab-opacity", "1");
|
|
122
130
|
}
|
|
123
|
-
|
|
124
|
-
tabsRef.current.style.setProperty("--hovered-tab-opacity", "
|
|
131
|
+
} else {
|
|
132
|
+
tabsRef.current.style.setProperty("--hovered-tab-opacity", "0");
|
|
133
|
+
prevHoveredKey.current = null;
|
|
125
134
|
}
|
|
126
|
-
}
|
|
127
|
-
|
|
128
|
-
|
|
129
|
-
}
|
|
130
|
-
}, [state?.selectedKey, hoveredKey, tabRefs.current]);
|
|
135
|
+
},
|
|
136
|
+
[state?.selectedKey, hoveredKey, tabRefs.current]
|
|
137
|
+
);
|
|
131
138
|
useEffect(() => {
|
|
132
139
|
updateCSSVariables();
|
|
133
140
|
}, [updateCSSVariables, tabRefs.current.size]);
|
|
134
141
|
useEffect(() => {
|
|
135
|
-
const handleResize = () => updateCSSVariables();
|
|
142
|
+
const handleResize = () => updateCSSVariables(false);
|
|
136
143
|
window.addEventListener("resize", handleResize);
|
|
137
144
|
return () => window.removeEventListener("resize", handleResize);
|
|
138
145
|
}, [updateCSSVariables]);
|
|
146
|
+
useEffect(() => {
|
|
147
|
+
if (typeof ResizeObserver === "undefined") {
|
|
148
|
+
return void 0;
|
|
149
|
+
}
|
|
150
|
+
let frame = null;
|
|
151
|
+
const scheduleUpdate = () => {
|
|
152
|
+
if (frame !== null) {
|
|
153
|
+
return;
|
|
154
|
+
}
|
|
155
|
+
frame = requestAnimationFrame(() => {
|
|
156
|
+
frame = null;
|
|
157
|
+
updateCSSVariables(false);
|
|
158
|
+
});
|
|
159
|
+
};
|
|
160
|
+
const resizeObserver = new ResizeObserver(scheduleUpdate);
|
|
161
|
+
const observeTabs = () => {
|
|
162
|
+
resizeObserver.disconnect();
|
|
163
|
+
for (const tab of tabRefs.current.values()) {
|
|
164
|
+
resizeObserver.observe(tab);
|
|
165
|
+
}
|
|
166
|
+
};
|
|
167
|
+
observeTabs();
|
|
168
|
+
const mutationObserver = new MutationObserver(() => {
|
|
169
|
+
observeTabs();
|
|
170
|
+
scheduleUpdate();
|
|
171
|
+
});
|
|
172
|
+
if (tabsRef.current) {
|
|
173
|
+
mutationObserver.observe(tabsRef.current, {
|
|
174
|
+
childList: true,
|
|
175
|
+
subtree: true
|
|
176
|
+
});
|
|
177
|
+
}
|
|
178
|
+
return () => {
|
|
179
|
+
if (frame !== null) {
|
|
180
|
+
cancelAnimationFrame(frame);
|
|
181
|
+
}
|
|
182
|
+
resizeObserver.disconnect();
|
|
183
|
+
mutationObserver.disconnect();
|
|
184
|
+
};
|
|
185
|
+
}, [updateCSSVariables, tabRefs, tabsRef]);
|
|
139
186
|
return /* @__PURE__ */ jsxs(Fragment, { children: [
|
|
140
187
|
/* @__PURE__ */ jsx("div", { className: classes.root }),
|
|
141
188
|
/* @__PURE__ */ jsx("div", { className: classes.hovered })
|
|
@@ -1 +1 @@
|
|
|
1
|
-
{"version":3,"file":"TabsIndicators.esm.js","sources":["../../../src/components/Tabs/TabsIndicators.tsx"],"sourcesContent":["/*\n * Copyright 2024 The Backstage Authors\n *\n * Licensed under the Apache License, Version 2.0 (the \"License\");\n * you may not use this file except in compliance with the License.\n * You may obtain a copy of the License at\n *\n * http://www.apache.org/licenses/LICENSE-2.0\n *\n * Unless required by applicable law or agreed to in writing, software\n * distributed under the License is distributed on an \"AS IS\" BASIS,\n * WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied.\n * See the License for the specific language governing permissions and\n * limitations under the License.\n */\n\nimport { TabListStateContext } from 'react-aria-components';\nimport { useDefinition } from '../../hooks/useDefinition';\nimport { TabsIndicatorsDefinition } from './definition';\nimport { useContext, useEffect, useCallback, useRef } from 'react';\nimport type { TabsIndicatorsProps } from './types';\n\n/**\n * A component that renders the indicators for the toolbar.\n *\n * @internal\n */\nexport const TabsIndicators = (props: TabsIndicatorsProps) => {\n const { ownProps } = useDefinition(TabsIndicatorsDefinition, props);\n const { classes, tabRefs, tabsRef, hoveredKey, prevHoveredKey } = ownProps;\n const state = useContext(TabListStateContext);\n const prevSelectedKey = useRef<string | null>(null);\n\n const updateCSSVariables = useCallback(() => {\n // When rendered inside CollectionBuilder's hidden tree (for collection\n // building), there is no TabListStateContext provider, so state is null.\n // Bail out to avoid overwriting CSS variables on the shared tabsRef DOM\n // element that the real instance also writes to.\n if (state == null) return;\n\n if (!tabsRef.current) return;\n\n const tabsRect = tabsRef.current.getBoundingClientRect();\n\n // Set active tab variables\n if (state?.selectedKey != null && state.selectedKey !== '') {\n const activeTab = tabRefs.current.get(state.selectedKey.toString());\n\n if (activeTab) {\n const activeRect = activeTab.getBoundingClientRect();\n const relativeLeft = activeRect.left - tabsRect.left;\n const relativeTop = activeRect.top - tabsRect.top;\n\n // Control transition timing based on whether this is the first time setting active tab\n const isFirstActiveTab = prevSelectedKey.current === null;\n\n if (isFirstActiveTab) {\n // First time setting active tab: no transitions for position\n tabsRef.current.style.setProperty(\n '--active-transition-duration',\n '0s',\n );\n // Enable transitions on next frame for future tab switches\n requestAnimationFrame(() => {\n if (tabsRef.current) {\n tabsRef.current.style.setProperty(\n '--active-transition-duration',\n '0.25s',\n );\n }\n });\n } else {\n // Switching between tabs: full transitions\n tabsRef.current.style.setProperty(\n '--active-transition-duration',\n '0.25s',\n );\n }\n\n // Update previous selected key for next time\n prevSelectedKey.current = state.selectedKey.toString();\n\n tabsRef.current.style.setProperty(\n '--active-tab-left',\n `${relativeLeft}px`,\n );\n tabsRef.current.style.setProperty(\n '--active-tab-right',\n `${relativeLeft + activeRect.width}px`,\n );\n tabsRef.current.style.setProperty(\n '--active-tab-top',\n `${relativeTop}px`,\n );\n tabsRef.current.style.setProperty(\n '--active-tab-bottom',\n `${relativeTop + activeRect.height}px`,\n );\n tabsRef.current.style.setProperty(\n '--active-tab-width',\n `${activeRect.width}px`,\n );\n tabsRef.current.style.setProperty(\n '--active-tab-height',\n `${activeRect.height}px`,\n );\n tabsRef.current.style.setProperty('--active-tab-opacity', '1');\n }\n } else {\n tabsRef.current.style.setProperty('--active-tab-opacity', '0');\n prevSelectedKey.current = null;\n }\n\n // Set hovered tab variables\n if (hoveredKey) {\n const hoveredTab = tabRefs.current.get(hoveredKey);\n if (hoveredTab) {\n const hoveredRect = hoveredTab.getBoundingClientRect();\n const relativeLeft = hoveredRect.left - tabsRect.left;\n const relativeTop = hoveredRect.top - tabsRect.top;\n\n tabsRef.current.style.setProperty(\n '--hovered-tab-left',\n `${relativeLeft}px`,\n );\n tabsRef.current.style.setProperty(\n '--hovered-tab-right',\n `${relativeLeft + hoveredRect.width}px`,\n );\n tabsRef.current.style.setProperty(\n '--hovered-tab-top',\n `${relativeTop}px`,\n );\n tabsRef.current.style.setProperty(\n '--hovered-tab-bottom',\n `${relativeTop + hoveredRect.height}px`,\n );\n tabsRef.current.style.setProperty(\n '--hovered-tab-width',\n `${hoveredRect.width}px`,\n );\n tabsRef.current.style.setProperty(\n '--hovered-tab-height',\n `${hoveredRect.height}px`,\n );\n // Control transition timing based on whether this is a new hover session\n const isNewHoverSession = prevHoveredKey.current === null;\n\n if (isNewHoverSession) {\n // Starting new hover session: no transitions for position\n tabsRef.current.style.setProperty(\n '--hovered-transition-duration',\n '0s',\n );\n // Enable transitions on next frame for future tab switches\n requestAnimationFrame(() => {\n if (tabsRef.current) {\n tabsRef.current.style.setProperty(\n '--hovered-transition-duration',\n '0.2s',\n );\n }\n });\n } else {\n // Moving between tabs in same session: full transitions\n tabsRef.current.style.setProperty(\n '--hovered-transition-duration',\n '0.2s',\n );\n }\n\n // Update previous hover key for next time\n prevHoveredKey.current = hoveredKey;\n\n tabsRef.current.style.setProperty('--hovered-tab-opacity', '1');\n }\n } else {\n // When not hovering, hide with opacity and reset for next hover session\n tabsRef.current.style.setProperty('--hovered-tab-opacity', '0');\n\n // Reset previous hover key so next hover is treated as new session\n prevHoveredKey.current = null;\n }\n }, [state?.selectedKey, hoveredKey, tabRefs.current]);\n\n useEffect(() => {\n updateCSSVariables();\n }, [updateCSSVariables, tabRefs.current.size]);\n\n useEffect(() => {\n const handleResize = () => updateCSSVariables();\n window.addEventListener('resize', handleResize);\n return () => window.removeEventListener('resize', handleResize);\n }, [updateCSSVariables]);\n\n return (\n <>\n <div className={classes.root} />\n <div className={classes.hovered} />\n </>\n );\n};\n"],"names":[],"mappings":"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"}
|
|
1
|
+
{"version":3,"file":"TabsIndicators.esm.js","sources":["../../../src/components/Tabs/TabsIndicators.tsx"],"sourcesContent":["/*\n * Copyright 2024 The Backstage Authors\n *\n * Licensed under the Apache License, Version 2.0 (the \"License\");\n * you may not use this file except in compliance with the License.\n * You may obtain a copy of the License at\n *\n * http://www.apache.org/licenses/LICENSE-2.0\n *\n * Unless required by applicable law or agreed to in writing, software\n * distributed under the License is distributed on an \"AS IS\" BASIS,\n * WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied.\n * See the License for the specific language governing permissions and\n * limitations under the License.\n */\n\nimport { TabListStateContext } from 'react-aria-components';\nimport { useDefinition } from '../../hooks/useDefinition';\nimport { TabsIndicatorsDefinition } from './definition';\nimport { useContext, useEffect, useCallback, useRef } from 'react';\nimport type { TabsIndicatorsProps } from './types';\n\n/**\n * A component that renders the indicators for the toolbar.\n *\n * @internal\n */\nexport const TabsIndicators = (props: TabsIndicatorsProps) => {\n const { ownProps } = useDefinition(TabsIndicatorsDefinition, props);\n const { classes, tabRefs, tabsRef, hoveredKey, prevHoveredKey } = ownProps;\n const state = useContext(TabListStateContext);\n const prevSelectedKey = useRef<string | null>(null);\n\n const updateCSSVariables = useCallback(\n (animate = true) => {\n // When rendered inside CollectionBuilder's hidden tree (for collection\n // building), there is no TabListStateContext provider, so state is null.\n // Bail out to avoid overwriting CSS variables on the shared tabsRef DOM\n // element that the real instance also writes to.\n if (state == null) return;\n\n if (!tabsRef.current) return;\n\n const tabsRect = tabsRef.current.getBoundingClientRect();\n\n // Set active tab variables\n if (state?.selectedKey != null && state.selectedKey !== '') {\n const activeTab = tabRefs.current.get(state.selectedKey.toString());\n\n if (activeTab) {\n const activeRect = activeTab.getBoundingClientRect();\n const relativeLeft = activeRect.left - tabsRect.left;\n const relativeTop = activeRect.top - tabsRect.top;\n\n // Control transition timing based on whether this is the first time setting active tab\n const isFirstActiveTab = prevSelectedKey.current === null;\n\n if (isFirstActiveTab || !animate) {\n // First time setting active tab or non-animated reposition (e.g.\n // tab resize): no transitions for position\n tabsRef.current.style.setProperty(\n '--active-transition-duration',\n '0s',\n );\n // Enable transitions on next frame for future tab switches, unless\n // more non-animated updates may follow (e.g. a resize in progress)\n if (animate) {\n requestAnimationFrame(() => {\n if (tabsRef.current) {\n tabsRef.current.style.setProperty(\n '--active-transition-duration',\n '0.25s',\n );\n }\n });\n }\n } else {\n // Switching between tabs: full transitions\n tabsRef.current.style.setProperty(\n '--active-transition-duration',\n '0.25s',\n );\n }\n\n // Update previous selected key for next time\n prevSelectedKey.current = state.selectedKey.toString();\n\n tabsRef.current.style.setProperty(\n '--active-tab-left',\n `${relativeLeft}px`,\n );\n tabsRef.current.style.setProperty(\n '--active-tab-right',\n `${relativeLeft + activeRect.width}px`,\n );\n tabsRef.current.style.setProperty(\n '--active-tab-top',\n `${relativeTop}px`,\n );\n tabsRef.current.style.setProperty(\n '--active-tab-bottom',\n `${relativeTop + activeRect.height}px`,\n );\n tabsRef.current.style.setProperty(\n '--active-tab-width',\n `${activeRect.width}px`,\n );\n tabsRef.current.style.setProperty(\n '--active-tab-height',\n `${activeRect.height}px`,\n );\n tabsRef.current.style.setProperty('--active-tab-opacity', '1');\n }\n } else {\n tabsRef.current.style.setProperty('--active-tab-opacity', '0');\n prevSelectedKey.current = null;\n }\n\n // Set hovered tab variables\n if (hoveredKey) {\n const hoveredTab = tabRefs.current.get(hoveredKey);\n if (hoveredTab) {\n const hoveredRect = hoveredTab.getBoundingClientRect();\n const relativeLeft = hoveredRect.left - tabsRect.left;\n const relativeTop = hoveredRect.top - tabsRect.top;\n\n tabsRef.current.style.setProperty(\n '--hovered-tab-left',\n `${relativeLeft}px`,\n );\n tabsRef.current.style.setProperty(\n '--hovered-tab-right',\n `${relativeLeft + hoveredRect.width}px`,\n );\n tabsRef.current.style.setProperty(\n '--hovered-tab-top',\n `${relativeTop}px`,\n );\n tabsRef.current.style.setProperty(\n '--hovered-tab-bottom',\n `${relativeTop + hoveredRect.height}px`,\n );\n tabsRef.current.style.setProperty(\n '--hovered-tab-width',\n `${hoveredRect.width}px`,\n );\n tabsRef.current.style.setProperty(\n '--hovered-tab-height',\n `${hoveredRect.height}px`,\n );\n // Control transition timing based on whether this is a new hover session\n const isNewHoverSession = prevHoveredKey.current === null;\n\n if (isNewHoverSession || !animate) {\n // Starting new hover session or non-animated reposition (e.g. tab\n // resize): no transitions for position\n tabsRef.current.style.setProperty(\n '--hovered-transition-duration',\n '0s',\n );\n // Enable transitions on next frame for future tab switches, unless\n // more non-animated updates may follow (e.g. a resize in progress)\n if (animate) {\n requestAnimationFrame(() => {\n if (tabsRef.current) {\n tabsRef.current.style.setProperty(\n '--hovered-transition-duration',\n '0.2s',\n );\n }\n });\n }\n } else {\n // Moving between tabs in same session: full transitions\n tabsRef.current.style.setProperty(\n '--hovered-transition-duration',\n '0.2s',\n );\n }\n\n // Update previous hover key for next time\n prevHoveredKey.current = hoveredKey;\n\n tabsRef.current.style.setProperty('--hovered-tab-opacity', '1');\n }\n } else {\n // When not hovering, hide with opacity and reset for next hover session\n tabsRef.current.style.setProperty('--hovered-tab-opacity', '0');\n\n // Reset previous hover key so next hover is treated as new session\n prevHoveredKey.current = null;\n }\n },\n [state?.selectedKey, hoveredKey, tabRefs.current],\n );\n\n useEffect(() => {\n updateCSSVariables();\n }, [updateCSSVariables, tabRefs.current.size]);\n\n useEffect(() => {\n const handleResize = () => updateCSSVariables(false);\n window.addEventListener('resize', handleResize);\n return () => window.removeEventListener('resize', handleResize);\n }, [updateCSSVariables]);\n\n useEffect(() => {\n // ResizeObserver is not available in some environments (e.g. Jest/jsdom)\n if (typeof ResizeObserver === 'undefined') {\n return undefined;\n }\n\n let frame: number | null = null;\n const scheduleUpdate = () => {\n if (frame !== null) {\n return;\n }\n\n frame = requestAnimationFrame(() => {\n frame = null;\n updateCSSVariables(false);\n });\n };\n\n const resizeObserver = new ResizeObserver(scheduleUpdate);\n const observeTabs = () => {\n resizeObserver.disconnect();\n for (const tab of tabRefs.current.values()) {\n resizeObserver.observe(tab);\n }\n };\n observeTabs();\n\n // Tab elements can be added, removed, or replaced without this component\n // re-rendering, so watch the DOM to keep the observed set in sync.\n const mutationObserver = new MutationObserver(() => {\n observeTabs();\n scheduleUpdate();\n });\n\n if (tabsRef.current) {\n mutationObserver.observe(tabsRef.current, {\n childList: true,\n subtree: true,\n });\n }\n\n return () => {\n if (frame !== null) {\n cancelAnimationFrame(frame);\n }\n resizeObserver.disconnect();\n mutationObserver.disconnect();\n };\n }, [updateCSSVariables, tabRefs, tabsRef]);\n\n return (\n <>\n <div className={classes.root} />\n <div className={classes.hovered} />\n </>\n );\n};\n"],"names":[],"mappings":"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"}
|