@versini/ui-button 12.0.4 → 12.1.1

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.
@@ -1,11 +1,12 @@
1
1
  /*!
2
- @versini/ui-button v12.0.4
2
+ @versini/ui-button v12.1.1
3
3
  © 2026 gizmette.com
4
4
  */
5
5
 
6
6
  import clsx from "clsx";
7
7
 
8
8
 
9
+
9
10
  const BUTTON_CLASSNAME = "av-button";
10
11
 
11
12
 
@@ -63,12 +64,10 @@ const ACTIVE_BG_DANGER_DARK_DARK_LIGHT = "active:bg-action-danger-dark-active da
63
64
  const ACTIVE_BG_DANGER_LIGHT_DARK_DARK = "active:bg-action-danger-light-active dark:active:bg-action-danger-dark-active";
64
65
  // Border classes
65
66
  const BORDER_MEDIUM = "border-border-medium";
66
- const BORDER_ACCENT = "border-border-accent";
67
67
  const BORDER_DANGER_MEDIUM = "border-border-danger-medium";
68
68
  const BORDER_DANGER_DARK = "border-border-danger-dark";
69
69
  // System mode compound border classes
70
- const BORDER_MEDIUM_DARK_ACCENT = "border-border-medium dark:border-border-accent";
71
- const BORDER_ACCENT_DARK_MEDIUM = "border-border-accent dark:border-border-medium";
70
+ const BORDER_DARK_MEDIUM = "border-border-medium dark:border-border-medium";
72
71
  const BORDER_DANGER_DARK_DARK_MEDIUM = "border-border-danger-dark dark:border-border-danger-medium";
73
72
  const BORDER_DANGER_MEDIUM_DARK_DARK = "border-border-danger-medium dark:border-border-danger-dark";
74
73
  // Focus classes
@@ -188,12 +187,17 @@ const getButtonTextCopyClasses = ({ mode, noBackground, truncate, variant })=>{
188
187
  return clsx(NOT_PLUME, TEXT_COPY_LIGHTER, truncateClass);
189
188
  }
190
189
  /* v8 ignore stop */ };
