dash-ui-kit 1.0.93 → 2.0.0-dev
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/README.md +52 -0
- package/dist/react/components/accordion/index.cjs.js +193 -0
- package/dist/react/components/accordion/index.cjs.js.map +1 -0
- package/dist/react/components/accordion/index.esm.js +169 -0
- package/dist/react/components/accordion/index.esm.js.map +1 -0
- package/dist/react/components/avatar/index.cjs.js +39 -0
- package/dist/react/components/avatar/index.cjs.js.map +1 -0
- package/dist/react/components/avatar/index.esm.js +34 -0
- package/dist/react/components/avatar/index.esm.js.map +1 -0
- package/dist/react/components/badge/index.cjs.js +92 -0
- package/dist/react/components/badge/index.cjs.js.map +1 -0
- package/dist/react/components/badge/index.esm.js +87 -0
- package/dist/react/components/badge/index.esm.js.map +1 -0
- package/dist/react/components/bigNumber/index.cjs.js +100 -0
- package/dist/react/components/bigNumber/index.cjs.js.map +1 -0
- package/dist/react/components/bigNumber/index.esm.js +95 -0
- package/dist/react/components/bigNumber/index.esm.js.map +1 -0
- package/dist/react/components/button/index.cjs.js +535 -0
- package/dist/react/components/button/index.cjs.js.map +1 -0
- package/dist/react/components/button/index.d.ts +3 -1
- package/dist/react/components/button/index.esm.js +530 -0
- package/dist/react/components/button/index.esm.js.map +1 -0
- package/dist/react/components/copyButton/index.cjs.js +95 -0
- package/dist/react/components/copyButton/index.cjs.js.map +1 -0
- package/dist/react/components/copyButton/index.esm.js +71 -0
- package/dist/react/components/copyButton/index.esm.js.map +1 -0
- package/dist/react/components/dashLogo/index.cjs.js +74 -0
- package/dist/react/components/dashLogo/index.cjs.js.map +1 -0
- package/dist/react/components/dashLogo/index.esm.js +69 -0
- package/dist/react/components/dashLogo/index.esm.js.map +1 -0
- package/dist/react/components/dateBlock/index.cjs.js +120 -0
- package/dist/react/components/dateBlock/index.cjs.js.map +1 -0
- package/dist/react/components/dateBlock/index.esm.js +115 -0
- package/dist/react/components/dateBlock/index.esm.js.map +1 -0
- package/dist/react/components/dialog/index.cjs.js +292 -0
- package/dist/react/components/dialog/index.cjs.js.map +1 -0
- package/dist/react/components/dialog/index.esm.js +270 -0
- package/dist/react/components/dialog/index.esm.js.map +1 -0
- package/dist/react/components/heading/index.cjs.js +60 -0
- package/dist/react/components/heading/index.cjs.js.map +1 -0
- package/dist/react/components/heading/index.esm.js +58 -0
- package/dist/react/components/heading/index.esm.js.map +1 -0
- package/dist/react/components/icons/index.cjs.js +1173 -0
- package/dist/react/components/icons/index.cjs.js.map +1 -0
- package/dist/react/components/icons/index.d.ts +2 -0
- package/dist/react/components/icons/index.esm.js +1128 -0
- package/dist/react/components/icons/index.esm.js.map +1 -0
- package/dist/react/components/identifier/index.cjs.js +286 -0
- package/dist/react/components/identifier/index.cjs.js.map +1 -0
- package/dist/react/components/identifier/index.esm.js +282 -0
- package/dist/react/components/identifier/index.esm.js.map +1 -0
- package/dist/react/components/index.cjs.js +101 -0
- package/dist/react/components/index.cjs.js.map +1 -0
- package/dist/react/components/index.d.ts +1 -1
- package/dist/react/components/index.esm.js +29 -0
- package/dist/react/components/index.esm.js.map +1 -0
- package/dist/react/components/input/index.cjs.js +237 -0
- package/dist/react/components/input/index.cjs.js.map +1 -0
- package/dist/react/components/input/index.esm.js +232 -0
- package/dist/react/components/input/index.esm.js.map +1 -0
- package/dist/react/components/list/index.cjs.js +49 -0
- package/dist/react/components/list/index.cjs.js.map +1 -0
- package/dist/react/components/list/index.esm.js +47 -0
- package/dist/react/components/list/index.esm.js.map +1 -0
- package/dist/react/components/notActive/index.cjs.js +40 -0
- package/dist/react/components/notActive/index.cjs.js.map +1 -0
- package/dist/react/components/notActive/index.esm.js +38 -0
- package/dist/react/components/notActive/index.esm.js.map +1 -0
- package/dist/react/components/overlayMenu/index.cjs.js +425 -0
- package/dist/react/components/overlayMenu/index.cjs.js.map +1 -0
- package/dist/react/components/overlayMenu/index.esm.js +420 -0
- package/dist/react/components/overlayMenu/index.esm.js.map +1 -0
- package/dist/react/components/overlaySelect/index.cjs.js +345 -0
- package/dist/react/components/overlaySelect/index.cjs.js.map +1 -0
- package/dist/react/components/overlaySelect/index.esm.js +340 -0
- package/dist/react/components/overlaySelect/index.esm.js.map +1 -0
- package/dist/react/components/progressStepBar/index.cjs.js +49 -0
- package/dist/react/components/progressStepBar/index.cjs.js.map +1 -0
- package/dist/react/components/progressStepBar/index.d.ts +2 -1
- package/dist/react/components/progressStepBar/index.esm.js +47 -0
- package/dist/react/components/progressStepBar/index.esm.js.map +1 -0
- package/dist/react/components/select/index.cjs.js +236 -0
- package/dist/react/components/select/index.cjs.js.map +1 -0
- package/dist/react/components/select/index.esm.js +212 -0
- package/dist/react/components/select/index.esm.js.map +1 -0
- package/dist/react/components/switch/index.cjs.js +184 -0
- package/dist/react/components/switch/index.cjs.js.map +1 -0
- package/dist/react/components/switch/index.esm.js +179 -0
- package/dist/react/components/switch/index.esm.js.map +1 -0
- package/dist/react/components/tabs/index.cjs.js +178 -0
- package/dist/react/components/tabs/index.cjs.js.map +1 -0
- package/dist/react/components/tabs/index.esm.js +154 -0
- package/dist/react/components/tabs/index.esm.js.map +1 -0
- package/dist/react/components/text/index.cjs.js +120 -0
- package/dist/react/components/text/index.cjs.js.map +1 -0
- package/dist/react/components/text/index.esm.js +115 -0
- package/dist/react/components/text/index.esm.js.map +1 -0
- package/dist/react/components/textarea/index.cjs.js +256 -0
- package/dist/react/components/textarea/index.cjs.js.map +1 -0
- package/dist/react/components/textarea/index.esm.js +251 -0
- package/dist/react/components/textarea/index.esm.js.map +1 -0
- package/dist/react/components/timeDelta/index.cjs.js +93 -0
- package/dist/react/components/timeDelta/index.cjs.js.map +1 -0
- package/dist/react/components/timeDelta/index.d.ts +1 -1
- package/dist/react/components/timeDelta/index.esm.js +88 -0
- package/dist/react/components/timeDelta/index.esm.js.map +1 -0
- package/dist/react/components/transactionStatusIcon/index.cjs.js +59 -0
- package/dist/react/components/transactionStatusIcon/index.cjs.js.map +1 -0
- package/dist/react/components/transactionStatusIcon/index.esm.js +54 -0
- package/dist/react/components/transactionStatusIcon/index.esm.js.map +1 -0
- package/dist/react/components/valueCard/index.cjs.js +176 -0
- package/dist/react/components/valueCard/index.cjs.js.map +1 -0
- package/dist/react/components/valueCard/index.d.ts +1 -1
- package/dist/react/components/valueCard/index.esm.js +171 -0
- package/dist/react/components/valueCard/index.esm.js.map +1 -0
- package/dist/react/contexts/ThemeContext.cjs.js +79 -0
- package/dist/react/contexts/ThemeContext.cjs.js.map +1 -0
- package/dist/react/contexts/ThemeContext.esm.js +76 -0
- package/dist/react/contexts/ThemeContext.esm.js.map +1 -0
- package/dist/react/contexts/index.cjs.js +11 -0
- package/dist/react/contexts/index.cjs.js.map +1 -0
- package/dist/react/contexts/index.esm.js +4 -0
- package/dist/react/contexts/index.esm.js.map +1 -0
- package/dist/react/hooks/useDebounce.cjs.js +83 -0
- package/dist/react/hooks/useDebounce.cjs.js.map +1 -0
- package/dist/react/hooks/useDebounce.esm.js +78 -0
- package/dist/react/hooks/useDebounce.esm.js.map +1 -0
- package/dist/react/index.cjs.js +99 -12811
- package/dist/react/index.cjs.js.map +1 -1
- package/dist/react/index.d.ts +1 -1
- package/dist/react/index.esm.js +27 -12725
- package/dist/react/index.esm.js.map +1 -1
- package/dist/react/shared/utils/datetime.cjs.js +59 -0
- package/dist/react/shared/utils/datetime.cjs.js.map +1 -0
- package/dist/react/shared/utils/datetime.esm.js +57 -0
- package/dist/react/shared/utils/datetime.esm.js.map +1 -0
- package/dist/react/utils/copyToClipboard.cjs.js +31 -0
- package/dist/react/utils/copyToClipboard.cjs.js.map +1 -0
- package/dist/react/utils/copyToClipboard.esm.js +26 -0
- package/dist/react/utils/copyToClipboard.esm.js.map +1 -0
- package/dist/react/utils/index.d.ts +1 -1
- package/dist/react-native/components/avatar/index.d.ts +26 -0
- package/dist/react-native/components/avatar/index.web.d.ts +24 -0
- package/dist/react-native/components/badge/index.d.ts +51 -0
- package/dist/react-native/components/bigNumber/index.d.ts +26 -0
- package/dist/react-native/components/button/index.d.ts +39 -0
- package/dist/react-native/components/copyButton/index.d.ts +22 -0
- package/dist/react-native/components/copyButton/index.web.d.ts +20 -0
- package/dist/react-native/components/dashLogo/index.d.ts +30 -0
- package/dist/react-native/components/heading/index.d.ts +25 -0
- package/dist/react-native/components/icons/index.d.ts +43 -0
- package/dist/react-native/components/identifier/index.d.ts +47 -0
- package/dist/react-native/components/index.d.ts +15 -0
- package/dist/react-native/components/input/index.d.ts +53 -0
- package/dist/react-native/components/notActive/index.d.ts +16 -0
- package/dist/react-native/components/tabs/index.d.ts +50 -0
- package/dist/react-native/components/text/index.d.ts +28 -0
- package/dist/react-native/components/transactionStatusIcon/index.d.ts +24 -0
- package/dist/react-native/components/valueCard/index.d.ts +43 -0
- package/dist/react-native/hooks/index.d.ts +1 -0
- package/dist/react-native/hooks/useDebounce.d.ts +43 -0
- package/dist/react-native/index.cjs.js +2856 -0
- package/dist/react-native/index.cjs.js.map +1 -0
- package/dist/react-native/index.d.ts +4 -0
- package/dist/react-native/index.esm.js +2808 -0
- package/dist/react-native/index.esm.js.map +1 -0
- package/dist/react-native/styles/index.d.ts +11 -0
- package/dist/react-native/styles/tokens.d.ts +308 -0
- package/dist/react-native/styles/utils.d.ts +65 -0
- package/dist/react-native/utils/clipboard.d.ts +27 -0
- package/dist/react-native/utils/index.d.ts +2 -0
- package/dist/react-native/utils/tw.d.ts +7 -0
- package/dist/shared/constants/colors.d.ts +25 -0
- package/dist/shared/constants/index.d.ts +2 -0
- package/dist/shared/constants/sizes.d.ts +49 -0
- package/dist/shared/index.cjs.js +171 -0
- package/dist/shared/index.cjs.js.map +1 -0
- package/dist/shared/index.d.ts +3 -0
- package/dist/shared/index.esm.js +161 -0
- package/dist/shared/index.esm.js.map +1 -0
- package/dist/shared/types/common.d.ts +33 -0
- package/dist/shared/types/index.d.ts +1 -0
- package/dist/shared/utils/index.d.ts +1 -0
- package/dist/styles.css +1 -1
- package/package.json +50 -8
- /package/dist/{react → shared}/utils/datetime.d.ts +0 -0
|
@@ -0,0 +1,256 @@
|
|
|
1
|
+
"use client";
|
|
2
|
+
|
|
3
|
+
'use strict';
|
|
4
|
+
|
|
5
|
+
Object.defineProperty(exports, '__esModule', { value: true });
|
|
6
|
+
|
|
7
|
+
var tslib = require('tslib');
|
|
8
|
+
var jsxRuntime = require('react/jsx-runtime');
|
|
9
|
+
var React = require('react');
|
|
10
|
+
var index = require('../button/index.cjs.js');
|
|
11
|
+
var classVarianceAuthority = require('class-variance-authority');
|
|
12
|
+
var ThemeContext = require('../../contexts/ThemeContext.cjs.js');
|
|
13
|
+
|
|
14
|
+
const textareaContainer = classVarianceAuthority.cva('relative flex items-baseline transition-all w-full', {
|
|
15
|
+
variants: {
|
|
16
|
+
theme: {
|
|
17
|
+
light: 'bg-white',
|
|
18
|
+
dark: 'bg-gray-800'
|
|
19
|
+
},
|
|
20
|
+
hasValue: {
|
|
21
|
+
true: '',
|
|
22
|
+
false: ''
|
|
23
|
+
},
|
|
24
|
+
colorScheme: {
|
|
25
|
+
default: 'focus-within:ring-blue-500/20',
|
|
26
|
+
brand: 'focus-within:ring-dash-brand/20',
|
|
27
|
+
error: 'focus-within:ring-red-500/20',
|
|
28
|
+
success: 'focus-within:ring-green-500/20'
|
|
29
|
+
},
|
|
30
|
+
size: {
|
|
31
|
+
sm: 'dash-block-sm',
|
|
32
|
+
md: 'dash-block-md',
|
|
33
|
+
xl: 'dash-block-xl'
|
|
34
|
+
},
|
|
35
|
+
variant: {
|
|
36
|
+
outlined: 'outline outline-1 outline-offset-[-1px]'
|
|
37
|
+
},
|
|
38
|
+
isValid: {
|
|
39
|
+
true: '',
|
|
40
|
+
false: '',
|
|
41
|
+
null: ''
|
|
42
|
+
},
|
|
43
|
+
disabled: {
|
|
44
|
+
false: '',
|
|
45
|
+
true: 'opacity-60 cursor-not-allowed'
|
|
46
|
+
}
|
|
47
|
+
},
|
|
48
|
+
compoundVariants: [
|
|
49
|
+
// Outlined variant colors
|
|
50
|
+
{
|
|
51
|
+
variant: 'outlined',
|
|
52
|
+
colorScheme: 'default',
|
|
53
|
+
class: 'outline-[rgba(17,17,17,0.32)] focus-within:outline-[rgba(17,17,17,0.6)]'
|
|
54
|
+
}, {
|
|
55
|
+
variant: 'outlined',
|
|
56
|
+
colorScheme: 'brand',
|
|
57
|
+
class: 'outline-dash-brand/30 focus-within:outline-dash-brand'
|
|
58
|
+
}, {
|
|
59
|
+
variant: 'outlined',
|
|
60
|
+
colorScheme: 'error',
|
|
61
|
+
isValid: false,
|
|
62
|
+
class: 'outline-red-500 focus-within:outline-red-500'
|
|
63
|
+
}, {
|
|
64
|
+
variant: 'outlined',
|
|
65
|
+
colorScheme: 'success',
|
|
66
|
+
isValid: true,
|
|
67
|
+
class: 'outline-green-500 focus-within:outline-green-500'
|
|
68
|
+
},
|
|
69
|
+
// Outlined variant with focus ring
|
|
70
|
+
{
|
|
71
|
+
variant: 'outlined',
|
|
72
|
+
class: 'focus-within:ring-2'
|
|
73
|
+
},
|
|
74
|
+
// Add extra padding for PASTE button when no value
|
|
75
|
+
{
|
|
76
|
+
hasValue: false,
|
|
77
|
+
size: 'sm',
|
|
78
|
+
class: 'pr-[70px]'
|
|
79
|
+
}, {
|
|
80
|
+
hasValue: false,
|
|
81
|
+
size: 'md',
|
|
82
|
+
class: 'pr-[70px]'
|
|
83
|
+
}, {
|
|
84
|
+
hasValue: false,
|
|
85
|
+
size: 'xl',
|
|
86
|
+
class: 'pr-[70px]'
|
|
87
|
+
}],
|
|
88
|
+
defaultVariants: {
|
|
89
|
+
theme: 'light',
|
|
90
|
+
hasValue: false,
|
|
91
|
+
colorScheme: 'default',
|
|
92
|
+
size: 'xl',
|
|
93
|
+
variant: 'outlined',
|
|
94
|
+
isValid: null,
|
|
95
|
+
disabled: false
|
|
96
|
+
}
|
|
97
|
+
});
|
|
98
|
+
const textarea = classVarianceAuthority.cva('w-full bg-transparent outline-none resize-none transition-all text-[0.875rem] leading-[1.0625rem] placeholder:text-opacity-60', {
|
|
99
|
+
variants: {
|
|
100
|
+
theme: {
|
|
101
|
+
light: 'text-[#111111] placeholder:text-[rgba(17,17,17,0.6)]',
|
|
102
|
+
dark: 'text-white placeholder:text-gray-400'
|
|
103
|
+
},
|
|
104
|
+
font: {
|
|
105
|
+
main: 'font-dash-main',
|
|
106
|
+
grotesque: 'font-dash-grotesque'
|
|
107
|
+
},
|
|
108
|
+
weight: {
|
|
109
|
+
light: 'font-light',
|
|
110
|
+
normal: 'font-normal',
|
|
111
|
+
medium: 'font-medium',
|
|
112
|
+
semibold: 'font-semibold',
|
|
113
|
+
bold: 'font-bold'
|
|
114
|
+
},
|
|
115
|
+
size: {
|
|
116
|
+
sm: '',
|
|
117
|
+
md: '',
|
|
118
|
+
xl: ''
|
|
119
|
+
},
|
|
120
|
+
disabled: {
|
|
121
|
+
false: '',
|
|
122
|
+
true: 'cursor-not-allowed'
|
|
123
|
+
}
|
|
124
|
+
},
|
|
125
|
+
defaultVariants: {
|
|
126
|
+
theme: 'light',
|
|
127
|
+
font: 'main',
|
|
128
|
+
weight: 'light',
|
|
129
|
+
size: 'xl',
|
|
130
|
+
disabled: false
|
|
131
|
+
}
|
|
132
|
+
});
|
|
133
|
+
/**
|
|
134
|
+
* A versatile textarea component that adapts to light/dark theme,
|
|
135
|
+
* supports various color schemes, sizes, variants, and states.
|
|
136
|
+
* Includes optional paste functionality and validation.
|
|
137
|
+
*
|
|
138
|
+
* @example
|
|
139
|
+
* <Textarea
|
|
140
|
+
* placeholder='Enter your message'
|
|
141
|
+
* colorScheme='brand'
|
|
142
|
+
* size='xl'
|
|
143
|
+
* font='grotesque'
|
|
144
|
+
* weight='medium'
|
|
145
|
+
* rows={4}
|
|
146
|
+
* showPasteButton={true}
|
|
147
|
+
* />
|
|
148
|
+
*/
|
|
149
|
+
const Textarea = _a => {
|
|
150
|
+
var _b, _c;
|
|
151
|
+
var {
|
|
152
|
+
className = '',
|
|
153
|
+
onChange,
|
|
154
|
+
showPasteButton = true,
|
|
155
|
+
validator = null,
|
|
156
|
+
rows = 3,
|
|
157
|
+
size = 'xl',
|
|
158
|
+
variant = 'outlined',
|
|
159
|
+
colorScheme = 'default',
|
|
160
|
+
font = 'main',
|
|
161
|
+
weight = 'light',
|
|
162
|
+
error = false,
|
|
163
|
+
success = false,
|
|
164
|
+
disabled = false
|
|
165
|
+
} = _a,
|
|
166
|
+
props = tslib.__rest(_a, ["className", "onChange", "showPasteButton", "validator", "rows", "size", "variant", "colorScheme", "font", "weight", "error", "success", "disabled"]);
|
|
167
|
+
const {
|
|
168
|
+
theme
|
|
169
|
+
} = ThemeContext.useTheme();
|
|
170
|
+
const [value, setValue] = React.useState((_c = (_b = props.value) !== null && _b !== void 0 ? _b : props.defaultValue) !== null && _c !== void 0 ? _c : '');
|
|
171
|
+
const [isValid, setIsValid] = React.useState(null);
|
|
172
|
+
const textareaRef = React.useRef(null);
|
|
173
|
+
const handleChange = e => {
|
|
174
|
+
const newValue = e.target.value;
|
|
175
|
+
setValue(newValue);
|
|
176
|
+
if (typeof onChange === 'function') {
|
|
177
|
+
onChange(newValue);
|
|
178
|
+
}
|
|
179
|
+
validateInput(newValue);
|
|
180
|
+
};
|
|
181
|
+
const validateInput = input => {
|
|
182
|
+
if (validator === null) {
|
|
183
|
+
setIsValid(null);
|
|
184
|
+
return;
|
|
185
|
+
}
|
|
186
|
+
if (typeof validator === 'function') {
|
|
187
|
+
setIsValid(validator(input));
|
|
188
|
+
} else {
|
|
189
|
+
setIsValid(Boolean(validator));
|
|
190
|
+
}
|
|
191
|
+
};
|
|
192
|
+
const handlePaste = () => {
|
|
193
|
+
navigator.clipboard.readText().then(text => {
|
|
194
|
+
if (text !== '') {
|
|
195
|
+
setValue(text);
|
|
196
|
+
if (textareaRef.current != null) {
|
|
197
|
+
textareaRef.current.value = text;
|
|
198
|
+
}
|
|
199
|
+
if (onChange != null) {
|
|
200
|
+
onChange(text);
|
|
201
|
+
}
|
|
202
|
+
validateInput(text);
|
|
203
|
+
}
|
|
204
|
+
}).catch(err => {
|
|
205
|
+
console.error('Failed to read clipboard contents: ', err);
|
|
206
|
+
});
|
|
207
|
+
};
|
|
208
|
+
const hasValue = value !== '';
|
|
209
|
+
// Determine color scheme based on state
|
|
210
|
+
let finalColorScheme = colorScheme;
|
|
211
|
+
let finalIsValid = isValid;
|
|
212
|
+
if (error) {
|
|
213
|
+
finalColorScheme = 'error';
|
|
214
|
+
finalIsValid = false;
|
|
215
|
+
} else if (success) {
|
|
216
|
+
finalColorScheme = 'success';
|
|
217
|
+
finalIsValid = true;
|
|
218
|
+
}
|
|
219
|
+
const containerClasses = textareaContainer({
|
|
220
|
+
theme,
|
|
221
|
+
hasValue,
|
|
222
|
+
colorScheme: finalColorScheme,
|
|
223
|
+
size,
|
|
224
|
+
variant,
|
|
225
|
+
isValid: finalIsValid,
|
|
226
|
+
disabled
|
|
227
|
+
});
|
|
228
|
+
const textareaClasses = textarea({
|
|
229
|
+
theme,
|
|
230
|
+
font,
|
|
231
|
+
weight,
|
|
232
|
+
size,
|
|
233
|
+
disabled
|
|
234
|
+
}) + ' ' + className;
|
|
235
|
+
return jsxRuntime.jsxs("div", {
|
|
236
|
+
className: containerClasses,
|
|
237
|
+
children: [jsxRuntime.jsx("textarea", Object.assign({
|
|
238
|
+
ref: textareaRef,
|
|
239
|
+
value: value,
|
|
240
|
+
onChange: handleChange,
|
|
241
|
+
rows: rows,
|
|
242
|
+
className: textareaClasses,
|
|
243
|
+
disabled: disabled
|
|
244
|
+
}, props)), showPasteButton && !hasValue && !disabled && jsxRuntime.jsx(index.Button, {
|
|
245
|
+
colorScheme: 'brand',
|
|
246
|
+
size: 'sm',
|
|
247
|
+
onClick: handlePaste,
|
|
248
|
+
className: `absolute top-1/2 !-translate-y-1/2 ${size === 'sm' ? 'right-3' : size === 'md' ? 'right-[1.125rem]' : 'right-[1.5625rem]'}`,
|
|
249
|
+
children: "PASTE"
|
|
250
|
+
})]
|
|
251
|
+
});
|
|
252
|
+
};
|
|
253
|
+
|
|
254
|
+
exports.Textarea = Textarea;
|
|
255
|
+
exports.default = Textarea;
|
|
256
|
+
//# sourceMappingURL=index.cjs.js.map
|
|
@@ -0,0 +1 @@
|
|
|
1
|
+
{"version":3,"file":"index.cjs.js","sources":["../../../../src/react/components/textarea/index.tsx"],"sourcesContent":["import React, { useState, useRef, TextareaHTMLAttributes } from 'react'\nimport { Button } from '../button'\nimport { cva, VariantProps } from 'class-variance-authority'\nimport { useTheme } from '../../contexts/ThemeContext'\n\nconst textareaContainer = cva(\n 'relative flex items-baseline transition-all w-full',\n {\n variants: {\n theme: {\n light: 'bg-white',\n dark: 'bg-gray-800'\n },\n hasValue: {\n true: '',\n false: ''\n },\n colorScheme: {\n default: 'focus-within:ring-blue-500/20',\n brand: 'focus-within:ring-dash-brand/20',\n error: 'focus-within:ring-red-500/20',\n success: 'focus-within:ring-green-500/20'\n },\n size: {\n sm: 'dash-block-sm',\n md: 'dash-block-md',\n xl: 'dash-block-xl'\n },\n variant: {\n outlined: 'outline outline-1 outline-offset-[-1px]'\n },\n isValid: {\n true: '',\n false: '',\n null: ''\n },\n disabled: {\n false: '',\n true: 'opacity-60 cursor-not-allowed'\n }\n },\n compoundVariants: [\n // Outlined variant colors\n {\n variant: 'outlined',\n colorScheme: 'default',\n class: 'outline-[rgba(17,17,17,0.32)] focus-within:outline-[rgba(17,17,17,0.6)]'\n },\n {\n variant: 'outlined',\n colorScheme: 'brand',\n class: 'outline-dash-brand/30 focus-within:outline-dash-brand'\n },\n {\n variant: 'outlined',\n colorScheme: 'error',\n isValid: false,\n class: 'outline-red-500 focus-within:outline-red-500'\n },\n {\n variant: 'outlined',\n colorScheme: 'success',\n isValid: true,\n class: 'outline-green-500 focus-within:outline-green-500'\n },\n // Outlined variant with focus ring\n {\n variant: 'outlined',\n class: 'focus-within:ring-2'\n },\n // Add extra padding for PASTE button when no value\n { hasValue: false, size: 'sm', class: 'pr-[70px]' },\n { hasValue: false, size: 'md', class: 'pr-[70px]' },\n { hasValue: false, size: 'xl', class: 'pr-[70px]' }\n ],\n defaultVariants: {\n theme: 'light',\n hasValue: false,\n colorScheme: 'default',\n size: 'xl',\n variant: 'outlined',\n isValid: null,\n disabled: false\n }\n }\n)\n\nconst textarea = cva(\n 'w-full bg-transparent outline-none resize-none transition-all text-[0.875rem] leading-[1.0625rem] placeholder:text-opacity-60',\n {\n variants: {\n theme: {\n light: 'text-[#111111] placeholder:text-[rgba(17,17,17,0.6)]',\n dark: 'text-white placeholder:text-gray-400'\n },\n font: {\n main: 'font-dash-main',\n grotesque: 'font-dash-grotesque'\n },\n weight: {\n light: 'font-light',\n normal: 'font-normal',\n medium: 'font-medium',\n semibold: 'font-semibold',\n bold: 'font-bold'\n },\n size: {\n sm: '',\n md: '',\n xl: ''\n },\n disabled: {\n false: '',\n true: 'cursor-not-allowed'\n }\n },\n defaultVariants: {\n theme: 'light',\n font: 'main',\n weight: 'light',\n size: 'xl',\n disabled: false\n }\n }\n)\n\ntype TextareaVariants = VariantProps<typeof textareaContainer>\n\nexport interface TextareaProps extends Omit<TextareaHTMLAttributes<HTMLTextAreaElement>, 'onChange' | 'size'>, Omit<TextareaVariants, 'theme' | 'hasValue' | 'isValid' | 'disabled'> {\n className?: string\n onChange?: (value: string) => void\n showPasteButton?: boolean\n validator?: ((value: string) => boolean) | boolean\n rows?: number\n font?: 'main' | 'grotesque'\n weight?: 'light' | 'normal' | 'medium' | 'semibold' | 'bold'\n error?: boolean\n success?: boolean\n}\n\n/**\n * A versatile textarea component that adapts to light/dark theme,\n * supports various color schemes, sizes, variants, and states.\n * Includes optional paste functionality and validation.\n *\n * @example\n * <Textarea\n * placeholder='Enter your message'\n * colorScheme='brand'\n * size='xl'\n * font='grotesque'\n * weight='medium'\n * rows={4}\n * showPasteButton={true}\n * />\n */\nexport const Textarea: React.FC<TextareaProps> = ({\n className = '',\n onChange,\n showPasteButton = true,\n validator = null,\n rows = 3,\n size = 'xl',\n variant = 'outlined',\n colorScheme = 'default',\n font = 'main',\n weight = 'light',\n error = false,\n success = false,\n disabled = false,\n ...props\n}) => {\n const { theme } = useTheme()\n const [value, setValue] = useState<string>((props.value as string) ?? (props.defaultValue as string) ?? '')\n const [isValid, setIsValid] = useState<boolean | null>(null)\n const textareaRef = useRef<HTMLTextAreaElement>(null)\n\n const handleChange = (e: React.ChangeEvent<HTMLTextAreaElement>): void => {\n const newValue = e.target.value\n setValue(newValue)\n\n if (typeof onChange === 'function') {\n onChange(newValue)\n }\n\n validateInput(newValue)\n }\n\n const validateInput = (input: string): void => {\n if (validator === null) {\n setIsValid(null)\n return\n }\n\n if (typeof validator === 'function') {\n setIsValid(validator(input))\n } else {\n setIsValid(Boolean(validator))\n }\n }\n\n const handlePaste = (): void => {\n navigator.clipboard.readText()\n .then((text) => {\n if (text !== '') {\n setValue(text)\n if (textareaRef.current != null) {\n textareaRef.current.value = text\n }\n if (onChange != null) {\n onChange(text)\n }\n validateInput(text)\n }\n })\n .catch((err) => {\n console.error('Failed to read clipboard contents: ', err)\n })\n }\n\n const hasValue = value !== ''\n\n // Determine color scheme based on state\n let finalColorScheme = colorScheme\n let finalIsValid: boolean | null = isValid\n \n if (error) {\n finalColorScheme = 'error'\n finalIsValid = false\n } else if (success) {\n finalColorScheme = 'success'\n finalIsValid = true\n }\n\n const containerClasses = textareaContainer({\n theme,\n hasValue,\n colorScheme: finalColorScheme,\n size,\n variant,\n isValid: finalIsValid,\n disabled\n })\n\n const textareaClasses = textarea({\n theme,\n font,\n weight,\n size,\n disabled\n }) + ' ' + className\n\n return (\n <div className={containerClasses}>\n <textarea\n ref={textareaRef}\n value={value}\n onChange={handleChange}\n rows={rows}\n className={textareaClasses}\n disabled={disabled}\n {...props}\n />\n\n {showPasteButton && !hasValue && !disabled && (\n <Button \n colorScheme='brand' \n size='sm' \n onClick={handlePaste}\n className={`absolute top-1/2 !-translate-y-1/2 ${\n size === 'sm' ? 'right-3' : \n size === 'md' ? 'right-[1.125rem]' : \n 'right-[1.5625rem]'\n }`}\n >\n PASTE\n </Button>\n )}\n </div>\n )\n}\n\nexport default Textarea "],"names":["textareaContainer","cva","variants","theme","light","dark","hasValue","true","false","colorScheme","default","brand","error","success","size","sm","md","xl","variant","outlined","isValid","null","disabled","compoundVariants","class","defaultVariants","textarea","font","main","grotesque","weight","normal","medium","semibold","bold","Textarea","_a","className","onChange","showPasteButton","validator","rows","props","__rest","useTheme","value","setValue","useState","_c","_b","defaultValue","setIsValid","textareaRef","useRef","handleChange","e","newValue","target","validateInput","input","Boolean","handlePaste","navigator","clipboard","readText","then","text","current","catch","err","console","finalColorScheme","finalIsValid","containerClasses","textareaClasses","_jsxs","children","_jsx","Object","assign","ref","Button","onClick"],"mappings":"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"}
|
|
@@ -0,0 +1,251 @@
|
|
|
1
|
+
"use client";
|
|
2
|
+
|
|
3
|
+
import { __rest } from 'tslib';
|
|
4
|
+
import { jsxs, jsx } from 'react/jsx-runtime';
|
|
5
|
+
import { useState, useRef } from 'react';
|
|
6
|
+
import { Button } from '../button/index.esm.js';
|
|
7
|
+
import { cva } from 'class-variance-authority';
|
|
8
|
+
import { useTheme } from '../../contexts/ThemeContext.esm.js';
|
|
9
|
+
|
|
10
|
+
const textareaContainer = cva('relative flex items-baseline transition-all w-full', {
|
|
11
|
+
variants: {
|
|
12
|
+
theme: {
|
|
13
|
+
light: 'bg-white',
|
|
14
|
+
dark: 'bg-gray-800'
|
|
15
|
+
},
|
|
16
|
+
hasValue: {
|
|
17
|
+
true: '',
|
|
18
|
+
false: ''
|
|
19
|
+
},
|
|
20
|
+
colorScheme: {
|
|
21
|
+
default: 'focus-within:ring-blue-500/20',
|
|
22
|
+
brand: 'focus-within:ring-dash-brand/20',
|
|
23
|
+
error: 'focus-within:ring-red-500/20',
|
|
24
|
+
success: 'focus-within:ring-green-500/20'
|
|
25
|
+
},
|
|
26
|
+
size: {
|
|
27
|
+
sm: 'dash-block-sm',
|
|
28
|
+
md: 'dash-block-md',
|
|
29
|
+
xl: 'dash-block-xl'
|
|
30
|
+
},
|
|
31
|
+
variant: {
|
|
32
|
+
outlined: 'outline outline-1 outline-offset-[-1px]'
|
|
33
|
+
},
|
|
34
|
+
isValid: {
|
|
35
|
+
true: '',
|
|
36
|
+
false: '',
|
|
37
|
+
null: ''
|
|
38
|
+
},
|
|
39
|
+
disabled: {
|
|
40
|
+
false: '',
|
|
41
|
+
true: 'opacity-60 cursor-not-allowed'
|
|
42
|
+
}
|
|
43
|
+
},
|
|
44
|
+
compoundVariants: [
|
|
45
|
+
// Outlined variant colors
|
|
46
|
+
{
|
|
47
|
+
variant: 'outlined',
|
|
48
|
+
colorScheme: 'default',
|
|
49
|
+
class: 'outline-[rgba(17,17,17,0.32)] focus-within:outline-[rgba(17,17,17,0.6)]'
|
|
50
|
+
}, {
|
|
51
|
+
variant: 'outlined',
|
|
52
|
+
colorScheme: 'brand',
|
|
53
|
+
class: 'outline-dash-brand/30 focus-within:outline-dash-brand'
|
|
54
|
+
}, {
|
|
55
|
+
variant: 'outlined',
|
|
56
|
+
colorScheme: 'error',
|
|
57
|
+
isValid: false,
|
|
58
|
+
class: 'outline-red-500 focus-within:outline-red-500'
|
|
59
|
+
}, {
|
|
60
|
+
variant: 'outlined',
|
|
61
|
+
colorScheme: 'success',
|
|
62
|
+
isValid: true,
|
|
63
|
+
class: 'outline-green-500 focus-within:outline-green-500'
|
|
64
|
+
},
|
|
65
|
+
// Outlined variant with focus ring
|
|
66
|
+
{
|
|
67
|
+
variant: 'outlined',
|
|
68
|
+
class: 'focus-within:ring-2'
|
|
69
|
+
},
|
|
70
|
+
// Add extra padding for PASTE button when no value
|
|
71
|
+
{
|
|
72
|
+
hasValue: false,
|
|
73
|
+
size: 'sm',
|
|
74
|
+
class: 'pr-[70px]'
|
|
75
|
+
}, {
|
|
76
|
+
hasValue: false,
|
|
77
|
+
size: 'md',
|
|
78
|
+
class: 'pr-[70px]'
|
|
79
|
+
}, {
|
|
80
|
+
hasValue: false,
|
|
81
|
+
size: 'xl',
|
|
82
|
+
class: 'pr-[70px]'
|
|
83
|
+
}],
|
|
84
|
+
defaultVariants: {
|
|
85
|
+
theme: 'light',
|
|
86
|
+
hasValue: false,
|
|
87
|
+
colorScheme: 'default',
|
|
88
|
+
size: 'xl',
|
|
89
|
+
variant: 'outlined',
|
|
90
|
+
isValid: null,
|
|
91
|
+
disabled: false
|
|
92
|
+
}
|
|
93
|
+
});
|
|
94
|
+
const textarea = cva('w-full bg-transparent outline-none resize-none transition-all text-[0.875rem] leading-[1.0625rem] placeholder:text-opacity-60', {
|
|
95
|
+
variants: {
|
|
96
|
+
theme: {
|
|
97
|
+
light: 'text-[#111111] placeholder:text-[rgba(17,17,17,0.6)]',
|
|
98
|
+
dark: 'text-white placeholder:text-gray-400'
|
|
99
|
+
},
|
|
100
|
+
font: {
|
|
101
|
+
main: 'font-dash-main',
|
|
102
|
+
grotesque: 'font-dash-grotesque'
|
|
103
|
+
},
|
|
104
|
+
weight: {
|
|
105
|
+
light: 'font-light',
|
|
106
|
+
normal: 'font-normal',
|
|
107
|
+
medium: 'font-medium',
|
|
108
|
+
semibold: 'font-semibold',
|
|
109
|
+
bold: 'font-bold'
|
|
110
|
+
},
|
|
111
|
+
size: {
|
|
112
|
+
sm: '',
|
|
113
|
+
md: '',
|
|
114
|
+
xl: ''
|
|
115
|
+
},
|
|
116
|
+
disabled: {
|
|
117
|
+
false: '',
|
|
118
|
+
true: 'cursor-not-allowed'
|
|
119
|
+
}
|
|
120
|
+
},
|
|
121
|
+
defaultVariants: {
|
|
122
|
+
theme: 'light',
|
|
123
|
+
font: 'main',
|
|
124
|
+
weight: 'light',
|
|
125
|
+
size: 'xl',
|
|
126
|
+
disabled: false
|
|
127
|
+
}
|
|
128
|
+
});
|
|
129
|
+
/**
|
|
130
|
+
* A versatile textarea component that adapts to light/dark theme,
|
|
131
|
+
* supports various color schemes, sizes, variants, and states.
|
|
132
|
+
* Includes optional paste functionality and validation.
|
|
133
|
+
*
|
|
134
|
+
* @example
|
|
135
|
+
* <Textarea
|
|
136
|
+
* placeholder='Enter your message'
|
|
137
|
+
* colorScheme='brand'
|
|
138
|
+
* size='xl'
|
|
139
|
+
* font='grotesque'
|
|
140
|
+
* weight='medium'
|
|
141
|
+
* rows={4}
|
|
142
|
+
* showPasteButton={true}
|
|
143
|
+
* />
|
|
144
|
+
*/
|
|
145
|
+
const Textarea = _a => {
|
|
146
|
+
var _b, _c;
|
|
147
|
+
var {
|
|
148
|
+
className = '',
|
|
149
|
+
onChange,
|
|
150
|
+
showPasteButton = true,
|
|
151
|
+
validator = null,
|
|
152
|
+
rows = 3,
|
|
153
|
+
size = 'xl',
|
|
154
|
+
variant = 'outlined',
|
|
155
|
+
colorScheme = 'default',
|
|
156
|
+
font = 'main',
|
|
157
|
+
weight = 'light',
|
|
158
|
+
error = false,
|
|
159
|
+
success = false,
|
|
160
|
+
disabled = false
|
|
161
|
+
} = _a,
|
|
162
|
+
props = __rest(_a, ["className", "onChange", "showPasteButton", "validator", "rows", "size", "variant", "colorScheme", "font", "weight", "error", "success", "disabled"]);
|
|
163
|
+
const {
|
|
164
|
+
theme
|
|
165
|
+
} = useTheme();
|
|
166
|
+
const [value, setValue] = useState((_c = (_b = props.value) !== null && _b !== void 0 ? _b : props.defaultValue) !== null && _c !== void 0 ? _c : '');
|
|
167
|
+
const [isValid, setIsValid] = useState(null);
|
|
168
|
+
const textareaRef = useRef(null);
|
|
169
|
+
const handleChange = e => {
|
|
170
|
+
const newValue = e.target.value;
|
|
171
|
+
setValue(newValue);
|
|
172
|
+
if (typeof onChange === 'function') {
|
|
173
|
+
onChange(newValue);
|
|
174
|
+
}
|
|
175
|
+
validateInput(newValue);
|
|
176
|
+
};
|
|
177
|
+
const validateInput = input => {
|
|
178
|
+
if (validator === null) {
|
|
179
|
+
setIsValid(null);
|
|
180
|
+
return;
|
|
181
|
+
}
|
|
182
|
+
if (typeof validator === 'function') {
|
|
183
|
+
setIsValid(validator(input));
|
|
184
|
+
} else {
|
|
185
|
+
setIsValid(Boolean(validator));
|
|
186
|
+
}
|
|
187
|
+
};
|
|
188
|
+
const handlePaste = () => {
|
|
189
|
+
navigator.clipboard.readText().then(text => {
|
|
190
|
+
if (text !== '') {
|
|
191
|
+
setValue(text);
|
|
192
|
+
if (textareaRef.current != null) {
|
|
193
|
+
textareaRef.current.value = text;
|
|
194
|
+
}
|
|
195
|
+
if (onChange != null) {
|
|
196
|
+
onChange(text);
|
|
197
|
+
}
|
|
198
|
+
validateInput(text);
|
|
199
|
+
}
|
|
200
|
+
}).catch(err => {
|
|
201
|
+
console.error('Failed to read clipboard contents: ', err);
|
|
202
|
+
});
|
|
203
|
+
};
|
|
204
|
+
const hasValue = value !== '';
|
|
205
|
+
// Determine color scheme based on state
|
|
206
|
+
let finalColorScheme = colorScheme;
|
|
207
|
+
let finalIsValid = isValid;
|
|
208
|
+
if (error) {
|
|
209
|
+
finalColorScheme = 'error';
|
|
210
|
+
finalIsValid = false;
|
|
211
|
+
} else if (success) {
|
|
212
|
+
finalColorScheme = 'success';
|
|
213
|
+
finalIsValid = true;
|
|
214
|
+
}
|
|
215
|
+
const containerClasses = textareaContainer({
|
|
216
|
+
theme,
|
|
217
|
+
hasValue,
|
|
218
|
+
colorScheme: finalColorScheme,
|
|
219
|
+
size,
|
|
220
|
+
variant,
|
|
221
|
+
isValid: finalIsValid,
|
|
222
|
+
disabled
|
|
223
|
+
});
|
|
224
|
+
const textareaClasses = textarea({
|
|
225
|
+
theme,
|
|
226
|
+
font,
|
|
227
|
+
weight,
|
|
228
|
+
size,
|
|
229
|
+
disabled
|
|
230
|
+
}) + ' ' + className;
|
|
231
|
+
return jsxs("div", {
|
|
232
|
+
className: containerClasses,
|
|
233
|
+
children: [jsx("textarea", Object.assign({
|
|
234
|
+
ref: textareaRef,
|
|
235
|
+
value: value,
|
|
236
|
+
onChange: handleChange,
|
|
237
|
+
rows: rows,
|
|
238
|
+
className: textareaClasses,
|
|
239
|
+
disabled: disabled
|
|
240
|
+
}, props)), showPasteButton && !hasValue && !disabled && jsx(Button, {
|
|
241
|
+
colorScheme: 'brand',
|
|
242
|
+
size: 'sm',
|
|
243
|
+
onClick: handlePaste,
|
|
244
|
+
className: `absolute top-1/2 !-translate-y-1/2 ${size === 'sm' ? 'right-3' : size === 'md' ? 'right-[1.125rem]' : 'right-[1.5625rem]'}`,
|
|
245
|
+
children: "PASTE"
|
|
246
|
+
})]
|
|
247
|
+
});
|
|
248
|
+
};
|
|
249
|
+
|
|
250
|
+
export { Textarea, Textarea as default };
|
|
251
|
+
//# sourceMappingURL=index.esm.js.map
|
|
@@ -0,0 +1 @@
|
|
|
1
|
+
{"version":3,"file":"index.esm.js","sources":["../../../../src/react/components/textarea/index.tsx"],"sourcesContent":["import React, { useState, useRef, TextareaHTMLAttributes } from 'react'\nimport { Button } from '../button'\nimport { cva, VariantProps } from 'class-variance-authority'\nimport { useTheme } from '../../contexts/ThemeContext'\n\nconst textareaContainer = cva(\n 'relative flex items-baseline transition-all w-full',\n {\n variants: {\n theme: {\n light: 'bg-white',\n dark: 'bg-gray-800'\n },\n hasValue: {\n true: '',\n false: ''\n },\n colorScheme: {\n default: 'focus-within:ring-blue-500/20',\n brand: 'focus-within:ring-dash-brand/20',\n error: 'focus-within:ring-red-500/20',\n success: 'focus-within:ring-green-500/20'\n },\n size: {\n sm: 'dash-block-sm',\n md: 'dash-block-md',\n xl: 'dash-block-xl'\n },\n variant: {\n outlined: 'outline outline-1 outline-offset-[-1px]'\n },\n isValid: {\n true: '',\n false: '',\n null: ''\n },\n disabled: {\n false: '',\n true: 'opacity-60 cursor-not-allowed'\n }\n },\n compoundVariants: [\n // Outlined variant colors\n {\n variant: 'outlined',\n colorScheme: 'default',\n class: 'outline-[rgba(17,17,17,0.32)] focus-within:outline-[rgba(17,17,17,0.6)]'\n },\n {\n variant: 'outlined',\n colorScheme: 'brand',\n class: 'outline-dash-brand/30 focus-within:outline-dash-brand'\n },\n {\n variant: 'outlined',\n colorScheme: 'error',\n isValid: false,\n class: 'outline-red-500 focus-within:outline-red-500'\n },\n {\n variant: 'outlined',\n colorScheme: 'success',\n isValid: true,\n class: 'outline-green-500 focus-within:outline-green-500'\n },\n // Outlined variant with focus ring\n {\n variant: 'outlined',\n class: 'focus-within:ring-2'\n },\n // Add extra padding for PASTE button when no value\n { hasValue: false, size: 'sm', class: 'pr-[70px]' },\n { hasValue: false, size: 'md', class: 'pr-[70px]' },\n { hasValue: false, size: 'xl', class: 'pr-[70px]' }\n ],\n defaultVariants: {\n theme: 'light',\n hasValue: false,\n colorScheme: 'default',\n size: 'xl',\n variant: 'outlined',\n isValid: null,\n disabled: false\n }\n }\n)\n\nconst textarea = cva(\n 'w-full bg-transparent outline-none resize-none transition-all text-[0.875rem] leading-[1.0625rem] placeholder:text-opacity-60',\n {\n variants: {\n theme: {\n light: 'text-[#111111] placeholder:text-[rgba(17,17,17,0.6)]',\n dark: 'text-white placeholder:text-gray-400'\n },\n font: {\n main: 'font-dash-main',\n grotesque: 'font-dash-grotesque'\n },\n weight: {\n light: 'font-light',\n normal: 'font-normal',\n medium: 'font-medium',\n semibold: 'font-semibold',\n bold: 'font-bold'\n },\n size: {\n sm: '',\n md: '',\n xl: ''\n },\n disabled: {\n false: '',\n true: 'cursor-not-allowed'\n }\n },\n defaultVariants: {\n theme: 'light',\n font: 'main',\n weight: 'light',\n size: 'xl',\n disabled: false\n }\n }\n)\n\ntype TextareaVariants = VariantProps<typeof textareaContainer>\n\nexport interface TextareaProps extends Omit<TextareaHTMLAttributes<HTMLTextAreaElement>, 'onChange' | 'size'>, Omit<TextareaVariants, 'theme' | 'hasValue' | 'isValid' | 'disabled'> {\n className?: string\n onChange?: (value: string) => void\n showPasteButton?: boolean\n validator?: ((value: string) => boolean) | boolean\n rows?: number\n font?: 'main' | 'grotesque'\n weight?: 'light' | 'normal' | 'medium' | 'semibold' | 'bold'\n error?: boolean\n success?: boolean\n}\n\n/**\n * A versatile textarea component that adapts to light/dark theme,\n * supports various color schemes, sizes, variants, and states.\n * Includes optional paste functionality and validation.\n *\n * @example\n * <Textarea\n * placeholder='Enter your message'\n * colorScheme='brand'\n * size='xl'\n * font='grotesque'\n * weight='medium'\n * rows={4}\n * showPasteButton={true}\n * />\n */\nexport const Textarea: React.FC<TextareaProps> = ({\n className = '',\n onChange,\n showPasteButton = true,\n validator = null,\n rows = 3,\n size = 'xl',\n variant = 'outlined',\n colorScheme = 'default',\n font = 'main',\n weight = 'light',\n error = false,\n success = false,\n disabled = false,\n ...props\n}) => {\n const { theme } = useTheme()\n const [value, setValue] = useState<string>((props.value as string) ?? (props.defaultValue as string) ?? '')\n const [isValid, setIsValid] = useState<boolean | null>(null)\n const textareaRef = useRef<HTMLTextAreaElement>(null)\n\n const handleChange = (e: React.ChangeEvent<HTMLTextAreaElement>): void => {\n const newValue = e.target.value\n setValue(newValue)\n\n if (typeof onChange === 'function') {\n onChange(newValue)\n }\n\n validateInput(newValue)\n }\n\n const validateInput = (input: string): void => {\n if (validator === null) {\n setIsValid(null)\n return\n }\n\n if (typeof validator === 'function') {\n setIsValid(validator(input))\n } else {\n setIsValid(Boolean(validator))\n }\n }\n\n const handlePaste = (): void => {\n navigator.clipboard.readText()\n .then((text) => {\n if (text !== '') {\n setValue(text)\n if (textareaRef.current != null) {\n textareaRef.current.value = text\n }\n if (onChange != null) {\n onChange(text)\n }\n validateInput(text)\n }\n })\n .catch((err) => {\n console.error('Failed to read clipboard contents: ', err)\n })\n }\n\n const hasValue = value !== ''\n\n // Determine color scheme based on state\n let finalColorScheme = colorScheme\n let finalIsValid: boolean | null = isValid\n \n if (error) {\n finalColorScheme = 'error'\n finalIsValid = false\n } else if (success) {\n finalColorScheme = 'success'\n finalIsValid = true\n }\n\n const containerClasses = textareaContainer({\n theme,\n hasValue,\n colorScheme: finalColorScheme,\n size,\n variant,\n isValid: finalIsValid,\n disabled\n })\n\n const textareaClasses = textarea({\n theme,\n font,\n weight,\n size,\n disabled\n }) + ' ' + className\n\n return (\n <div className={containerClasses}>\n <textarea\n ref={textareaRef}\n value={value}\n onChange={handleChange}\n rows={rows}\n className={textareaClasses}\n disabled={disabled}\n {...props}\n />\n\n {showPasteButton && !hasValue && !disabled && (\n <Button \n colorScheme='brand' \n size='sm' \n onClick={handlePaste}\n className={`absolute top-1/2 !-translate-y-1/2 ${\n size === 'sm' ? 'right-3' : \n size === 'md' ? 'right-[1.125rem]' : \n 'right-[1.5625rem]'\n }`}\n >\n PASTE\n </Button>\n )}\n </div>\n )\n}\n\nexport default Textarea "],"names":["textareaContainer","cva","variants","theme","light","dark","hasValue","true","false","colorScheme","default","brand","error","success","size","sm","md","xl","variant","outlined","isValid","null","disabled","compoundVariants","class","defaultVariants","textarea","font","main","grotesque","weight","normal","medium","semibold","bold","Textarea","_a","className","onChange","showPasteButton","validator","rows","props","__rest","useTheme","value","setValue","useState","_c","_b","defaultValue","setIsValid","textareaRef","useRef","handleChange","e","newValue","target","validateInput","input","Boolean","handlePaste","navigator","clipboard","readText","then","text","current","catch","err","console","finalColorScheme","finalIsValid","containerClasses","textareaClasses","_jsxs","children","_jsx","Object","assign","ref","Button","onClick"],"mappings":"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"}
|
|
@@ -0,0 +1,93 @@
|
|
|
1
|
+
"use client";
|
|
2
|
+
|
|
3
|
+
'use strict';
|
|
4
|
+
|
|
5
|
+
Object.defineProperty(exports, '__esModule', { value: true });
|
|
6
|
+
|
|
7
|
+
var jsxRuntime = require('react/jsx-runtime');
|
|
8
|
+
var React = require('react');
|
|
9
|
+
var classVarianceAuthority = require('class-variance-authority');
|
|
10
|
+
var index = require('../notActive/index.cjs.js');
|
|
11
|
+
var datetime = require('../../shared/utils/datetime.cjs.js');
|
|
12
|
+
var ThemeContext = require('../../contexts/ThemeContext.cjs.js');
|
|
13
|
+
|
|
14
|
+
const wrapperStyles = classVarianceAuthority.cva('inline', {
|
|
15
|
+
variants: {
|
|
16
|
+
theme: {
|
|
17
|
+
light: 'text-gray-900',
|
|
18
|
+
dark: 'text-gray-100'
|
|
19
|
+
}
|
|
20
|
+
},
|
|
21
|
+
defaultVariants: {
|
|
22
|
+
theme: 'light'
|
|
23
|
+
}
|
|
24
|
+
});
|
|
25
|
+
/**
|
|
26
|
+
* TimeDelta component renews a human-readable delta string periodically,
|
|
27
|
+
* and optionally wraps it in a tooltip showing the exact date/time.
|
|
28
|
+
* Supports light/dark theme.
|
|
29
|
+
*/
|
|
30
|
+
const TimeDelta = ({
|
|
31
|
+
startDate,
|
|
32
|
+
endDate,
|
|
33
|
+
// showTimestampTooltip = true,
|
|
34
|
+
// tooltipDate,
|
|
35
|
+
format = 'default'
|
|
36
|
+
}) => {
|
|
37
|
+
const {
|
|
38
|
+
theme
|
|
39
|
+
} = ThemeContext.useTheme();
|
|
40
|
+
const [timeDelta, setTimeDelta] = React.useState(null);
|
|
41
|
+
// const tooltipDateObj = new Date(tooltipDate ?? endDate)
|
|
42
|
+
React.useEffect(() => {
|
|
43
|
+
if (endDate == null) {
|
|
44
|
+
setTimeDelta(null);
|
|
45
|
+
return;
|
|
46
|
+
}
|
|
47
|
+
let timeoutId;
|
|
48
|
+
const updateDelta = () => {
|
|
49
|
+
const start = startDate != null ? new Date(startDate) : new Date();
|
|
50
|
+
const end = new Date(endDate);
|
|
51
|
+
setTimeDelta(datetime.getTimeDelta(start, end, format));
|
|
52
|
+
const now = new Date();
|
|
53
|
+
const diffMs = Math.abs(end.getTime() - now.getTime());
|
|
54
|
+
if (diffMs > 60000) {
|
|
55
|
+
const msToNextMinute = (60 - now.getSeconds()) * 1000 - now.getMilliseconds();
|
|
56
|
+
timeoutId = setTimeout(updateDelta, msToNextMinute);
|
|
57
|
+
} else {
|
|
58
|
+
timeoutId = setTimeout(updateDelta, 1000);
|
|
59
|
+
}
|
|
60
|
+
};
|
|
61
|
+
updateDelta();
|
|
62
|
+
return () => clearTimeout(timeoutId);
|
|
63
|
+
}, [startDate, endDate, format]);
|
|
64
|
+
if (timeDelta == null || timeDelta === '') {
|
|
65
|
+
return jsxRuntime.jsx(index.NotActive, {});
|
|
66
|
+
}
|
|
67
|
+
// const showTooltip = showTimestampTooltip && format !== 'detailed' && !isNaN(tooltipDateObj.getTime())
|
|
68
|
+
const content = jsxRuntime.jsx("span", {
|
|
69
|
+
className: wrapperStyles({
|
|
70
|
+
theme
|
|
71
|
+
}),
|
|
72
|
+
children: timeDelta
|
|
73
|
+
});
|
|
74
|
+
// if (showTooltip) {
|
|
75
|
+
// return (
|
|
76
|
+
// <Tooltip
|
|
77
|
+
// placement="top"
|
|
78
|
+
// content={
|
|
79
|
+
// <span className={tooltipContentStyles()}>
|
|
80
|
+
// {tooltipDateObj.toLocaleDateString()}{' '}{tooltipDateObj.toLocaleTimeString()}
|
|
81
|
+
// </span>
|
|
82
|
+
// }
|
|
83
|
+
// >
|
|
84
|
+
// {content}
|
|
85
|
+
// </Tooltip>
|
|
86
|
+
// )
|
|
87
|
+
// }
|
|
88
|
+
return content;
|
|
89
|
+
};
|
|
90
|
+
|
|
91
|
+
exports.TimeDelta = TimeDelta;
|
|
92
|
+
exports.default = TimeDelta;
|
|
93
|
+
//# sourceMappingURL=index.cjs.js.map
|
|
@@ -0,0 +1 @@
|
|
|
1
|
+
{"version":3,"file":"index.cjs.js","sources":["../../../../src/react/components/timeDelta/index.tsx"],"sourcesContent":["import React, { useEffect, useState } from 'react'\nimport { cva } from 'class-variance-authority'\nimport { NotActive } from '../notActive'\nimport { getTimeDelta, type TimeDeltaFormat } from '../../../shared/utils/datetime'\nimport { useTheme } from '../../contexts/ThemeContext'\n\nconst wrapperStyles = cva(\n 'inline',\n {\n variants: {\n theme: {\n light: 'text-gray-900',\n dark: 'text-gray-100'\n }\n },\n defaultVariants: {\n theme: 'light'\n }\n }\n)\n\nexport interface TimeDeltaProps {\n /** Start date for delta calculation, defaults to now */\n startDate?: Date | number | string\n /** End date or timestamp */\n endDate: Date | number | string\n /** Show tooltip with exact timestamp */\n showTimestampTooltip?: boolean\n /** Override date for tooltip instead of endDate */\n tooltipDate?: Date | number | string\n /** Format mode: 'default' shows delta and tooltip, 'detailed' suppresses tooltip */\n format?: TimeDeltaFormat\n}\n\n/**\n * TimeDelta component renews a human-readable delta string periodically,\n * and optionally wraps it in a tooltip showing the exact date/time.\n * Supports light/dark theme.\n */\nexport const TimeDelta: React.FC<TimeDeltaProps> = ({\n startDate,\n endDate,\n // showTimestampTooltip = true,\n // tooltipDate,\n format = 'default'\n}) => {\n const { theme } = useTheme()\n const [timeDelta, setTimeDelta] = useState<string | null>(null)\n // const tooltipDateObj = new Date(tooltipDate ?? endDate)\n\n useEffect(() => {\n if (endDate == null) {\n setTimeDelta(null)\n return\n }\n\n let timeoutId: ReturnType<typeof setTimeout>\n\n const updateDelta = (): void => {\n const start = startDate != null ? new Date(startDate) : new Date()\n const end = new Date(endDate)\n setTimeDelta(getTimeDelta(start, end, format))\n\n const now = new Date()\n const diffMs = Math.abs(end.getTime() - now.getTime())\n if (diffMs > 60_000) {\n const msToNextMinute = (60 - now.getSeconds()) * 1000 - now.getMilliseconds()\n timeoutId = setTimeout(updateDelta, msToNextMinute)\n } else {\n timeoutId = setTimeout(updateDelta, 1000)\n }\n }\n\n updateDelta()\n\n return () => clearTimeout(timeoutId)\n }, [startDate, endDate, format])\n\n if (timeDelta == null || timeDelta === '') {\n return <NotActive />\n }\n\n // const showTooltip = showTimestampTooltip && format !== 'detailed' && !isNaN(tooltipDateObj.getTime())\n const content = <span className={wrapperStyles({ theme })}>{timeDelta}</span>\n\n // if (showTooltip) {\n // return (\n // <Tooltip\n // placement=\"top\"\n // content={\n // <span className={tooltipContentStyles()}>\n // {tooltipDateObj.toLocaleDateString()}{' '}{tooltipDateObj.toLocaleTimeString()}\n // </span>\n // }\n // >\n // {content}\n // </Tooltip>\n // )\n // }\n\n return content\n}\n\nexport default TimeDelta "],"names":["wrapperStyles","cva","variants","theme","light","dark","defaultVariants","TimeDelta","startDate","endDate","format","useTheme","timeDelta","setTimeDelta","useState","useEffect","timeoutId","updateDelta","start","Date","end","getTimeDelta","now","diffMs","Math","abs","getTime","msToNextMinute","getSeconds","getMilliseconds","setTimeout","clearTimeout","_jsx","NotActive","content","className","children"],"mappings":";;;;;;;;;;;;;AAMA,MAAMA,aAAa,GAAGC,0BAAG,CACvB,QAAQ,EACR;AACEC,EAAAA,QAAQ,EAAE;AACRC,IAAAA,KAAK,EAAE;AACLC,MAAAA,KAAK,EAAE,eAAe;AACtBC,MAAAA,IAAI,EAAE;AACP;GACF;AACDC,EAAAA,eAAe,EAAE;AACfH,IAAAA,KAAK,EAAE;AACR;AACF,CAAA,CACF;AAeD;;;;AAIG;AACI,MAAMI,SAAS,GAA6BA,CAAC;EAClDC,SAAS;EACTC,OAAO;AACP;AACA;AACAC,EAAAA,MAAM,GAAG;AACV,CAAA,KAAI;EACH,MAAM;AAAEP,IAAAA;GAAO,GAAGQ,qBAAQ,EAAE;EAC5B,MAAM,CAACC,SAAS,EAAEC,YAAY,CAAC,GAAGC,cAAQ,CAAgB,IAAI,CAAC;AAC/D;AAEAC,EAAAA,eAAS,CAAC,MAAK;IACb,IAAIN,OAAO,IAAI,IAAI,EAAE;MACnBI,YAAY,CAAC,IAAI,CAAC;AAClB,MAAA;AACF;AAEA,IAAA,IAAIG,SAAwC;IAE5C,MAAMC,WAAW,GAAGA,MAAW;AAC7B,MAAA,MAAMC,KAAK,GAAGV,SAAS,IAAI,IAAI,GAAG,IAAIW,IAAI,CAACX,SAAS,CAAC,GAAG,IAAIW,IAAI,EAAE;AAClE,MAAA,MAAMC,GAAG,GAAG,IAAID,IAAI,CAACV,OAAO,CAAC;MAC7BI,YAAY,CAACQ,qBAAY,CAACH,KAAK,EAAEE,GAAG,EAAEV,MAAM,CAAC,CAAC;AAE9C,MAAA,MAAMY,GAAG,GAAG,IAAIH,IAAI,EAAE;AACtB,MAAA,MAAMI,MAAM,GAAGC,IAAI,CAACC,GAAG,CAACL,GAAG,CAACM,OAAO,EAAE,GAAGJ,GAAG,CAACI,OAAO,EAAE,CAAC;MACtD,IAAIH,MAAM,GAAG,KAAM,EAAE;AACnB,QAAA,MAAMI,cAAc,GAAG,CAAC,EAAE,GAAGL,GAAG,CAACM,UAAU,EAAE,IAAI,IAAI,GAAGN,GAAG,CAACO,eAAe,EAAE;AAC7Eb,QAAAA,SAAS,GAAGc,UAAU,CAACb,WAAW,EAAEU,cAAc,CAAC;AACrD,OAAC,MAAM;AACLX,QAAAA,SAAS,GAAGc,UAAU,CAACb,WAAW,EAAE,IAAI,CAAC;AAC3C;KACD;AAEDA,IAAAA,WAAW,EAAE;AAEb,IAAA,OAAO,MAAMc,YAAY,CAACf,SAAS,CAAC;GACrC,EAAE,CAACR,SAAS,EAAEC,OAAO,EAAEC,MAAM,CAAC,CAAC;AAEhC,EAAA,IAAIE,SAAS,IAAI,IAAI,IAAIA,SAAS,KAAK,EAAE,EAAE;AACzC,IAAA,OAAOoB,cAAA,CAACC,eAAS,EAAA,EAAA,CAAG;AACtB;AAEA;AACA,EAAA,MAAMC,OAAO,GAAGF,cAAM,CAAA,MAAA,EAAA;IAAAG,SAAS,EAAEnC,aAAa,CAAC;AAAEG,MAAAA;AAAO,KAAA,CAAC;AAAGiC,IAAAA,QAAA,EAAAxB;IAAiB;AAE7E;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AAEA,EAAA,OAAOsB,OAAO;AAChB;;;;;"}
|