useActionState
는 폼 액션의 결과를 기반으로 State를 업데이트할 수 있도록 제공하는 Hook입니다.
const [state, formAction, isPending] = useActionState(fn, initialState, permalink?);
레퍼런스
useActionState(action, initialState, permalink?)
useActionState
를 컴포넌트의 최상위 레벨에서 호출하여 폼 액션이 실행될 때 업데이트되는 컴포넌트 State를 생성하세요. useActionState
는 기존의 폼 액션 함수와 초기 State를 전달받고, 폼에서 사용할 새로운 액션을 반환합니다. 또한 최신 폼 State와 액션이 대기 중인지 여부(isPending
)도 반환합니다. 이때 최신 폼 State는 useActionState
에 전달한 함수에도 함께 전달됩니다.
import { useActionState } from "react";
async function increment(previousState, formData) {
return previousState + 1;
}
function StatefulForm({}) {
const [state, formAction] = useActionState(increment, 0);
return (
<form>
{state}
<button formAction={formAction}>Increment</button>
</form>
);
}
폼 State는 폼을 마지막으로 제출했을 때 액션에서 반환되는 값입니다. 아직 폼을 제출하지 않았다면, initialState
로 설정됩니다.
서버 함수Server Function와 함께 사용하는 경우, useActionState
를 통해 하이드레이션Hydration이 끝나기 전에도 폼 제출에 대한 서버 응답을 표시할 수 있습니다.
매개변수
fn
: 폼이 제출되거나 버튼이 눌렸을 때 호출되는 함수입니다. 함수가 호출되면 첫 번째 인수로 폼의 이전 State(처음에는 전달한initialState
, 이후에는 이전 반환값)가 전달되고, 그 뒤로는 폼 액션이 일반적으로 받는 인수들이 전달됩니다.initialState
: State가 처음에 가지기를 원하는 값입니다. 이는 직렬화 가능한 값이면 무엇이든 될 수 있습니다. 이 인수는 액션이 처음 호출된 후에는 무시됩니다.- optional
permalink
: 이 폼이 수정하는 고유한 페이지 URL을 포함하는 문자열입니다. 동적 콘텐츠가 있는 페이지(예: 피드)에서 점진적 향상Progressive Enhancement과 함께 사용됩니다. 만약fn
이 서버 함수이고, 폼이 자바스크립트 번들이 로드되기 전에 제출되면, 브라우저는 현재 페이지의 URL 대신 지정된 영구 링크Permalink URL로 이동합니다. React가 State를 전달하는 방법을 알 수 있도록, 동일한 폼 컴포넌트가 대상 페이지에 렌더링되도록 해야 합니다. (동일한 액션fn
과permalink
포함.) 폼이 하이드레이션된 후, 이 매개변수는 더 이상 효과가 없습니다.
반환값
useActionState
는 다음 세 가지 값을 담은 배열을 반환합니다.
-
현재 State입니다. 첫 렌더링 시에는
initialState
와 일치하며, 액션이 실행된 후에는 액션이 반환한 값과 일치합니다. -
<form>
컴포넌트의action
Prop이나 폼 내부<button>
컴포넌트의formAction
Prop에 전달할 수 있는 새 액션입니다. -
폼 액션이 대기 중인지 여부를 알려주는
isPending
플래그입니다. -
The current state. During the first render, it will match the
initialState
you have passed. After the action is invoked, it will match the value returned by the action. -
A new action that you can pass as the
action
prop to yourform
component orformAction
prop to anybutton
component within the form. The action can also be called manually withinstartTransition
. -
The
isPending
flag that tells you whether there is a pending Transition.
주의 사항
- React 서버 컴포넌트를 지원하는 프레임워크에서
useActionState
를 사용하면, 클라이언트 자바스크립트 실행 전에도 폼과 상호작용할 수 있습니다. 만약 서버 컴포넌트를 사용하지 않는다면, 이는 단순히 컴포넌트 지역 State와 동일하게 동작합니다. useActionState
에 전달된 함수는 첫 번째 인수로 이전 또는 초기 State를 추가로 받습니다. 즉, 직접 폼 액션을 사용했을 때와 비교해 함수의 시그니처가 달라질 수 있습니다.
사용법
폼 액션에서 반환된 정보 사용하기
컴포넌트의 최상위 레벨에서 useActionState
를 호출하면, 폼이 마지막으로 제출되었을 때 액션이 반환한 값에 접근할 수 있습니다.
import { useActionState } from 'react';
import { action } from './actions.js';
function MyComponent() {
const [state, formAction] = useActionState(action, null);
// ...
return (
<form action={formAction}>
{/* ... */}
</form>
);
}
useActionState
가 반환하는 배열은 다음과 같은 요소를 갖습니다.
- 폼의 현재 State입니다. 처음에는 전달한 초기 State로 설정되며, 폼이 제출된 후에는 전달한 액션의 반환값으로 설정됩니다.
<form>
의action
Prop에 전달할 새로운 액션입니다.- 액션이 처리되는 동안 사용할 수 있는 대기Pending State입니다.
- The current state of the form, which is initially set to the initial state you provided, and after the form is submitted is set to the return value of the action you provided.
- A new action that you pass to
<form>
as itsaction
prop or call manually withinstartTransition
. - A pending state that you can utilise while your action is processing.
폼 제출 시, 액션 함수가 호출되고 그 반환값이 폼의 새로운 현재 State가 됩니다.
또한 액션 함수는 새롭게 추가된 첫 번째 인수로 폼의 현재 State를 받습니다. (처음엔 초기 State, 그 후에는 직전 반환값) 나머지 인수들은 일반 폼 액션 호출과 동일합니다.
function action(currentState, formData) {
// ...
return 'next state';
}
예시 1 of 2: 오류 표시하기
서버 함수Server Function에서 반환된 오류 메시지나 토스트 메시지를 표시하려면, 해당 액션을 useActionState
로 감싸주세요.
import { useActionState } from "react"; import { addToCart } from "./actions.js"; function AddToCartForm({itemID, itemTitle}) { const [message, formAction, isPending] = useActionState(addToCart, null); return ( <form action={formAction}> <h2>{itemTitle}</h2> <input type="hidden" name="itemID" value={itemID} /> <button type="submit">Add to Cart</button> {isPending ? "Loading..." : message} </form> ); } export default function App() { return ( <> <AddToCartForm itemID="1" itemTitle="JavaScript: The Definitive Guide" /> <AddToCartForm itemID="2" itemTitle="JavaScript: The Good Parts" /> </> ); }