2013-05-29 17:03:01 +00:00
|
|
|
var util = require('util');
|
2013-05-28 05:20:30 +00:00
|
|
|
/**
|
|
|
|
*
|
|
|
|
* @param newClass
|
|
|
|
* @param baseClass
|
|
|
|
*/
|
2013-05-29 17:03:01 +00:00
|
|
|
exports.inherits = function (newClass, baseClass, options) {
|
2014-01-24 17:09:53 +00:00
|
|
|
util.inherits(newClass, baseClass);
|
2013-05-29 17:03:01 +00:00
|
|
|
|
2014-01-24 17:09:53 +00:00
|
|
|
options = options || {
|
|
|
|
staticProperties: true,
|
|
|
|
override: false
|
|
|
|
};
|
2013-05-29 17:03:01 +00:00
|
|
|
|
2014-01-24 17:09:53 +00:00
|
|
|
if (options.staticProperties) {
|
|
|
|
Object.keys(baseClass).forEach(function (classProp) {
|
|
|
|
if (classProp !== 'super_' && (!newClass.hasOwnProperty(classProp)
|
|
|
|
|| options.override)) {
|
|
|
|
var pd = Object.getOwnPropertyDescriptor(baseClass, classProp);
|
|
|
|
Object.defineProperty(newClass, classProp, pd);
|
|
|
|
}
|
|
|
|
});
|
|
|
|
}
|
2011-10-10 13:22:51 +00:00
|
|
|
};
|
|
|
|
|
2013-05-28 05:20:30 +00:00
|
|
|
/**
|
2013-10-02 05:14:21 +00:00
|
|
|
* Mix in the a class into the new class
|
|
|
|
* @param newClass The target class to receive the mixin
|
|
|
|
* @param mixinClass The class to be mixed in
|
2013-05-28 05:20:30 +00:00
|
|
|
* @param options
|
|
|
|
*/
|
2013-10-02 05:14:21 +00:00
|
|
|
exports.mixin = function (newClass, mixinClass, options) {
|
2014-01-24 17:09:53 +00:00
|
|
|
if (Array.isArray(newClass._mixins)) {
|
|
|
|
if (newClass._mixins.indexOf(mixinClass) !== -1) {
|
|
|
|
return;
|
2013-10-02 05:14:21 +00:00
|
|
|
}
|
2014-01-24 17:09:53 +00:00
|
|
|
newClass._mixins.push(mixinClass);
|
|
|
|
} else {
|
|
|
|
newClass._mixins = [mixinClass];
|
|
|
|
}
|
2013-10-02 05:14:21 +00:00
|
|
|
|
2014-01-24 17:09:53 +00:00
|
|
|
options = options || {
|
|
|
|
staticProperties: true,
|
|
|
|
instanceProperties: true,
|
|
|
|
override: false,
|
|
|
|
proxyFunctions: false
|
|
|
|
};
|
2013-05-28 05:20:30 +00:00
|
|
|
|
2014-01-24 17:09:53 +00:00
|
|
|
if (options.staticProperties === undefined) {
|
|
|
|
options.staticProperties = true;
|
|
|
|
}
|
2013-12-04 21:44:25 +00:00
|
|
|
|
2014-01-24 17:09:53 +00:00
|
|
|
if (options.instanceProperties === undefined) {
|
|
|
|
options.instanceProperties = true;
|
|
|
|
}
|
2013-12-04 21:44:25 +00:00
|
|
|
|
2014-01-24 17:09:53 +00:00
|
|
|
if (options.staticProperties) {
|
2014-02-21 01:25:07 +00:00
|
|
|
mixInto(mixinClass, newClass, options);
|
2014-01-24 17:09:53 +00:00
|
|
|
}
|
2013-05-28 05:20:30 +00:00
|
|
|
|
2014-02-21 01:25:07 +00:00
|
|
|
if (options.instanceProperties && mixinClass.prototype) {
|
|
|
|
mixInto(mixinClass.prototype, newClass.prototype, options);
|
2014-01-24 17:09:53 +00:00
|
|
|
}
|
2014-02-21 01:25:07 +00:00
|
|
|
|
2014-01-24 17:09:53 +00:00
|
|
|
return newClass;
|
2013-05-28 05:20:30 +00:00
|
|
|
};
|
|
|
|
|
2014-02-21 01:25:07 +00:00
|
|
|
function mixInto(sourceScope, targetScope, options) {
|
|
|
|
Object.keys(sourceScope).forEach(function (propertyName, options) {
|
|
|
|
var targetPropertyExists = targetScope.hasOwnProperty(propertyName);
|
|
|
|
var sourceProperty = Object.getOwnPropertyDescriptor(sourceScope, propertyName);
|
|
|
|
var targetProperty = targetPropertyExists && Object.getOwnPropertyDescriptor(targetScope, propertyName);
|
|
|
|
var sourceIsFunc = typeof sourceProperty.value === 'function';
|
2014-05-20 20:44:25 +00:00
|
|
|
var isFunc = targetPropertyExists && typeof targetProperty.value === 'function';
|
|
|
|
var isDelegate = isFunc && targetProperty.value._delegate;
|
|
|
|
var shouldOverride = options.override || !targetPropertyExists || isDelegate;
|
2014-02-21 01:25:07 +00:00
|
|
|
|
|
|
|
if (shouldOverride) {
|
2014-05-20 20:44:25 +00:00
|
|
|
if (sourceIsFunc) {
|
|
|
|
sourceProperty.value = sourceProperty.value;
|
|
|
|
}
|
|
|
|
|
2014-02-21 01:25:07 +00:00
|
|
|
Object.defineProperty(targetScope, propertyName, sourceProperty);
|
|
|
|
}
|
|
|
|
});
|
|
|
|
}
|