Ecosyste.ms: Awesome

An open API service indexing awesome lists of open source software.

Awesome Lists | Featured Topics | Projects

https://github.com/mano-08/type


https://github.com/mano-08/type

Last synced: 27 days ago
JSON representation

Awesome Lists containing this project

README

        

# Perfect form challenge

I love react-hook-form. It's a great library for managing form state, and I use it in most of my projects. One of the perks of react-hook-form when used with typescript is that it will complain if you type to register a field that is not present in the defined form fields.

As an example, typescript will catch that "bingbong" is not within the formSchema and will complain about it. This is a huge help, since you can rely on typescript to let you know if you are correctly adhering to your form or not.

```typescript
import { z } from "zod";
import { useForm, FormProvider } from "react-hook-form";
import { zodResolver } from "@hookform/resolvers/zod";

const formSchema = z.object({
firstName: z.string().min(2).max(50),
lastName: z.string().min(2).max(50),
});

export default function Home() {
type FormValues = z.infer;
const methods = useForm({
resolver: zodResolver(formSchema),
defaultValues: {
firstName: "",
lastName: "",
},
});

function onSubmit(values: z.infer) {
console.log(values);
}

return (

{/* valid name */}

{/* invalid name, caught by ts */}


);
}
```

I'd love to build out a component library with form input elements that have custom UI. This is already possible with the great work done by ShadCN. See below for an example of shadCN based component.

```typescript
// shadCN component
type FormFieldContextValue<
TFieldValues extends FieldValues = FieldValues,
TName extends FieldPath = FieldPath
> = {
name: TName;
};

const FormFieldContext = React.createContext(
{} as FormFieldContextValue
);

const FormField = <
TFieldValues extends FieldValues = FieldValues,
TName extends FieldPath = FieldPath
>({
...props
}: ControllerProps) => {
return (



);
};
```

```typescript
"use client";

import { useForm, FormProvider } from "react-hook-form";
import { zodResolver } from "@hookform/resolvers/zod";
import { z } from "zod";
import { FormField, FormItem, FormLabel, FormControl } from "@/components/Form";

const formSchema = z.object({
firstName: z.string().min(2).max(50),
lastName: z.string().min(2).max(50),
});

export default function Home() {
type FormValues = z.infer;
const methods = useForm({
resolver: zodResolver(formSchema),
defaultValues: {
firstName: "",
lastName: "",
},
});

function onSubmit(values: z.infer) {
console.log(values);
}

return (


(

First name




)}
/>
Submit


);
}
```

In this example, typescript will complain within the FormField component letting us know that `name` of _bingBong_ is not a key in the form schema. This is great, and lets us create custom components that are typesafe.

### The problem

While ShadCN is great, I don't really love how you have to pass `control={methods.control}` over and over again for every component you use in the form. React-hook-form has a hook called `useFormContext` that lets us define our form in a provider then subscribe to it's state. This means we should be able to do something like this:

```typescript
type TextInputProps = {
name: FieldPath;
};

const TextInput = ({
name,
}: TextInputProps) => {
const {
control,
formState: { errors },
} = useFormContext();

return (


(

)}
/>
{errors[name] &&

{errors[name]?.message?.toString()}

}

);
};
```

```typescript
"use client";
import TextInput from "@/components/TextInput";

// same as before
// commenting out to save space
// const formSchema = ...

export default function Home() {
// type FormValues = z.infer;
// const methods = useForm({
// ...
// });

//function onSubmit ...

return (



Submit


);
}
```

This component feels much better and it's much lighter weight to use. However, the type checking DOES NOT WORK! Feature wise, it will work. We are able to grab the control from `useFormContext` and the component works just fine. But when we use it, we can pass invalid names into the name prop and typescript does not complain about it.

The challenge here is to create a `` component that feels just as good to use as the example above AND the type-checking should work so typescript complains if you try to include a value for the name prop that does not exist within the schema. A winning solution does not include the same value as a prop across multiple instances of the component. For example, in shadCN I would have to do something like this:

```typescript
(

First name




)}
/>
(

Last name




)}
/>
```

I'm passing the same value `methods.control` again and again for each instance of the component. I would instead like to do something like:

```typescript


```

where nothing is duplicated, it is instead found within the component using Context.

I've included 5 attempts in the `components` folder above. None of them work correctly. One solution that works is `TextInput.tsx` however, you have to pass the types as a generic to the component which looks horrible and you'd end up with the same problem of having the pass that generic to each instance of the component. Somehow, the ShadCN solution is able to get around this problem, but I think it may be because you are also passing control which is giving it the context it needs.

```typescript
name="bingBong" />
```

If this is impossible, I'm sorry, but I'm not a typescript wizard. My grail component would look like `` and would be typesafe.