<form> - This feature is available in the latest Canary

Canary

React의 익스텐션인 <form>은 현재 React Canary(카나리)와 실험적 채널에서만 사용할 수 있습니다. React에서의 안정적인 배포를 위해 <form>내장 브라우저 HTML 컴포넌트로서만 동작합니다. 더 자세한 사항은 React 배포 채널을 확인하세요.

내장 브라우저 <form> 컴포넌트는 정보 제출을 위한 대화형 컨트롤을 만들 수 있습니다.

<form action={search}>
<input name="query" />
<button type="submit">Search</button>
</form>

레퍼런스

<form>

정보 제출을 위한 대화형 컨트롤을 생성하기 위해, 내장된 브라우저 <form> 컴포넌트를 렌더링하세요.

<form action={search}>
<input name="query" />
<button type="submit">Search</button>
</form>

아래 예시를 보세요.

Props

<form>은 모든 공통 엘리먼트 props를 지원합니다.

action: URL 혹은 함수. URL이 action을 통해 전달되면, 폼은 HTML 폼 컴포넌트처럼 행동합니다. 함수가 action을 통해 전달되면, 해당 함수는 폼 제출처럼 동작합니다. action을 통한 함수는 비동기로 동작할 수 있으며, 폼을 통해 제출된 formData를 포함한 단일 인수로 호출됩니다. action의 프로퍼티는 formAction의 속성인 <button>, <input type="submit">, 혹은 <input type="image">로 재정의될 수 있습니다.

주의

  • 함수가 action이나 formAction에 전달되면, HTTP 메서드는 method 프로퍼티의 값과 관계없이 POST로 처리됩니다.

사용법

클라이언트에서 폼 제출 처리하기

폼이 제출될 때 함수를 실행하기 위해, 폼의 action 프로퍼티에 함수를 전달하세요. formData가 함수에 인자로 전달되어, 폼에서 전달된 데이터에 접근할 수 있습니다. 이 점이 URL만 받던 기존 HTML action과의 차이점입니다.

export default function Search() {
  function search(formData) {
    const query = formData.get("query");
    alert(`You searched for '${query}'`);
  }
  return (
    <form action={search}>
      <input name="query" />
      <button type="submit">Search</button>
    </form>
  );
}

Server Action에서 폼 제출 처리하기

입력과 제출 버튼과 함께 <form>을 렌더링하세요. 폼이 제출될 때 해당 함수를 실행하기 위해 Server Action('use server'가 표시된 함수)를 폼의 action의 프로퍼티로 전달하세요.

<form action>에 Server Action을 전달하면 자바스크립트가 활성화되기 전이나 코드가 로드되기 전에 사용자가 폼을 제출할 수 있습니다. 이는 연결 상태나 기계가 느리거나 자바스크립트가 비활성화된 사용자에게 유용하고, action 프로퍼티에 URL이 전달될 때와 폼이 동작하는 방식은 비슷합니다.

<form>의 액션에 데이터를 제공하기 위해 폼 필드의 hidden을 사용할 수 있습니다. Server Action은 formData 대신 hidden이 적용된 폼 필드 데이터를 불러 올 수 있습니다.

import { updateCart } from './lib.js';

function AddToCart({productId}) {
async function addToCart(formData) {
'use server'
const productId = formData.get('productId')
await updateCart(productId)
}
return (
<form action={addToCart}>
<input type="hidden" name="productId" value={productId} />
<button type="submit">Add to Cart</button>
</form>

);
}

폼 액션에 따른 데이터를 제공하기 위해 hidden 폼 필드를 사용하는 대신에 bind를 호출해 추가 인자를 제공할 수 있습니다. 이렇게 하면 함수에 인수로 전달되는 formData 외에 새 인수((productId))가 함수에 바인딩됩니다.

import { updateCart } from './lib.js';

function AddToCart({productId}) {
async function addToCart(productId, formData) {
"use server";
await updateCart(productId)
}
const addProductToCart = addToCart.bind(null, productId);
return (
<form action={addProductToCart}>
<button type="submit">Add to Cart</button>
</form>
);
}

<form>서버컴포넌트에 의해 렌더링되고 Server Action<form>action 프로퍼티에 전달 되면, 폼은 점진적으로 향상됩니다.

폼이 제출되는 동안 pending 상태 보여주기

폼이 제출되는 동안 pending 상태를 보여주기 위해, <form>이 렌더되는 컴포넌트 안에서 useFormStatus hook을 호출해 반환된 pending 프로퍼티를 읽어낼 수 있습니다.

여기 폼이 제출되고 있음을 나타내기 위해 pending 프로퍼티를 사용해 보았습니다.

import { useFormStatus } from "react-dom";
import { submitForm } from "./actions.js";

function Submit() {
  const { pending } = useFormStatus();
  return (
    <button type="submit" disabled={pending}>
      {pending ? "Submitting..." : "Submit"}
    </button>
  );
}

function Form({ action }) {
  return (
    <form action={action}>
      <Submit />
    </form>
  );
}

export default function App() {
  return <Form action={submitForm} />;
}

useFormStatus hook에 대해 더 알고 싶다면, 참고 문서를 확인하세요.

낙관적으로 폼 데이터 업데이트하기

useOptimistic hook은 네트워크 요청과 같은 백그라운드의 작업이 끝나기 전에 사용자 인터페이스에 낙관적으로 업데이트하는 방법을 제공합니다. 폼의 맥락에서 이 기술은 앱을 더욱 반응형으로 느끼게 해줍니다. 사용자가 폼을 제출하면 인터페이스는 사용자가 기대하는 결과물로 즉시 업데이트됩니다.

