ES6 Intro

This is a quick intro to get up-to-speed with the new class syntax introduced in ES6. We’ll also cover ES modules as they’re kind of unavoidable in frontend now.

Classes

Syntax

The ES6 class syntax is an easier way to create object-oriented structures in JavaScript. Previously you had to rely on prototypical inheritance with constructor functions.

(If you’re curious about inheritance and prototypes you could check out MDN’s guide, but it’s not strictly necessary to continue)

Here’s an example:

1
2
3
4
5
6
7
8
9
10
function Animal(species) {
this.species = species;
}

Animal.prototype.getSpecies = function() {
return this.species;
};

const tiger = new Animal('tiger');
tiger.getSpecies(); // tiger

Any new instance of Animal has access to whatever we put on its prototype.

This is kind of verbose and awkward (having to refer to the prototype all the time), so classes provide a new syntax for achieving the same result:

1
2
3
4
5
6
7
8
9
10
11
class Animal {
constructor(species) {
this.species = species;
}
getSpecies() {
return this.species;
}
}

const tiger = new Animal('tiger');
tiger.getSpecies(); // tiger

The outcome here is the same. getSpecies is still on Animal‘s prototype, but we used a nicer syntax to get there.

Constructor

The constructor method is the equivalent of the defining function in the first example. It takes whatever arguments you call the class with when you instantiate it with the new keyword.

Inheritance

Classes make it easy to inherit functionality from other classes. We can use the extends keyword to access properties on the base class:

1
2
3
4
5
6
7
8
9
10
11
12
13
class Dog extends Animal {
constructor(name) {
super('dog');
this.name = name;
}
getName() {
return this.name;
}
}

const spot = new Dog('Spot');
spot.getSpecies(); // dog
spot.getName(); // Spot

super refers to the base class you’re extending. So super('dog') is like calling new Animal('dog'), only from within our new class.

Modules

ES Modules are similar to Node’s require syntax, but are a standardised part of the JS language. They let you split your code up and share it across multiple smaller files, as well as easily import external dependencies.

Node 10 has experimental support for ES Modules (but you have to pass the --experimental-modules flag and use the .mjs file extension). The newest browsers now have initial support, but generally we need to use a tool called a bundler to parse all our imports and “bundle” them into a single file that all browsers will understand.

We’ll cover this in more detail later—for now this project is set up so the imports should work fine.

Exports

Files can have two types of exports: default and named. Generally you use default exports if there’s only one thing in a file that needs to be accessible outside of it. You’d use named exports to export multiple things (e.g. from a collection of utility functions). This is similar to how you might do module.exports = a for a single Node export, or module.exports = { a, b } to export an object with multiple properties.

This is how you default export something:

1
2
const a = 1;
export default a;

And this is how you named export something:

1
2
const a = 1;
export { a };

You can only default export a single thing, but you can have as many named exports as you like:

1
2
3
const a = 1;
const b = 2;
export { a, b };

You’ll also see this briefer version of named exports:

1
2
export const a = 1;
export const b = 2;

Imports

There are also two kinds of imports: default and named. This is how you import something that was default-exported:

1
2
import a from './a';
console.log(a); // 1;

This is how you import named-exports:

1
2
import { a } from './a';
console.log(a); // 1;

You can import as many named-exports as you like on the same line:

1
2
import { a, b } from './a';
console.log(a, b); // 1 2

Important: when you import a default-export you can call it whatever you want. You’re effectively creating a new variable and assigning the default-export to it. In contrast named-exports have to imported with the correct name—otherwise JS would have no idea which export you wanted.

Also important: unlike Node’s require ES Modules are not dynamic. This means you cannot put them inside your code and import things conditionally. You also cannot use a variable in an import path. E.g.

1
2
3
4
5
6
7
const myVar = 'a';

const a = require(`./${myVar}`); // would work
import a from `./${myVar}`; // would not

const a = myVar ? require('./a') : require('something'); // would work
const a = myVar ? import('./a') : import './something'; //would not

Exercise

  1. Clone the project and run npm i
  2. npm t to run the test watcher
  3. Rewrite Character and Hero to use ES6 classes and ES Modules
  4. You may need to change index.js too
  5. Keep all the tests passing!