Create autocomplete helper which allows for arbitrary values
Sometimes you run into an issue where you want some autocomplete, but not total autocomplete. For instance, here, we have a React component, which takes in a prop, which is a size
. And this size here
can be either sm
or xs
.
But imagine that we design an API where you can either pass in sm
, xs
, or any arbitrary value. What you might think is the right thing to do is to add string
to the IconSize
union type.
This seems to make sense, except that you lose all the autocomplete on the size
prop, which isn't really what you want. The reason this happens is that TypeScript is quite clever with union types, and it tries to kind of pull in all of the types that it can.
For instance, if you specify the same type twice, like xs
, then it won't show up twice in the autocomplete.
The reason this is happening with a string, is that TypeScript is reasoning since xs
and sm
are both strings, and you're also specifying string
, that it means it should just removed xs
and sm
and only offer string. Preventing your from getting autocomplete.
The way we can get around this is we can actually use the Omit
utility type, and we can omit xs
or sm
from the union. And now TypeScript won't collapse. these three things, and you'll have autocomplete again.
We can even turn this into a type helper if we want to.
This is a way of just taking this information up in our IconSize
type and turning it into a type helper. And now we can just wrap sm
and xs
in this type helper.
And it's gonna work for us. So again, xs
, sm
, and any arbitrary value.
Transcript
Sometimes, you run into an issue where you want some autocomplete, but not total autocomplete. For instance, here, we have a React component which takes in a prop which is a size. This size here can be either SM or XS.
Imagine that we design an API where you can either pass in SM, XS, or can you pass in any arbitrary value. Here, what you might think is the right thing to do is go "or string" here. This is a union type, so it has SM or XS or string. This seems to make sense, except that you lose all the autocomplete on the size prop, which isn't really what you want.
The reason this happens is that TypeScript is quite clever with union types. It tries to pull in all of the types that it can. For instance, if you specify the same type twice, like XS, then it won't show up twice in the autocomplete. You'll just have SM and XS.
The reason that this is happening with the string is it's saying, "OK, SM is a string, and XS is also a string. What we need to do is remove these two, and then only offer string," which is why you don't get any autocomplete. The way we can get around this is we can actually use the omit utility type.
We can omit XS or SM from the union. Now, TypeScript won't collapse these three things, and you'll have the autocomplete here. You'll have SM, XS, and you'll be able to pass any other arbitrary value. We can even turn this into a type helper, if we want to.
We can say type loose autocomplete with a T, and we're going to say extend string, just because we're dealing with strings here, and we're going to say T, or we can have omit string T. This is a way of just taking this information up here and turning it into type helper.
Now, we can just wrap this with loose autocomplete, and it's going to work for us. Again, XS, SM, and any arbitrary value.
Ever wanted just a bit of autocomplete? Here, we create a TypeScript helper called LooseAutocomplete which gives us autocomplete while also allowing arbitrary values.
Picked up this tip from @GavinRayDev - worth a follow!