TanStack Form is a headless library, offering you complete flexibility to style it as you see fit. It's compatible with a wide range of UI libraries, including Tailwind, Material UI, Mantine, shadcn/ui, or even plain CSS.
This guide focuses on Material UI, Mantine, and shadcn/ui, but the concepts are applicable to any UI library of your choice.
Before integrating TanStack Form with a UI library, ensure the necessary dependencies are installed in your project:
Note: While you can mix and match libraries, it's generally advisable to stick with one to maintain consistency and minimize bloat.
Here's an example demonstrating the integration of TanStack Form with Mantine components:
import { TextInput, Checkbox } from '@mantine/core' import { useForm } from '@tanstack/react-form' export default function App() { const { Field, handleSubmit, state } = useForm({ defaultValues: { name: '', isChecked: false, }, onSubmit: async ({ value }) => { // Handle form submission console.log(value) }, }) return ( <> <form onSubmit={(e) => { e.preventDefault() handleSubmit() }} > <Field name="name" children={({ state, handleChange, handleBlur }) => ( <TextInput defaultValue={state.value} onChange={(e) => handleChange(e.target.value)} onBlur={handleBlur} placeholder="Enter your name" /> )} /> <Field name="isChecked" children={({ state, handleChange, handleBlur }) => ( <Checkbox onChange={(e) => handleChange(e.target.checked)} onBlur={handleBlur} checked={state.value} /> )} /> </form> <div> <pre>{JSON.stringify(state.values, null, 2)}</pre> </div> </> ) }
import { TextInput, Checkbox } from '@mantine/core' import { useForm } from '@tanstack/react-form' export default function App() { const { Field, handleSubmit, state } = useForm({ defaultValues: { name: '', isChecked: false, }, onSubmit: async ({ value }) => { // Handle form submission console.log(value) }, }) return ( <> <form onSubmit={(e) => { e.preventDefault() handleSubmit() }} > <Field name="name" children={({ state, handleChange, handleBlur }) => ( <TextInput defaultValue={state.value} onChange={(e) => handleChange(e.target.value)} onBlur={handleBlur} placeholder="Enter your name" /> )} /> <Field name="isChecked" children={({ state, handleChange, handleBlur }) => ( <Checkbox onChange={(e) => handleChange(e.target.checked)} onBlur={handleBlur} checked={state.value} /> )} /> </form> <div> <pre>{JSON.stringify(state.values, null, 2)}</pre> </div> </> ) }
The process for integrating MaterialĀ UI components is similar. Here's an example using TextField and Checkbox from MaterialĀ UI:
<Field name="name" children={({ state, handleChange, handleBlur }) => { return ( <TextField id="filled-basic" label="Filled" variant="filled" defaultValue={state.value} onChange={(e) => handleChange(e.target.value)} onBlur={handleBlur} placeholder="Enter your name" /> ); }} /> <Field name="isMuiCheckBox" children={({ state, handleChange, handleBlur }) => { return ( <MuiCheckbox onChange={(e) => handleChange(e.target.checked)} onBlur={handleBlur} checked={state.value} /> ); }} />
<Field name="name" children={({ state, handleChange, handleBlur }) => { return ( <TextField id="filled-basic" label="Filled" variant="filled" defaultValue={state.value} onChange={(e) => handleChange(e.target.value)} onBlur={handleBlur} placeholder="Enter your name" /> ); }} /> <Field name="isMuiCheckBox" children={({ state, handleChange, handleBlur }) => { return ( <MuiCheckbox onChange={(e) => handleChange(e.target.checked)} onBlur={handleBlur} checked={state.value} /> ); }} />
The process for integrating shadcn/ui components is similar. Here's an example using Input and Checkbox from shadcn/ui:
<Field name="name" children={({ state, handleChange, handleBlur }) => ( <Input value={state.value} onChange={(e) => handleChange(e.target.value)} onBlur={handleBlur} placeholder="Enter your name" /> )} /> <Field name="isChecked" children={({ state, handleChange, handleBlur }) => ( <Checkbox onCheckedChange={(checked) => handleChange(checked === true)} onBlur={handleBlur} checked={state.value} /> )} />
<Field name="name" children={({ state, handleChange, handleBlur }) => ( <Input value={state.value} onChange={(e) => handleChange(e.target.value)} onBlur={handleBlur} placeholder="Enter your name" /> )} /> <Field name="isChecked" children={({ state, handleChange, handleBlur }) => ( <Checkbox onCheckedChange={(checked) => handleChange(checked === true)} onBlur={handleBlur} checked={state.value} /> )} />
Your weekly dose of JavaScript news. Delivered every Monday to over 100,000 devs, for free.
Your weekly dose of JavaScript news. Delivered every Monday to over 100,000 devs, for free.