@adamosuiteservices/ui 1.8.13 → 1.9.14

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.
Files changed (144) hide show
  1. package/dist/components/ui/icon/icon.d.ts +11 -0
  2. package/dist/components/ui/icon/icon.stories.d.ts +55 -0
  3. package/dist/components/ui/icon/index.d.ts +1 -0
  4. package/dist/components/ui/input-group/input-group.d.ts +1 -1
  5. package/dist/custom-layered-styles.css +1 -1
  6. package/dist/icon.cjs +6 -0
  7. package/dist/icon.js +36 -0
  8. package/dist/icons.css +1 -0
  9. package/dist/styles.css +1 -1
  10. package/docs/AI-GUIDE.md +321 -321
  11. package/docs/components/layout/sidebar.md +404 -404
  12. package/docs/components/layout/toaster.md +436 -436
  13. package/docs/components/ui/accordion-rounded.md +583 -583
  14. package/docs/components/ui/accordion.md +267 -267
  15. package/docs/components/ui/alert.md +671 -671
  16. package/docs/components/ui/avatar.md +588 -588
  17. package/docs/components/ui/badge.md +1024 -1024
  18. package/docs/components/ui/button-group.md +1002 -1002
  19. package/docs/components/ui/button.md +1078 -1078
  20. package/docs/components/ui/calendar.md +1159 -1159
  21. package/docs/components/ui/card.md +1265 -1265
  22. package/docs/components/ui/checkbox.md +292 -292
  23. package/docs/components/ui/collapsible.md +320 -320
  24. package/docs/components/ui/command.md +454 -454
  25. package/docs/components/ui/context-menu.md +540 -540
  26. package/docs/components/ui/dialog.md +628 -628
  27. package/docs/components/ui/dropdown-menu.md +731 -731
  28. package/docs/components/ui/field.md +706 -706
  29. package/docs/components/ui/hover-card.md +446 -446
  30. package/docs/components/ui/icon.md +502 -0
  31. package/docs/components/ui/input-group.md +509 -509
  32. package/docs/components/ui/input.md +362 -362
  33. package/docs/components/ui/kbd.md +434 -434
  34. package/docs/components/ui/label.md +359 -359
  35. package/docs/components/ui/pagination.md +650 -650
  36. package/docs/components/ui/popover.md +536 -536
  37. package/docs/components/ui/progress.md +182 -182
  38. package/docs/components/ui/radio-group.md +311 -311
  39. package/docs/components/ui/select.md +352 -352
  40. package/docs/components/ui/separator.md +214 -214
  41. package/docs/components/ui/sheet.md +142 -142
  42. package/docs/components/ui/skeleton.md +140 -140
  43. package/docs/components/ui/slider.md +341 -341
  44. package/docs/components/ui/spinner.md +170 -170
  45. package/docs/components/ui/switch.md +402 -402
  46. package/docs/components/ui/table.md +183 -183
  47. package/docs/components/ui/tabs-underline.md +106 -106
  48. package/docs/components/ui/tabs.md +122 -122
  49. package/docs/components/ui/textarea.md +243 -243
  50. package/docs/components/ui/toggle.md +243 -243
  51. package/docs/components/ui/tooltip.md +320 -320
  52. package/docs/components/ui/typography.md +191 -191
  53. package/package.json +6 -5
  54. package/dist/components/icons/account-balance-icon.d.ts +0 -3
  55. package/dist/components/icons/account-icon.d.ts +0 -3
  56. package/dist/components/icons/add-circle-icon.d.ts +0 -3
  57. package/dist/components/icons/alarm-icon.d.ts +0 -3
  58. package/dist/components/icons/archive-icon.d.ts +0 -3
  59. package/dist/components/icons/arrow-back-icon.d.ts +0 -3
  60. package/dist/components/icons/arrow-circle-up-icon.d.ts +0 -3
  61. package/dist/components/icons/arrow-forward-icon.d.ts +0 -3
  62. package/dist/components/icons/arrow-outward-icon.d.ts +0 -3
  63. package/dist/components/icons/article-icon.d.ts +0 -3
  64. package/dist/components/icons/attach-file-icon.d.ts +0 -3
  65. package/dist/components/icons/autorenew-icon.d.ts +0 -3
  66. package/dist/components/icons/bookmark-icon.d.ts +0 -3
  67. package/dist/components/icons/calculate-icon.d.ts +0 -3
  68. package/dist/components/icons/calendar-today-icon.d.ts +0 -3
  69. package/dist/components/icons/call-split-icon.d.ts +0 -3
  70. package/dist/components/icons/cancel-filled-icon.d.ts +0 -3
  71. package/dist/components/icons/cancel-icon.d.ts +0 -3
  72. package/dist/components/icons/check-circle-icon.d.ts +0 -3
  73. package/dist/components/icons/check-icon.d.ts +0 -3
  74. package/dist/components/icons/chevron-back-icon.d.ts +0 -3
  75. package/dist/components/icons/chevron-down-icon.d.ts +0 -3
  76. package/dist/components/icons/chevron-forward-icon.d.ts +0 -3
  77. package/dist/components/icons/chevron-up-icon.d.ts +0 -3
  78. package/dist/components/icons/clarify-icon.d.ts +0 -3
  79. package/dist/components/icons/clock-icon.d.ts +0 -3
  80. package/dist/components/icons/close-icon.d.ts +0 -3
  81. package/dist/components/icons/confirmation-number-icon.d.ts +0 -3
  82. package/dist/components/icons/contacts-icon.d.ts +0 -3
  83. package/dist/components/icons/contract-delete-icon.d.ts +0 -3
  84. package/dist/components/icons/copy-icon.d.ts +0 -3
  85. package/dist/components/icons/do-not-touch-icon.d.ts +0 -3
  86. package/dist/components/icons/download-icon.d.ts +0 -3
  87. package/dist/components/icons/dragger-icon.d.ts +0 -3
  88. package/dist/components/icons/edit-icon.d.ts +0 -3
  89. package/dist/components/icons/edit-square-icon.d.ts +0 -3
  90. package/dist/components/icons/exclamation-icon.d.ts +0 -3
  91. package/dist/components/icons/expand-circle-right-icon.d.ts +0 -3
  92. package/dist/components/icons/feature-search-icon.d.ts +0 -3
  93. package/dist/components/icons/filter-icon.d.ts +0 -3
  94. package/dist/components/icons/folder-icon.d.ts +0 -3
  95. package/dist/components/icons/folder-open-icon.d.ts +0 -3
  96. package/dist/components/icons/format-list-bulleted-icon.d.ts +0 -3
  97. package/dist/components/icons/hamburger-menu-icon.d.ts +0 -3
  98. package/dist/components/icons/help-icon.d.ts +0 -3
  99. package/dist/components/icons/hide-pass-icon.d.ts +0 -3
  100. package/dist/components/icons/home-icon.d.ts +0 -3
  101. package/dist/components/icons/id-card-icon.d.ts +0 -3
  102. package/dist/components/icons/index.d.ts +0 -88
  103. package/dist/components/icons/info-icon.d.ts +0 -3
  104. package/dist/components/icons/kid-star-icon.d.ts +0 -3
  105. package/dist/components/icons/language-icon.d.ts +0 -3
  106. package/dist/components/icons/last-page-icon.d.ts +0 -3
  107. package/dist/components/icons/layers-icon.d.ts +0 -3
  108. package/dist/components/icons/location-icon.d.ts +0 -3
  109. package/dist/components/icons/mail-icon.d.ts +0 -3
  110. package/dist/components/icons/manage-search-icon.d.ts +0 -3
  111. package/dist/components/icons/menu-icon.d.ts +0 -3
  112. package/dist/components/icons/message-icon.d.ts +0 -3
  113. package/dist/components/icons/metrics-icon.d.ts +0 -3
  114. package/dist/components/icons/mic-icon.d.ts +0 -3
  115. package/dist/components/icons/minus-icon.d.ts +0 -3
  116. package/dist/components/icons/mode-comment-icon.d.ts +0 -3
  117. package/dist/components/icons/money-icon.d.ts +0 -3
  118. package/dist/components/icons/monitoring-icon.d.ts +0 -3
  119. package/dist/components/icons/more-icon.d.ts +0 -3
  120. package/dist/components/icons/notifications-icon.d.ts +0 -3
  121. package/dist/components/icons/open-in-new-icon.d.ts +0 -3
  122. package/dist/components/icons/palette-icon.d.ts +0 -3
  123. package/dist/components/icons/password-icon.d.ts +0 -3
  124. package/dist/components/icons/pending-icon.d.ts +0 -3
  125. package/dist/components/icons/person-add-icon.d.ts +0 -3
  126. package/dist/components/icons/person-search-icon.d.ts +0 -3
  127. package/dist/components/icons/photo-icon.d.ts +0 -3
  128. package/dist/components/icons/plus-icon.d.ts +0 -3
  129. package/dist/components/icons/policy-icon.d.ts +0 -3
  130. package/dist/components/icons/publish-icon.d.ts +0 -3
  131. package/dist/components/icons/ready-icon.d.ts +0 -3
  132. package/dist/components/icons/receipt-icon.d.ts +0 -3
  133. package/dist/components/icons/receive-icon.d.ts +0 -3
  134. package/dist/components/icons/refresh-icon.d.ts +0 -3
  135. package/dist/components/icons/search-icon.d.ts +0 -3
  136. package/dist/components/icons/see-icon.d.ts +0 -3
  137. package/dist/components/icons/send-icon.d.ts +0 -3
  138. package/dist/components/icons/settings-icon.d.ts +0 -3
  139. package/dist/components/icons/shield-icon.d.ts +0 -3
  140. package/dist/components/icons/swap-horiz-icon.d.ts +0 -3
  141. package/dist/components/icons/tag-icon.d.ts +0 -3
  142. package/dist/components/icons/trash-icon.d.ts +0 -3
  143. package/dist/icons.cjs +0 -1
  144. package/dist/icons.js +0 -1507