191
- const getButtonBackgroundClasses = ({ mode, noBackground, variant })=>{
190
+ // Aqua-style split background overlay:
191
+ // - Top half: flat white overlay (light, uniform)
192
+ // - Bottom half: starts darker, gradually lightens toward the bottom edge
193
+ const BG_SPLIT_OVERLAY = "bg-[linear-gradient(to_bottom,oklch(1_0_0/0.2)_50%,oklch(0_0_0/0.1)_50%,oklch(1_0_0/0.05)_100%)]";
194
+ const getButtonBackgroundClasses = ({ mode, noBackground, variant, splitBackground })=>{
192
195
  if (noBackground) {
193
196
  return;
194
197
  }
198
+ const splitClass = splitBackground ? BG_SPLIT_OVERLAY : "";
195
199
  if (isPrimary(variant)) {
196
- return clsx({
200
+ return clsx(splitClass, {
197
201
  [BG_ACTION_LIGHT]: isLight(mode),
198
202
  [BG_ACTION_DARK]: isDark(mode),
199
203
  [BG_LIGHT_DARK_DARK]: isSystem(mode),
@@ -201,7 +205,7 @@ const getButtonBackgroundClasses = ({ mode, noBackground, variant })=>{
201
205
  });
202
206
  }
203
207
  if (isSecondary(variant)) {
204
- return clsx({
208
+ return clsx(splitClass, {
205
209
  [BG_ACTION_DARK]: isLight(mode),
206
210
  [BG_ACTION_LIGHT]: isDark(mode),
207
211
  [BG_DARK_DARK_LIGHT]: isSystem(mode),
@@ -209,7 +213,7 @@ const getButtonBackgroundClasses = ({ mode, noBackground, variant })=>{
209
213
  });
210
214
  }
211
215
  if (isDanger(variant)) {
212
- return clsx({
216
+ return clsx(splitClass, {
213
217
  [BG_DANGER_DARK]: isDark(mode),
214
218
  [BG_DANGER_LIGHT]: isLight(mode),
215
219
  [BG_DANGER_DARK_DARK_LIGHT]: isSystem(mode),
@@ -217,7 +221,7 @@ const getButtonBackgroundClasses = ({ mode, noBackground, variant })=>{
217
221
  });
218
222
  }
219
223
  /* v8 ignore start - selected variant edge case */ if (isSelected(variant)) {
220
- return "bg-action-selected-dark";
224
+ return clsx(splitClass, "bg-action-selected-dark");
221
225
  }
222
226
  /* v8 ignore stop */ };
223
227
  const getButtonRadiusClasses = ({ radius })=>{
@@ -313,18 +317,14 @@ const getButtonBorderClasses = ({ mode, noBorder, variant })=>{
313
317
  }
314
318
  if (isPrimary(variant)) {
315
319
  return clsx("border", {
316
- [BORDER_MEDIUM]: isLight(mode),
317
- [BORDER_ACCENT]: isDark(mode),
318
- [BORDER_MEDIUM_DARK_ACCENT]: isSystem(mode),
319
- [BORDER_ACCENT_DARK_MEDIUM]: isAltSystem(mode)
320
+ [BORDER_MEDIUM]: isLight(mode) || isDark(mode),
321
+ [BORDER_DARK_MEDIUM]: isAltSystem(mode) || isSystem(mode)
320
322
  });
321
323
  }
322
324
  if (isSecondary(variant)) {
323
325
  return clsx("border", {
324
- [BORDER_ACCENT]: isLight(mode),
325
- [BORDER_MEDIUM]: isDark(mode),
326
- [BORDER_ACCENT_DARK_MEDIUM]: isSystem(mode),
327
- [BORDER_MEDIUM_DARK_ACCENT]: isAltSystem(mode)
326
+ [BORDER_MEDIUM]: isDark(mode) || isLight(mode),
327
+ [BORDER_DARK_MEDIUM]: isSystem(mode) || isAltSystem(mode)
328
328
  });
329
329
  }
330
330
  if (isDanger(variant)) {
@@ -412,7 +412,7 @@ const getBadgeClasses = ({ size, badge })=>{
412
412
  displayValue
413
413
  };
414
414
  };
415
- const getButtonClasses = ({ type, className, raw, mode, focusMode, disabled, fullWidth, size, noBorder, labelRight, labelLeft, noBackground, variant, truncate, align, radius, animated, badge })=>{
415
+ const getButtonClasses = ({ type, className, raw, mode, focusMode, disabled, fullWidth, size, noBorder, labelRight, labelLeft, noBackground, variant, truncate, align, radius, animated, badge, splitBackground })=>{
416
416
  if (!variant) {
417
417
  variant = "primary";
418
418
  }
@@ -425,7 +425,8 @@ const getButtonClasses = ({ type, className, raw, mode, focusMode, disabled, ful
425
425
  }), getButtonBackgroundClasses({
426
426
  mode,
427
427
  noBackground,
428
- variant
428
+ variant,
429
+ splitBackground
429
430
  }), getButtonRadiusClasses({
430
431
  radius
431
432
  }), getButtonSizesClasses({
@@ -464,4 +465,5 @@ const getButtonClasses = ({ type, className, raw, mode, focusMode, disabled, ful
464
465
  }), className);
465
466
  };
466
467
 
467
- export { TYPE_BUTTON, TYPE_ICON, TYPE_LINK, getBadgeClasses, getButtonClasses, getButtonIconLabelClasses, getIconClasses };
468
+ export { TYPE_BUTTON, TYPE_ICON, TYPE_LINK, clsx, getBadgeClasses, getButtonClasses, getButtonIconLabelClasses, getIconClasses };
469
+ export { jsx, jsxs } from "react/jsx-runtime";
package/dist/370.js ADDED
@@ -0,0 +1,220 @@
1
+ /*!
2
+ @versini/ui-button v12.1.1
3
+ © 2026 gizmette.com
4
+ */
5
+
6
+ import { useMergeRefs } from "@versini/ui-hooks/use-merge-refs";
7
+ import { useResizeObserver } from "@versini/ui-hooks/use-resize-observer";
8
+ import { jsxs, getButtonClasses, getIconClasses, getButtonIconLabelClasses, TYPE_ICON, getBadgeClasses, jsx } from "./188.js";
9
+ import { useLayoutEffect, useRef, useEffect, BaseButton_private } from "./795.js";
10
+
11
+
12
+
13
+
14
+
15
+
16
+
17
+
18
+
19
+ const WIDTH = {
20
+ small: 24,
21
+ medium: 32,
22
+ large: 48
23
+ };
24
+ const PADDINGS = {
25
+ small: 8 * 2,
26
+ medium: 12 * 2,
27
+ large: 16 * 2
28
+ };
29
+ const BORDERS = 2; // border x 2
30
+ const ANIMATION_DURATION = 300; // ms - should match the CSS transition duration
31
+ function ButtonIcon({ children, disabled = false, mode = "system", focusMode = "system", fullWidth = false, className, type = "button", raw = false, noBorder = false, "aria-label": ariaLabel, label, size = "medium", labelRight, labelLeft, noBackground = false, align = "center", radius = "large", variant = "secondary", iconClassName, animated = false, badge, splitBackground = false, ref, ...otherProps }) {
32
+ const buttonClass = getButtonClasses({
33
+ type: TYPE_ICON,
34
+ mode,
35
+ focusMode,
36
+ fullWidth,
37
+ disabled,
38
+ raw,
39
+ className,
40
+ noBorder,
41
+ size,
42
+ labelRight,
43
+ labelLeft,
44
+ noBackground,
45
+ align,
46
+ radius,
47
+ variant,
48
+ animated,
49
+ badge,
50
+ splitBackground
51
+ });
52
+ const iconClass = getIconClasses({
53
+ mode,
54
+ raw,
55
+ iconClassName,
56
+ variant
57
+ });
58
+ const labelClass = getButtonIconLabelClasses({
59
+ animated
60
+ });
61
+ // Create a combined class for the button content wrapper
62
+ const contentWrapperClass = "flex items-center justify-center relative w-full h-full overflow-hidden";
63
+ const [labelRightRef, rectR] = useResizeObserver();
64
+ const [labelLeftRef, rectL] = useResizeObserver();
65
+ const [iconRef, rectIcon] = useResizeObserver();
66
+ const bufferRef = useRef(0);
67
+ const buttonRef = useRef(null);
68
+ const timeoutRef = useRef(null);
69
+ const mergedRef = useMergeRefs([
70
+ ref,
71
+ buttonRef
72
+ ]);
73
+ /**
74
+ * Effect to calculate the buffer to add to the width of the button to
75
+ * account for the icon, paddings and borders.
76
+ */ useLayoutEffect(()=>{
77
+ /* v8 ignore start */ if (iconRef && iconRef.current && animated) {
78
+ bufferRef.current = rectIcon.width + PADDINGS[size] + (noBorder ? 0 : BORDERS);
79
+ // Set initial button width if it hasn't been set yet
80
+ if (buttonRef.current && !buttonRef.current.style.width) {
81
+ buttonRef.current.style.width = `${WIDTH[size]}px`;
82
+ }
83
+ }
84
+ /* v8 ignore stop */ }, [
85
+ rectIcon,
86
+ iconRef,
87
+ size,
88
+ noBorder,
89
+ animated
90
+ ]);
91
+ /**
92
+ * Effect to update the width of the button based on the visibility of
93
+ * the right and left labels.
94
+ */ useLayoutEffect(()=>{
95
+ /* v8 ignore start */ if (buttonRef && buttonRef.current && animated) {
96
+ // Calculate the target width first
97
+ let newWidth = WIDTH[size];
98
+ if (labelRight && labelRightRef && rectR.width > 0) {
99
+ newWidth = rectR.width + bufferRef.current;
100
+ } else if (labelLeft && labelLeftRef && rectL.width > 0) {
101
+ newWidth = rectL.width + bufferRef.current;
102
+ }
103
+ // Clear any existing timeout to prevent race conditions
104
+ if (timeoutRef.current) {
105
+ clearTimeout(timeoutRef.current);
106
+ }
107
+ // Start transition - expand button first
108
+ if (newWidth !== parseInt(buttonRef.current.style.width || "0", 10)) {
109
+ // Make sure labels are hidden during width transition
110
+ if (labelRightRef.current) {
111
+ labelRightRef.current.style.opacity = "0";
112
+ }
113
+ if (labelLeftRef.current) {
114
+ labelLeftRef.current.style.opacity = "0";
115
+ }
116
+ // Set the width to trigger the transition
117
+ buttonRef.current.style.width = `${newWidth}px`;
118
+ // After button width transition completes, show the label
119
+ if (newWidth > WIDTH[size]) {
120
+ timeoutRef.current = setTimeout(()=>{
121
+ if (labelRightRef.current && labelRight) {
122
+ labelRightRef.current.style.opacity = "1";
123
+ }
124
+ if (labelLeftRef.current && labelLeft) {
125
+ labelLeftRef.current.style.opacity = "1";
126
+ }
127
+ timeoutRef.current = null;
128
+ }, ANIMATION_DURATION * 0.8); // Wait for most of the width transition to complete
129
+ }
130
+ }
131
+ // If transitioning to icon-only state, hide labels immediately
132
+ if (newWidth === WIDTH[size]) {
133
+ if (labelRightRef.current) {
134
+ labelRightRef.current.style.opacity = "0";
135
+ }
136
+ if (labelLeftRef.current) {
137
+ labelLeftRef.current.style.opacity = "0";
138
+ }
139
+ }
140
+ }
141
+ /* v8 ignore stop */ }, [
142
+ rectR,
143
+ labelRight,
144
+ labelRightRef,
145
+ rectL,
146
+ labelLeft,
147
+ labelLeftRef,
148
+ size,
149
+ animated
150
+ ]);
151
+ // Clean up timeout on unmount
152
+ /* v8 ignore start */ useEffect(()=>{
153
+ return ()=>{
154
+ if (timeoutRef.current) {
155
+ clearTimeout(timeoutRef.current);
156
+ }
157
+ };
158
+ }, []);
159
+ /* v8 ignore stop */ const badgeInfo = getBadgeClasses({
160
+ size,
161
+ badge
162
+ });
163
+ return /*#__PURE__*/ jsxs(BaseButton_private, {
164
+ ref: mergedRef,
165
+ className: buttonClass,
166
+ disabled: disabled,
167
+ type: type,
168
+ "aria-label": ariaLabel || label,
169
+ ...otherProps,
170
+ children: [
171
+ /*#__PURE__*/ jsxs("div", {
172
+ className: contentWrapperClass,
173
+ children: [
174
+ /*#__PURE__*/ jsx(ButtonLabel, {
175
+ label: labelLeft,
176
+ labelRef: labelLeftRef,
177
+ labelClass: labelClass,
178
+ labelInnerClass: "pr-2",
179
+ initiallyHidden: animated
180
+ }),
181
+ /*#__PURE__*/ jsx("span", {
182
+ ref: iconRef,
183
+ className: iconClass,
184
+ children: children
185
+ }),
186
+ /*#__PURE__*/ jsx(ButtonLabel, {
187
+ label: labelRight,
188
+ labelRef: labelRightRef,
189
+ labelClass: labelClass,
190
+ labelInnerClass: "pl-2",
191
+ initiallyHidden: animated
192
+ })
193
+ ]
194
+ }),
195
+ badgeInfo && /*#__PURE__*/ jsx("span", {
196
+ className: badgeInfo.className,
197
+ "aria-hidden": "true",
198
+ children: badgeInfo.displayValue
199
+ })
200
+ ]
201
+ });
202
+ }
203
+ const ButtonLabel = ({ labelRef, labelClass, label, labelInnerClass, initiallyHidden = false })=>{
204
+ if (!label && !initiallyHidden) {
205
+ return null;
206
+ }
207
+ return /*#__PURE__*/ jsx("span", {
208
+ ref: labelRef,
209
+ className: labelClass,
210
+ style: initiallyHidden ? {
211
+ opacity: 0
212
+ } : undefined,
213
+ children: label && /*#__PURE__*/ jsx("span", {
214
+ className: labelInnerClass,
215
+ children: label
216
+ })
217
+ });
218
+ };
219
+
220
+ export { ButtonIcon };
@@ -1,11 +1,9 @@
1
1
  /*!
2
- @versini/ui-button v12.0.4
2
+ @versini/ui-button v12.1.1
3
3
  © 2026 gizmette.com
4
4
  */
5
5
 
6
- import { jsx } from "react/jsx-runtime";
7
- import "react";
8
-
6
+ import { jsx } from "./188.js";
9
7
 
10
8
 
11
9
 
@@ -42,3 +40,4 @@ import "react";
42
40
  }
43
41
 
44
42
  export { BaseButton_private };
43
+ export { useEffect, useLayoutEffect, useRef, useState } from "react";
@@ -1,2 +1,2 @@
1
1
  import type { ButtonTypes } from "@versini/ui-types";
2
- export declare function Button({ children, disabled, mode, focusMode, fullWidth, className, size, raw, noBorder, variant, truncate, radius, ref, ...otherProps }: ButtonTypes.Props): import("react/jsx-runtime").JSX.Element;
2
+ export declare function Button({ children, disabled, mode, focusMode, fullWidth, className, size, raw, noBorder, variant, truncate, radius, splitBackground, ref, ...otherProps }: ButtonTypes.Props): import("react/jsx-runtime").JSX.Element;
@@ -1,17 +1,15 @@
1
1
  /*!
2
- @versini/ui-button v12.0.4
2
+ @versini/ui-button v12.1.1
3
3
  © 2026 gizmette.com
4
4
  */
5
5
 
6
- import { jsx } from "react/jsx-runtime";
7
- import { getButtonClasses, TYPE_BUTTON } from "../../287.js";
8
- import { BaseButton_private } from "../../285.js";
6
+ import { getButtonClasses, TYPE_BUTTON, jsx } from "../../188.js";
7
+ import { BaseButton_private } from "../../795.js";
9
8
 
10
9
 
11
10
 
12
11
 
13
-
14
- function Button({ children, disabled = false, mode = "system", focusMode = "system", fullWidth = false, className, size = "medium", raw = false, noBorder = false, variant = "primary", truncate = false, radius = "large", ref, ...otherProps }) {
12
+ function Button({ children, disabled = false, mode = "system", focusMode = "system", fullWidth = false, className, size = "medium", raw = false, noBorder = false, variant = "primary", truncate = false, radius = "large", splitBackground = false, ref, ...otherProps }) {
15
13
  const buttonClass = getButtonClasses({
16
14
  type: TYPE_BUTTON,
17
15
  mode,
@@ -24,7 +22,8 @@ function Button({ children, disabled = false, mode = "system", focusMode = "syst
24
22
  noBorder,
25
23
  variant,
26
24
  truncate,
27
- radius
25
+ radius,
26
+ splitBackground
28
27
  });
29
28
  return /*#__PURE__*/ jsx(BaseButton_private, {
30
29
  ref: ref,
@@ -1,14 +1,12 @@
1
1
  /*!
2
- @versini/ui-button v12.0.4
2
+ @versini/ui-button v12.1.1
3
3
  © 2026 gizmette.com
4
4
  */
5
5
 
6
- import { jsx } from "react/jsx-runtime";
7
6
  import { IconCopied, IconCopy } from "@versini/ui-icons";
8
- import { useEffect, useState } from "react";
9
- import { ButtonIcon } from "./ButtonIcon.js";
10
-
11
-
7
+ import { jsx } from "../../188.js";
8
+ import { useState, useEffect } from "../../795.js";
9
+ import { ButtonIcon } from "../../370.js";
12
10
 
13
11
 
14
12
 
@@ -1,2 +1,2 @@
1
1
  import type { ButtonIconTypes } from "@versini/ui-types";
2
- export declare function ButtonIcon({ children, disabled, mode, focusMode, fullWidth, className, type, raw, noBorder, "aria-label": ariaLabel, label, size, labelRight, labelLeft, noBackground, align, radius, variant, iconClassName, animated, badge, ref, ...otherProps }: ButtonIconTypes.Props): import("react/jsx-runtime").JSX.Element;
2
+ export declare function ButtonIcon({ children, disabled, mode, focusMode, fullWidth, className, type, raw, noBorder, "aria-label": ariaLabel, label, size, labelRight, labelLeft, noBackground, align, radius, variant, iconClassName, animated, badge, splitBackground, ref, ...otherProps }: ButtonIconTypes.Props): import("react/jsx-runtime").JSX.Element;
@@ -1,223 +1,7 @@
1
1
  /*!
2
- @versini/ui-button v12.0.4
2
+ @versini/ui-button v12.1.1
3
3
  © 2026 gizmette.com
4
4
  */
5
5
 
6
- import { jsx, jsxs } from "react/jsx-runtime";
7
- import { useMergeRefs } from "@versini/ui-hooks/use-merge-refs";
8
- import { useResizeObserver } from "@versini/ui-hooks/use-resize-observer";
9
- import { useEffect, useLayoutEffect, useRef } from "react";
10
- import { getButtonClasses, getIconClasses, getButtonIconLabelClasses, TYPE_ICON, getBadgeClasses } from "../../287.js";
11
- import { BaseButton_private } from "../../285.js";
12
6
 
13
-
14
-
15
-
16
-
17
-
18
-
19
-
20
-
21
-
22
-
23
- const WIDTH = {
24
- small: 24,
25
- medium: 32,
26
- large: 48
27
- };
28
- const PADDINGS = {
29
- small: 8 * 2,
30
- medium: 12 * 2,
31
- large: 16 * 2
32
- };
33
- const BORDERS = 2; // border x 2
34
- const ANIMATION_DURATION = 300; // ms - should match the CSS transition duration
35
- function ButtonIcon({ children, disabled = false, mode = "system", focusMode = "system", fullWidth = false, className, type = "button", raw = false, noBorder = false, "aria-label": ariaLabel, label, size = "medium", labelRight, labelLeft, noBackground = false, align = "center", radius = "large", variant = "secondary", iconClassName, animated = false, badge, ref, ...otherProps }) {
36
- const buttonClass = getButtonClasses({
37
- type: TYPE_ICON,
38
- mode,
39
- focusMode,
40
- fullWidth,
41
- disabled,
42
- raw,
43
- className,
44
- noBorder,
45
- size,
46
- labelRight,
47
- labelLeft,
48
- noBackground,
49
- align,
50
- radius,
51
- variant,
52
- animated,
53
- badge
54
- });
55
- const iconClass = getIconClasses({
56
- mode,
57
- raw,
58
- iconClassName,
59
- variant
60
- });
61
- const labelClass = getButtonIconLabelClasses({
62
- animated
63
- });
64
- // Create a combined class for the button content wrapper
65
- const contentWrapperClass = "flex items-center justify-center relative w-full h-full overflow-hidden";
66
- const [labelRightRef, rectR] = useResizeObserver();
67
- const [labelLeftRef, rectL] = useResizeObserver();
68
- const [iconRef, rectIcon] = useResizeObserver();
69
- const bufferRef = useRef(0);
70
- const buttonRef = useRef(null);
71
- const timeoutRef = useRef(null);
72
- const mergedRef = useMergeRefs([
73
- ref,
74
- buttonRef
75
- ]);
76
- /**
77
- * Effect to calculate the buffer to add to the width of the button to
78
- * account for the icon, paddings and borders.
79
- */ useLayoutEffect(()=>{
80
- /* v8 ignore start */ if (iconRef && iconRef.current && animated) {
81
- bufferRef.current = rectIcon.width + PADDINGS[size] + (noBorder ? 0 : BORDERS);
82
- // Set initial button width if it hasn't been set yet
83
- if (buttonRef.current && !buttonRef.current.style.width) {
84
- buttonRef.current.style.width = `${WIDTH[size]}px`;
85
- }
86
- }
87
- /* v8 ignore stop */ }, [
88
- rectIcon,
89
- iconRef,
90
- size,
91
- noBorder,
92
- animated
93
- ]);
94
- /**
95
- * Effect to update the width of the button based on the visibility of
96
- * the right and left labels.
97
- */ useLayoutEffect(()=>{
98
- /* v8 ignore start */ if (buttonRef && buttonRef.current && animated) {
99
- // Calculate the target width first
100
- let newWidth = WIDTH[size];
101
- if (labelRight && labelRightRef && rectR.width > 0) {
102
- newWidth = rectR.width + bufferRef.current;
103
- } else if (labelLeft && labelLeftRef && rectL.width > 0) {
104
- newWidth = rectL.width + bufferRef.current;
105
- }
106
- // Clear any existing timeout to prevent race conditions
107
- if (timeoutRef.current) {
108
- clearTimeout(timeoutRef.current);
109
- }
110
- // Start transition - expand button first
111
- if (newWidth !== parseInt(buttonRef.current.style.width || "0", 10)) {
112
- // Make sure labels are hidden during width transition
113
- if (labelRightRef.current) {
114
- labelRightRef.current.style.opacity = "0";
115
- }
116
- if (labelLeftRef.current) {
117
- labelLeftRef.current.style.opacity = "0";
118
- }
119
- // Set the width to trigger the transition
120
- buttonRef.current.style.width = `${newWidth}px`;
121
- // After button width transition completes, show the label
122
- if (newWidth > WIDTH[size]) {
123
- timeoutRef.current = setTimeout(()=>{
124
- if (labelRightRef.current && labelRight) {
125
- labelRightRef.current.style.opacity = "1";
126
- }
127
- if (labelLeftRef.current && labelLeft) {
128
- labelLeftRef.current.style.opacity = "1";
129
- }
130
- timeoutRef.current = null;
131
- }, ANIMATION_DURATION * 0.8); // Wait for most of the width transition to complete
132
- }
133
- }
134
- // If transitioning to icon-only state, hide labels immediately
135
- if (newWidth === WIDTH[size]) {
136
- if (labelRightRef.current) {
137
- labelRightRef.current.style.opacity = "0";
138
- }
139
- if (labelLeftRef.current) {
140
- labelLeftRef.current.style.opacity = "0";
141
- }
142
- }
143
- }
144
- /* v8 ignore stop */ }, [
145
- rectR,
146
- labelRight,
147
- labelRightRef,
148
- rectL,
149
- labelLeft,
150
- labelLeftRef,
151
- size,
152
- animated
153
- ]);
154
- // Clean up timeout on unmount
155
- /* v8 ignore start */ useEffect(()=>{
156
- return ()=>{
157
- if (timeoutRef.current) {
158
- clearTimeout(timeoutRef.current);
159
- }
160
- };
161
- }, []);
162
- /* v8 ignore stop */ const badgeInfo = getBadgeClasses({
163
- size,
164
- badge
165
- });
166
- return /*#__PURE__*/ jsxs(BaseButton_private, {
167
- ref: mergedRef,
168
- className: buttonClass,
169
- disabled: disabled,
170
- type: type,
171
- "aria-label": ariaLabel || label,
172
- ...otherProps,
173
- children: [
174
- /*#__PURE__*/ jsxs("div", {
175
- className: contentWrapperClass,
176
- children: [
177
- /*#__PURE__*/ jsx(ButtonLabel, {
178
- label: labelLeft,
179
- labelRef: labelLeftRef,
180
- labelClass: labelClass,
181
- labelInnerClass: "pr-2",
182
- initiallyHidden: animated
183
- }),
184
- /*#__PURE__*/ jsx("span", {
185
- ref: iconRef,
186
- className: iconClass,
187
- children: children
188
- }),
189
- /*#__PURE__*/ jsx(ButtonLabel, {
190
- label: labelRight,
191
- labelRef: labelRightRef,
192
- labelClass: labelClass,
193
- labelInnerClass: "pl-2",
194
- initiallyHidden: animated
195
- })
196
- ]
197
- }),
198
- badgeInfo && /*#__PURE__*/ jsx("span", {
199
- className: badgeInfo.className,
200
- "aria-hidden": "true",
201
- children: badgeInfo.displayValue
202
- })
203
- ]
204
- });
205
- }
206
- const ButtonLabel = ({ labelRef, labelClass, label, labelInnerClass, initiallyHidden = false })=>{
207
- if (!label && !initiallyHidden) {
208
- return null;
209
- }
210
- return /*#__PURE__*/ jsx("span", {
211
- ref: labelRef,
212
- className: labelClass,
213
- style: initiallyHidden ? {
214
- opacity: 0
215
- } : undefined,
216
- children: label && /*#__PURE__*/ jsx("span", {
217
- className: labelInnerClass,
218
- children: label
219
- })
220
- });
221
- };
222
-
223
- export { ButtonIcon };
7
+ export { ButtonIcon } from "../../370.js";
@@ -1,2 +1,2 @@
1
1
  import type { ButtonLinkTypes } from "@versini/ui-types";
2
- export declare function ButtonLink({ children, mode, focusMode, fullWidth, className, size, raw, noBorder, target, truncate, noNewWindowIcon, radius, ref, ...otherProps }: ButtonLinkTypes.Props): import("react/jsx-runtime").JSX.Element;
2
+ export declare function ButtonLink({ children, mode, focusMode, fullWidth, className, size, raw, noBorder, target, truncate, noNewWindowIcon, radius, splitBackground, ref, ...otherProps }: ButtonLinkTypes.Props): import("react/jsx-runtime").JSX.Element;
@@ -1,18 +1,14 @@
1
1
  /*!
2
- @versini/ui-button v12.0.4
2
+ @versini/ui-button v12.1.1
3
3
  © 2026 gizmette.com
4
4
  */
5
5
 
6
- import { jsx, jsxs } from "react/jsx-runtime";
7
- import clsx from "clsx";
8
- import { getButtonClasses, TYPE_LINK } from "../../287.js";
6
+ import { jsxs, getButtonClasses, TYPE_LINK, clsx, jsx } from "../../188.js";
9
7
 
10
8
 
11
9
 
12
10
 
13
-
14
-
15
- function ButtonLink({ children, mode = "system", focusMode = "system", fullWidth = false, className, size = "small", raw = false, noBorder = false, target, truncate = false, noNewWindowIcon = false, radius = "large", ref, ...otherProps }) {
11
+ function ButtonLink({ children, mode = "system", focusMode = "system", fullWidth = false, className, size = "small", raw = false, noBorder = false, target, truncate = false, noNewWindowIcon = false, radius = "large", splitBackground = false, ref, ...otherProps }) {
16
12
  const buttonClass = getButtonClasses({
17
13
  type: TYPE_LINK,
18
14
  mode,
@@ -24,7 +20,8 @@ function ButtonLink({ children, mode = "system", focusMode = "system", fullWidth
24
20
  size,
25
21
  noBorder,
26
22
  truncate,
27
- radius
23
+ radius,
24
+ splitBackground
28
25
  });
29
26
  const newWindow = target === "_blank";
30
27
  const extraProps = {
@@ -13,5 +13,5 @@ export declare const getBadgeClasses: ({ size, badge, }: Pick<ButtonIconTypes.Pr
13
13
  className: string;
14
14
  displayValue: string | null;
15
15
  } | null;
16
- export declare const getButtonClasses: ({ type, className, raw, mode, focusMode, disabled, fullWidth, size, noBorder, labelRight, labelLeft, noBackground, variant, truncate, align, radius, animated, badge, }: GetButtonClassesProps) => string;
16
+ export declare const getButtonClasses: ({ type, className, raw, mode, focusMode, disabled, fullWidth, size, noBorder, labelRight, labelLeft, noBackground, variant, truncate, align, radius, animated, badge, splitBackground, }: GetButtonClassesProps) => string;
17
17
  export {};
package/package.json CHANGED
@@ -1,6 +1,6 @@
1
1
  {
2
2
  "name": "@versini/ui-button",
3
- "version": "12.0.4",
3
+ "version": "12.1.1",
4
4
  "license": "MIT",
5
5
  "author": "Arno Versini",
6
6
  "publishConfig": {
@@ -59,7 +59,7 @@
59
59
  },
60
60
  "devDependencies": {
61
61
  "@testing-library/jest-dom": "6.9.1",
62
- "@versini/ui-types": "8.3.0"
62
+ "@versini/ui-types": "8.4.0"
63
63
  },
64
64
  "dependencies": {
65
65
  "@versini/ui-hooks": "6.1.1",
@@ -70,5 +70,5 @@
70
70
  "sideEffects": [
71
71
  "**/*.css"
72
72
  ],
73
- "gitHead": "6b50d466dbc1261a8539f469016ab438e4e8e51c"
73
+ "gitHead": "61a2209a12d41c6f0787a5a303b3fa067756d233"
74
74
  }