Daniel Earwicker Chief Software Architect
FISCAL Technologies Ltd

Box 'em! - Property references for TypeScript


TYPESCRIPT MOBX REACT BOXM 2017-01-11

This concerns quite an abstract, simple building block, but it is a neat tool for use with React and MobX. In MobX there's a utility observable.box (docs). But I don't want to use that create all my properties and have to put .get() after every read access. I want to use the cool @observable decorator and just fetch my properties directly, and assign new values with =. What I need is a way to box a property. Oh, and it better be statically type checked in TypeScript.

For the overall idea, see the project page, or just look at the takeaway:

// MobX model:
class Person {
    @observable firstName: string;
    @observable lastName: string;
    @observable dateOfBirth: Date;
}

// Two-way binding in React component:
const { firstName, lastName, dateOfBirth } = box(props.person);
return (
    <div>
        <label>First name: <TextInput value={firstName} /></label>
        <label>Last name: <TextInput value={lastName} /></label>
        <label>Date of birth: <DateInput value={dateOfBirth} /></label>
    </div>
);

That is, incredibly easy, tidy, declarative and statically-typed two-way binding. I'm working on another project called bidi-mobx that will exploit this idea and provide examples (including validation).

Here I want to mention how it uses a few new-ish TypeScript features, and where it calls for possible future extensions to the language.

Way back when Roslyn first went open source and I got excited about how hackable it was, I added A new kind of managed lvalue pointer. Just like you can get a reference to a method (called a delegate in C# - and, wonder of wonders, it's always properly bound to the right this), why not a reference to a specific mutable property? It's a bundling together of two methods: a getter and a setter. The pain is having no neat syntax to construct the reference, specifying the object and the property just once each.

And as if in a demonstration of how TypeScript is way better than C#, you can now achieve the same end without having to change the compiler! It's statically type checked, although there is a limitation to this (and I'm not entirely sure how it would be fixed, but I wouldn't bet against TS fixing it soon).

It's thanks to three separate features that play together very nicely. First, keyof T is a union of string literal types, in which each string is the name of a property of T. So it's a compile-time analogue of Object.keys.

Second, if K is a string literal type and T has a property of that name, the type expression T[K] gives us the type of that property. This is a type indexer. The syntax looks just like the runtime JavaScript indexing operation where you pass a string property name to get its value. One way to think of this is that an interface is a collection (a dictionary or map) of types, keyed by names, and now we have a way to fetch an item from the collection. Maybe next we'll have conditional types, and then who knows where we might end up. Conditions need booleans to steer them, but TypeScript already has true and false as types. Hold onto your hats…

One slight wrinkle is that T[K] identifies a property, which might have the readonly modifier. Type indexing seems to ignore that modifier so that information is lost. This leads to the type hole I mentioned above. We want to target mutable properties only, but we can't tell if a property is readonly (or do much about it, even if we could tell - this is one place where conditional types would come in handy).

Finally, there are now mapped types, which is a way of declaring a set of properties with a single declaration, e.g. there's now a standard definition in the TS core library:

type Readonly<T> = {
    readonly [P in keyof T]: T[P];
}

The magic part is the [P in keyof T] which roughly means "repeat this line for each property of T", so if T has five properties then so will Readonly<T>. This is like Array#map, but operating on types at compile time instead of values at runtime. It might grow into a compile-time form of list comprehension.

My use of these features is structurally very similar to Readonly<T>, except I wrap each property in another type:

export type BoxedObject<T> = {
    readonly [K in keyof T]: BoxedValue<T[K]>;
};

And that type is:

export interface BoxedValue<P> { 
    get(): P;
    set(v: P): void;
}

So a BoxedObject<T> has properties that have the same name as the properties of T, and each property type P is "amplified" (wrapped) to become BoxedValue<P>. By the way, BoxedValue has been designed to match the shape of those boxed objects available at a low level in MobX. So if you define React components to accept BoxedValue props, they will be broadly compatible.

The only other thing is to look at the implementation. Actually there are two. Implementing BoxedValue is trivial, but BoxedObject must seem to have a property when you ask it for one. The fast way to do this (I've timed it) is to use Proxy.

const handler: ProxyHandler<any> = {
    get(target: any, key: PropertyKey) {
        return makeBoxedValue(target, key);
    }
}

How neat is that? Unfortunately no version of IE has Proxy, so there's a fallback that just generates an object that has every property of the underlying object created on it.

Time reversible events 2023-04-07
Language Smackdown: Java vs. C# 2023-03-07
Domesday '86 Reloaded (Reloaded) 2021-02-07
The Blob Lottery 2020-09-27
Abstraction is a Thing 2020-03-07
Unfortunate Bifurcations 2019-11-24
Two Cheers for SQL 2019-08-26
Factory Injection in C# 2019-07-02
Hangfire - A Tale of Several Queues 2019-05-24
How Does Auth work? 2018-11-24
From Ember to React, Part 2: Baby, Bathwater, Routing, etc. 2018-03-18
From Ember to React, Part 1: Why Not Ember? 2017-11-07
json-mobx - Like React, but for Data (Part 2) 2017-02-15
Redux in Pieces 2017-01-28
Box 'em! - Property references for TypeScript 2017-01-11
TypeScript - What's up with this? 2017-01-01
MobX - Like React, but for Data 2016-12-28
Eventless - XAML Flavoured 2016-12-24
Immuto - Epilogue 2016-12-20
Immuto - Radical Unification 2016-09-22
Immuto - Working with React (An Example) 2016-09-16
Immuto - Strongly Typed Redux Composition 2016-09-11
TypeScript - What is a class? 2016-09-11
TypeScript and runtime typing - EPISODE II 2016-09-10
TypeScript and runtime typing 2016-09-04
What's good about Redux 2016-07-24
TypeScript multicast functions 2016-03-13
Introducing doop 2016-03-08
TypeScript is not really a superset of JavaScript and that is a Good Thing 2015-07-11
A new kind of managed lvalue pointer 2014-04-27
Using pointer syntax as a shorthand for IEnumerable 2014-04-26
Adding crazily powerful operator overloading to C# 6 2014-04-23
Introducing Carota 2013-11-04
Want to comment on anything? Create an issue!