A utility type for widening types

cryogenicplanet

Rahul Tarak

Posted on August 18, 2021

A utility type for widening types

I love typescript, it makes my life super easy and makes my dev experience far superior. That said one gripe I've always had in complex typescript codebase is with nested types, especially using things like Pick or Omit you had to check the type definition, 4-5 levels deep to actually know what the type is, this can be quite annoying.

Recently I found a solution a great solution to this called Type Widening and it can be super easily implemented.

If you just want the basic type, here you go. That said I will break down how this works and how you can improve it below. You can also find a more advanced version of this with recursion and enum guards at bottom of the post.

export type ToPrimitive<T> = 
  : T extends string
  ? string
  : T extends number
  ? number
  : T extends boolean
  ? boolean
  : T extends (..._args: any[]) => any
  ? (..._args: Parameters<T>) => ReturnType<T>
  : T

/**
 * Expands a type so you can nicely seem the primitives
 */
export type Widen<T> = {
  [key in keyof T]: ToPrimitive<T[key]>
}
Enter fullscreen mode Exit fullscreen mode

Usage

Usage is very simple just wrap the type you want with Widen<MyType> below is an embedded example which allows you to hover over the two types to see the difference

Extending the use of Widen

Enums

We use Enum quite a bit in our codebase and unfortunately there is no T extends enum in typescript, so to hack around it you much create a Blacklist type which you can skip over

// Unfortunately this is the only way I could find right now
type EnumBlackList = DogBread | CatBread | OtherEnums 
Enter fullscreen mode Exit fullscreen mode

Recursion

We can modify ToPrimitive to make it recursive like this

type ToPrimitive<T> = T extends string
  ? string
  : T extends number
  ? number
  : T extends boolean
  ? boolean
  : T extends (..._args: any[]) => any
  ? (..._args: Parameters<T>) => ReturnType<T>:
    T extends object // Check if object and call itself
    ? { [key in keyof T]: ToPrimitive<T[key]> }
  : T;
Enter fullscreen mode Exit fullscreen mode

You can play with a demo below, which has all three different iterations setup for you

How it works

Not going into too much detail here, just a brief summary

This uses a combination of unity types from Typescript with conditional types to enable this

T extends string checks if the type is based on type string, which can be used as a conditional check. This allows us to check our base primitives like string number and boolean

To add support for functions, we need to check if the type has the structure of a function which we are doing by checking T extends (...args: any[]) => any Then we are returning the function with its correct parameters and return type using the Parameter and ReturnType utility type

Final Type

type EnumBlacklist = // your enums go here like Enum1 | Enum2 | Enum 3

type Widen<T> = { [key in keyof T]: ToPrimitive<T[key]> };

type ToPrimitive<T> = 
    T extends EnumBlackList ? T :
    T extends string
  ? string
  : T extends number
  ? number
  : T extends boolean
  ? boolean
  : T extends (..._args: any[]) => any
  ? (..._args: Parameters<T>) => ReturnType<T>
    T extends object
    ? { [key in keyof T]: ToPrimitive<T[key]> }
  : T;

type Test = Widen<MyType>
Enter fullscreen mode Exit fullscreen mode

Feel free to email me any questions or suggestion on how I can improve this type or post at rahul@modfy.video

💖 💪 🙅 🚩
cryogenicplanet
Rahul Tarak

Posted on August 18, 2021

Join Our Newsletter. No Spam, Only the good stuff.

Sign up to receive the latest update from our blog.

Related