예를 들어, 사용자가 폼에 메시지를 입력하고 “제출” 버튼을 클릭하면 useOptimistic hook은 “제출중…” 라벨과 함께 메시지가 서버에 보내지기 전에 리스트에 즉시 보입니다. 이러한 ‘낙관적인’ 접근 방식은 속도와 반응성이 뛰어나다는 인상을 줍니다. 그다음 폼은 실제로 백그라운드에 메시지 보내기를 시도합니다. 서버에 메시지가 잘 도착하면, “제출중…” 라벨은 사라집니다.

import { useOptimistic, useState, useRef } from "react";
import { deliverMessage } from "./actions.js";

function Thread({ messages, sendMessage }) {
  const formRef = useRef();
  async function formAction(formData) {
    addOptimisticMessage(formData.get("message"));
    formRef.current.reset();
    await sendMessage(formData);
  }
  const [optimisticMessages, addOptimisticMessage] = useOptimistic(
    messages,
    (state, newMessage) => [
      ...state,
      {
        text: newMessage,
        sending: true
      }
    ]
  );

  return (
    <>
      {optimisticMessages.map((message, index) => (
        <div key={index}>
          {message.text}
          {!!message.sending && <small> (Sending...)</small>}
        </div>
      ))}
      <form action={formAction} ref={formRef}>
        <input type="text" name="message" placeholder="Hello!" />
        <button type="submit">Send</button>
      </form>
    </>
  );
}

export default function App() {
  const [messages, setMessages] = useState([
    { text: "Hello there!", sending: false, key: 1 }
  ]);
  async function sendMessage(formData) {
    const sentMessage = await deliverMessage(formData.get("message"));
    setMessages([...messages, { text: sentMessage }]);
  }
  return <Thread messages={messages} sendMessage={sendMessage} />;
}

useOptimistic hook에 대해 더 알고 싶다면 참고 문서를 확인해 보세요.

폼 제출 에러 처리하기

<form>action 프로퍼티로 전달된 어떤 함수는 에러를 던지기도 합니다. 이런 에러를 <form>에 에러 바운더리를 감싸 해결할 수 있습니다. 만약 <form>action 프로퍼티에서 호출된 함수가 에러를 던진다면 에러 바운더리의 fallback이 보이게 됩니다.

import { ErrorBoundary } from "react-error-boundary";

export default function Search() {
  function search() {
    throw new Error("search error");
  }
  return (
    <ErrorBoundary
      fallback={<p>폼 제출 중에 에러가 발생했습니다.</p>}
    >
      <form action={search}>
        <input name="query" />
        <button type="submit">Search</button>
      </form>
    </ErrorBoundary>
  );
}

자바스크립트 없이 폼 제출 에러 보여주기

점진적 향상을 위해 자바스크립트 번들이 로드되기 전 에러 메시지를 보여주기 위해 다음 요소들이 지켜져야 합니다.

  1. <form>서버 컴포넌트에서 렌더.
  2. <form>action 프로퍼티로 전달된 함수가 Server Action.
  3. useActionState hook이 에러메시지를 보여주기 위해 사용.

useActionStateServer Action과 초기 상태 값이라는 두 개의 매개변수를 가집니다. useActionState는 상태 변수와 액션이라는 두 개의 값을 반환합니다. useActionState를 통해 반환된 액션은 폼의 action 프로퍼티에 전달될 수 있습니다. useActionState를 통해 반환된 상태 변수는 에러 메시지를 보여주는 데 사용됩니다. useActionState에 전달된 Server Action에서 반환된 값은 상태 변수를 업데이트하는 데 사용됩니다.

import { useActionState } from "react";
import { signUpNewUser } from "./api";

export default function Page() {
  async function signup(prevState, formData) {
    "use server";
    const email = formData.get("email");
    try {
      await signUpNewUser(email);
      alert(`Added "${email}"`);
    } catch (err) {
      return err.toString();
    }
  }
  const [message, signupAction] = useActionState(signup, null);
  return (
    <>
      <h1>Signup for my newsletter</h1>
      <p>Signup with the same email twice to see an error</p>
      <form action={signupAction} id="signup-form">
        <label htmlFor="email">Email: </label>
        <input name="email" id="email" placeholder="react@example.com" />
        <button>Sign up</button>
        {!!message && <p>{message}</p>}
      </form>
    </>
  );
}

useActionState문서를 통해 폼 작업에서 상태를 업데이트하는 방법에 대해 자세히 알아보세요.

다양한 제출 타입 처리하기

사용자가 누른 버튼에 따라 여러 제출 작업을 처리하도록 폼을 설계할 수 있습니다. 폼 내부의 각 버튼은 formAction 프로퍼티를 설정하여 고유한 동작 또는 동작과 연결할 수 있습니다.

사용자가 특정 버튼을 클릭하면 폼이 제출되고 해당 버튼의 속성 및 동작으로 정의된 해당 동작이 실행됩니다. 예를 들어, 폼은 기본적으로 검토를 위해 문서를 제출하지만 formAction이 설정된 별도의 버튼이 있어 문서를 초안으로 저장할 수 있습니다.

export default function Search() {
  function publish(formData) {
    const content = formData.get("content");
    const button = formData.get("button");
    alert(`'${content}' was published with the '${button}' button`);
  }

  function save(formData) {
    const content = formData.get("content");
    alert(`Your draft of '${content}' has been saved!`);
  }

  return (
    <form action={publish}>
      <textarea name="content" rows={4} cols={40} />
      <br />
      <button type="submit" name="button" value="submit">Publish</button>
      <button formAction={save}>Save draft</button>
    </form>
  );
}