Object.extend Black Magic
Just an interesting inheritance mixin I use from time to time. Haven't seen it around on the web,
thought you might enjoy it.
function extend () {
return [].reduce.call(arguments, Object.create, this);
}
Create yourself a prototype and add in the ye olde extend magic.
Let's just use the classic example of person/say name we all know and love.
var person = {
sayName: function () {
console.log("Hello, my name is " + this.name);
return this;
},
name: "John Doe",
extend: extend
};
Now let's create another "prototype", but not too specific. Just add in some extra functionality.
Note this is formatting using Object.create's propertiesObject argument.
var ninja = {
hide: {
value: function () {
this.hidden = true;
return this;
}
}
};
Maybe another for good measure? Be sure to note here that these don't inherit from person, it's just some behavior or data we might want to add to a person.
var wizard = {
fireball: {
value: function () {
console.log("Pew Pew!");
return this;
}
}
};
And now: a god-damned ninja-wizard.
var ninjaWizard = person.extend(ninja, wizard, {
name: {
value: "Ninja Wizard"
},
wtf: {
value: function () {
console.log("RWAWRRRGGGGG!");
return this;
}
}
});
ninjaWizard.sayName().wtf().fireball().hide();
Maybe create a son of Ninja Wizard?
var shaggar = ninjaWizard.extend({
name: {
value: "Shaggar, Son of Dolff"
},
feedToGoats: {
value: function () {
if (this.hasCutoffManhood) {
//Do things
}
}
}
});
shaggar.sayName().feedToGoats();
The above works with as many prototypes as you'd like to throw at it. If you want to do a little
formatting to avoid the verbosity of Object.create, go for it. I personally don't mind it.
EDIT: For those of you interested, I've created a very small library with this functionality: metatype.js.
Written by Chris Coniglio
Related protips
7 Responses
data:image/s3,"s3://crabby-images/ef6ac/ef6acc2555671f3a527908dfe646d3f70a787e7a" alt=""
Tags: "wizards", "ninjas". Niiiice.
data:image/s3,"s3://crabby-images/57f2d/57f2d8a7fa5636322d6f1cf7ccbce6b19566ca03" alt=""
This is pretty sick, nice trick
data:image/s3,"s3://crabby-images/ab470/ab470f023c2e19e6edc466195e1ac85a3149adeb" alt=""
hi chris, that looks great. does it have any disatvangtes compared to the standard extend method (like underscore's)?
data:image/s3,"s3://crabby-images/8d25f/8d25fbee3435ca907b0230f16d54a6b16f24be1f" alt=""
Awesome!!!
data:image/s3,"s3://crabby-images/de041/de041b6c1a608117eb20b981d2752fd746e9be0e" alt=""
This should be in standard.
data:image/s3,"s3://crabby-images/3d1a6/3d1a60b0dd21b2072dacb7a240fb624fe74d43a2" alt=""
Hey @oronm, this is a very different type of functionality than underscore's extend. The extend in underscore simply copies properties from one object to another. This is creating a new object in which the prototype chain extends from the object on which the method was called, includes each argument passed, and ends with the new object.
data:image/s3,"s3://crabby-images/0e014/0e0147854ba82b18fc2aa10ec26b0403f4f3eff3" alt=""
"a god-damned ninja-wizard" code that is :)
Awesome tip!