Moved test-class-extend into suite as Class/{Extend,InstanceSafety}Test
More refactoring to come for ExtendTest at some pointperfodd
parent
ae367964a3
commit
2b5bcaf02d
|
@ -0,0 +1,431 @@
|
||||||
|
/**
|
||||||
|
* Tests class module extend() method
|
||||||
|
*
|
||||||
|
* Copyright (C) 2010, 2011, 2013 Mike Gerwitz
|
||||||
|
*
|
||||||
|
* This file is part of GNU ease.js.
|
||||||
|
*
|
||||||
|
* ease.js is free software: you can redistribute it and/or modify
|
||||||
|
* it under the terms of the GNU General Public License as published by
|
||||||
|
* the Free Software Foundation, either version 3 of the License, or
|
||||||
|
* (at your option) any later version.
|
||||||
|
*
|
||||||
|
* This program is distributed in the hope that it will be useful,
|
||||||
|
* but WITHOUT ANY WARRANTY; without even the implied warranty of
|
||||||
|
* MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the
|
||||||
|
* GNU General Public License for more details.
|
||||||
|
*
|
||||||
|
* You should have received a copy of the GNU General Public License
|
||||||
|
* along with this program. If not, see <http://www.gnu.org/licenses/>.
|
||||||
|
*
|
||||||
|
* Note that these tests all use the `new' keyword for instantiating
|
||||||
|
* classes, even though it is not required with ease.js; this is both for
|
||||||
|
* historical reasons (when `new' was required during early development) and
|
||||||
|
* because we are not testing (and do want to depend upon) that feature.
|
||||||
|
*/
|
||||||
|
|
||||||
|
require( 'common' ).testCase(
|
||||||
|
{
|
||||||
|
caseSetUp: function()
|
||||||
|
{
|
||||||
|
this.test_props = {
|
||||||
|
one: 1,
|
||||||
|
two: 2,
|
||||||
|
};
|
||||||
|
|
||||||
|
this.Sut = this.require( 'class' );
|
||||||
|
|
||||||
|
// there are two different means of extending; we want to test them
|
||||||
|
// both (this will be denoted Foo)
|
||||||
|
this.classes = [
|
||||||
|
this.Sut.extend( this.test_props ),
|
||||||
|
this.Sut( this.test_props ),
|
||||||
|
];
|
||||||
|
},
|
||||||
|
|
||||||
|
|
||||||
|
/**
|
||||||
|
* All classes can be easily extended via an extend method, although it
|
||||||
|
* is not necessarily recommended to be used directly, as you must
|
||||||
|
* ensure that the object is an ease.js class and the resulting class
|
||||||
|
* will be anonymous.
|
||||||
|
*/
|
||||||
|
'@each(classes) Created class contains extend method': function( C )
|
||||||
|
{
|
||||||
|
this.assertOk( typeof C.extend === 'function' );
|
||||||
|
},
|
||||||
|
|
||||||
|
|
||||||
|
/**
|
||||||
|
* It would make sense that a subtype returned is an object, since it
|
||||||
|
* cannot be a class if it isn't.
|
||||||
|
*/
|
||||||
|
'@each(classes) Subtype is returned as an object': function( C )
|
||||||
|
{
|
||||||
|
this.assertOk( C.extend() instanceof Object );
|
||||||
|
},
|
||||||
|
|
||||||
|
|
||||||
|
/**
|
||||||
|
* Subtypes should inherit all properties of the supertype into their
|
||||||
|
* prototype chain.
|
||||||
|
*/
|
||||||
|
'@each(classes) Subtype inherits parent properties': function( C )
|
||||||
|
{
|
||||||
|
var SubFoo = C.extend();
|
||||||
|
|
||||||
|
for ( var prop in this.test_props )
|
||||||
|
{
|
||||||
|
this.assertEqual(
|
||||||
|
this.test_props[ prop ],
|
||||||
|
SubFoo.prototype[ prop ],
|
||||||
|
"Missing property: " + prop
|
||||||
|
);
|
||||||
|
}
|
||||||
|
},
|
||||||
|
|
||||||
|
|
||||||
|
/**
|
||||||
|
* A subtype should obvious contain the properties that were a part of
|
||||||
|
* its definition.
|
||||||
|
*/
|
||||||
|
'@each(classes) Subtype contains its own properties': function( C )
|
||||||
|
{
|
||||||
|
var sub_props = {
|
||||||
|
three: 3,
|
||||||
|
four: 4,
|
||||||
|
};
|
||||||
|
|
||||||
|
var sub_foo = new C.extend( sub_props )();
|
||||||
|
|
||||||
|
// and ensure that the subtype's properties were included
|
||||||
|
for ( var prop in sub_props )
|
||||||
|
{
|
||||||
|
this.assertEqual(
|
||||||
|
sub_props[ prop ],
|
||||||
|
sub_foo[ prop ],
|
||||||
|
"Missing property: " + prop
|
||||||
|
);
|
||||||
|
}
|
||||||
|
},
|
||||||
|
|
||||||
|
|
||||||
|
/**
|
||||||
|
* In addition to the core functions provided by ease.js for checking
|
||||||
|
* instances, we try to ease into the protype model the best we can in
|
||||||
|
* order to work with other prototypes; therefore, instances should be
|
||||||
|
* recognized as instances of their parent classes even by the
|
||||||
|
* ECMAScript `instanceof' operator.
|
||||||
|
*/
|
||||||
|
'@each(classes) Subtypes are ECMAScript instances of their supertypes':
|
||||||
|
function( C )
|
||||||
|
{
|
||||||
|
this.assertOk( C.extend()() instanceof C );
|
||||||
|
},
|
||||||
|
|
||||||
|
|
||||||
|
/**
|
||||||
|
* Even though this can be checked using the instanceof operator,
|
||||||
|
* ease.js has a more complex type system (e.g. supporting of
|
||||||
|
* interfaces) and so we want to provide a consistent alternative.
|
||||||
|
*/
|
||||||
|
'@each(classes) Subtypes are easejs instances of their supertypes':
|
||||||
|
function( C )
|
||||||
|
{
|
||||||
|
var SubFoo = C.extend(),
|
||||||
|
sub_instance = new SubFoo();
|
||||||
|
|
||||||
|
this.assertOk( sub_instance.isInstanceOf( SubFoo ) );
|
||||||
|
},
|
||||||
|
|
||||||
|
|
||||||
|
/*
|
||||||
|
* Foo
|
||||||
|
* |
|
||||||
|
* SubFoo
|
||||||
|
* / \
|
||||||
|
* SubSubFoo SubSubFoo2
|
||||||
|
*
|
||||||
|
/
|
||||||
|
|
||||||
|
/**
|
||||||
|
* Objects should be considered instances of any classes that their
|
||||||
|
* instantiating class inherits from, since they inherit their API and
|
||||||
|
* are interchangable, provided that only the common subset of the API
|
||||||
|
* is used.
|
||||||
|
*/
|
||||||
|
'@each(classes) Objects are instances of their super-supertypes':
|
||||||
|
function( C )
|
||||||
|
{
|
||||||
|
var sub_sub_instance = new ( C.extend().extend() )();
|
||||||
|
|
||||||
|
this.assertOk(
|
||||||
|
( ( sub_sub_instance instanceof C )
|
||||||
|
&& sub_sub_instance.isInstanceOf( C )
|
||||||
|
)
|
||||||
|
);
|
||||||
|
},
|
||||||
|
|
||||||
|
|
||||||
|
/**
|
||||||
|
* It would not make sense that an object is considered to be an
|
||||||
|
* instance of any possible subtypes---that is, if C inherits B, then an
|
||||||
|
* instance of B is not of type C; C could introduce an incompatible
|
||||||
|
* interface.
|
||||||
|
*/
|
||||||
|
'@each(classes) Objects are not instances of subtypes': function( C )
|
||||||
|
{
|
||||||
|
var SubFoo = C.extend(),
|
||||||
|
SubSubFoo = SubFoo.extend(),
|
||||||
|
sub_inst = new SubFoo();
|
||||||
|
|
||||||
|
this.assertOk(
|
||||||
|
( !( sub_inst instanceof SubSubFoo )
|
||||||
|
&& !( sub_inst.isInstanceOf( SubSubFoo ) )
|
||||||
|
)
|
||||||
|
);
|
||||||
|
},
|
||||||
|
|
||||||
|
|
||||||
|
/**
|
||||||
|
* Two classes that inherit from a common parent are not compatible, as
|
||||||
|
* they can introduce their own distinct interfaces.
|
||||||
|
*/
|
||||||
|
'@each(classes) Objects are not instances of sibling types':
|
||||||
|
function( C )
|
||||||
|
{
|
||||||
|
var SubFoo = C.extend(),
|
||||||
|
SubSubFoo = SubFoo.extend(),
|
||||||
|
SubSubFoo2 = SubFoo.extend(),
|
||||||
|
|
||||||
|
sub_sub2_inst = new SubSubFoo2();
|
||||||
|
|
||||||
|
this.assertOk(
|
||||||
|
( !( sub_sub2_inst instanceof SubSubFoo )
|
||||||
|
&& !( sub_sub2_inst.isInstanceOf( SubSubFoo ) )
|
||||||
|
)
|
||||||
|
);
|
||||||
|
},
|
||||||
|
|
||||||
|
|
||||||
|
/**
|
||||||
|
* We support extending existing prototypes (that is, inherit from
|
||||||
|
* constructors that were not created using ease.js).
|
||||||
|
*/
|
||||||
|
'Constructor prototype is copied to subclass': function()
|
||||||
|
{
|
||||||
|
var Ctor = function() {};
|
||||||
|
Ctor.prototype = { foo: {} };
|
||||||
|
|
||||||
|
this.assertStrictEqual(
|
||||||
|
this.Sut.extend( Ctor, {} ).prototype.foo,
|
||||||
|
Ctor.prototype.foo
|
||||||
|
);
|
||||||
|
},
|
||||||
|
|
||||||
|
|
||||||
|
/**
|
||||||
|
* This should go without saying---we're aiming for consistency here and
|
||||||
|
* subclassing doesn't make much sense if it doesn't work.
|
||||||
|
*/
|
||||||
|
'Subtype of constructor should contain extended members': function()
|
||||||
|
{
|
||||||
|
var Ctor = function() {};
|
||||||
|
|
||||||
|
this.assertNotEqual(
|
||||||
|
( new this.Sut.extend( Ctor, { foo: {} } )() ).foo,
|
||||||
|
undefined
|
||||||
|
);
|
||||||
|
},
|
||||||
|
|
||||||
|
|
||||||
|
/**
|
||||||
|
* If a subtype provides a property of the same name as its parent, then
|
||||||
|
* it should act as a reassignment.
|
||||||
|
*/
|
||||||
|
'Subtypes can override parent property values': function()
|
||||||
|
{
|
||||||
|
var expect = 'ok',
|
||||||
|
C = this.Sut.extend( { p: null } ).extend( { p: expect } );
|
||||||
|
|
||||||
|
this.assertEqual( C().p, expect );
|
||||||
|
},
|
||||||
|
|
||||||
|
|
||||||
|
/**
|
||||||
|
* Prevent overriding the internal method that initializes property
|
||||||
|
* values upon instantiation.
|
||||||
|
*/
|
||||||
|
'__initProps() cannot be declared (internal method)': function()
|
||||||
|
{
|
||||||
|
var _self = this;
|
||||||
|
|
||||||
|
this.assertThrows( function()
|
||||||
|
{
|
||||||
|
_self.Sut.extend(
|
||||||
|
{
|
||||||
|
__initProps: function() {},
|
||||||
|
} );
|
||||||
|
}, Error );
|
||||||
|
},
|
||||||
|
|
||||||
|
|
||||||
|
// TODO: move me into a more appropriate test case (this may actually be
|
||||||
|
// tested elsewhere)
|
||||||
|
/**
|
||||||
|
* If using the short-hand extend, an object is required to represent
|
||||||
|
* the class defintiion.
|
||||||
|
*/
|
||||||
|
'Invoking class module requires object as argument if extending':
|
||||||
|
function()
|
||||||
|
{
|
||||||
|
var _self = this;
|
||||||
|
|
||||||
|
// these tests can be run in the browser in pre-ES5 environments, so
|
||||||
|
// no forEach()
|
||||||
|
var chk = [ 5, false, undefined ],
|
||||||
|
i = chk.length;
|
||||||
|
|
||||||
|
while ( i-- )
|
||||||
|
{
|
||||||
|
this.assertThrows( function()
|
||||||
|
{
|
||||||
|
_self.Sut( chk[ i ] );
|
||||||
|
},
|
||||||
|
TypeError
|
||||||
|
);
|
||||||
|
}
|
||||||
|
},
|
||||||
|
|
||||||
|
|
||||||
|
/**
|
||||||
|
* We provide a useful default toString() method, but one may wish to
|
||||||
|
* override it
|
||||||
|
*/
|
||||||
|
'Can override toString() method': function()
|
||||||
|
{
|
||||||
|
var str = 'foomookittypoo',
|
||||||
|
result = ''
|
||||||
|
;
|
||||||
|
|
||||||
|
result = this.Sut( 'FooToStr',
|
||||||
|
{
|
||||||
|
toString: function()
|
||||||
|
{
|
||||||
|
return str;
|
||||||
|
},
|
||||||
|
} )().toString();
|
||||||
|
|
||||||
|
this.assertEqual( result, str );
|
||||||
|
},
|
||||||
|
|
||||||
|
|
||||||
|
/**
|
||||||
|
* In ease.js's initial design, keywords were not included. This meant
|
||||||
|
* that duplicate member definitions were not possible---it'd throw a
|
||||||
|
* parse error (maybe). However, with keywords, it is now possible to
|
||||||
|
* redeclare a member with the same name in the same class definition.
|
||||||
|
* Since this doesn't make much sense, we must disallow it.
|
||||||
|
*/
|
||||||
|
'Cannot provide duplicate member definitions using unique keys':
|
||||||
|
function()
|
||||||
|
{
|
||||||
|
var _self = this;
|
||||||
|
|
||||||
|
this.assertThrows( function()
|
||||||
|
{
|
||||||
|
_self.Sut(
|
||||||
|
{
|
||||||
|
// declare as protected first so that we won't get a visibility
|
||||||
|
// de-escalation error with the below re-definition
|
||||||
|
'protected foo': '',
|
||||||
|
|
||||||
|
// should fail; redefinition
|
||||||
|
'public foo': '',
|
||||||
|
} );
|
||||||
|
}, Error );
|
||||||
|
|
||||||
|
this.assertThrows( function()
|
||||||
|
{
|
||||||
|
_self.Sut(
|
||||||
|
{
|
||||||
|
// declare as protected first so that we won't get a visibility
|
||||||
|
// de-escalation error with the below re-definition
|
||||||
|
'protected foo': function() {},
|
||||||
|
|
||||||
|
// should fail; redefinition
|
||||||
|
'public foo': function() {},
|
||||||
|
} );
|
||||||
|
}, Error );
|
||||||
|
},
|
||||||
|
|
||||||
|
|
||||||
|
/**
|
||||||
|
* To understand this test, one must understand how "inheritance" works
|
||||||
|
* with prototypes. We must create a new instance of the ctor (class)
|
||||||
|
* and add that instance to the prototype chain (if we added an
|
||||||
|
* un-instantiated constructor, then the members in the prototype would
|
||||||
|
* be accessible only though ctor.prototype). Therefore, when we
|
||||||
|
* instantiate this class for use in the prototype, we must ensure the
|
||||||
|
* constructor is not invoked, since our intent is not to create a new
|
||||||
|
* instance of the class.
|
||||||
|
*/
|
||||||
|
'__construct should not be called when extending class': function()
|
||||||
|
{
|
||||||
|
var called = false,
|
||||||
|
Foo = this.Sut( {
|
||||||
|
'public __construct': function()
|
||||||
|
{
|
||||||
|
called = true;
|
||||||
|
}
|
||||||
|
} ).extend( {} );
|
||||||
|
|
||||||
|
this.assertEqual( called, false );
|
||||||
|
},
|
||||||
|
|
||||||
|
|
||||||
|
/**
|
||||||
|
* Previously, when attempting to extend from an invalid supertype,
|
||||||
|
* you'd get a CALL_NON_FUNCTION_AS_CONSTRUCTOR error, which is not very
|
||||||
|
* helpful to someone who is not familiar with the ease.js internals.
|
||||||
|
* Let's provide a more useful error that clearly states what's going
|
||||||
|
* on.
|
||||||
|
*/
|
||||||
|
'Extending from non-ctor or non-class provides useful error': function()
|
||||||
|
{
|
||||||
|
try
|
||||||
|
{
|
||||||
|
// invalid supertype
|
||||||
|
this.Sut.extend( 'oops', {} );
|
||||||
|
}
|
||||||
|
catch ( e )
|
||||||
|
{
|
||||||
|
this.assertOk( e.message.search( 'extend from' ),
|
||||||
|
"Error message for extending from non-ctor or class " +
|
||||||
|
"makes sense"
|
||||||
|
);
|
||||||
|
|
||||||
|
return;
|
||||||
|
}
|
||||||
|
|
||||||
|
this.assertFail(
|
||||||
|
"Attempting to extend from non-ctor or class should " +
|
||||||
|
"throw exception"
|
||||||
|
);
|
||||||
|
},
|
||||||
|
|
||||||
|
|
||||||
|
/**
|
||||||
|
* If we attempt to extend an object (rather than a constructor), we
|
||||||
|
* should simply use that as the prototype directly rather than
|
||||||
|
* attempting to instantiate it.
|
||||||
|
*/
|
||||||
|
'Extending object will not attempt instantiation': function()
|
||||||
|
{
|
||||||
|
var obj = { foo: 'bar' };
|
||||||
|
|
||||||
|
this.assertEqual( obj.foo, this.Sut.extend( obj, {} )().foo,
|
||||||
|
"Should be able to use object as prototype"
|
||||||
|
);
|
||||||
|
},
|
||||||
|
} );
|
|
@ -0,0 +1,87 @@
|
||||||
|
/**
|
||||||
|
* Tests safety of class instances
|
||||||
|
*
|
||||||
|
* Copyright (C) 2010, 2011, 2013, 2014 Mike Gerwitz
|
||||||
|
*
|
||||||
|
* This file is part of GNU ease.js.
|
||||||
|
*
|
||||||
|
* ease.js is free software: you can redistribute it and/or modify
|
||||||
|
* it under the terms of the GNU General Public License as published by
|
||||||
|
* the Free Software Foundation, either version 3 of the License, or
|
||||||
|
* (at your option) any later version.
|
||||||
|
*
|
||||||
|
* This program is distributed in the hope that it will be useful,
|
||||||
|
* but WITHOUT ANY WARRANTY; without even the implied warranty of
|
||||||
|
* MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the
|
||||||
|
* GNU General Public License for more details.
|
||||||
|
*
|
||||||
|
* You should have received a copy of the GNU General Public License
|
||||||
|
* along with this program. If not, see <http://www.gnu.org/licenses/>.
|
||||||
|
*/
|
||||||
|
|
||||||
|
require( 'common' ).testCase(
|
||||||
|
{
|
||||||
|
caseSetUp: function()
|
||||||
|
{
|
||||||
|
this.Sut = this.require( 'class' );
|
||||||
|
},
|
||||||
|
|
||||||
|
|
||||||
|
/**
|
||||||
|
* Ensure that we're not getting/setting values of the prototype, which
|
||||||
|
* would have disasterous implications (=== can also be used to test for
|
||||||
|
* references, but this test demonstrates the functionality that we're
|
||||||
|
* looking to ensure)
|
||||||
|
*/
|
||||||
|
'Multiple instances of same class do not share array references':
|
||||||
|
function()
|
||||||
|
{
|
||||||
|
var C = this.Sut.extend( { arr: [] } ),
|
||||||
|
obj1 = new C(),
|
||||||
|
obj2 = new C();
|
||||||
|
|
||||||
|
obj1.arr.push( 'one' );
|
||||||
|
obj2.arr.push( 'two' );
|
||||||
|
|
||||||
|
// if the arrays are distinct, then each will have only one element
|
||||||
|
this.assertEqual( obj1.arr[ 0 ], 'one' );
|
||||||
|
this.assertEqual( obj2.arr[ 0 ], 'two' );
|
||||||
|
this.assertEqual( obj1.arr.length, 1 );
|
||||||
|
this.assertEqual( obj2.arr.length, 1 );
|
||||||
|
},
|
||||||
|
|
||||||
|
|
||||||
|
/**
|
||||||
|
* Same concept as above, but with objects instead of arrays.
|
||||||
|
*/
|
||||||
|
'Multiple instances of same class do not share object references':
|
||||||
|
function()
|
||||||
|
{
|
||||||
|
var C = this.Sut.extend( { obj: {} } ),
|
||||||
|
obj1 = new C(),
|
||||||
|
obj2 = new C();
|
||||||
|
|
||||||
|
obj1.obj.a = true;
|
||||||
|
obj2.obj.b = true;
|
||||||
|
|
||||||
|
this.assertEqual( obj1.obj.a, true );
|
||||||
|
this.assertEqual( obj1.obj.b, undefined );
|
||||||
|
|
||||||
|
this.assertEqual( obj2.obj.a, undefined );
|
||||||
|
this.assertEqual( obj2.obj.b, true );
|
||||||
|
},
|
||||||
|
|
||||||
|
|
||||||
|
/**
|
||||||
|
* Ensure that the above checks extend to subtypes.
|
||||||
|
*/
|
||||||
|
'Instances of subtypes do not share property references': function()
|
||||||
|
{
|
||||||
|
var C2 = this.Sut.extend( { arr: [], obj: {} } ).extend( {} ),
|
||||||
|
obj1 = new C2(),
|
||||||
|
obj2 = new C2();
|
||||||
|
|
||||||
|
this.assertNotEqual( obj1.arr !== obj2.arr );
|
||||||
|
this.assertNotEqual( obj1.obj !== obj2.obj );
|
||||||
|
},
|
||||||
|
} );
|
|
@ -116,6 +116,11 @@ module.exports = function( test_case )
|
||||||
|
|
||||||
if ( method === 'each' )
|
if ( method === 'each' )
|
||||||
{
|
{
|
||||||
|
if ( !( context[ prop ] ) )
|
||||||
|
{
|
||||||
|
throw Error( "Unknown @each context: " + prop );
|
||||||
|
}
|
||||||
|
|
||||||
count = context[ prop ].length;
|
count = context[ prop ].length;
|
||||||
args = [];
|
args = [];
|
||||||
|
|
||||||
|
|
|
@ -1,463 +0,0 @@
|
||||||
/**
|
|
||||||
* Tests class module extend() method
|
|
||||||
*
|
|
||||||
* Copyright (C) 2010, 2011, 2013 Mike Gerwitz
|
|
||||||
*
|
|
||||||
* This file is part of GNU ease.js.
|
|
||||||
*
|
|
||||||
* ease.js is free software: you can redistribute it and/or modify
|
|
||||||
* it under the terms of the GNU General Public License as published by
|
|
||||||
* the Free Software Foundation, either version 3 of the License, or
|
|
||||||
* (at your option) any later version.
|
|
||||||
*
|
|
||||||
* This program is distributed in the hope that it will be useful,
|
|
||||||
* but WITHOUT ANY WARRANTY; without even the implied warranty of
|
|
||||||
* MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the
|
|
||||||
* GNU General Public License for more details.
|
|
||||||
*
|
|
||||||
* You should have received a copy of the GNU General Public License
|
|
||||||
* along with this program. If not, see <http://www.gnu.org/licenses/>.
|
|
||||||
*/
|
|
||||||
|
|
||||||
var common = require( './common' ),
|
|
||||||
assert = require( 'assert' ),
|
|
||||||
Class = common.require( 'class' );
|
|
||||||
|
|
||||||
var foo_props = {
|
|
||||||
one: 1,
|
|
||||||
two: 2,
|
|
||||||
},
|
|
||||||
|
|
||||||
// there are two different means of extending; we want to test them both
|
|
||||||
classes = [
|
|
||||||
Class.extend( foo_props ),
|
|
||||||
Class( foo_props ),
|
|
||||||
],
|
|
||||||
|
|
||||||
class_count = classes.length,
|
|
||||||
|
|
||||||
// will hold the class being tested
|
|
||||||
Foo = null
|
|
||||||
;
|
|
||||||
|
|
||||||
|
|
||||||
// Run all tests for both. This will ensure that, regardless of how the class is
|
|
||||||
// created, it operates as it should. Fortunately, these tests are fairly quick.
|
|
||||||
for ( var i = 0; i < class_count; i++ )
|
|
||||||
{
|
|
||||||
Foo = classes[ i ];
|
|
||||||
|
|
||||||
assert.ok(
|
|
||||||
( Foo.extend instanceof Function ),
|
|
||||||
"Created class contains extend method"
|
|
||||||
);
|
|
||||||
|
|
||||||
var sub_props = {
|
|
||||||
three: 3,
|
|
||||||
four: 4,
|
|
||||||
},
|
|
||||||
|
|
||||||
SubFoo = Foo.extend( sub_props ),
|
|
||||||
sub_foo = SubFoo()
|
|
||||||
;
|
|
||||||
|
|
||||||
assert.ok(
|
|
||||||
( SubFoo instanceof Object ),
|
|
||||||
"Subtype is returned as an object"
|
|
||||||
);
|
|
||||||
|
|
||||||
// ensure properties were inherited from supertype
|
|
||||||
for ( var prop in foo_props )
|
|
||||||
{
|
|
||||||
assert.equal(
|
|
||||||
foo_props[ prop ],
|
|
||||||
SubFoo.prototype[ prop ],
|
|
||||||
"Subtype inherits parent properties: " + prop
|
|
||||||
);
|
|
||||||
}
|
|
||||||
|
|
||||||
// and ensure that the subtype's properties were included
|
|
||||||
for ( var prop in sub_props )
|
|
||||||
{
|
|
||||||
assert.equal(
|
|
||||||
sub_props[ prop ],
|
|
||||||
sub_foo[ prop ],
|
|
||||||
"Subtype contains its own properties: " + prop
|
|
||||||
);
|
|
||||||
}
|
|
||||||
|
|
||||||
|
|
||||||
var sub_instance = new SubFoo();
|
|
||||||
|
|
||||||
assert.ok(
|
|
||||||
( sub_instance instanceof Foo ),
|
|
||||||
"Subtypes are considered to be instances of their supertypes " +
|
|
||||||
"(via instanceof operator)"
|
|
||||||
);
|
|
||||||
|
|
||||||
assert.ok(
|
|
||||||
sub_instance.isInstanceOf( SubFoo ),
|
|
||||||
"Subtypes are considered to be instances of their supertypes (via " +
|
|
||||||
"isInstanceOf method)"
|
|
||||||
);
|
|
||||||
|
|
||||||
|
|
||||||
// Foo
|
|
||||||
// |
|
|
||||||
// SubFoo
|
|
||||||
// / \
|
|
||||||
// SubSubFoo SubSubFoo2
|
|
||||||
//
|
|
||||||
var SubSubFoo = SubFoo.extend(),
|
|
||||||
SubSubFoo2 = SubFoo.extend(),
|
|
||||||
|
|
||||||
sub_sub_instance = new SubSubFoo(),
|
|
||||||
sub_sub2_instance = new SubSubFoo2();
|
|
||||||
|
|
||||||
assert.ok(
|
|
||||||
( ( sub_sub_instance instanceof Foo )
|
|
||||||
&& sub_sub_instance.isInstanceOf( Foo )
|
|
||||||
),
|
|
||||||
"Sub-subtypes should be instances of their super-supertype"
|
|
||||||
);
|
|
||||||
|
|
||||||
assert.ok(
|
|
||||||
( !( sub_instance instanceof SubSubFoo )
|
|
||||||
&& !( sub_instance.isInstanceOf( SubSubFoo ) )
|
|
||||||
),
|
|
||||||
"Supertypes should not be considered instances of their subtypes"
|
|
||||||
);
|
|
||||||
|
|
||||||
assert.ok(
|
|
||||||
( !( sub_sub2_instance instanceof SubSubFoo )
|
|
||||||
&& !( sub_sub2_instance.isInstanceOf( SubSubFoo ) )
|
|
||||||
),
|
|
||||||
"Subtypes should not be considered instances of their siblings"
|
|
||||||
);
|
|
||||||
|
|
||||||
|
|
||||||
// to test inheritance of classes that were not previously created via the
|
|
||||||
// Class.extend() method
|
|
||||||
var OtherClass = function() {};
|
|
||||||
OtherClass.prototype =
|
|
||||||
{
|
|
||||||
foo: 'bla',
|
|
||||||
};
|
|
||||||
|
|
||||||
var SubOther = Class.extend( OtherClass,
|
|
||||||
{
|
|
||||||
newFoo: 2,
|
|
||||||
});
|
|
||||||
|
|
||||||
|
|
||||||
assert.equal(
|
|
||||||
SubOther.prototype.foo,
|
|
||||||
OtherClass.prototype.foo,
|
|
||||||
"Prototype of existing class should be copied to subclass"
|
|
||||||
);
|
|
||||||
|
|
||||||
assert.notEqual(
|
|
||||||
SubOther().newFoo,
|
|
||||||
undefined,
|
|
||||||
"Subtype should contain extended members"
|
|
||||||
);
|
|
||||||
|
|
||||||
|
|
||||||
assert['throws']( function()
|
|
||||||
{
|
|
||||||
Class.extend( OtherClass,
|
|
||||||
{
|
|
||||||
foo: function() {},
|
|
||||||
});
|
|
||||||
}, TypeError, "Cannot override property with a method" );
|
|
||||||
|
|
||||||
|
|
||||||
var AnotherFoo = Class.extend(
|
|
||||||
{
|
|
||||||
arr: [],
|
|
||||||
obj: {},
|
|
||||||
});
|
|
||||||
|
|
||||||
var Obj1 = new AnotherFoo(),
|
|
||||||
Obj2 = new AnotherFoo();
|
|
||||||
|
|
||||||
Obj1.arr.push( 'one' );
|
|
||||||
Obj2.arr.push( 'two' );
|
|
||||||
|
|
||||||
Obj1.obj.a = true;
|
|
||||||
Obj2.obj.b = true;
|
|
||||||
|
|
||||||
// to ensure we're not getting/setting values of the prototype (=== can also be
|
|
||||||
// used to test for references, but this test demonstrates the functionality
|
|
||||||
// that we're looking to ensure)
|
|
||||||
assert.ok(
|
|
||||||
( ( Obj1.arr[ 0 ] === 'one' ) && ( Obj2.arr[ 0 ] === 'two' ) ),
|
|
||||||
"Multiple instances of the same class do not share array references"
|
|
||||||
);
|
|
||||||
|
|
||||||
assert.ok(
|
|
||||||
( ( ( Obj1.obj.a === true ) && ( Obj1.obj.b === undefined ) )
|
|
||||||
&& ( ( Obj2.obj.a === undefined ) && ( Obj2.obj.b === true ) )
|
|
||||||
),
|
|
||||||
"Multiple instances of the same class do not share object references"
|
|
||||||
);
|
|
||||||
|
|
||||||
var arr_val = 1;
|
|
||||||
var SubAnotherFoo = AnotherFoo.extend(
|
|
||||||
{
|
|
||||||
arr: [ arr_val ],
|
|
||||||
});
|
|
||||||
|
|
||||||
var SubObj1 = new SubAnotherFoo(),
|
|
||||||
SubObj2 = new SubAnotherFoo();
|
|
||||||
|
|
||||||
assert.ok(
|
|
||||||
( ( SubObj1.arr !== SubObj2.arr ) && ( SubObj1.obj !== SubObj2.obj ) ),
|
|
||||||
"Instances of subtypes do not share property references"
|
|
||||||
);
|
|
||||||
|
|
||||||
assert.ok(
|
|
||||||
( ( SubObj1.arr[ 0 ] === arr_val ) && ( SubObj2.arr[ 0 ] === arr_val ) ),
|
|
||||||
"Subtypes can override parent property values"
|
|
||||||
);
|
|
||||||
|
|
||||||
assert['throws']( function()
|
|
||||||
{
|
|
||||||
Class.extend(
|
|
||||||
{
|
|
||||||
__initProps: function() {},
|
|
||||||
});
|
|
||||||
}, Error, "__initProps() cannot be declared (internal method)" );
|
|
||||||
|
|
||||||
|
|
||||||
var SubSubAnotherFoo = AnotherFoo.extend(),
|
|
||||||
SubSubObj1 = new SubSubAnotherFoo(),
|
|
||||||
SubSubObj2 = new SubSubAnotherFoo();
|
|
||||||
|
|
||||||
// to ensure the effect is recursive
|
|
||||||
assert.ok(
|
|
||||||
( ( SubSubObj1.arr !== SubSubObj2.arr )
|
|
||||||
&& ( SubSubObj1.obj !== SubSubObj2.obj )
|
|
||||||
),
|
|
||||||
"Instances of subtypes do not share property references"
|
|
||||||
);
|
|
||||||
}
|
|
||||||
|
|
||||||
|
|
||||||
( function testInvokingClassModuleRequiresObjectAsArgumentIfCreating()
|
|
||||||
{
|
|
||||||
assert['throws']( function()
|
|
||||||
{
|
|
||||||
Class( 'moo' );
|
|
||||||
Class( 5 );
|
|
||||||
Class( false );
|
|
||||||
Class();
|
|
||||||
},
|
|
||||||
TypeError,
|
|
||||||
"Invoking class module requires object as argument if extending " +
|
|
||||||
"from base class"
|
|
||||||
);
|
|
||||||
|
|
||||||
var args = [ {}, 'one', 'two', 'three' ];
|
|
||||||
|
|
||||||
// we must only provide one argument if the first argument is an object (the
|
|
||||||
// class definition)
|
|
||||||
try
|
|
||||||
{
|
|
||||||
Class.apply( null, args );
|
|
||||||
|
|
||||||
// if all goes well, we don't get to this line
|
|
||||||
assert.fail(
|
|
||||||
"Only one argument for class definitions is permitted"
|
|
||||||
);
|
|
||||||
}
|
|
||||||
catch ( e )
|
|
||||||
{
|
|
||||||
assert.notEqual(
|
|
||||||
e.message.match( args.length + ' given' ),
|
|
||||||
null,
|
|
||||||
"Class invocation should give argument count on error"
|
|
||||||
);
|
|
||||||
}
|
|
||||||
} )();
|
|
||||||
|
|
||||||
|
|
||||||
/**
|
|
||||||
* We provide a useful default toString() method, but one may wish to override
|
|
||||||
* it
|
|
||||||
*/
|
|
||||||
( function testCanOverrideToStringMethod()
|
|
||||||
{
|
|
||||||
var str = 'foomookittypoo',
|
|
||||||
result = ''
|
|
||||||
;
|
|
||||||
|
|
||||||
result = Class( 'FooToStr',
|
|
||||||
{
|
|
||||||
toString: function()
|
|
||||||
{
|
|
||||||
return str;
|
|
||||||
},
|
|
||||||
bla: function() {},
|
|
||||||
})().toString();
|
|
||||||
|
|
||||||
assert.equal(
|
|
||||||
result,
|
|
||||||
str,
|
|
||||||
"Can override default toString() method of class"
|
|
||||||
);
|
|
||||||
} )();
|
|
||||||
|
|
||||||
|
|
||||||
/**
|
|
||||||
* In ease.js's initial design, keywords were not included. This meant that
|
|
||||||
* duplicate member definitions were not possible - it'd throw a parse error.
|
|
||||||
* However, with keywords, it is now possible to redeclare a member with the
|
|
||||||
* same name in the same class definition. Since this doesn't make much sense,
|
|
||||||
* we must disallow it.
|
|
||||||
*/
|
|
||||||
( function testCannotProvideDuplicateMemberDefintions()
|
|
||||||
{
|
|
||||||
assert['throws']( function()
|
|
||||||
{
|
|
||||||
Class(
|
|
||||||
{
|
|
||||||
// declare as protected first so that we won't get a visibility
|
|
||||||
// de-escalation error with the below re-definition
|
|
||||||
'protected foo': '',
|
|
||||||
|
|
||||||
// should fail; redefinition
|
|
||||||
'public foo': '',
|
|
||||||
} );
|
|
||||||
}, Error, "Cannot redeclare property in same class definition" );
|
|
||||||
|
|
||||||
assert['throws']( function()
|
|
||||||
{
|
|
||||||
Class(
|
|
||||||
{
|
|
||||||
// declare as protected first so that we won't get a visibility
|
|
||||||
// de-escalation error with the below re-definition
|
|
||||||
'protected foo': function() {},
|
|
||||||
|
|
||||||
// should fail; redefinition
|
|
||||||
'public foo': function() {},
|
|
||||||
} );
|
|
||||||
}, Error, "Cannot redeclare method in same class definition" );
|
|
||||||
} )();
|
|
||||||
|
|
||||||
|
|
||||||
/**
|
|
||||||
* To understand this test, one must understand how "inheritance" works
|
|
||||||
* with prototypes. We must create a new instance of the ctor (class) and add
|
|
||||||
* that instance to the prototype chain (if we added an un-instantiated
|
|
||||||
* constructor, then the members in the prototype would be accessible only
|
|
||||||
* though ctor.prototype). Therefore, when we instantiate this class for use in
|
|
||||||
* the prototype, we must ensure the constructor is not invoked, since our
|
|
||||||
* intent is not to create a new instance of the class.
|
|
||||||
*/
|
|
||||||
( function testConstructorShouldNotBeCalledWhenExtendingClass()
|
|
||||||
{
|
|
||||||
var called = false,
|
|
||||||
Foo = Class( {
|
|
||||||
'public __construct': function()
|
|
||||||
{
|
|
||||||
called = true;
|
|
||||||
}
|
|
||||||
} ).extend( {} );
|
|
||||||
|
|
||||||
assert.equal( called, false,
|
|
||||||
"Constructor should not be called when extending a class"
|
|
||||||
);
|
|
||||||
} )();
|
|
||||||
|
|
||||||
|
|
||||||
/**
|
|
||||||
* Previously, when attempting to extend from an invalid supertype, you'd get a
|
|
||||||
* CALL_NON_FUNCTION_AS_CONSTRUCTOR error, which is not very helpful to someone
|
|
||||||
* who is not familiar with the ease.js internals. Let's provide a more useful
|
|
||||||
* error that clearly states what's going on.
|
|
||||||
*/
|
|
||||||
( function testExtendingFromNonCtorOrClassProvidesUsefulError()
|
|
||||||
{
|
|
||||||
try
|
|
||||||
{
|
|
||||||
// invalid supertype
|
|
||||||
Class.extend( 'oops', {} );
|
|
||||||
}
|
|
||||||
catch ( e )
|
|
||||||
{
|
|
||||||
assert.ok( e.message.search( 'extend from' ),
|
|
||||||
"Error message for extending from non-ctor or class makes sense"
|
|
||||||
);
|
|
||||||
|
|
||||||
return;
|
|
||||||
}
|
|
||||||
|
|
||||||
assert.fail(
|
|
||||||
"Attempting to extend from non-ctor or class should throw exception"
|
|
||||||
);
|
|
||||||
} )();
|
|
||||||
|
|
||||||
|
|
||||||
/**
|
|
||||||
* Only virtual methods may be overridden.
|
|
||||||
*/
|
|
||||||
( function testCannotOverrideNonVirtualMethod()
|
|
||||||
{
|
|
||||||
try
|
|
||||||
{
|
|
||||||
var Foo = Class(
|
|
||||||
{
|
|
||||||
// non-virtual
|
|
||||||
'public foo': function() {},
|
|
||||||
} ),
|
|
||||||
|
|
||||||
SubFoo = Foo.extend(
|
|
||||||
{
|
|
||||||
// should fail (cannot override non-virtual method)
|
|
||||||
'override public foo': function() {},
|
|
||||||
} );
|
|
||||||
}
|
|
||||||
catch ( e )
|
|
||||||
{
|
|
||||||
assert.ok( e.message.search( 'foo' ),
|
|
||||||
"Non-virtual override error message should contain name of method"
|
|
||||||
);
|
|
||||||
|
|
||||||
return;
|
|
||||||
}
|
|
||||||
|
|
||||||
assert.fail( "Should not be permitted to override non-virtual method" );
|
|
||||||
} )();
|
|
||||||
|
|
||||||
|
|
||||||
/**
|
|
||||||
* If we attempt to extend an object (rather than a constructor), we should
|
|
||||||
* simply use that as the prototype directly rather than attempting to
|
|
||||||
* instantiate it.
|
|
||||||
*/
|
|
||||||
( function testExtendingObjectWillNotAttemptInstantiation()
|
|
||||||
{
|
|
||||||
var obj = { foo: 'bar' };
|
|
||||||
|
|
||||||
assert.equal( obj.foo, Class.extend( obj, {} )().foo,
|
|
||||||
'Should be able to use object as prototype'
|
|
||||||
);
|
|
||||||
} )();
|
|
||||||
|
|
||||||
|
|
||||||
/**
|
|
||||||
* It only makes sense to extend from an object or function (constructor, more
|
|
||||||
* specifically)
|
|
||||||
*
|
|
||||||
* We could also test to ensure that the return value of the constructor is an
|
|
||||||
* object, but that is unnecessary for the time being.
|
|
||||||
*/
|
|
||||||
( function testWillThrowExceptionIfNonObjectOrCtorIsProvided()
|
|
||||||
{
|
|
||||||
assert['throws']( function()
|
|
||||||
{
|
|
||||||
Class.extend( 'foo', {} );
|
|
||||||
}, TypeError, 'Should not be able to extend from non-object or non-ctor' );
|
|
||||||
} )();
|
|
||||||
|
|
Loading…
Reference in New Issue