1
0
Fork 0
easejs/lib/interface.js

204 lines
5.4 KiB
JavaScript

/**
* Contains interface module
*
* 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 core
*/
var util = require( './util' ),
member_builder = require( './member_builder' ),
Class = require( './class' );
/**
* Creates an interface
*
* @return {Interface} extended interface
*/
exports.extend = function()
{
return extend.apply( this, arguments );
};
/**
* Default interface implementation
*
* @return {undefined}
*/
function Interface() {}
var extend = ( function( extending )
{
return function extend()
{
// ensure we'll be permitted to instantiate interfaces for the base
extending = true;
var args = Array.prototype.slice.call( arguments ),
props = args.pop() || {},
base = args.pop() || Interface,
prototype = new base(),
members = member_builder.initMembers(
prototype, prototype, prototype
)
;
// sanity check
inheritCheck( prototype );
var new_interface = createInterface();
util.propParse( props, {
property: function()
{
throw TypeError(
"Properties are not permitted within Interface " +
"definitions"
);
},
getter: function()
{
throw TypeError(
"Getters are not permitted within Interface definitions"
);
},
setter: function()
{
throw TypeError(
"Setters are not permitted within Interface definitions"
);
},
method: function( name, value, is_abstract, keywords )
{
if ( !is_abstract )
{
throw TypeError(
"Only abstract methods are permitted within " +
"Interface definitions"
);
}
member_builder.buildMethod(
members, null, name, value, keywords
);
},
} );
attachExtend( new_interface );
new_interface.prototype = prototype;
new_interface.constructor = new_interface;
// freeze the interface (preventing additions), if supported
util.freeze( new_interface );
// we're done; let's not allow interfaces to be instantiated anymore
extending = false;
return new_interface;
};
/**
* Creates a new interface constructor function
*
* @return {function()}
*/
function createInterface()
{
return function()
{
// allows us to extend the interface without throwing an exception
// (since the prototype requires an instance)
if ( !extending )
{
// only called if someone tries to create a new instance of an
// interface
throw Error( "Interfaces cannot be instantiated" );
}
};
}
} )( false );
/**
* Assures that the parent object is a valid object to inherit from
*
* This method allows inheriting from any object (note that it will likely cause
* errors if not an interface), but will place restrictions on objects like
* Classes that do not make sense to inherit from. This will provide a more
* friendly error, with suggestions on how to resolve the issue, rather than a
* cryptic error resulting from inheritance problems.
*
* This method will throw an exception if there is a violation.
*
* @param {Object} prototype prototype to check for inheritance flaws
*
* @return {undefined}
*/
function inheritCheck( prototype )
{
// if we're inheriting from another interface, then we're good
if ( prototype instanceof Interface )
{
return;
}
if ( Class.isClassInstance( prototype ) )
{
throw new TypeError(
"Interfaces cannot extend from classes. Try creating an " +
"abstract class instead."
);
}
}
/**
* Attaches extend method to the given function (interface)
*
* @param {Function} func function (interface) to attach method to
*
* @return {undefined}
*/
function attachExtend( func )
{
/**
* Shorthand for extending interfaces
*
* This method can be invoked on the object, rather than having to call
* Interface.extend( this ).
*
* @param {Object} props properties to add to extended interface
*
* @return {Object} extended interface
*/
util.defineSecureProp( func, 'extend', function( props )
{
return extend( this, props );
});
}