DEV Community

Cover image for Canvas art with TypeScript
Chris Mytton
Chris Mytton

Posted on • Originally published at chrismytton.uk

Canvas art with TypeScript

I ported a JavaScript "random art generator" I created a few years ago over to TypeScript. This post contains some notes from the process.


I've been learning TypeScript recently. As part of that I wanted to dig out some of my old JavaScript projects and port them to TypeScript. I find porting code to a new language is a good way to learn.

I found an old project that I created in 2014 which uses JavaScript and the Canvas API to create random pieces of art which change on every page load. The code itself is fairly straightforward, so it seemed like a good candidate for porting to TypeScript.

Classes in TypeScript

TypeScript classes build on ES6 classes. The major difference is that in TypeScript you need to explicitly declare the types of properties you're expecting on the class.

class CanvasPainting {
  viewport: HTMLCanvasElement;
  ctx: CanvasRenderingContext2D;
}

If you try to use this.viewport without declaring the property TypeScript will tell you:

Property 'viewport' does not exist on type 'CanvasPainting'.

I find having these properties explicitly defined actually makes it easier to figure out what the class might be doing internally. They provide useful documentation about what kind of data you can expect to find in each property.

Index signatures

This is something I struggled with for a while. I was trying to dynamically call methods on my Shapes class using this[shape], but TypeScript was giving this warning:

Element implicitly has an 'any' type because expression of type 'string' can't be used to index type 'Shapes'.

No index signature with a parameter of type 'string' was found on type 'Shapes'.

After reading various bits of the TypeScript documentation and some Stack Overflow answers I eventually figured out that I needed to define an index signature on my shapes class. This tells TypeScript what can be used to index this class, and what the indexing might return.

class Shapes extends CanvasPainting {
  [key: string]: any;

  // ...
}

This tells TypeScript that you can index this class with a string, and it will return something. There might be a way to avoid using any here, but I haven't figured that out yet.

Dealing with null values

TypeScript knows which methods can return null and will make you deal with those cases. If you try to call a method on a value that might be null then TypeScript will give you a warning.

You can avoid this warning by explicitly checking for null before using the value.

class CanvasPainting {
  viewport: HTMLCanvasElement;
  ctx: CanvasRenderingContext2D;

  constructor(viewport: HTMLCanvasElement) {
    this.viewport = viewport;
    const ctx = viewport.getContext('2d');

    // getContext can return null, so check for that case.
    if (!ctx) {
      throw new Error("getContext('2d') failed");
    }
    this.ctx = ctx;
  }
}

Without this check for null TypeScript warns you that Type 'null' is not assignable to type 'CanvasRenderingContext2D'..

It's easy to forget to check for null when writing JavaScript. TypeScript protects you from this class of error at the compile stage.

Summary

I really like what I've seen so far with TypeScript. Working on little projects like this is a great way to get some experience with new tools.

I'd forgotten all about this fun little art project. It's nice to have given it a bit of polish and released it into the world, rather than it gathering dust on my hard drive.

See the project: www.chrismytton.uk/art.

Check out the code: chrismytton/art on GitHub.

Top comments (0)