An introduction to ES6 classes
// Base class
class Animal {
constructor({hasClaws}) {
this._hasClaws = hasClaws;
}
hasClaws() {
return this._hasClaws;
}
}
// Derived class
class Cat extends Animal {
constructor({hasClaws, purrs}) {
super({hasClaws});
this._purrs = purrs;
}
purrs() {
return this._purrs;
}
}
const chaos = new Cat({hasClaws: true, purrs: true});
chaos.hasClaws();
chaos.purrs();
chaos instanceof Cat; // true
chaos instanceof Animal; // true
new Animal(); // ReferenceError
class Animal {
prop: 6, // SyntaxError
constructor({hasClaws}) {
this._hasClaws = hasClaws;
}, // SyntaxError
; // ignored
hasClaws() { // required syntax
return this._hasClaws;
}
} // no semi-colon
typeof Animal // "function" - just like an ES5 constructor function..
Animal(); // TypeError (but you have to use new)
new Animal(); // ✅
(BUT DIFFERENT)
// anonymous
const Walrus = class Walrus {
hasTusks(){
return true;
}
};
const wilfred = new Walrus();
// named
const Walrus = class Walrus {
name(){
return Walrus.name;
}
};
const wilfred = new Walrus();
wilfred.name(); // 'Walrus'
const dynamic = 'methodName';
class Foo {
constructor(prop) {
this.prop = prop;
}
static staticMethod() {
return 'classy';
}
prototypeMethod() {
return 'prototypical';
}
[dynamic]() {
return 'snazzy';
}
}
const foo = new Foo(123);
Foo.staticMethod(); // 'classy'
foo.prototypeMethod(); // 'prototypical'
typeof foo.methodName; // 'function'
class Cat extends Animal {
constructor({hasClaws, purrs}) {
// note the change in syntax
// super() only works in the constructor
// you must call super before using this
super({hasClaws});
this._purrs = purrs;
}
hasFur() {
// here we must use super[prop]
return super.hasFur() && !this.isSphinx;
}
}
// The prototype of a subclass is the superclass:
Object.getPrototypeOf(Cat) === Animal; // true
It is an error to define a constructor in a derived class but leave it empty (not calling super).
class Cat extends Animal {
purrs = true;
#hasFur;
constructor({hasClaws, hasFur}) {
super({hasClaws});
this.#hasFur = hasFur;
}
}
https://github.com/tc39/proposal-class-fields
// loudNoises is a function. Base is a variable, not an actual class
const loudNoises = Base => class extends Base {
shout() { ··· }
roar() { ··· }
growl() { ··· }
};
class Cat extends loudNoises(Animal) {}
// This can keep going and going
class Cat extends climbsThings(movesQuickly(loudNoises(Animal))) {}
Mixins are not natively supported in ES6. There are a couple of different approaches we can use to implement them. This is the one we're going to use.