Note: Custom types used to be referred to as “union types” in Elm. Names from other communities include tagged unions and ADTs.

Custom Types

So far we have seen a bunch of types like Bool, Int, and String. But how do we define our own?

Say we are making a chat room. Everyone needs a name, but maybe some users do not have a permanent account. They just give a name each time they show up.

We can describe this situation by defining a UserStatus type, listing all the possible variations:

type UserStatus = Regular | Visitor

The UserStatus type has two variants. Someone can be a Regular or a Visitor. So we could represent a user as a record like this:

type UserStatus
  = Regular
  | Visitor

type alias User =
  { status : UserStatus
  , name : String
  }

thomas = { status = Regular, name = "Thomas" }
kate95 = { status = Visitor, name = "kate95" }

So now we can track if someone is a Regular with an account or a Visitor who is just passing through. It is not too tough, but we can make it simpler!

Rather than creating a custom type and a type alias, we can represent all this with just a single custom type. The Regular and Visitor variants each have an associated data. In our case, the associated data is a String value:

type User
  = Regular String
  | Visitor String

thomas = Regular "Thomas"
kate95 = Visitor "kate95"

The data is attached directly to the variant, so there is no need for the record anymore.

Another benefit of this approach is that each variant can have different associated data. Say that Regular users gave their age when they signed up. There is no nice way to capture that with records, but when you define your own custom type it is no problem. Let's add some associated data to the Regular variant in an interactive example:

> type User | = Regular String Int | | Visitor String | > Regular <function> : String -> Int -> User > Visitor <function> : String -> User > Regular "Thomas" 44 Regular "Thomas" 44 : User > Visitor "kate95" Visitor "kate95" : User

>

Try defining a Regular visitor with a name and age ⬆️

We only added an age, but variants of a type can diverge quite dramatically. For example, maybe we want to add location for Regular users so we can suggest regional chat rooms. Add more associated data! Or maybe we want to have anonymous users. Add a third variant called Anonymous. Maybe we end up with:

type User
  = Regular String Int Location
  | Visitor String
  | Anonymous

No problem! Let’s see some other examples now.

Messages

In the architecture section, we saw a couple of examples of defining a Msg type. This sort of type is extremely common in Elm. In our chat room, we might define a Msg type like this:

type Msg
  = PressedEnter
  | ChangedDraft String
  | ReceivedMessage { user : User, message : String }
  | ClickedExit

We have four variants. Some variants have no associated data, others have a bunch. Notice that ReceivedMessage actually has a record as associated data. That is totally fine. Any type can be associated data! This allows you to describe interactions in your application very precisely.

Modeling

Custom types become extremely powerful when you start modeling situations very precisely. For example, if you are waiting for some data to load, you might want to model it with a custom type like this:

type Profile
  = Failure
  | Loading
  | Success { name : String, description : String }

So you can start in the Loading state and then transition to Failure or Success depending on what happens. This makes it really simple to write a view function that always shows something reasonable when data is loading.

Now we know how to create custom types, the next section will show how to use them!

Note: Custom types are the most important feature in Elm. They have a lot of depth, especially once you get in the habit of trying to model scenarios more precisely. I tried to share some of this depth in Types as Sets and Types as Bits in the appendix. I hope you find them helpful!

results matching ""

    No results matching ""