Your Modules Should be Deep, Generics Will Help You with That.

Table of Contents

First, what is meaning of a deep module? the term "Deep Module" is used by John Ousterhout in his book A Philosophy of Software Design:

The best modules are those that provide powerful functionality yet have simple interfaces. I use the term deep to describe such modules.

The best modules are deep: they have a lot of functionality hidden behind a simple interface. A deep module is a good abstraction because only a small fraction of its internal complexity is visible to its users.

A good example of this is MUI's Autocomplete component, the most basic usage of it is quite simple:

<Autocomplete
  options={top100Films}
  value={film}
  onChange={setFilm}
  renderInput={(params) => <TextField {...params} label="Movie" />}
/>

You probably should create your own Autocomplete wrapper component so you don't have to specify renderInput on every call, but still, the interface is quite simple, even though the capabilities of this component are quite huge and versatile (you can give the documentation page a quick read to see what it supports).

so where does Generics help? Say you want to support multiple selection, you can use multiple prop for this:

<Autocomplete
  options={top100Films}
  value={film}
  onChange={setFilm}
  renderInput={(params) => <TextField {...params} label="Movie" />}
  multiple
/>

You will get the following type error:

Type 'string' is not assignable to type 'string[]'.

this won't be possible to do without Generics (this specific case is also possible to do with discrimination unions), the idea is, you would like to have a simple interface, while also supporting all use cases with minimum cognitive load, your interface should help its users to provide all correct parameters without them needing to remember all details about the interface, and that's what Generics will help us with.

How Generics Help You Design Simple Interfaces

To drive this home, what would this interface looks like without generics, I can think of two options:

  • Create another component, called MultiAutocomplete which only accepts array value, but now say you want to also have freeSolo so you have to create FreeSoloMultiAutocomplete.... you get the idea.
  • Don't bother with typing this and provide the value type as string | string[], and you can see how complex the interface will become.

With generics, by providing multipe={true}, it will reflect on other props, requiring the user to provide arrays for value prop, I find this interface to be a good example of advanced generic usage, simple yet powerful, we will try to re-implement this interface, bit by bit.

Re-implementing AutocompleteProps

before we start the implementation, let's look at the library's generic signature:

export interface AutocompleteProps<
  Value,
  Multiple extends boolean | undefined,
  DisableClearable extends boolean | undefined,
  FreeSolo extends boolean | undefined,
  ChipComponent extends React.ElementType = ChipTypeMap['defaultComponent'],
>

We can skip ChipComponent as we can't easily see how it will change the interface, let's go through each parameter one by one, we will also only observe their effects on options and value props for brevity.

Value

the autocomplete value can be anything, so we won't use generic constraints (extends) for the Value parameter:

interface AutocompleteProps<Value> {
  options: ReadonlyArray<Value>;
  value?: Value | null;
}

ReadonlyArray is needed so our component can accept readonly options.

Multiple

If truevalue must be an array and the menu will support multiple selections. the value of this parameter matches multiple prop:

interface AutocompleteProps<Value, Multiple extends boolean | undefined> {
  options: ReadonlyArray<Value>;
  value?: Value | null;
  multiple?: Multiple;
}

Now, if multiple={true} then value should be an array, to achieve this we need Conditional Types:

interface AutocompleteProps<Value, Multiple extends boolean | undefined> {
  options: ReadonlyArray<Value>;
  value?: Multiple extends true ? Array<Value> : Value | null;
  multiple?: Multiple;
}

DisableClearable

If true, the input can't be cleared (can't be null). similarly to Multiple conditional type:

interface AutocompleteProps<
  Value,
  Multiple extends boolean | undefined,
  DisableClearable extends boolean | undefined,
> {
  options: ReadonlyArray<Value>;
  value?: Multiple extends true
    ? Value[]
    : DisableClearable extends true
      ? NonNullable<Value>
      : Value | null;
  multiple?: Multiple;
  disableClearable?: DisableClearable;
}

FreeSolo

If true, the user input is not bounded to the provided options and the user can type in anything, thus the type will become Value | string, how can we do this conditionally? it's possible to do it this way:

FreeSolo extends true ? Value | string : Value

