Skip to content

Commit

Permalink
- rework the protected variables functionality
Browse files Browse the repository at this point in the history
  • Loading branch information
wizard04wsu committed Oct 30, 2019
1 parent fb6f9c8 commit 528d441
Show file tree
Hide file tree
Showing 5 changed files with 86 additions and 201 deletions.
90 changes: 16 additions & 74 deletions README.md
Original file line number Diff line number Diff line change
Expand Up @@ -40,8 +40,8 @@ let Rectangle = Class.extend({
Super();
this.width = width||0;
this.height = height||0;
Object.defineProperty(this, "area", { get:function (){ return Math.abs(this.width * this.height); }, enumerable:true, configurable:true });
Object.defineProperty(this, "whatAmI", { get:function (){ return "I am a rectangle."; }, enumerable:true, configurable:true });
this.area = function (){ return Math.abs(this.width * this.height); };
this.whatAmI = function (){ return "I am a rectangle."; };
},
returnFn:function (width, height){
return Math.abs((width||0) * (height||0));
Expand All @@ -53,8 +53,8 @@ let Square = Rectangle.extend({
constructorFn:function (Super, width){
Super(width, width);
Object.defineProperty(this, "height", { get:function (){ return this.width; }, set:function (val){ this.width = 1*val||0; }, enumerable:true, configurable:true });
let iAm = [this.whatAmI, "I am a square."].join(" ");
Object.defineProperty(this, "whatAmI", { get:function (){ return iAm; }, enumerable:true, configurable:true });
let iAm = [this.whatAmI(), "I am a square."].join(" ");
this.whatAmI = function (){ return iAm; };
},
returnFn:function (width){
return Math.pow(width||0, 2);
Expand All @@ -64,22 +64,16 @@ let Square = Rectangle.extend({
let s = new Square(3);
s.toString(); //[instance of Square]
s.area; //9
s.area(); //9
s.height = 4;
s.area; //16
s.whatAmI; //I am a rectangle. I am a square.
s.area(); //16
s.whatAmI(); //I am a rectangle. I am a square.
```

### Protected members

Additionally, descendant classes can be given protected access to items in a super-class' constructor. This is done by providing getters and setters that are inherited. Once <code>*Super*()</code> is called within the constructor, the protected properties are made available as static properties of <code>*Super*</code>. The function also gains a method that allows you to grant protected access to (or revoke access from) descendant classes.

**<samp>*Super*.defineProtectedMember(*name*[, *options*])</samp>**

Adds a getter and/or setter that will be accessible within the constructors of descendant classes. If neither is specified, the protected member is removed so that it is not accessible from any descendants of this class.

<code>*options*</code> is an object with two optional methods, <code>get</code> and <code>set</code>.
Additionally, a class can give its descendants protected access to its private variables. Once <code>*Super*()</code> is called within the constructor, the protected properties of its parent class are made available via <code>*Super*.protected</code>. This object will be available to child classes as well; any additions to or deletions of its members that are made here in the constructor will be reflected in the class' descendants.

#### Example

Expand All @@ -89,15 +83,15 @@ let Alpha = Class.extend({
constructorFn:function (Super){
Super();
let randomInstanceID = Math.random();
Super.defineProtectedMember("rando", { get:function(){return randomInstanceID} });
Super.protected.rando = randomInstanceID;
}
});
let Bravo = Alpha.extend({
className:"Bravo",
constructorFn:function (Super){
Super();
this.foo = "My ID is "+Super.rando;
this.foo = "My ID is "+Super.protected.rando;
}
});
Expand All @@ -112,82 +106,30 @@ b.foo; //My ID is ...

### Private members

A WeakMap or a symbol can be used to implement private members for the class, allowing functions defined both inside and outside of the constructor to share data. This can also be used to pass along access to the protected members.
A WeakMap or a symbol can be used to implement private members for class instances, allowing functions defined both inside and outside of the constructor to share data.

#### Example using a WeakMap

```
let Alpha = Class.extend({
constructorFn:function (Super){
Super();
let foo = "foo";
Super.defineProtectedMember("foo", { get:function(){return foo} });
}
});
let Bravo = (function (){
let Cuber = (function (){
const private = new WeakMap();
function cube(){ return Math.pow(private.get(this).val, 3); }
return Alpha.extend({
constructorFn: function (Super, myVal){
Super();
let that = this;
private.set(this, {
val: myVal,
square: function (){ return Math.pow(private.get(that).val, 2); },
protected: Super
});
this.cube = cube;
this.test = function (){ console.log(private.get(this).val, private.get(this).square(), this.cube(), private.get(this).protected.foo); };
}
});
})();
let b = new Bravo(5);
b.test() //5 25 125 "foo"
```

#### Example using a symbol

```
let Alpha = Class.extend({
constructorFn:function (Super){
Super();
let foo = "foo";
Super.defineProtectedMember("foo", { get:function(){return foo} });
}
});
let Bravo = (function (){
const private = Symbol();
function cube(){ return Math.pow(this[private].val, 3); }
return Alpha.extend({
return Class.extend({
constructorFn: function (Super, myVal){
Super();
let that = this;
this[private] = {
val: myVal,
square: function (){ return Math.pow(that[private].val, 2); },
protected: Super
};
private.set(this, { val: myVal });
this.cube = cube;
this.test = function (){ console.log(this[private].val, this[private].square(), this.cube(), this[private].protected.foo); };
}
});
})();
let b = new Bravo(5);
let c = new Cuber(5);
b.test() //5 25 125 "foo"
c.cube(); //125
```


Expand Down
105 changes: 20 additions & 85 deletions src/Class.js
Original file line number Diff line number Diff line change
Expand Up @@ -36,93 +36,28 @@
};
let _extendToString = function toString(){ return "function extend() { [custom code] }"; };

function _defineAccessor(propertyName, accessModifier, getter, setter){

let classConstructor = this.constructor,
classScope = _scopes.get(classConstructor);

if(classScope && classScope.hasOwnProperty(propertyName) && classScope[propertyName].accessModifier !== accessModifier){
throw new Error("the access modifier of an existing property cannot be modified");
}
if(!accessModifier || !xor(accessModifier & PRIVATE, accessModifier & PROTECTED)){
throw new Error("invalid access modifier");
}

if(!classScope){
classScope = {};
_scopes.set(classConstructor, classScope);
}
classScope[propertyName].accessModifier = accessModifier;
if(accessModifier & STATIC){
Object.defineProperty(classScope[propertyName], "value", {
get:getter, set:setter, enumerable:true, configurable:true
});
}
else{
classScope[propertyName].instances = new WeakMap();
Object.defineProperty(classScope[propertyName].instances[this], "value", {
get:getter, set:setter, enumerable:true, configurable:true
function _generateProtectedAccessorsForSubclass(protectedAccessors_parent = {}){
let protectedAccessors_child = {};
for(let key in protectedAccessors_parent){
Object.defineProperty(protectedAccessors_child, key, {
get: ()=>protectedAccessors_parent[key],
set: value=>(protectedAccessors_parent[key] = value),
enumerable:true, configurable:true
});
}

}
function _getAccessors(instance){

if(instance === void 0){
instance = this;
}
else if(!(instance instanceof _baseClass)){
throw new Error("argument is not an instance of Class");
}

let requestedScope = _scopes.get(instance.constructor),
returnedScope = {};

for(let key in requestedScope){
let accessModifier = requestedScope[key].accessModifier;
if( ((accessModifier & PRIVATE) && this.constructor === instance.constructor)
|| ((accessModifier & PROTECTED) && this instanceof instance.constructor) ){
if(accessModifier & STATIC){
Object.defineProperty(returnedScope, key, {
get:function (){ return requestedScope[key].value; },
set:function (val){
requestedScope[key].value = val;
return requestedScope[key].value;
},
enumerable:true, configurable:true
});
}
else{
Object.defineProperty(returnedScope, key, {
get:function (){ return requestedScope[key].instances[instance].value; },
set:function (val){
requestedScope[key].instances[instance].value = val;
return requestedScope[key].instances[instance].value;
},
enumerable:true, configurable:true
});
}
}
}

return returnedScope;

//protectedAccessors_child.foo = 'test';
return protectedAccessors_child;
}


/*** base class ***/

//the base Class constructor; it will have two static methods, 'extend' and 'noConflict'
let _baseClass = function Class(){}
let _baseClass = function Class(){};

defineProperty(_baseClass.prototype, "toString", _instanceToString, true, false, true);
defineProperty(_baseClass, "toString", _classToString, true, false, true);

defineProperty(_baseClass, "accessModifier", {}, false, false, true);
defineProperty(_baseClass.accessModifier, "PRIVATE", PRIVATE, false, false, true);
defineProperty(_baseClass.accessModifier, "PROTECTED", PROTECTED, false, false, true);
defineProperty(_baseClass.accessModifier, "STATIC", STATIC, false, false, true);


/**
* Creates a new class that inherits from the parent class.
Expand Down Expand Up @@ -162,22 +97,25 @@

defineProperty(newInstance, "constructor", newClass, true, false, true);
}

let protectedMembers,
let protectedAccessors,
superFnCalled = false;
let superFn = function Super(){

if(superFnCalled) return; //don't initialize it more than once
superFnCalled = true;

//initialize the instance using the parent class
protectedMembers = newClass.prototype.constructor.apply(newInstance, arguments) || {};
protectedAccessors = newClass.prototype.constructor.apply(newInstance, arguments) || {};

defineProperty(superFn, "getAccessors", _getAccessors.bind(newInstance), false, false, true);
defineProperty(superFn, "defineAccessor", _defineAccessor.bind(newInstance), false, false, true);
//add protected value accessors to the Super function
Object.defineProperty(superFn, "protected", {
get: ()=>protectedAccessors,
enumerable:false, configurable:false
});

}

//construct the new instance
_initializing = true;
//$constructorFn.bind(newInstance, superFn).apply(null, arguments);
Expand All @@ -192,14 +130,11 @@
//this function is the constructor of the new instance

_initializing = false;

//In case the 'Super' argument gets referenced elsewhere, remove this since it's not allowed to be used outside of the constructor anyway.
delete superFn.defineProtectedMember;
}
else{
//this function is the constructor of a super-class

return protectedMembers;
return _generateProtectedAccessorsForSubclass(protectedAccessors);
}
//else return this

Expand Down
42 changes: 25 additions & 17 deletions test/Class testing.htm
Original file line number Diff line number Diff line change
Expand Up @@ -62,11 +62,12 @@

console.group("Delta class");
Delta = Class.extend({className:"Delta",
returnFn:function (){return "foo"},
extensions:{
bar:function (){return "bar"},
baz:function (){return this.bar()}
}
constructorFn:function (Super){
Super();
this.bar = function (){return "bar"};
this.baz = function (){return this.bar()};
},
returnFn:function (){return "foo"}
});
console.dir(Delta);
console.groupEnd();
Expand All @@ -93,10 +94,10 @@

console.group("Echo class");
Echo = Class.extend({className:"Echo",
returnFn:function (){return this.foo},
extensions:{
foo:function (){return "foo"}
}
constructorFn:function (){
this.foo = function (){return "foo"};
},
returnFn:function (){return this.foo}
});
console.dir(Echo);
console.groupEnd();
Expand Down Expand Up @@ -206,7 +207,12 @@
Super();
let foo="bar";
this.bop = function (){return foo};
Super.defineProtectedMember("foo", { get:function (){return foo}, set:function(v){foo=v} }); //subclasses of Lima will have access to the 'foo' variable
Object.defineProperty(Super.protected, "foo", {
get:function (){return foo},
set:function(v){foo=v},
enumerable:true, configurable:true
}); //subclasses of Lima will have access to the 'foo' variable
Super.protected.foo = foo;
}
});
console.dir(Lima);
Expand All @@ -216,11 +222,13 @@
Mike = Lima.extend({className:"Mike",
constructorFn:function(Super){
Super();
Super.defineProtectedMember("foo"); //subclasses of Mike will not have access to the protected property
console.assert(Super.foo === "bar", Super.foo); //Mike constructor has access to the protected property
Super.foo = "baz";
console.assert(Super.foo === "baz", Super.foo); //protected property can be changed via the Mike constructor
let protected = Super.protected;
console.log('Super.protected', protected);
console.assert(protected.foo === "bar", protected.foo); //Mike constructor has access to the protected foo value
protected.foo = "baz";
console.assert(protected.foo === "baz", protected.foo); //protected foo value can be changed via the Mike constructor
console.assert(this.bop() === "baz", this.bop()); //confirms that the value in the Lima constructor's variable is what changed
delete protected.foo; //subclasses of Mike will not have access to the protected foo value
}
});
console.dir(Mike);
Expand All @@ -229,15 +237,15 @@
console.group("Mike instance");
mike1 = new Mike();
console.dir(mike1);
console.assert(mike1.foo === void 0, mike1.foo); //instance doesn't have access to the protected property
console.assert(mike1.bop() === "baz", mike1.bop()); //instance's constructor-created method does have access to the protected property
console.assert(mike1.foo === void 0, mike1.foo); //instance doesn't have access to the protected foo value
console.assert(mike1.bop() === "baz", mike1.bop()); //instance's constructor-created method does have access to the protected foo value
console.groupEnd();

console.group("November class");
November = Mike.extend({className:"November",
constructorFn:function(Super){
Super();
console.assert(Super.foo === void 0, Super.foo); //class November doesn't have access to the protected property
console.assert(Super.protected.foo === void 0, Super.protected.foo); //class November doesn't have access to the protected foo value
console.assert(this.bop() === "baz", this.bop()); //inherited function still has access
}
});
Expand Down
Loading

0 comments on commit 528d441

Please sign in to comment.