2010-12-29 23:33:30 -05:00
|
|
|
/**
|
2010-12-30 09:54:53 -05:00
|
|
|
* Tests class interface implement method
|
2010-12-29 23:33:30 -05:00
|
|
|
*
|
|
|
|
* Copyright (C) 2010 Mike Gerwitz
|
|
|
|
*
|
|
|
|
* This file is part of ease.js.
|
|
|
|
*
|
|
|
|
* ease.js is free software: you can redistribute it and/or modify it under the
|
|
|
|
* terms of the GNU Lesser 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 Lesser General Public License
|
|
|
|
* for more details.
|
|
|
|
*
|
|
|
|
* You should have received a copy of the GNU Lesser General Public License
|
|
|
|
* along with this program. If not, see <http://www.gnu.org/licenses/>.
|
|
|
|
*
|
|
|
|
* @author Mike Gerwitz
|
|
|
|
* @package test
|
|
|
|
*/
|
|
|
|
|
|
|
|
var common = require( './common' ),
|
|
|
|
assert = require( 'assert' ),
|
|
|
|
Class = common.require( 'class' ),
|
|
|
|
Interface = common.require( 'interface' );
|
|
|
|
|
|
|
|
|
|
|
|
var Type = Interface.extend( {
|
2011-01-11 19:03:30 -05:00
|
|
|
'abstract foo': [],
|
|
|
|
}),
|
2010-12-29 23:33:30 -05:00
|
|
|
|
2011-01-11 19:03:30 -05:00
|
|
|
Type2 = Interface.extend( {
|
|
|
|
'abstract foo2': [],
|
|
|
|
}),
|
2010-12-29 23:33:30 -05:00
|
|
|
|
2011-01-11 19:03:30 -05:00
|
|
|
Foo = {},
|
|
|
|
PlainFoo = Class.extend(),
|
|
|
|
PlainFoo2 = {}
|
|
|
|
;
|
2010-12-29 23:33:30 -05:00
|
|
|
|
|
|
|
|
2011-01-11 19:03:30 -05:00
|
|
|
( function testClassExportsContainImplementMethodToExtendFromNoBaseClass()
|
|
|
|
{
|
|
|
|
assert.ok(
|
|
|
|
( Class.implement instanceof Function ),
|
|
|
|
"Class provides method to implement interfaces"
|
|
|
|
);
|
|
|
|
} )();
|
2011-01-10 19:56:09 -05:00
|
|
|
|
2010-12-29 23:33:30 -05:00
|
|
|
|
2011-01-11 19:03:30 -05:00
|
|
|
( function testClassObjectsContainImplementMethodToImplementUsingItselfAsABase()
|
2010-12-29 23:33:30 -05:00
|
|
|
{
|
2011-01-11 19:03:30 -05:00
|
|
|
assert.ok(
|
|
|
|
( PlainFoo.implement instanceof Function ),
|
|
|
|
"Classes contain an implement() method"
|
|
|
|
);
|
|
|
|
} )();
|
|
|
|
|
|
|
|
|
|
|
|
( function testCanImplementInterfaceFromAnEmptyBase()
|
2011-01-10 19:56:09 -05:00
|
|
|
{
|
2011-01-11 19:03:30 -05:00
|
|
|
assert.doesNotThrow( function()
|
2011-01-04 00:37:54 -05:00
|
|
|
{
|
2011-01-11 19:03:30 -05:00
|
|
|
Foo = Class.implement( Type, Type2 );
|
|
|
|
}, Error, "Class can implement interfaces" );
|
|
|
|
|
|
|
|
assert.ok(
|
|
|
|
( Class.isClass( Foo ) ),
|
|
|
|
"Class returned from implementing interfaces on an empty base is a " +
|
|
|
|
"valid class"
|
|
|
|
);
|
|
|
|
} )();
|
|
|
|
|
|
|
|
|
|
|
|
( function testAbstractMethodsCopiedIntoNewClassUsingEmptyBase()
|
|
|
|
{
|
|
|
|
assert.ok(
|
|
|
|
( ( Foo.prototype.foo instanceof Function )
|
|
|
|
&& ( Foo.prototype.foo2 instanceof Function )
|
|
|
|
),
|
|
|
|
"Abstract methods are copied into the new class prototype (empty base)"
|
|
|
|
);
|
|
|
|
} )();
|
|
|
|
|
|
|
|
|
|
|
|
( function testCanImplementInterfaceAtopAnExistingClass()
|
|
|
|
{
|
|
|
|
assert.doesNotThrow( function()
|
|
|
|
{
|
|
|
|
PlainFoo2 = PlainFoo.implement( Type, Type2 );
|
|
|
|
}, Error, "Classes can implement interfaces" );
|
|
|
|
|
|
|
|
assert.ok(
|
|
|
|
( Class.isClass( PlainFoo2 ) ),
|
|
|
|
"Class returned from implementing interfaces on an existing base is a " +
|
|
|
|
"valid class"
|
|
|
|
);
|
|
|
|
} )();
|
|
|
|
|
|
|
|
|
|
|
|
( function testAbstractMethodsCopiedIntoNewClassUsingExistingBase()
|
|
|
|
{
|
|
|
|
assert.ok(
|
|
|
|
( ( PlainFoo2.prototype.foo instanceof Function )
|
|
|
|
&& ( PlainFoo2.prototype.foo2 instanceof Function )
|
|
|
|
),
|
|
|
|
"Abstract methods are copied into the new class prototype (concrete base)"
|
|
|
|
);
|
|
|
|
} )();
|
2011-01-04 00:37:54 -05:00
|
|
|
|
|
|
|
|
2011-01-11 19:03:30 -05:00
|
|
|
( function testClassesImplementingInterfacesAreConsideredAbstractByDefault()
|
|
|
|
{
|
|
|
|
assert.equal(
|
|
|
|
( Foo.isAbstract() && PlainFoo2.isAbstract() ),
|
|
|
|
true,
|
|
|
|
"Classes that implements interface(s) are considered abstract if the " +
|
|
|
|
"implemented methods have no concrete implementations"
|
|
|
|
);
|
|
|
|
} ) ();
|
|
|
|
|
|
|
|
|
|
|
|
( function testAbstractMethodListUpdatedWhenInterfaceImplemented()
|
|
|
|
{
|
|
|
|
// no base
|
|
|
|
assert.equal(
|
|
|
|
Foo.abstractMethods.length,
|
|
|
|
2,
|
|
|
|
"Abstract methods list is updated when interface is implemented " +
|
|
|
|
"(empty base)"
|
|
|
|
);
|
|
|
|
|
|
|
|
// PlainFoo base
|
|
|
|
assert.equal(
|
|
|
|
PlainFoo2.abstractMethods.length,
|
|
|
|
2,
|
|
|
|
"Abstract methods list is updated when interface is implemented " +
|
|
|
|
"(concrete base)"
|
|
|
|
);
|
|
|
|
} )();
|
|
|
|
|
|
|
|
|
|
|
|
( function testProperAbstractMethodsAreCopiedFromInterface()
|
|
|
|
{
|
|
|
|
assert.ok(
|
|
|
|
( ( Foo.abstractMethods[ 0 ] == 'foo' )
|
|
|
|
&& ( Foo.abstractMethods[ 1 ] == 'foo2' )
|
|
|
|
)
|
|
|
|
&& ( ( PlainFoo2.abstractMethods[ 0 ] == 'foo' )
|
|
|
|
&& ( PlainFoo2.abstractMethods[ 1 ] == 'foo2' )
|
|
|
|
),
|
|
|
|
"Abstract methods list contains names of implemented methods"
|
|
|
|
);
|
|
|
|
} )();
|
|
|
|
|
|
|
|
|
|
|
|
( function testInstancesOfClassesAreInstancesOfTheirImplementedInterfaces()
|
|
|
|
{
|
|
|
|
// concrete implementation so that we can instantiate it
|
|
|
|
var ConcreteFoo = Foo.extend(
|
|
|
|
{
|
|
|
|
foo: function() {},
|
|
|
|
foo2: function() {},
|
|
|
|
}),
|
|
|
|
|
|
|
|
concrete_inst = new ConcreteFoo();
|
|
|
|
|
|
|
|
assert.ok(
|
|
|
|
( concrete_inst.isInstanceOf( Type )
|
|
|
|
&& concrete_inst.isInstanceOf( Type2 )
|
|
|
|
),
|
|
|
|
"Instances of classes implementing interfaces are considered to be " +
|
|
|
|
"instances of the implemented interfaces"
|
|
|
|
);
|
|
|
|
} )();
|
2011-01-04 00:37:54 -05:00
|
|
|
|