2010-12-01 19:27:40 -05:00
|
|
|
/**
|
|
|
|
* 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
|
|
|
|
*/
|
|
|
|
|
2011-01-24 23:35:45 -05:00
|
|
|
var util = require( './util' ),
|
|
|
|
member_builder = require( './member_builder' ),
|
|
|
|
Class = require( './class' );
|
2010-12-01 19:27:40 -05:00
|
|
|
|
|
|
|
|
2011-03-03 19:08:24 -05:00
|
|
|
/**
|
|
|
|
* This module may be invoked in order to provide a more natural looking
|
|
|
|
* interface definition
|
|
|
|
*
|
|
|
|
* Only new interfaces may be created using this method. They cannot be
|
|
|
|
* extended. To extend an existing interface, call its extend() method, or use
|
|
|
|
* the extend() method of this module.
|
|
|
|
*
|
|
|
|
* @param {Object} def interface definition
|
|
|
|
*
|
|
|
|
* @return {Interface} new interface
|
|
|
|
*/
|
|
|
|
module.exports = function( def )
|
|
|
|
{
|
|
|
|
// if the first argument is an object, then we are declaring an interface
|
|
|
|
if ( typeof def !== 'object' )
|
|
|
|
{
|
|
|
|
throw TypeError(
|
|
|
|
"Must provide interface definition when declaring interface"
|
|
|
|
);
|
|
|
|
}
|
|
|
|
|
|
|
|
// ensure we have the proper number of arguments (if they passed in too
|
|
|
|
// many, it may signify that they don't know what they're doing, and likely
|
|
|
|
// they're not getting the result they're looking for)
|
|
|
|
if ( arguments.length > 1 )
|
|
|
|
{
|
|
|
|
throw Error(
|
|
|
|
"Expecting one argument for Interface definition; " +
|
|
|
|
arguments.length + " given."
|
|
|
|
);
|
|
|
|
}
|
|
|
|
|
|
|
|
return extend( def );
|
|
|
|
};
|
|
|
|
|
|
|
|
|
2010-12-01 19:27:40 -05:00
|
|
|
/**
|
|
|
|
* Creates an interface
|
|
|
|
*
|
|
|
|
* @return {Interface} extended interface
|
|
|
|
*/
|
2011-03-03 19:08:24 -05:00
|
|
|
module.exports.extend = function()
|
2010-12-01 19:27:40 -05:00
|
|
|
{
|
|
|
|
return extend.apply( this, arguments );
|
2010-12-28 22:10:12 -05:00
|
|
|
};
|
2010-12-01 19:27:40 -05:00
|
|
|
|
|
|
|
|
|
|
|
/**
|
|
|
|
* Default interface implementation
|
|
|
|
*
|
|
|
|
* @return {undefined}
|
|
|
|
*/
|
|
|
|
function Interface() {}
|
|
|
|
|
|
|
|
|
2010-12-29 22:40:23 -05:00
|
|
|
var extend = ( function( extending )
|
2010-12-01 19:27:40 -05:00
|
|
|
{
|
2010-12-29 22:40:23 -05:00
|
|
|
return function extend()
|
|
|
|
{
|
|
|
|
// ensure we'll be permitted to instantiate interfaces for the base
|
|
|
|
extending = true;
|
2010-12-01 19:27:40 -05:00
|
|
|
|
2010-12-29 22:40:23 -05:00
|
|
|
var args = Array.prototype.slice.call( arguments ),
|
|
|
|
props = args.pop() || {},
|
|
|
|
base = args.pop() || Interface,
|
2011-01-24 23:35:45 -05:00
|
|
|
prototype = new base(),
|
|
|
|
|
|
|
|
members = member_builder.initMembers(
|
|
|
|
prototype, prototype, prototype
|
|
|
|
)
|
|
|
|
;
|
2010-12-28 21:56:55 -05:00
|
|
|
|
2010-12-29 22:40:23 -05:00
|
|
|
// sanity check
|
|
|
|
inheritCheck( prototype );
|
2010-12-01 23:01:20 -05:00
|
|
|
|
2010-12-29 22:40:23 -05:00
|
|
|
var new_interface = createInterface();
|
|
|
|
|
2011-01-24 23:35:45 -05:00
|
|
|
util.propParse( props, {
|
|
|
|
property: function()
|
2010-12-28 20:13:50 -05:00
|
|
|
{
|
2011-01-24 23:35:45 -05:00
|
|
|
throw TypeError(
|
|
|
|
"Properties are not permitted within Interface " +
|
2011-03-01 09:17:24 -05:00
|
|
|
"definitions (did you forget the 'abstract' keyword?)"
|
2011-01-24 23:35:45 -05:00
|
|
|
);
|
|
|
|
},
|
|
|
|
|
2011-01-24 23:56:54 -05:00
|
|
|
getter: function()
|
|
|
|
{
|
|
|
|
throw TypeError(
|
|
|
|
"Getters are not permitted within Interface definitions"
|
|
|
|
);
|
|
|
|
},
|
|
|
|
|
|
|
|
setter: function()
|
|
|
|
{
|
|
|
|
throw TypeError(
|
|
|
|
"Setters are not permitted within Interface definitions"
|
|
|
|
);
|
|
|
|
},
|
|
|
|
|
2011-01-24 23:35:45 -05:00
|
|
|
method: function( name, value, is_abstract, keywords )
|
|
|
|
{
|
|
|
|
if ( !is_abstract )
|
2010-12-29 22:40:23 -05:00
|
|
|
{
|
|
|
|
throw TypeError(
|
2011-01-24 23:35:45 -05:00
|
|
|
"Only abstract methods are permitted within " +
|
|
|
|
"Interface definitions"
|
2010-12-29 22:40:23 -05:00
|
|
|
);
|
|
|
|
}
|
2011-01-24 23:35:45 -05:00
|
|
|
|
|
|
|
member_builder.buildMethod(
|
|
|
|
members, null, name, value, keywords
|
|
|
|
);
|
2010-12-29 22:40:23 -05:00
|
|
|
},
|
|
|
|
} );
|
|
|
|
|
|
|
|
attachExtend( new_interface );
|
|
|
|
new_interface.prototype = prototype;
|
|
|
|
new_interface.constructor = new_interface;
|
|
|
|
|
|
|
|
// freeze the interface (preventing additions), if supported
|
|
|
|
util.freeze( new_interface );
|
2010-12-01 23:01:20 -05:00
|
|
|
|
2010-12-29 22:40:23 -05:00
|
|
|
// we're done; let's not allow interfaces to be instantiated anymore
|
|
|
|
extending = false;
|
2010-12-01 19:27:40 -05:00
|
|
|
|
2010-12-29 22:40:23 -05:00
|
|
|
return new_interface;
|
|
|
|
};
|
2010-12-01 19:27:40 -05:00
|
|
|
|
2010-12-29 22:40:23 -05:00
|
|
|
|
|
|
|
/**
|
|
|
|
* 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 );
|
2010-12-01 19:27:40 -05:00
|
|
|
|
2010-12-01 19:38:30 -05:00
|
|
|
|
2010-12-28 21:56:55 -05:00
|
|
|
/**
|
|
|
|
* 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."
|
|
|
|
);
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
|
2010-12-01 23:27:31 -05:00
|
|
|
/**
|
|
|
|
* Attaches extend method to the given function (interface)
|
|
|
|
*
|
|
|
|
* @param {Function} func function (interface) to attach method to
|
|
|
|
*
|
|
|
|
* @return {undefined}
|
|
|
|
*/
|
2010-12-28 22:08:30 -05:00
|
|
|
function attachExtend( func )
|
2010-12-01 23:27:31 -05:00
|
|
|
{
|
|
|
|
/**
|
|
|
|
* 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 );
|
|
|
|
});
|
|
|
|
}
|
|
|
|
|