Not sure about negation, but humble
Typescript lets you use union and intersection types.
I am hooked on intersection types
in Typescript. It is like mixin heaven.
I have a WithId type I use alot that I can tack on to any Firebase record type.
If I had the motivation I could write a generic firebase getter that returns documents as a “T & WithId” (means type T with an additional id:string field)
It decreases the number of types you need to define for one. And lets you lego up various interfaces.
Yeah Typescript is great. It doesn't have a negation type per se, but its pretty easy to derive one using the never type [1]. The author of this essay actually explicitly calls out Typescript several times. Some quotes from the paper's conclusion:
> In this essay I tried to survey the multiple advantages and usages of set-theoretic types in programming.
> ...the need of set-theoretic types naturally arises when trying to fit type-systems on dynamic languages
> The development of languages such as Flow, TypeScript, and Typed Racket is the latest witness of this fact.
I called it HasId, there's a withId RxJS operator to go with it that selects from an IdRecord<T> based on an inner observable that emits an ID string.
Due to nominal typing in TS you don't need to explicitly implement HasId anywhere either, meaning you don't end up with it cluttering up every single type with an ID field (plus imports).
Nominal typing means that two types are different if they have the same structure but a different name. Structural typing is the case where they are considered the same type if they have the same structure.
You need some sort of a tag if you want to pattern match at runtime, which is the usual use case for tagged unions. Now, a tagged union like
type Option a = Some a | None # here Some and None are tags, not types themselves
could also be written as an untagged union by using types as tags:
type Some a # a wrapper type for an a
type None # a unit type
type Option a = Some a | None
but for that to be useful you still need to be able to check at runtime whether you have a Some or a None… which means that the tags are still there, just hidden by the language.
But the more typical use case for untagged unions is different: given a type such as
type StringOrInt = String | Int
you're not even supposed to be able to check whether you in fact have a String or an Int, rather the language only allows you to use the subset of operations defined for both String and Int. In other words, slightly ironically, the type StringOrInt exposes the intersection of String API and Int API.
So (untagged) Union-type of Haskell really (almost) means INTERSECTION.
But it is not an intersection in set-theoretic sense if we think of the values as elements of the sets, because it allows values of both types, but you can only use an intersection of the APIs they provide.
- Their values can be discriminated at runtime (it's depending on the implementation of union types whether that's possible. In C a union is just a structure large enough to hold all constituents)
- You can sum the same type (union(Int, Int) simply is Int)
- The tag breaks recursion. You can build recursive sum types because of it, e.g. 'a List = Empty | Cons('a, 'a List)
I am hooked on intersection types in Typescript. It is like mixin heaven.
I have a WithId type I use alot that I can tack on to any Firebase record type.
If I had the motivation I could write a generic firebase getter that returns documents as a “T & WithId” (means type T with an additional id:string field)
It decreases the number of types you need to define for one. And lets you lego up various interfaces.