Submit Button
Form submission button that automatically manages loading and disabled states based on form validation and submission status.
npx shadcn@latest add https://shuip.plvo.dev/r/tsf-submit-button.json
pnpm dlx shadcn@latest add https://shuip.plvo.dev/r/tsf-submit-button.json
bun x shadcn@latest add https://shuip.plvo.dev/r/tsf-submit-button.json
'use client';import type { VariantProps } from 'class-variance-authority';import { Loader2Icon } from 'lucide-react';import { Button, type buttonVariants } from '@/components/ui/button';import { useFormContext } from '@/components/ui/shuip/tanstack-form/form-context';type ButtonProps = React.ComponentProps<'button'> &VariantProps<typeof buttonVariants> & {asChild?: boolean;};export interface SubmitButtonProps {children?: React.ReactNode;props?: ButtonProps;}export function SubmitButton({ children = 'Submit', props }: SubmitButtonProps) {const form = useFormContext();return (<form.Subscribe selector={(state) => [state.isSubmitting, state.canSubmit]}>{([isSubmitting, canSubmit]) => (<Button type='submit' disabled={isSubmitting || !canSubmit} className='transition-all duration-300' {...props}>{isSubmitting && <Loader2Icon role='status' aria-label='Loading' className='size-4 animate-spin' />}{children}</Button>)}</form.Subscribe>);}
Submit buttons need to prevent double submissions and show loading feedback. SubmitButton reads the surrounding form via useFormContext and subscribes to its state via form.Subscribe to track isSubmitting and canSubmit, automatically disabling the button when the form is invalid or already submitting.
The component displays a Loader2Icon spinner when isSubmitting is true, providing visual feedback without manual state management. It accepts a props parameter for Button variants (outline, destructive, etc.) and all native button attributes.
Because it consumes the form via context, render it inside <form.AppForm> instead of passing the form down as a prop.
Built-in features
- Context-bound form state via
useFormContext— no prop drilling - Auto-disabled when
!canSubmitorisSubmitting - Loading spinner via
Loader2Iconduring submission - Button variants via
props.variantandprops.size
Setup
Field and form components are bound via React context. In your project, create lib/form.ts once:
// lib/form.ts
import { createFormHook } from '@tanstack/react-form';
import { fieldContext, formContext } from '@/components/ui/shuip/tanstack-form/form-context';
import { InputField } from '@/components/ui/shuip/tanstack-form/input-field';
import { SubmitButton } from '@/components/ui/shuip/tanstack-form/submit-button';
export const { useAppForm } = createFormHook({
fieldContext,
formContext,
fieldComponents: { InputField },
formComponents: { SubmitButton },
});See the form-context item for details.
Usage with validation
import { useAppForm } from '@/lib/form';
const form = useAppForm({
defaultValues: { email: '' },
onSubmit: async ({ value }) => {
await saveData(value);
},
});
<form.AppField
name='email'
validators={{
onChange: ({ value }) => !value.includes('@') ? 'Invalid' : undefined,
}}
children={(field) => (
<field.InputField label='Email' />
)}
/>
{/* Button disabled until email is valid */}
<form.AppForm>
<form.SubmitButton>Submit</form.SubmitButton>
</form.AppForm>Custom variants
<form.AppForm>
<form.SubmitButton props={{ variant: 'outline' }}>
Save Draft
</form.SubmitButton>
</form.AppForm>
<form.AppForm>
<form.SubmitButton props={{ variant: 'destructive', size: 'lg' }}>
Delete Account
</form.SubmitButton>
</form.AppForm>Examples
Default
'use client';import { createFormHook } from '@tanstack/react-form';import { fieldContext, formContext } from '@/components/ui/shuip/tanstack-form/form-context';import { InputField } from '@/components/ui/shuip/tanstack-form/input-field';import { SubmitButton } from '@/components/ui/shuip/tanstack-form/submit-button';const { useAppForm } = createFormHook({fieldContext,formContext,fieldComponents: { InputField },formComponents: { SubmitButton },});export default function TsfSubmitButtonExample() {const form = useAppForm({defaultValues: {email: '',},onSubmit: async ({ value }) => {await new Promise((resolve) => setTimeout(resolve, 2000));alert(`Subscribed: ${value.email}`);},});return (<formonSubmit={(e) => {e.preventDefault();form.handleSubmit();}}className='space-y-4 max-w-md'><form.AppFieldname='email'validators={{onChange: ({ value }) => (!value ? 'Email is required' : !value.includes('@') ? 'Invalid email' : undefined),}}children={(field) => (<field.InputFieldlabel='Email'description='Subscribe to our newsletter'props={{ type: 'email', placeholder: 'Email' }}/>)}/><form.AppForm><form.SubmitButton>Subscribe</form.SubmitButton></form.AppForm></form>);}
Props
Prop
Type