Text Fields
Text Field Textarea
여러 줄의 긴 텍스트를 입력받고 자동으로 높이를 조절하는 컴포넌트입니다.
import { TextField, TextFieldTextarea } from "seed-design/ui/text-field";
export default function MultilineTextFieldPreview() {
return (
<TextField label="라벨">
<TextFieldTextarea autoFocus />
</TextField>
);
}Installation
npx @seed-design/cli@latest add ui:text-fieldProps
TextField
Prop
Type
TextFieldTextarea
Prop
Type
Examples
State
Enabled
import { HStack } from "@seed-design/react";
import { TextField, TextFieldTextarea } from "seed-design/ui/text-field";
export default function TextFieldTextareaEnabled() {
return (
<HStack width="full" gap="x3">
<TextField label="라벨" description="설명을 써주세요">
<TextFieldTextarea placeholder="플레이스홀더" />
</TextField>
<TextField
label="라벨"
description="설명을 써주세요"
invalid
errorMessage="오류가 발생한 이유를 써주세요"
>
<TextFieldTextarea placeholder="플레이스홀더" />
</TextField>
</HStack>
);
}Disabled
import { HStack } from "@seed-design/react";
import { TextField, TextFieldTextarea } from "seed-design/ui/text-field";
export default function TextFieldTextareaDisabled() {
return (
<HStack width="full" gap="x3">
<TextField label="라벨" description="설명을 써주세요" disabled>
<TextFieldTextarea placeholder="플레이스홀더" />
</TextField>
<TextField
label="라벨"
description="설명을 써주세요"
disabled
invalid
errorMessage="오류가 발생한 이유를 써주세요"
>
<TextFieldTextarea placeholder="플레이스홀더" />
</TextField>
</HStack>
);
}Read Only
import { HStack } from "@seed-design/react";
import { TextField, TextFieldTextarea } from "seed-design/ui/text-field";
export default function TextFieldTextareaReadOnly() {
return (
<HStack width="full" gap="x3">
<TextField label="라벨" description="설명을 써주세요" readOnly>
<TextFieldTextarea placeholder="플레이스홀더" />
</TextField>
<TextField
label="라벨"
description="설명을 써주세요"
readOnly
invalid
errorMessage="오류가 발생한 이유를 써주세요"
>
<TextFieldTextarea placeholder="플레이스홀더" />
</TextField>
</HStack>
);
}Sizing
Fixed Height
import { TextField, TextFieldTextarea } from "seed-design/ui/text-field";
export default function TextFieldTextareaSpecifiedHeight() {
return (
<TextField label="라벨" description="설명을 써주세요">
<TextFieldTextarea placeholder="플레이스홀더" style={{ height: "250px" }} />
</TextField>
);
}Auto Height with Constraints
import { TextField, TextFieldTextarea } from "seed-design/ui/text-field";
export default function TextFieldTextareaConstraints() {
return (
<TextField label="라벨" description="설명을 써주세요">
<TextFieldTextarea
placeholder="플레이스홀더"
style={{ minHeight: "200px", maxHeight: "300px" }}
/>
</TextField>
);
}Customizable Parts
Indicator
indicator 또는 showRequiredIndicator prop을 사용할 수 있습니다.
import { HStack } from "@seed-design/react";
import { TextField, TextFieldTextarea } from "seed-design/ui/text-field";
export default function TextFieldTextareaIndicator() {
return (
<HStack gap="x3" width="full">
<TextField
label="선택 필드"
labelWeight="bold"
description="설명을 써주세요"
indicator="선택"
>
<TextFieldTextarea placeholder="플레이스홀더" />
</TextField>
<TextField label="필수 필드" description="설명을 써주세요" required>
<TextFieldTextarea placeholder="플레이스홀더" />
</TextField>
<TextField label="필수 필드" description="설명을 써주세요" required showRequiredIndicator>
<TextFieldTextarea placeholder="플레이스홀더" />
</TextField>
</HStack>
);
}Grapheme Count
import { TextField, TextFieldTextarea } from "seed-design/ui/text-field";
export default function TextFieldTextareaGraphemeCount() {
return (
<TextField label="라벨" description="설명을 써주세요" maxGraphemeCount={8}>
<TextFieldTextarea placeholder="플레이스홀더" />
</TextField>
);
}자소 단위로 쪼개진 value에 관한 정보를 onValueChange 콜백에서 graphemes와 slicedGraphemes로 제공합니다.
자소 분리는 unicode-segmenter를 통해 이루어집니다.
import { TextField, TextFieldTextarea } from "seed-design/ui/text-field";
import { useState } from "react";
import { Text, VStack } from "@seed-design/react";
export default function TextFieldTextareaGraphemeControlled() {
const [value, setValue] = useState("");
const [graphemes, setGraphemes] = useState<string[]>([]);
return (
<VStack gap="x4" width="full" align="center">
<TextField
label="라벨"
description="국기 이모지 🇰🇷 를 추가해보세요."
maxGraphemeCount={100}
value={value}
onValueChange={({ slicedValue, slicedGraphemes }) => {
setValue(slicedValue);
setGraphemes(slicedGraphemes);
}}
>
<TextFieldTextarea placeholder="플레이스홀더" />
</TextField>
<div
style={{
display: "grid",
gridTemplateColumns: "1fr 1fr",
rowGap: "16px",
columnGap: "32px",
padding: "16px",
}}
>
<Text textStyle="t3Medium">
<code>graphemes.length</code>: {graphemes.length}
</Text>
<Text textStyle="t3Medium">
<code>value.length</code>: {value.length}
</Text>
<Text textStyle="t3Medium">
<code>graphemes</code>: {JSON.stringify(graphemes)}
</Text>
<Text textStyle="t3Medium" style={{ whiteSpace: "pre-wrap" }}>
<code>value</code>: {value}
</Text>
</div>
</VStack>
);
}Use Cases
Form
import { HStack, VStack } from "@seed-design/react";
import { useCallback, useState, type FormEvent } from "react";
import { ActionButton } from "seed-design/ui/action-button";
import { TextField, TextFieldTextarea } from "seed-design/ui/text-field";
interface FormValues {
bio: string;
address: string;
}
type FieldErrors = Record<keyof FormValues, string | null>;
export default function TextFieldTextareaForm() {
const [formValues, setFormValues] = useState<FormValues>({
bio: "",
address: "",
});
const [fieldErrors, setFieldStates] = useState<FieldErrors>({
bio: null,
address: null,
});
const validateForm = useCallback((): boolean => {
let isValid = true;
const newFieldErrors: FieldErrors = {
bio: null,
address: null,
};
// Name validation
if (!formValues.bio) {
newFieldErrors.bio = "필수 입력 항목입니다";
isValid = false;
}
if (!formValues.address.startsWith("대한민국")) {
newFieldErrors.address = "대한민국으로 시작해주세요";
isValid = false;
}
if (!formValues.address) {
newFieldErrors.address = "필수 입력 항목입니다";
isValid = false;
}
setFieldStates(newFieldErrors);
return isValid;
}, [formValues]);
const handleSubmit = useCallback(
(event: FormEvent) => {
event.preventDefault();
if (validateForm()) {
window.alert(JSON.stringify(formValues, null, 2));
}
},
[formValues, validateForm],
);
const handleReset = useCallback((event: FormEvent) => {
event.preventDefault();
setFormValues({ bio: "", address: "" });
setFieldStates({ bio: null, address: null });
}, []);
const handleNameChange = (value: string) => {
setFormValues((prev) => ({ ...prev, bio: value }));
setFieldStates((prev) => ({ ...prev, name: null }));
};
const handleAddressChange = (value: string) => {
setFormValues((prev) => ({ ...prev, address: value }));
setFieldStates((prev) => ({ ...prev, address: null }));
};
return (
<VStack gap="x3" width="full" as="form" onSubmit={handleSubmit} onReset={handleReset}>
<HStack gap="x2">
<TextField
label="자기소개"
description="자기소개를 써주세요"
required
showRequiredIndicator
value={formValues.bio}
onValueChange={({ value }) => handleNameChange(value)}
{...(fieldErrors.bio && { invalid: true, errorMessage: fieldErrors.bio })}
>
<TextFieldTextarea placeholder="저는…" />
</TextField>
<TextField
label="주소"
description="주소를 써주세요"
maxGraphemeCount={30}
required
showRequiredIndicator
value={formValues.address}
onValueChange={({ slicedValue }) => handleAddressChange(slicedValue)}
{...(fieldErrors.address && { invalid: true, errorMessage: fieldErrors.address })}
>
<TextFieldTextarea placeholder="대한민국" />
</TextField>
</HStack>
<HStack gap="x2">
<ActionButton type="reset" variant="neutralWeak">
초기화
</ActionButton>
<ActionButton type="submit" flexGrow={1}>
제출
</ActionButton>
</HStack>
</VStack>
);
}React Hook Form
import { HStack, VStack } from "@seed-design/react";
import { useCallback, type FormEvent, type KeyboardEvent } from "react";
import { useController, useForm } from "react-hook-form";
import { ActionButton } from "seed-design/ui/action-button";
import { TextField, TextFieldTextarea } from "seed-design/ui/text-field";
interface FormValues {
bio: string;
address: string;
}
export default function TextFieldTextareaReactHookForm() {
const { handleSubmit, reset, control } = useForm<FormValues>({
defaultValues: {
bio: "",
address: "",
},
shouldFocusError: true,
});
const {
field: { onChange: bioOnChange, ...bioField },
fieldState: bioFieldState,
} = useController({
name: "bio",
control,
rules: {
required: "필수 입력 항목입니다",
},
});
const {
field: { onChange: addressOnChange, ...addressField },
fieldState: addressFieldState,
} = useController({
name: "address",
control,
rules: {
required: "필수 입력 항목입니다",
pattern: { value: /^대한민국/, message: "대한민국으로 시작해주세요" },
},
});
const onValid = useCallback((data: FormValues) => {
window.alert(JSON.stringify(data, null, 2));
}, []);
const onReset = useCallback(
(event: FormEvent) => {
event.preventDefault();
reset();
},
[reset],
);
const onMetaReturn = useCallback(
(event: KeyboardEvent) => {
if (event.key === "Enter" && (event.metaKey || event.ctrlKey)) {
event.preventDefault();
handleSubmit(onValid)();
}
},
[handleSubmit, onValid],
);
return (
<VStack gap="x3" width="full" as="form" onSubmit={handleSubmit(onValid)} onReset={onReset}>
<HStack gap="x2">
<TextField
label="자기소개"
description="자기소개를 써주세요"
invalid={bioFieldState.invalid}
errorMessage={bioFieldState.error?.message}
onValueChange={({ value }) => bioOnChange(value)}
required
showRequiredIndicator
{...bioField}
>
<TextFieldTextarea placeholder="저는…" onKeyDown={onMetaReturn} />
</TextField>
<TextField
label="주소"
description="주소를 써주세요"
invalid={addressFieldState.invalid}
errorMessage={addressFieldState.error?.message}
maxGraphemeCount={30}
onValueChange={({ slicedValue }) => addressOnChange(slicedValue)}
required
showRequiredIndicator
{...addressField}
>
<TextFieldTextarea placeholder="대한민국" onKeyDown={onMetaReturn} />
</TextField>
</HStack>
<HStack gap="x2">
<ActionButton type="reset" variant="neutralWeak">
초기화
</ActionButton>
<ActionButton type="submit" flexGrow={1}>
제출
</ActionButton>
</HStack>
</VStack>
);
}Formatting
import { useMemo, useState } from "react";
import { TextField, TextFieldTextarea } from "seed-design/ui/text-field";
export default function TextFieldTextareaFormatting() {
const [value, setValue] = useState("");
const formattedValue = useMemo(
() =>
value
.split("")
.filter((char) => char !== " ")
.join(""),
[value],
);
return (
<TextField
label="레이블"
description="공백을 입력할 수 없어요"
value={formattedValue}
onValueChange={({ value }) => setValue(value)}
>
<TextFieldTextarea placeholder="공백을 입력해보세요" />
</TextField>
);
}Slicing
import { useState } from "react";
import { TextField, TextFieldTextarea } from "seed-design/ui/text-field";
export default function TextFieldTextareaSlicing() {
const [value, setValue] = useState("");
return (
<TextField
label="라벨"
description="6글자까지 입력 가능합니다"
maxGraphemeCount={6}
value={value}
onValueChange={({ slicedValue }) => setValue(slicedValue)}
>
<TextFieldTextarea placeholder="플레이스홀더" />
</TextField>
);
}Last updated on