213 lines
6.3 KiB
JavaScript
213 lines
6.3 KiB
JavaScript
/**
|
|
* Tests extending a class that mixes in traits
|
|
*
|
|
* Copyright (C) 2014 Free Software Foundation, Inc.
|
|
*
|
|
* 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( 'Trait' );
|
|
this.Class = this.require( 'class' );
|
|
},
|
|
|
|
|
|
/**
|
|
* The supertype should continue to work as it would without the
|
|
* subtype, which means that the supertype's traits should still be
|
|
* available. Note that ease.js does not (at least at the time of
|
|
* writing this test) check to see if a trait is no longer accessible
|
|
* due to overrides, and so a supertype's traits will always be
|
|
* instantiated.
|
|
*/
|
|
'Subtype instantiates traits of supertype': function()
|
|
{
|
|
var called = false;
|
|
|
|
var T = this.Sut(
|
|
{
|
|
foo: function() { called = true; },
|
|
} );
|
|
|
|
// C is a subtype of a class that mixes in T
|
|
var C = this.Class.use( T ).extend( {} )
|
|
.extend(
|
|
{
|
|
// ensure that there is no ctor-dependent trait stuff
|
|
__construct: function() {},
|
|
} );
|
|
|
|
C().foo();
|
|
this.assertOk( called );
|
|
},
|
|
|
|
|
|
/**
|
|
* Just as subtypes inherit the same polymorphisms with respect to
|
|
* interfaces, so too should subtypes inherit supertypes' mixed in
|
|
* traits' types.
|
|
*/
|
|
'Subtype has same polymorphic qualities of parent mixins': function()
|
|
{
|
|
var T = this.Sut( {} ),
|
|
o = this.Class.use( T ).extend( {} ).extend( {} )();
|
|
|
|
// o's supertype mixes in T
|
|
this.assertOk( this.Class.isA( T, o ) );
|
|
},
|
|
|
|
|
|
/**
|
|
* Subtyping should impose no limits on mixins (except for the obvious
|
|
* API compatibility restrictions inherent in OOP).
|
|
*/
|
|
'Subtype can mix in additional traits': function()
|
|
{
|
|
var a = false,
|
|
b = false;
|
|
|
|
var Ta = this.Sut(
|
|
{
|
|
'public ta': function() { a = true; },
|
|
} ),
|
|
Tb = this.Sut(
|
|
{
|
|
'public tb': function() { b = true; },
|
|
} ),
|
|
C = null;
|
|
|
|
var _self = this;
|
|
this.assertDoesNotThrow( function()
|
|
{
|
|
var sup = _self.Class.use( Ta ).extend( {} );
|
|
|
|
// mixes in Tb; supertype already mixed in Ta
|
|
C = _self.Class.use( Tb ).extend( sup, {} );
|
|
} );
|
|
|
|
this.assertDoesNotThrow( function()
|
|
{
|
|
// ensures that instantiation does not throw an error and that
|
|
// the methods both exist
|
|
var o = C();
|
|
o.ta();
|
|
o.tb();
|
|
} );
|
|
|
|
// ensure both were properly called
|
|
this.assertOk( a );
|
|
this.assertOk( b );
|
|
},
|
|
|
|
|
|
/**
|
|
* As a sanity check, ensure that subtyping does not override parent
|
|
* type data with respect to traits.
|
|
*
|
|
* Note that this test makes the preceding test redundant, but the
|
|
* separation is useful for debugging any potential regressions.
|
|
*/
|
|
'Subtype trait types do not overwrite supertype types': function()
|
|
{
|
|
var Ta = this.Sut( {} ),
|
|
Tb = this.Sut( {} ),
|
|
C = this.Class.use( Ta ).extend( {} ),
|
|
o = this.Class.use( Tb ).extend( C, {} )();
|
|
|
|
// o's supertype mixes in Ta
|
|
this.assertOk( this.Class.isA( Ta, o ) );
|
|
|
|
// o mixes in Tb
|
|
this.assertOk( this.Class.isA( Tb, o ) );
|
|
},
|
|
|
|
|
|
/**
|
|
* This alternative syntax mixes a trait directly into a base class and
|
|
* then omits the base class as an argument to the extend method; this
|
|
* syntax is most familiar with named classes, but we are not testing
|
|
* named classes here.
|
|
*/
|
|
'Can mix in traits directly atop of existing class': function()
|
|
{
|
|
var called_foo = false,
|
|
called_bar = false,
|
|
called_baz = false;
|
|
|
|
var C = this.Class(
|
|
{
|
|
'public foo': function() { called_foo = true; },
|
|
} );
|
|
|
|
var T = this.Sut(
|
|
{
|
|
'public bar': function() { called_bar = true; },
|
|
} );
|
|
|
|
// we must ensure not only that we have mixed in the trait, but that
|
|
// we have also maintained C's interface and can further extend it
|
|
var inst = C.use( T ).extend(
|
|
{
|
|
'public baz': function() { called_baz = true; },
|
|
} )();
|
|
|
|
inst.foo();
|
|
inst.bar();
|
|
inst.baz();
|
|
|
|
this.assertOk( called_foo );
|
|
this.assertOk( called_bar );
|
|
this.assertOk( called_baz );
|
|
},
|
|
|
|
|
|
/**
|
|
* This test ensures that we can mix in traits using the syntax
|
|
* C.use(T1).use(T2), and so on; this may be necessary to disambiguate
|
|
* overrides if T1 and T2 provide definitions for the same method (and
|
|
* so the syntax C.use(T1, T2) cannot be used). This syntax is also
|
|
* important for the concept of stackable traits (see
|
|
* LinearizationTest).
|
|
*
|
|
* Note that this differs from C.use(T1).use(T2).extend({}); we're
|
|
* talking about C.extend({}).use(T1).use(T2). Therefore, this can be
|
|
* considered to be syntatic sugar for
|
|
* C.use( T1 ).extend( {} ).use( T2 ).
|
|
*/
|
|
'Can chain use calls': function()
|
|
{
|
|
var T1 = this.Sut( { foo: function() {} } ),
|
|
T2 = this.Sut( { bar: function() {} } ),
|
|
C = null;
|
|
|
|
var Class = this.Class;
|
|
this.assertDoesNotThrow( function()
|
|
{
|
|
C = Class.extend( {} ).use( T1 ).use( T2 );
|
|
} );
|
|
|
|
// ensure that the methods were actually mixed in
|
|
this.assertDoesNotThrow( function()
|
|
{
|
|
C().foo();
|
|
C().bar();
|
|
} );
|
|
},
|
|
} );
|