Integrating with the interface:

interface AutocompleteProps<
  Value,
  Multiple extends boolean | undefined,
  DisableClearable extends boolean | undefined,
  FreeSolo extends boolean | undefined,
> {
  options: ReadonlyArray<Value>;
  value?: Multiple extends true
    ? Array<FreeSolo extends true ? Value | string : Value>
    : DisableClearable extends true
      ? NonNullable<FreeSolo extends true ? Value | string : Value>
      : (FreeSolo extends true ? Value | string : Value) | null;
  multiple?: Multiple;
  disableClearable?: DisableClearable;
  freeSolo?: FreeSolo;
}

Okay this doesn't look good, we probably should take FreeSolo condition to a separated utility type, a quick refresher: Type | never is the same as Type, we can make use of this:

export type AutocompleteFreeSoloValueMapping<FreeSolo> =
 FreeSolo extends true ? string : never;

interface AutocompleteProps<
  Value,
  Multiple extends boolean | undefined,
  DisableClearable extends boolean | undefined,
  FreeSolo extends boolean | undefined,
> {
  options: ReadonlyArray<Value>;
  value?: Multiple extends true
    ? Array<Value | AutocompleteFreeSoloValueMapping<FreeSolo>>
    : DisableClearable extends true
      ? NonNullable<Value | AutocompleteFreeSoloValueMapping<FreeSolo>>
      : Value | AutocompleteFreeSoloValueMapping<FreeSolo> | null;
  multiple?: Multiple;
  disableClearable?: DisableClearable;
  freeSolo?: FreeSolo;
}

a little bit better now... okay, it still looks pretty complex (nested ternaries are always hard to read) but again, this is something that's hidden inside your component, your users won't ever see this. It's part of the "internal complexity" that deep modules hides.

Tying It all Together

export type AutocompleteFreeSoloValueMapping<FreeSolo> =
 FreeSolo extends true ? string : never;
interface AutocompleteProps<
  Value,
  Multiple extends boolean | undefined,
  DisableClearable extends boolean | undefined,
  FreeSolo extends boolean | undefined,
> {
  options: ReadonlyArray<Value>;
  value?: Multiple extends true
    ? Array<Value | AutocompleteFreeSoloValueMapping<FreeSolo>>
    : DisableClearable extends true
      ? NonNullable<Value | AutocompleteFreeSoloValueMapping<FreeSolo>>
      : Value | AutocompleteFreeSoloValueMapping<FreeSolo> | null;
  multiple?: Multiple;
  disableClearable?: DisableClearable;
  freeSolo?: FreeSolo;
}

function Autocomplete<
  Value,
  Multiple extends boolean | undefined = false,
  DisableClearable extends boolean | undefined = false,
  FreeSolo extends boolean | undefined = false,
>({}: AutocompleteProps<Value, Multiple, DisableClearable, FreeSolo>) {
  // ...
}

You can play around with the Autocomplete and see how it reflects the correct value type for each boolean prop

NOTE

You can notice that we're using false as a default value (undefined also works)

function Autocomplete<
  Value,
  Multiple extends boolean | undefined = false,
  DisableClearable extends boolean | undefined = false,
  FreeSolo extends boolean | undefined = false,
({}: AutocompleteProps<Value, Multiple, DisableClearable, FreeSolo) {
// ...
}

This is needed because the generic parameters cannot be inferred if the props that they depend on are not defined, it will only work if you pass false or undefined explicitly:

<Autocomplete multiple={false} />

This why the default values are needed.

Deep Modules and Generics

Your module can be a component, a class, or even an utility function, you should always try to hide complexity and only ask your users for the minimum amount of parameters, while still allowing advanced usage if needed.

Generics can help you reduce the cognitive load that your users may have in their attempt to provide valid parameters for advanced module usage, all without even running the code. It may seems like a waste of time just writing a type this complex, but if you're implementing something that is supposed to be used extensively and it needs to support multiple use-cases, then it may worth doing this investment, and I also claim that it can help you think more thoroughly about your module interface design (props in the case of components) and as a result design better interfaces.

If you're interested, you can see the full type implementation for AutocompleteProps in mui/material-ui Github repository.