@@ -1,243 +1,243 @@
1
- # Textarea
2
-
3
- Campo de texto multilínea con auto-resize. Wrapper nativo de `<textarea>` HTML.
4
-
5
- ## Descripción
6
-
7
- El componente `Textarea` proporciona un campo de entrada de texto multilínea.
8
-
9
- ## Importación
10
-
11
- ```typescript
12
- import { Textarea } from "@adamosuiteservices/ui/textarea";
13
- ```
14
-
15
- ## Anatomía
16
-
17
- ```tsx
18
- <Textarea placeholder="Type your message here." />
19
- ```
20
-
21
- **Componentes**: 1 (Textarea)
22
-
23
- ## Props
24
-
25
- | Prop | Tipo | Descripción |
26
- | -------------- | -------------------------- | ---------------------------- |
27
- | `placeholder` | `string` | Texto placeholder |
28
- | `value` | `string` | Valor controlado |
29
- | `defaultValue` | `string` | Valor inicial (uncontrolled) |
30
- | `onChange` | `(e: ChangeEvent) => void` | Callback al cambiar |
31
- | `disabled` | `boolean` | Deshabilita el textarea |
32
- | `readOnly` | `boolean` | Solo lectura |
33
- | `required` | `boolean` | Campo requerido |
34
- | `rows` | `number` | Número de filas |
35
- | `maxLength` | `number` | Longitud máxima |
36
- | `className` | `string` | Clases CSS adicionales |
37
-
38
- **Nota**: Acepta todas las props HTML de `<textarea>`
39
-
40
- ## Patrones de Uso
41
-
42
- ### Básico
43
-
44
- ```tsx
45
- import { Textarea } from "@adamosuiteservices/ui/textarea";
46
-
47
- <Textarea placeholder="Type your message here." />;
48
- ```
49
-
50
- ### Con Label
51
-
52
- ```tsx
53
- import { Label } from "@adamosuiteservices/ui/label";
54
-
55
- <div className="grid w-full gap-2">
56
- <Label htmlFor="message">Your message</Label>
57
- <Textarea placeholder="Type your message here." id="message" />
58
- </div>;
59
- ```
60
-
61
- ### Controlado con Contador
62
-
63
- ```tsx
64
- import { useState } from "react";
65
-
66
- function App() {
67
- const [value, setValue] = useState("");
68
-
69
- return (
70
- <div className="grid w-full gap-2">
71
- <Label htmlFor="feedback">
72
- Your feedback ({value.length}/500 characters)
73
- </Label>
74
- <Textarea
75
- id="feedback"
76
- placeholder="Share your thoughts..."
77
- value={value}
78
- onChange={(e) => setValue(e.target.value)}
79
- maxLength={500}
80
- />
81
- <div className="flex justify-between text-sm text-muted-foreground">
82
- <span>{value.length} characters</span>
83
- <span>{500 - value.length} remaining</span>
84
- </div>
85
- </div>
86
- );
87
- }
88
- ```
89
-
90
- ### Con Validación
91
-
92
- ```tsx
93
- import { Button } from "@adamosuiteservices/ui/button";
94
- import {
95
- Card,
96
- CardContent,
97
- CardFooter,
98
- CardHeader,
99
- CardTitle,
100
- } from "@adamosuiteservices/ui/card";
101
-
102
- function CommentForm() {
103
- const [comment, setComment] = useState("");
104
- const [error, setError] = useState("");
105
-
106
- const handleSubmit = () => {
107
- if (comment.trim().length < 10) {
108
- setError("Comment must be at least 10 characters long");
109
- return;
110
- }
111
- setError("");
112
- // Submit logic
113
- };
114
-
115
- return (
116
- <Card>
117
- <CardHeader>
118
- <CardTitle>Leave a Comment</CardTitle>
119
- </CardHeader>
120
- <CardContent className="space-y-2">
121
- <Textarea
122
- placeholder="What did you think?"
123
- value={comment}
124
- onChange={(e) => {
125
- setComment(e.target.value);
126
- if (error) setError("");
127
- }}
128
- aria-invalid={!!error}
129
- rows={4}
130
- />
131
- {error && <p className="text-destructive text-sm">{error}</p>}
132
- </CardContent>
133
- <CardFooter>
134
- <Button onClick={handleSubmit}>Submit Comment</Button>
135
- </CardFooter>
136
- </Card>
137
- );
138
- }
139
- ```
140
-
141
- ### Chat Input con Enter
142
-
143
- ```tsx
144
- function ChatInput() {
145
- const [message, setMessage] = useState("");
146
-
147
- const handleSend = () => {
148
- if (message.trim()) {
149
- // Send message logic
150
- setMessage("");
151
- }
152
- };
153
-
154
- const handleKeyDown = (e: React.KeyboardEvent) => {
155
- if (e.key === "Enter" && !e.shiftKey) {
156
- e.preventDefault();
157
- handleSend();
158
- }
159
- };
160
-
161
- return (
162
- <div className="relative">
163
- <Textarea
164
- placeholder="Type a message... (Enter to send, Shift+Enter for new line)"
165
- value={message}
166
- onChange={(e) => setMessage(e.target.value)}
167
- onKeyDown={handleKeyDown}
168
- rows={3}
169
- className="pr-12 resize-none"
170
- />
171
- <Button
172
- size="sm"
173
- className="absolute bottom-2 right-2"
174
- onClick={handleSend}
175
- >
176
- Send
177
- </Button>
178
- </div>
179
- );
180
- }
181
- ```
182
-
183
- ### Resize Options
184
-
185
- ```tsx
186
- // Auto-resize (default con field-sizing-content)
187
- <Textarea placeholder="Auto-resize..." className="min-h-[80px] resize-none" />
188
-
189
- // Manual resize (ambas direcciones)
190
- <Textarea placeholder="Drag corner to resize..." className="min-h-[80px] resize-both" />
191
-
192
- // Vertical resize only
193
- <Textarea placeholder="Resize vertically..." className="min-h-[80px] resize-y" />
194
-
195
- // No resize
196
- <Textarea placeholder="Fixed size..." rows={4} className="resize-none" />
197
- ```
198
-
199
- ## Casos de Uso
200
-
201
- **Comentarios**: Feedback, reviews, mensajes
202
- **Formularios**: Descripción, bio, notas
203
- **Chat**: Mensajes multilínea
204
- **Code editor**: Snippets de código
205
- **Feedback**: Formularios de opinión
206
-
207
- ## Estilos Base
208
-
209
- - **Min height**: `min-h-16` default
210
- - **Border**: `border-input` con `shadow-xs`
211
- - **Focus**: `ring-ring/50` con `ring-[3px]`
212
- - **Padding**: `px-3 py-2`
213
- - **Font**: `text-base` en móvil, `text-sm` en md+
214
- - **Field sizing**: `field-sizing-content` para auto-resize
215
- - **Invalid**: `border-destructive` con `ring-destructive/20`
216
-
217
- ## Accesibilidad
218
-
219
- - ✅ **Label**: Asociar con `htmlFor` e `id`
220
- - ✅ **ARIA**: `aria-invalid` para errores
221
- - ✅ **Placeholder**: No usar como label única
222
- - ✅ **Error messages**: Describir errores claramente
223
- - ✅ **Keyboard**: Enter, Tab, Shift+Enter funcionan nativamente
224
-
225
- ## Notas de Implementación
226
-
227
- - **HTML nativo**: Wrapper de `<textarea>` HTML nativo
228
- - **Field sizing**: `field-sizing-content` permite auto-resize basado en contenido
229
- - **No external library**: Sin dependencias de Radix UI
230
- - **Data attribute**: `data-slot="textarea"` para identificación
231
-
232
- ## Troubleshooting
233
-
234
- **No auto-resize**: Verifica `field-sizing-content` en className (Chrome 123+)
235
- **Resize manual no funciona**: Usa `resize-y`, `resize-both`, o remueve `resize-none`
236
- **Height fijo**: Usa prop `rows` para altura específica
237
- **Contador de caracteres**: Usa `value.length` y `maxLength`
238
- **Enter no funciona en chat**: Usa `onKeyDown` con `e.key === "Enter" && !e.shiftKey`
239
-
240
- ## Referencias
241
-
242
- - **MDN textarea**: <https://developer.mozilla.org/en-US/docs/Web/HTML/Element/textarea>
243
- - **shadcn/ui Textarea**: <https://ui.shadcn.com/docs/components/textarea>
1
+ # Textarea
2
+
3
+ Campo de texto multilínea con auto-resize. Wrapper nativo de `<textarea>` HTML.
4
+
5
+ ## Descripción
6
+
7
+ El componente `Textarea` proporciona un campo de entrada de texto multilínea.
8
+
9
+ ## Importación
10
+
11
+ ```typescript
12
+ import { Textarea } from "@adamosuiteservices/ui/textarea";
13
+ ```
14
+
15
+ ## Anatomía
16
+
17
+ ```tsx
18
+ <Textarea placeholder="Type your message here." />
19
+ ```
20
+
21
+ **Componentes**: 1 (Textarea)
22
+
23
+ ## Props
24
+
25
+ | Prop | Tipo | Descripción |
26
+ | -------------- | -------------------------- | ---------------------------- |
27
+ | `placeholder` | `string` | Texto placeholder |
28
+ | `value` | `string` | Valor controlado |
29
+ | `defaultValue` | `string` | Valor inicial (uncontrolled) |
30
+ | `onChange` | `(e: ChangeEvent) => void` | Callback al cambiar |
31
+ | `disabled` | `boolean` | Deshabilita el textarea |
32
+ | `readOnly` | `boolean` | Solo lectura |
33
+ | `required` | `boolean` | Campo requerido |
34
+ | `rows` | `number` | Número de filas |
35
+ | `maxLength` | `number` | Longitud máxima |
36
+ | `className` | `string` | Clases CSS adicionales |
37
+
38
+ **Nota**: Acepta todas las props HTML de `<textarea>`
39
+
40
+ ## Patrones de Uso
41
+
42
+ ### Básico
43
+
44
+ ```tsx
45
+ import { Textarea } from "@adamosuiteservices/ui/textarea";
46
+
47
+ <Textarea placeholder="Type your message here." />;
48
+ ```
49
+
50
+ ### Con Label
51
+
52
+ ```tsx
53
+ import { Label } from "@adamosuiteservices/ui/label";
54
+
55
+ <div className="grid w-full gap-2">
56
+ <Label htmlFor="message">Your message</Label>
57
+ <Textarea placeholder="Type your message here." id="message" />
58
+ </div>;
59
+ ```
60
+
61
+ ### Controlado con Contador
62
+
63
+ ```tsx
64
+ import { useState } from "react";
65
+
66
+ function App() {
67
+ const [value, setValue] = useState("");
68
+
69
+ return (
70
+ <div className="grid w-full gap-2">
71
+ <Label htmlFor="feedback">
72
+ Your feedback ({value.length}/500 characters)
73
+ </Label>
74
+ <Textarea
75
+ id="feedback"
76
+ placeholder="Share your thoughts..."
77
+ value={value}
78
+ onChange={(e) => setValue(e.target.value)}
79
+ maxLength={500}
80
+ />
81
+ <div className="flex justify-between text-sm text-muted-foreground">
82
+ <span>{value.length} characters</span>
83
+ <span>{500 - value.length} remaining</span>
84
+ </div>
85
+ </div>
86
+ );
87
+ }
88
+ ```
89
+
90
+ ### Con Validación
91
+
92
+ ```tsx
93
+ import { Button } from "@adamosuiteservices/ui/button";
94
+ import {
95
+ Card,
96
+ CardContent,
97
+ CardFooter,
98
+ CardHeader,
99
+ CardTitle,
100
+ } from "@adamosuiteservices/ui/card";
101
+
102
+ function CommentForm() {
103
+ const [comment, setComment] = useState("");
104
+ const [error, setError] = useState("");
105
+
106
+ const handleSubmit = () => {
107
+ if (comment.trim().length < 10) {
108
+ setError("Comment must be at least 10 characters long");
109
+ return;
110
+ }
111
+ setError("");
112
+ // Submit logic
113
+ };
114
+
115
+ return (
116
+ <Card>
117
+ <CardHeader>
118
+ <CardTitle>Leave a Comment</CardTitle>
119
+ </CardHeader>
120
+ <CardContent className="space-y-2">
121
+ <Textarea
122
+ placeholder="What did you think?"
123
+ value={comment}
124
+ onChange={(e) => {
125
+ setComment(e.target.value);
126
+ if (error) setError("");
127
+ }}
128
+ aria-invalid={!!error}
129
+ rows={4}
130
+ />
131
+ {error && <p className="text-destructive text-sm">{error}</p>}
132
+ </CardContent>
133
+ <CardFooter>
134
+ <Button onClick={handleSubmit}>Submit Comment</Button>
135
+ </CardFooter>
136
+ </Card>
137
+ );
138
+ }
139
+ ```
140
+
141
+ ### Chat Input con Enter
142
+
143
+ ```tsx
144
+ function ChatInput() {
145
+ const [message, setMessage] = useState("");
146
+
147
+ const handleSend = () => {
148
+ if (message.trim()) {
149
+ // Send message logic
150
+ setMessage("");
151
+ }
152
+ };
153
+
154
+ const handleKeyDown = (e: React.KeyboardEvent) => {
155
+ if (e.key === "Enter" && !e.shiftKey) {
156
+ e.preventDefault();
157
+ handleSend();
158
+ }
159
+ };
160
+
161
+ return (
162
+ <div className="relative">
163
+ <Textarea
164
+ placeholder="Type a message... (Enter to send, Shift+Enter for new line)"
165
+ value={message}
166
+ onChange={(e) => setMessage(e.target.value)}
167
+ onKeyDown={handleKeyDown}
168
+ rows={3}
169
+ className="pr-12 resize-none"
170
+ />
171
+ <Button
172
+ size="sm"
173
+ className="absolute bottom-2 right-2"
174
+ onClick={handleSend}
175
+ >
176
+ Send
177
+ </Button>
178
+ </div>
179
+ );
180
+ }
181
+ ```
182
+
183
+ ### Resize Options
184
+
185
+ ```tsx
186
+ // Auto-resize (default con field-sizing-content)
187
+ <Textarea placeholder="Auto-resize..." className="min-h-[80px] resize-none" />
188
+
189
+ // Manual resize (ambas direcciones)
190
+ <Textarea placeholder="Drag corner to resize..." className="min-h-[80px] resize-both" />
191
+
192
+ // Vertical resize only
193
+ <Textarea placeholder="Resize vertically..." className="min-h-[80px] resize-y" />
194
+
195
+ // No resize
196
+ <Textarea placeholder="Fixed size..." rows={4} className="resize-none" />
197
+ ```
198
+
199
+ ## Casos de Uso
200
+
201
+ **Comentarios**: Feedback, reviews, mensajes
202
+ **Formularios**: Descripción, bio, notas
203
+ **Chat**: Mensajes multilínea
204
+ **Code editor**: Snippets de código
205
+ **Feedback**: Formularios de opinión
206
+
207
+ ## Estilos Base
208
+
209
+ - **Min height**: `min-h-16` default
210
+ - **Border**: `border-input` con `shadow-xs`
211
+ - **Focus**: `ring-ring/50` con `ring-[3px]`
212
+ - **Padding**: `px-3 py-2`
213
+ - **Font**: `text-base` en móvil, `text-sm` en md+
214
+ - **Field sizing**: `field-sizing-content` para auto-resize
215
+ - **Invalid**: `border-destructive` con `ring-destructive/20`
216
+
217
+ ## Accesibilidad
218
+
219
+ - ✅ **Label**: Asociar con `htmlFor` e `id`
220
+ - ✅ **ARIA**: `aria-invalid` para errores
221
+ - ✅ **Placeholder**: No usar como label única
222
+ - ✅ **Error messages**: Describir errores claramente
223
+ - ✅ **Keyboard**: Enter, Tab, Shift+Enter funcionan nativamente
224
+
225
+ ## Notas de Implementación
226
+
227
+ - **HTML nativo**: Wrapper de `<textarea>` HTML nativo
228
+ - **Field sizing**: `field-sizing-content` permite auto-resize basado en contenido
229
+ - **No external library**: Sin dependencias de Radix UI
230
+ - **Data attribute**: `data-slot="textarea"` para identificación
231
+
232
+ ## Troubleshooting
233
+
234
+ **No auto-resize**: Verifica `field-sizing-content` en className (Chrome 123+)
235
+ **Resize manual no funciona**: Usa `resize-y`, `resize-both`, o remueve `resize-none`
236
+ **Height fijo**: Usa prop `rows` para altura específica
237
+ **Contador de caracteres**: Usa `value.length` y `maxLength`
238
+ **Enter no funciona en chat**: Usa `onKeyDown` con `e.key === "Enter" && !e.shiftKey`
239
+
240
+ ## Referencias
241
+
242
+ - **MDN textarea**: <https://developer.mozilla.org/en-US/docs/Web/HTML/Element/textarea>
243
+ - **shadcn/ui Textarea**: <https://ui.shadcn.com/docs/components/textarea>