Claude-skill-registry formik
Manages React form state with Formik using validation, field arrays, and form context. Use when building complex forms in React, handling form submission, or integrating with Yup validation.
install
source · Clone the upstream repo
git clone https://github.com/majiayu000/claude-skill-registry
Claude Code · Install into ~/.claude/skills/
T=$(mktemp -d) && git clone --depth=1 https://github.com/majiayu000/claude-skill-registry "$T" && mkdir -p ~/.claude/skills && cp -r "$T/skills/data/formik" ~/.claude/skills/majiayu000-claude-skill-registry-formik && rm -rf "$T"
manifest:
skills/data/formik/SKILL.mdsource content
Formik
Form state management library for React with built-in validation, submission handling, and field management.
Quick Start
npm install formik yup
import { Formik, Form, Field, ErrorMessage } from 'formik'; import * as Yup from 'yup'; const schema = Yup.object({ email: Yup.string().email('Invalid email').required('Required'), password: Yup.string().min(8, 'Too short').required('Required'), }); function LoginForm() { return ( <Formik initialValues={{ email: '', password: '' }} validationSchema={schema} onSubmit={(values, { setSubmitting }) => { console.log(values); setSubmitting(false); }} > {({ isSubmitting }) => ( <Form> <Field name="email" type="email" /> <ErrorMessage name="email" component="div" /> <Field name="password" type="password" /> <ErrorMessage name="password" component="div" /> <button type="submit" disabled={isSubmitting}> Submit </button> </Form> )} </Formik> ); }
useFormik Hook
import { useFormik } from 'formik'; function Form() { const formik = useFormik({ initialValues: { email: '', password: '', }, validationSchema: schema, onSubmit: (values) => { console.log(values); }, }); return ( <form onSubmit={formik.handleSubmit}> <input name="email" type="email" onChange={formik.handleChange} onBlur={formik.handleBlur} value={formik.values.email} /> {formik.touched.email && formik.errors.email && ( <div>{formik.errors.email}</div> )} <input name="password" type="password" {...formik.getFieldProps('password')} /> {formik.touched.password && formik.errors.password && ( <div>{formik.errors.password}</div> )} <button type="submit">Submit</button> </form> ); }
Field Component
Basic Fields
<Field name="firstName" /> <Field name="email" type="email" /> <Field name="message" as="textarea" /> <Field name="color" as="select"> <option value="">Select</option> <option value="red">Red</option> <option value="blue">Blue</option> </Field>
With Custom Component
<Field name="phone"> {({ field, meta }) => ( <div> <input {...field} className={meta.error ? 'error' : ''} /> {meta.touched && meta.error && <span>{meta.error}</span>} </div> )} </Field>
Custom Input Component
const TextInput = ({ label, ...props }) => { const [field, meta] = useField(props); return ( <div> <label htmlFor={props.name}>{label}</label> <input {...field} {...props} /> {meta.touched && meta.error && <div className="error">{meta.error}</div>} </div> ); }; // Usage <TextInput label="Email" name="email" type="email" />
Form Component
<Formik initialValues={...} onSubmit={...}> <Form> {/* Form fields */} </Form> </Formik>
Or with render prop:
<Formik initialValues={...} onSubmit={...}> {(formikProps) => ( <form onSubmit={formikProps.handleSubmit}> {/* Access formik state */} </form> )} </Formik>
Validation
With Yup
import * as Yup from 'yup'; const validationSchema = Yup.object({ firstName: Yup.string() .max(15, 'Must be 15 characters or less') .required('Required'), lastName: Yup.string() .max(20, 'Must be 20 characters or less') .required('Required'), email: Yup.string() .email('Invalid email') .required('Required'), }); <Formik validationSchema={validationSchema} ... />
Custom Validate Function
<Formik validate={(values) => { const errors = {}; if (!values.email) { errors.email = 'Required'; } else if (!/^[A-Z0-9._%+-]+@[A-Z0-9.-]+\.[A-Z]{2,}$/i.test(values.email)) { errors.email = 'Invalid email'; } return errors; }} ... />
Field-Level Validation
const validateUsername = async (value) => { if (!value) return 'Required'; const taken = await checkUsername(value); if (taken) return 'Username taken'; }; <Field name="username" validate={validateUsername} />
Field Arrays
import { FieldArray } from 'formik'; <Formik initialValues={{ friends: [''], }} onSubmit={...} > {({ values }) => ( <Form> <FieldArray name="friends"> {({ push, remove }) => ( <div> {values.friends.map((friend, index) => ( <div key={index}> <Field name={`friends.${index}`} /> <button type="button" onClick={() => remove(index)}> Remove </button> </div> ))} <button type="button" onClick={() => push('')}> Add Friend </button> </div> )} </FieldArray> </Form> )} </Formik>
Array of Objects
<Formik initialValues={{ contacts: [{ name: '', email: '' }], }} ... > {({ values }) => ( <FieldArray name="contacts"> {({ push, remove }) => ( <> {values.contacts.map((contact, index) => ( <div key={index}> <Field name={`contacts.${index}.name`} placeholder="Name" /> <Field name={`contacts.${index}.email`} placeholder="Email" /> <button onClick={() => remove(index)}>Remove</button> </div> ))} <button onClick={() => push({ name: '', email: '' })}> Add Contact </button> </> )} </FieldArray> )} </Formik>
Formik Context
useFormikContext
import { useFormikContext } from 'formik'; function SubmitButton() { const { isSubmitting, isValid } = useFormikContext(); return ( <button type="submit" disabled={isSubmitting || !isValid}> {isSubmitting ? 'Submitting...' : 'Submit'} </button> ); }
Accessing Form State
function FormDebug() { const { values, errors, touched, isValid, dirty } = useFormikContext(); return ( <pre> {JSON.stringify({ values, errors, touched, isValid, dirty }, null, 2)} </pre> ); }
Form State
Formik Props
<Formik ...> {({ // Values values, // Form values initialValues, // Initial values errors, // Validation errors touched, // Touched fields // Status isSubmitting, // Submit in progress isValid, // No validation errors isValidating, // Validation in progress dirty, // Values changed from initial // Handlers handleSubmit, // Form submit handler handleChange, // Input change handler handleBlur, // Input blur handler handleReset, // Reset form // Helpers setFieldValue, // Set specific field setFieldTouched, // Mark field as touched setFieldError, // Set field error setValues, // Set all values setErrors, // Set all errors setTouched, // Set all touched setStatus, // Set form status setSubmitting, // Set submitting state resetForm, // Reset to initial validateForm, // Trigger validation validateField, // Validate specific field // Getters getFieldProps, // Get field props helper getFieldMeta, // Get field meta (error, touched) getFieldHelpers, // Get field helpers }) => ( <Form>...</Form> )} </Formik>
Submission
Basic Submit
<Formik onSubmit={(values, { setSubmitting }) => { submitToServer(values).then(() => { setSubmitting(false); }); }} ... />
Async Submit
<Formik onSubmit={async (values, { setSubmitting, setStatus, resetForm }) => { try { await api.submit(values); resetForm(); setStatus({ success: true }); } catch (error) { setStatus({ error: error.message }); } finally { setSubmitting(false); } }} ... />
With Server Errors
<Formik onSubmit={async (values, { setErrors }) => { try { await api.submit(values); } catch (error) { if (error.validationErrors) { setErrors(error.validationErrors); } } }} ... />
Enable/Disable Reinitialize
<Formik initialValues={userData} enableReinitialize={true} // Update when initialValues change ... />
TypeScript
interface FormValues { email: string; password: string; } const initialValues: FormValues = { email: '', password: '', }; <Formik<FormValues> initialValues={initialValues} onSubmit={(values: FormValues) => { console.log(values.email); }} ... />
Typed Custom Field
import { useField, FieldHookConfig } from 'formik'; interface TextInputProps extends FieldHookConfig<string> { label: string; } const TextInput: React.FC<TextInputProps> = ({ label, ...props }) => { const [field, meta] = useField(props); return ( <div> <label>{label}</label> <input {...field} {...props} /> {meta.touched && meta.error && <div>{meta.error}</div>} </div> ); };
See references/patterns.md for advanced patterns and recipes.