| 
									
										
										
										
											2014-12-05 00:10:01 +03:00
										 |  |  | /********************************************************************** | 
					
						
							|  |  |  | *  | 
					
						
							|  |  |  | * | 
					
						
							|  |  |  | * | 
					
						
							|  |  |  | **********************************************************************/ | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | define(function(require){ var module = {} | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | //var DEBUG = DEBUG != null ? DEBUG : true
 | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2015-12-31 10:37:21 +03:00
										 |  |  | args2array = require('lib/util').args2array | 
					
						
							| 
									
										
										
										
											2015-12-31 22:23:32 +03:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2014-12-09 18:02:24 +03:00
										 |  |  | actions = require('lib/actions') | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2014-12-05 00:10:01 +03:00
										 |  |  | 
 | 
					
						
							|  |  |  | /*********************************************************************/ | 
					
						
							|  |  |  | //
 | 
					
						
							|  |  |  | // Feature attributes:
 | 
					
						
							|  |  |  | // 	.tag			- feature tag (string)
 | 
					
						
							|  |  |  | // 					  this is used to identify the feature, its event handlers
 | 
					
						
							|  |  |  | // 					  and DOM elements.
 | 
					
						
							|  |  |  | //
 | 
					
						
							|  |  |  | // 	.title			- feature name (string | null)
 | 
					
						
							|  |  |  | // 	.doc			- feature description (string | null)
 | 
					
						
							|  |  |  | //
 | 
					
						
							|  |  |  | // 	.priority		- feature priority
 | 
					
						
							|  |  |  | // 					  can be:
 | 
					
						
							|  |  |  | // 					  	- 'high' (99) | 'medium' (0) | 'low' (-99)
 | 
					
						
							|  |  |  | // 					  	- number
 | 
					
						
							|  |  |  | // 					  	- null (0, default)
 | 
					
						
							|  |  |  | // 					  features with higher priority will be setup first,
 | 
					
						
							|  |  |  | // 					  features with the same priority will be run in order of
 | 
					
						
							|  |  |  | // 					  occurrence.
 | 
					
						
							| 
									
										
										
										
											2014-12-09 16:34:14 +03:00
										 |  |  | // 	.suggested		- list of optional suggested features, these are not 
 | 
					
						
							|  |  |  | // 					  required but setup if available.
 | 
					
						
							|  |  |  | // 					  This is useful for defining meta features but without
 | 
					
						
							|  |  |  | // 					  making each sub-feature a strict dependency.
 | 
					
						
							| 
									
										
										
										
											2014-12-05 00:10:01 +03:00
										 |  |  | // 	.depends		- feature dependencies -- tags of features that must setup
 | 
					
						
							|  |  |  | // 					  before the feature (list | null)
 | 
					
						
							|  |  |  | // 	.exclusive		- feature exclusivity tags (list | null)
 | 
					
						
							|  |  |  | // 					  an exclusivity group enforces that only one feature in
 | 
					
						
							|  |  |  | // 					  it will be run, i.e. the first / highest priority.
 | 
					
						
							|  |  |  | //
 | 
					
						
							|  |  |  | // 	.actions		- action object containing feature actions (ActionSet | null)
 | 
					
						
							|  |  |  | // 					  this will be mixed into the base object on .setup()
 | 
					
						
							|  |  |  | // 					  and mixed out on .remove()
 | 
					
						
							|  |  |  | // 	.config			- feature configuration, will be merged with base 
 | 
					
						
							|  |  |  | // 					  object's .config
 | 
					
						
							|  |  |  | // 	.handlers		- feature event handlers (list | null)
 | 
					
						
							|  |  |  | // 
 | 
					
						
							|  |  |  | //
 | 
					
						
							|  |  |  | //
 | 
					
						
							|  |  |  | // .handlers format:
 | 
					
						
							|  |  |  | // 	[
 | 
					
						
							|  |  |  | // 		[ <event-spec>, <handler-function> ],
 | 
					
						
							|  |  |  | // 		...
 | 
					
						
							|  |  |  | // 	]
 | 
					
						
							|  |  |  | //
 | 
					
						
							|  |  |  | // NOTE: both <event-spec> and <handler-function> must be compatible with
 | 
					
						
							|  |  |  | // 		Action.on(..)
 | 
					
						
							|  |  |  | //
 | 
					
						
							| 
									
										
										
										
											2014-12-06 15:26:20 +03:00
										 |  |  | //
 | 
					
						
							|  |  |  | // Feature applicability:
 | 
					
						
							|  |  |  | // 	If feature.isApplicable(..) returns true then the feature will not be
 | 
					
						
							|  |  |  | // 	considered on setup...
 | 
					
						
							|  |  |  | //
 | 
					
						
							|  |  |  | //
 | 
					
						
							| 
									
										
										
										
											2014-12-05 00:10:01 +03:00
										 |  |  | // XXX this could install the handlers in two locations:
 | 
					
						
							|  |  |  | // 		- mixin if available...
 | 
					
						
							|  |  |  | // 		- base object (currently implemented)
 | 
					
						
							|  |  |  | // 		should the first be done?
 | 
					
						
							|  |  |  | var FeatureProto = | 
					
						
							|  |  |  | module.FeatureProto = { | 
					
						
							|  |  |  | 	tag: null, | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | 	isApplicable: function(actions){ | 
					
						
							|  |  |  | 		return true | 
					
						
							|  |  |  | 	}, | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | 	getPriority: function(){ | 
					
						
							|  |  |  | 		var res = this.priority || 0 | 
					
						
							|  |  |  | 		return res == 'high' ? 99 | 
					
						
							|  |  |  | 			: res == 'low' ? -99 | 
					
						
							|  |  |  | 			: res == 'medium' ? 0 | 
					
						
							|  |  |  | 			: res | 
					
						
							|  |  |  | 	}, | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | 	setup: function(actions){ | 
					
						
							|  |  |  | 		var that = this | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | 		// mixin actions...
 | 
					
						
							|  |  |  | 		if(this.actions != null){ | 
					
						
							|  |  |  | 			actions.mixin(this.actions) | 
					
						
							|  |  |  | 		} | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | 		// install handlers...
 | 
					
						
							|  |  |  | 		if(this.handlers != null){ | 
					
						
							|  |  |  | 			this.handlers.forEach(function(h){ | 
					
						
							|  |  |  | 				actions.on(h[0], that.tag, h[1]) | 
					
						
							|  |  |  | 			}) | 
					
						
							|  |  |  | 		} | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | 		// merge config...
 | 
					
						
							|  |  |  | 		// XXX should this use inheritance???
 | 
					
						
							| 
									
										
										
										
											2015-12-20 08:12:56 +03:00
										 |  |  | 		// XXX do we need to clone .config?
 | 
					
						
							| 
									
										
										
										
											2014-12-05 00:10:01 +03:00
										 |  |  | 		if(this.config != null  | 
					
						
							|  |  |  | 				|| (this.actions != null  | 
					
						
							|  |  |  | 					&& this.actions.config != null)){ | 
					
						
							| 
									
										
										
										
											2015-11-29 20:46:15 +03:00
										 |  |  | 			var config = this.config = this.config || this.actions.config | 
					
						
							| 
									
										
										
										
											2014-12-05 00:10:01 +03:00
										 |  |  | 
 | 
					
						
							|  |  |  | 			if(actions.config == null){ | 
					
						
							|  |  |  | 				actions.config = {} | 
					
						
							|  |  |  | 			} | 
					
						
							|  |  |  | 			Object.keys(config).forEach(function(n){ | 
					
						
							| 
									
										
										
										
											2015-12-20 04:29:44 +03:00
										 |  |  | 				// NOTE: this will overwrite existing values...
 | 
					
						
							|  |  |  | 				actions.config[n] = config[n] | 
					
						
							| 
									
										
										
										
											2014-12-05 00:10:01 +03:00
										 |  |  | 			}) | 
					
						
							|  |  |  | 		} | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | 		// custom setup...
 | 
					
						
							|  |  |  | 		// XXX is this the correct way???
 | 
					
						
							|  |  |  | 		if(this.hasOwnProperty('setup') && this.setup !== FeatureProto.setup){ | 
					
						
							|  |  |  | 			this.setup(actions) | 
					
						
							|  |  |  | 		} | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | 		return this | 
					
						
							|  |  |  | 	}, | 
					
						
							|  |  |  | 	remove: function(actions){ | 
					
						
							|  |  |  | 		if(this.actions != null){ | 
					
						
							|  |  |  | 			actions.mixout(this.actions) | 
					
						
							|  |  |  | 		} | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | 		if(this.handlers != null){ | 
					
						
							|  |  |  | 			actions.off('*', this.tag) | 
					
						
							|  |  |  | 		} | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | 		if(this.hasOwnProperty('remove') && this.setup !== FeatureProto.remove){ | 
					
						
							|  |  |  | 			this.remove(actions) | 
					
						
							|  |  |  | 		} | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | 		// remove feature DOM elements...
 | 
					
						
							|  |  |  | 		actions.ribbons.viewer.find('.' + this.tag).remove() | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | 		return this | 
					
						
							|  |  |  | 	}, | 
					
						
							|  |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | // XXX is hard-coded default feature-set a good way to go???
 | 
					
						
							| 
									
										
										
										
											2014-12-09 18:02:24 +03:00
										 |  |  | //
 | 
					
						
							|  |  |  | // 	Feature(obj)
 | 
					
						
							|  |  |  | // 		-> feature
 | 
					
						
							|  |  |  | //
 | 
					
						
							|  |  |  | // 	Feature(feature-set, obj)
 | 
					
						
							|  |  |  | // 		-> feature
 | 
					
						
							|  |  |  | //
 | 
					
						
							|  |  |  | // 	Feature(tag, obj)
 | 
					
						
							|  |  |  | // 		-> feature
 | 
					
						
							|  |  |  | //
 | 
					
						
							|  |  |  | //
 | 
					
						
							|  |  |  | // 	Feature(tag, actions)
 | 
					
						
							|  |  |  | // 		-> feature
 | 
					
						
							|  |  |  | //
 | 
					
						
							|  |  |  | // 	Feature(feature-set, tag, actions)
 | 
					
						
							|  |  |  | // 		-> feature
 | 
					
						
							|  |  |  | //
 | 
					
						
							| 
									
										
										
										
											2014-12-05 00:10:01 +03:00
										 |  |  | var Feature = | 
					
						
							|  |  |  | module.Feature = | 
					
						
							| 
									
										
										
										
											2014-12-09 18:02:24 +03:00
										 |  |  | function Feature(feature_set, tag, obj){ | 
					
						
							|  |  |  | 	if(arguments.length == 2){ | 
					
						
							|  |  |  | 		// Feature(<tag>, <obj>)
 | 
					
						
							|  |  |  | 		if(typeof(feature_set) == typeof('str')){ | 
					
						
							|  |  |  | 			obj = tag | 
					
						
							|  |  |  | 			tag = feature_set | 
					
						
							|  |  |  | 			feature_set = Features | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | 		// Feature(<feature-set>, <obj>)
 | 
					
						
							|  |  |  | 		} else { | 
					
						
							|  |  |  | 			obj = tag | 
					
						
							|  |  |  | 			tag = null | 
					
						
							|  |  |  | 		} | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | 	// Feature(<obj>)
 | 
					
						
							|  |  |  | 	} else if(arguments.length == 1){ | 
					
						
							| 
									
										
										
										
											2014-12-05 00:10:01 +03:00
										 |  |  | 		obj = feature_set | 
					
						
							|  |  |  | 		feature_set = Features | 
					
						
							| 
									
										
										
										
											2014-12-09 18:02:24 +03:00
										 |  |  | 	} | 
					
						
							| 
									
										
										
										
											2014-12-09 16:34:14 +03:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2014-12-09 18:02:24 +03:00
										 |  |  | 	if(tag != null && obj.tag != null && obj.tag != tag){ | 
					
						
							|  |  |  | 		throw 'Error: tag and obj.tag mismatch, either use one or both must match.' | 
					
						
							| 
									
										
										
										
											2014-12-05 00:10:01 +03:00
										 |  |  | 	} | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2014-12-09 18:02:24 +03:00
										 |  |  | 	// action...
 | 
					
						
							|  |  |  | 	if(obj instanceof actions.Action){ | 
					
						
							|  |  |  | 		if(tag == null){ | 
					
						
							|  |  |  | 			throw 'Error: need a tag to make a feature out of an action' | 
					
						
							|  |  |  | 		} | 
					
						
							|  |  |  | 		var f = { | 
					
						
							|  |  |  | 			tag: tag, | 
					
						
							|  |  |  | 			actions: obj, | 
					
						
							|  |  |  | 		} | 
					
						
							|  |  |  | 		obj = f | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | 	// meta-feature...
 | 
					
						
							|  |  |  | 	} else if(obj.constructor === Array){ | 
					
						
							|  |  |  | 		if(tag == null){ | 
					
						
							|  |  |  | 			throw 'Error: need a tag to make a meta-feature' | 
					
						
							|  |  |  | 		} | 
					
						
							|  |  |  | 		var f = { | 
					
						
							|  |  |  | 			tag: tag, | 
					
						
							|  |  |  | 			suggested: obj, | 
					
						
							|  |  |  | 		} | 
					
						
							|  |  |  | 		obj = f | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | 	// feature...
 | 
					
						
							|  |  |  | 	} else { | 
					
						
							|  |  |  | 		obj.__proto__ = FeatureProto | 
					
						
							|  |  |  | 	} | 
					
						
							| 
									
										
										
										
											2014-12-05 00:10:01 +03:00
										 |  |  | 
 | 
					
						
							|  |  |  | 	if(feature_set){ | 
					
						
							|  |  |  | 		feature_set[obj.tag] = obj | 
					
						
							|  |  |  | 	} | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | 	return obj | 
					
						
							|  |  |  | } | 
					
						
							|  |  |  | Feature.prototype = FeatureProto | 
					
						
							|  |  |  | Feature.prototype.constructor = Feature | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | var FeatureSet = | 
					
						
							|  |  |  | module.FeatureSet = { | 
					
						
							| 
									
										
										
										
											2015-11-05 02:27:26 +03:00
										 |  |  | 	__feature__: Feature, | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2015-12-13 19:31:03 +03:00
										 |  |  | 	// if true, .setup(..) will report things it's doing... 
 | 
					
						
							|  |  |  | 	__verbose__: null, | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2016-01-03 04:49:00 +03:00
										 |  |  | 	// List of registered features...
 | 
					
						
							|  |  |  | 	get features(){ | 
					
						
							|  |  |  | 		var that = this | 
					
						
							|  |  |  | 		return Object.keys(this) | 
					
						
							|  |  |  | 			.filter(function(e){  | 
					
						
							|  |  |  | 				return e != 'features'  | 
					
						
							|  |  |  | 					&& that[e] instanceof Feature })  | 
					
						
							|  |  |  | 	}, | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2014-12-06 15:37:09 +03:00
										 |  |  | 	// Build feature list...
 | 
					
						
							|  |  |  | 	//
 | 
					
						
							|  |  |  | 	// 	Build a list of all registered features
 | 
					
						
							|  |  |  | 	// 	.buildFeatureList()
 | 
					
						
							|  |  |  | 	// 	.buildFeatureList(actions)
 | 
					
						
							|  |  |  | 	// 		-> list
 | 
					
						
							|  |  |  | 	//
 | 
					
						
							|  |  |  | 	// 	Build a list of given features
 | 
					
						
							|  |  |  | 	// 	.buildFeatureList(null, list)
 | 
					
						
							|  |  |  | 	// 	.buildFeatureList(actions, list)
 | 
					
						
							|  |  |  | 	// 		-> list
 | 
					
						
							|  |  |  | 	//
 | 
					
						
							|  |  |  | 	//
 | 
					
						
							|  |  |  | 	// NOTE: some feature .isApplicable(..) may expect the action set thus
 | 
					
						
							|  |  |  | 	// 		making it required for building a feature list.
 | 
					
						
							|  |  |  | 	// NOTE: this will try and keep the order as close as possible to the
 | 
					
						
							|  |  |  | 	// 		original as possible, this if the list is correctly ordered 
 | 
					
						
							|  |  |  | 	// 		it will not be affected...
 | 
					
						
							|  |  |  | 	// NOTE: this will fix most dependency ordering errors except for two:
 | 
					
						
							| 
									
										
										
										
											2014-12-06 15:26:20 +03:00
										 |  |  | 	// 		- cyclic dependencies
 | 
					
						
							|  |  |  | 	// 			e.g. a -> b and b -> a, here there is no way to reorder
 | 
					
						
							|  |  |  | 	// 				a and b to resolve this.
 | 
					
						
							|  |  |  | 	// 		- dependency / priority conflict
 | 
					
						
							|  |  |  | 	// 			e.g. a -> b but a has a higher priority that b thus 
 | 
					
						
							|  |  |  | 	// 				making it impossible to order the two without 
 | 
					
						
							|  |  |  | 	// 				breaking either the dependency or priority ordering.
 | 
					
						
							| 
									
										
										
										
											2014-12-08 17:37:29 +03:00
										 |  |  | 	//
 | 
					
						
							|  |  |  | 	//
 | 
					
						
							|  |  |  | 	// Dependency sorting:
 | 
					
						
							|  |  |  | 	//
 | 
					
						
							|  |  |  | 	// These are order dependencies, i.e. for a dependency to be 
 | 
					
						
							|  |  |  | 	// resolved it must satisfy ALL of the folowing:
 | 
					
						
							|  |  |  | 	// 	- all dependencies must exist in the list.
 | 
					
						
							|  |  |  | 	// 	- all dependencies must be positiond/setup before the dependant.
 | 
					
						
							|  |  |  | 	//
 | 
					
						
							|  |  |  | 	// The general algorithm is as folows:
 | 
					
						
							|  |  |  | 	// 	1) place the dependencies befeore the dependant for each element
 | 
					
						
							|  |  |  | 	// 	2) remove the duplicate features except fot the first occurance
 | 
					
						
							|  |  |  | 	// 	3) repeat 1 and 2 for 2 to depth times or until the feature list
 | 
					
						
							|  |  |  | 	// 		stabelizes, i.e. no new features are added on the last run.
 | 
					
						
							|  |  |  | 	//
 | 
					
						
							|  |  |  | 	// NOTE: if auto_include is true (default) this will add dependencies
 | 
					
						
							|  |  |  | 	// 		as they are needed...
 | 
					
						
							|  |  |  | 	// 		This is useful for "meta-features" that do nothing other than
 | 
					
						
							|  |  |  | 	// 		depend/include sets of other features, for exmale: 'ui', 
 | 
					
						
							|  |  |  | 	// 		'core', 'browser', ...etc.
 | 
					
						
							|  |  |  | 	// NOTE: dependency chains larger than depth will be dropped, this 
 | 
					
						
							|  |  |  | 	// 		can be fixed by setting a greater depth (default: 8)...
 | 
					
						
							|  |  |  | 	// NOTE: conflicts that can occur and can not be recovered from:
 | 
					
						
							|  |  |  | 	// 		- cyclic dependency
 | 
					
						
							|  |  |  | 	// 			X will be before one of its dependencies...
 | 
					
						
							|  |  |  | 	// 		- dependency / priority conflict
 | 
					
						
							|  |  |  | 	// 			X will have higher priority than one of its dependencies...
 | 
					
						
							| 
									
										
										
										
											2015-11-05 02:27:26 +03:00
										 |  |  | 	// NOTE: feature that depend in unapplicable features are considered
 | 
					
						
							|  |  |  | 	// 		unapplicable.
 | 
					
						
							|  |  |  | 	// 		XXX not sure if this is 100% correct...
 | 
					
						
							|  |  |  | 	// NOTE: child high priority features will push their dependencies up
 | 
					
						
							|  |  |  | 	// 		to precede them.
 | 
					
						
							|  |  |  | 	// 		...this will not resolve all the possible conflicts so be 
 | 
					
						
							|  |  |  | 	// 		careful.
 | 
					
						
							|  |  |  | 	//
 | 
					
						
							|  |  |  | 	// XXX make suggested feature expansion recursive...
 | 
					
						
							| 
									
										
										
										
											2016-01-03 04:49:00 +03:00
										 |  |  | 	// XXX this appears to be very slow if lst not passed...
 | 
					
						
							| 
									
										
										
										
											2016-04-03 06:30:30 +03:00
										 |  |  | 	// XXX add ability to remove features by prefixing a '-' to its name
 | 
					
						
							|  |  |  | 	// 		...if one -<feature> is present, remove all instances of 
 | 
					
						
							|  |  |  | 	// 		<feature>
 | 
					
						
							| 
									
										
										
										
											2016-04-20 00:43:20 +03:00
										 |  |  | 	// XXX .buildFeatureList() is slow and can be a bottleneck for large
 | 
					
						
							|  |  |  | 	// 		numbers of features... might be a good idea to take a look at
 | 
					
						
							|  |  |  | 	// 		this sometime...
 | 
					
						
							| 
									
										
										
										
											2014-12-08 17:37:29 +03:00
										 |  |  | 	buildFeatureList: function(obj, lst, auto_include, depth){ | 
					
						
							| 
									
										
										
										
											2016-01-03 04:49:00 +03:00
										 |  |  | 		var that = this | 
					
						
							|  |  |  | 		obj = obj || {} | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | 		lst = lst == null ? this.features : lst | 
					
						
							| 
									
										
										
										
											2014-12-05 00:10:01 +03:00
										 |  |  | 		lst = lst.constructor !== Array ? [lst] : lst | 
					
						
							| 
									
										
										
										
											2016-01-03 04:49:00 +03:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2014-12-08 17:37:29 +03:00
										 |  |  | 		auto_include = auto_include == null ? true : false | 
					
						
							|  |  |  | 		depth = depth || 8 | 
					
						
							| 
									
										
										
										
											2014-12-05 00:10:01 +03:00
										 |  |  | 
 | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2015-11-05 02:27:26 +03:00
										 |  |  | 		var missing = {} | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2014-12-08 17:37:29 +03:00
										 |  |  | 		// helpers...
 | 
					
						
							|  |  |  | 		// NOTE: _skipMissing(..) will add missing dependencies to missing...
 | 
					
						
							|  |  |  | 		var _skipMissing = function(feature, deps, missing){ | 
					
						
							|  |  |  | 			return deps.filter(function(d){  | 
					
						
							|  |  |  | 				if(lst.indexOf(d) < 0){ | 
					
						
							| 
									
										
										
										
											2015-11-05 02:27:26 +03:00
										 |  |  | 					missing[d] = missing[d] != null ? missing[d] : [] | 
					
						
							| 
									
										
										
										
											2014-12-08 17:37:29 +03:00
										 |  |  | 					if(missing[d].indexOf(feature) < 0){ | 
					
						
							|  |  |  | 						missing[d].push(feature) | 
					
						
							|  |  |  | 					} | 
					
						
							|  |  |  | 				} | 
					
						
							|  |  |  | 				return missing[d] == null | 
					
						
							|  |  |  | 			}) | 
					
						
							|  |  |  | 		} | 
					
						
							| 
									
										
										
										
											2015-11-05 02:27:26 +03:00
										 |  |  | 		var _sortDep = function(lst, missing, depth){ | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | 			do { | 
					
						
							|  |  |  | 				var res = [] | 
					
						
							|  |  |  | 				var l = lst.length | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | 				lst.forEach(function(n){ | 
					
						
							|  |  |  | 					var e = that[n] | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2015-12-17 03:34:20 +03:00
										 |  |  | 					if(!e){ | 
					
						
							| 
									
										
										
										
											2015-12-31 10:37:21 +03:00
										 |  |  | 						//console.warn('%s: feature is not loaded.', n)
 | 
					
						
							|  |  |  | 						// XXX should we break here???
 | 
					
						
							|  |  |  | 						throw (n + ': feature is not loaded.') | 
					
						
							| 
									
										
										
										
											2015-12-17 03:34:20 +03:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2015-11-05 02:27:26 +03:00
										 |  |  | 					// no dependencies...
 | 
					
						
							| 
									
										
										
										
											2015-12-31 10:37:21 +03:00
										 |  |  | 					} else if(e.depends == null || e.depends.length == 0){ | 
					
						
							| 
									
										
										
										
											2015-11-05 02:27:26 +03:00
										 |  |  | 						res.push(n) | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2014-12-08 17:37:29 +03:00
										 |  |  | 					} else { | 
					
						
							| 
									
										
										
										
											2015-11-05 02:27:26 +03:00
										 |  |  | 						// auto-include dependencies...
 | 
					
						
							|  |  |  | 						if(auto_include){ | 
					
						
							|  |  |  | 							var deps = e.depends | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | 						// skip dependencies that are not in list...
 | 
					
						
							|  |  |  | 						} else { | 
					
						
							|  |  |  | 							var deps = _skipMissing(n, e.depends, missing) | 
					
						
							|  |  |  | 						} | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | 						// place dependencies before the depended...
 | 
					
						
							|  |  |  | 						res = res.concat(deps) | 
					
						
							|  |  |  | 						res.push(n) | 
					
						
							| 
									
										
										
										
											2014-12-08 17:37:29 +03:00
										 |  |  | 					} | 
					
						
							| 
									
										
										
										
											2015-11-05 02:27:26 +03:00
										 |  |  | 				}) | 
					
						
							|  |  |  | 				lst = res | 
					
						
							|  |  |  | 				depth -= 1 | 
					
						
							|  |  |  | 			} while(lst.length != l && depth > 0) | 
					
						
							| 
									
										
										
										
											2014-12-08 14:38:42 +03:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2015-11-05 02:27:26 +03:00
										 |  |  | 			return lst | 
					
						
							|  |  |  | 		} | 
					
						
							|  |  |  | 		var _getSuggested = function(featureset, feature, suggested, missing){ | 
					
						
							|  |  |  | 			suggested = suggested || [] | 
					
						
							|  |  |  | 			 | 
					
						
							|  |  |  | 			var s = (feature.suggested || []) | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | 			s | 
					
						
							|  |  |  | 				// remove the already visited suggenstions...
 | 
					
						
							|  |  |  | 				.filter(function(e){ return suggested.indexOf(e) < 0 })  | 
					
						
							|  |  |  | 				// add unloaded features to missing...
 | 
					
						
							|  |  |  | 				.filter(function(e){  | 
					
						
							|  |  |  | 					if(featureset[e] == null){ | 
					
						
							|  |  |  | 						missing[e] = missing[e] != null ? missing[e] : [] | 
					
						
							|  |  |  | 						if(missing[e].indexOf(feature.tag) < 0){ | 
					
						
							|  |  |  | 							missing[e].push(feature.tag) | 
					
						
							|  |  |  | 						} | 
					
						
							|  |  |  | 						return false | 
					
						
							|  |  |  | 					} | 
					
						
							|  |  |  | 					return true | 
					
						
							|  |  |  | 				})  | 
					
						
							|  |  |  | 				// load new suggenstions...
 | 
					
						
							|  |  |  | 				.forEach(function(n){ | 
					
						
							|  |  |  | 					var e = featureset[n] | 
					
						
							|  |  |  | 					if(e != null && e.suggested != null){ | 
					
						
							|  |  |  | 						suggested = suggested | 
					
						
							|  |  |  | 							.concat(_getSuggested(featureset, e, suggested, missing)) | 
					
						
							|  |  |  | 							.unique() | 
					
						
							|  |  |  | 					} | 
					
						
							|  |  |  | 					suggested.push(n) | 
					
						
							|  |  |  | 				}) | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | 			return suggested | 
					
						
							| 
									
										
										
										
											2014-12-08 15:01:10 +03:00
										 |  |  | 		} | 
					
						
							| 
									
										
										
										
											2014-12-08 14:38:42 +03:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2014-12-09 16:34:14 +03:00
										 |  |  | 		// expand optional "suggested" features...
 | 
					
						
							| 
									
										
										
										
											2015-11-05 02:27:26 +03:00
										 |  |  | 		// XXX make this recursive...
 | 
					
						
							| 
									
										
										
										
											2014-12-09 16:34:14 +03:00
										 |  |  | 		var res = [] | 
					
						
							|  |  |  | 		lst.forEach(function(n){ | 
					
						
							|  |  |  | 			var e = that[n] | 
					
						
							|  |  |  | 			if(e != null && e.suggested != null){ | 
					
						
							| 
									
										
										
										
											2015-11-05 02:27:26 +03:00
										 |  |  | 				//res = res.concat(e.suggested)
 | 
					
						
							|  |  |  | 				res = res.concat(_getSuggested(that, e, res, missing)) | 
					
						
							| 
									
										
										
										
											2014-12-09 16:34:14 +03:00
										 |  |  | 			} | 
					
						
							|  |  |  | 			res.push(n) | 
					
						
							|  |  |  | 		}) | 
					
						
							|  |  |  | 		lst = res | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2014-12-08 17:37:29 +03:00
										 |  |  | 		// expand and sort dependencies...
 | 
					
						
							|  |  |  | 		// 	2+ times untill depth is 0 or length stabelizes...
 | 
					
						
							| 
									
										
										
										
											2015-11-05 02:27:26 +03:00
										 |  |  | 		lst = _sortDep(lst, missing, depth).unique() | 
					
						
							| 
									
										
										
										
											2014-12-06 15:26:20 +03:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2014-12-06 02:46:43 +03:00
										 |  |  | 		// sort features via priority keeping the order as close to 
 | 
					
						
							|  |  |  | 		// manual as possible...
 | 
					
						
							|  |  |  | 		var l = lst.length | 
					
						
							|  |  |  | 		lst = lst | 
					
						
							| 
									
										
										
										
											2014-12-08 14:38:42 +03:00
										 |  |  | 			// remove undefined and non-features...
 | 
					
						
							|  |  |  | 			.filter(function(e){ return that[e] != null  | 
					
						
							|  |  |  | 				&& that[e] instanceof Feature }) | 
					
						
							| 
									
										
										
										
											2014-12-06 02:46:43 +03:00
										 |  |  | 			// build the sort table: [ <priority>, <rev-index>, <elem> ]
 | 
					
						
							|  |  |  | 			// NOTE: <rev-index> is element number from the tail...
 | 
					
						
							|  |  |  | 			.map(function(e, i){ return [ -that[e].getPriority(), i, e ] }) | 
					
						
							|  |  |  | 			// do the sort...
 | 
					
						
							| 
									
										
										
										
											2014-12-08 14:38:42 +03:00
										 |  |  | 			// NOTE: for some reason JS compares lists as strings so we
 | 
					
						
							|  |  |  | 			// 		have to comare the list manually...
 | 
					
						
							| 
									
										
										
										
											2014-12-06 02:46:43 +03:00
										 |  |  | 			.sort(function(a, b){ return a[0] - b[0] || a[1] - b[1] }) | 
					
						
							| 
									
										
										
										
											2015-11-05 02:27:26 +03:00
										 |  |  | 			// cleanup -- drop the table...
 | 
					
						
							| 
									
										
										
										
											2014-12-06 02:46:43 +03:00
										 |  |  | 			.map(function(e){ return e[2] }) | 
					
						
							| 
									
										
										
										
											2014-12-05 00:10:01 +03:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2015-11-05 02:27:26 +03:00
										 |  |  | 		// sort dependencies again...
 | 
					
						
							|  |  |  | 		// NOTE: this bubles the "priority" up the dependency tree...
 | 
					
						
							|  |  |  | 		// NOTE: this will not resolve all the conflicts...
 | 
					
						
							|  |  |  | 		lst = _sortDep(lst, missing, depth).unique() | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2014-12-06 15:26:20 +03:00
										 |  |  | 		// clasify features...
 | 
					
						
							| 
									
										
										
										
											2014-12-05 00:10:01 +03:00
										 |  |  | 		var unapplicable = [] | 
					
						
							|  |  |  | 		var conflicts = {} | 
					
						
							| 
									
										
										
										
											2015-12-29 07:28:19 +03:00
										 |  |  | 		var that = this | 
					
						
							| 
									
										
										
										
											2014-12-05 00:10:01 +03:00
										 |  |  | 		lst = lst.filter(function(n, i){ | 
					
						
							|  |  |  | 			var e = that[n] | 
					
						
							|  |  |  | 			if(e == null){ | 
					
						
							|  |  |  | 				return true | 
					
						
							|  |  |  | 			} | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | 			// check applicability...
 | 
					
						
							| 
									
										
										
										
											2015-12-29 07:28:19 +03:00
										 |  |  | 			if(!e.isApplicable.call(that, obj)){ | 
					
						
							| 
									
										
										
										
											2014-12-05 00:10:01 +03:00
										 |  |  | 				unapplicable.push(n) | 
					
						
							|  |  |  | 				return false | 
					
						
							|  |  |  | 			} | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | 			// no dependencies...
 | 
					
						
							| 
									
										
										
										
											2014-12-06 15:26:20 +03:00
										 |  |  | 			if(e.depends == null || e.depends.length == 0){ | 
					
						
							| 
									
										
										
										
											2014-12-05 00:10:01 +03:00
										 |  |  | 				return true | 
					
						
							|  |  |  | 			} | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2015-11-05 02:27:26 +03:00
										 |  |  | 			// mark feature unapplicable if it depends on an unapplicable...
 | 
					
						
							|  |  |  | 			// NOTE: we need to do this once as features at this point
 | 
					
						
							|  |  |  | 			// 		are sorted by dependencies...
 | 
					
						
							|  |  |  | 			if(e.depends.filter(function(dep){ | 
					
						
							|  |  |  | 						return unapplicable.indexOf(dep) > -1  | 
					
						
							|  |  |  | 					}).length > 0){ | 
					
						
							|  |  |  | 				unapplicable.push(n) | 
					
						
							|  |  |  | 				return false | 
					
						
							|  |  |  | 			} | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2014-12-05 00:10:01 +03:00
										 |  |  | 			// keep only conflicting...
 | 
					
						
							|  |  |  | 			var deps = e.depends.filter(function(dep){ | 
					
						
							|  |  |  | 				dep = lst.indexOf(dep) | 
					
						
							|  |  |  | 				return dep == -1 || dep > i | 
					
						
							|  |  |  | 			}) | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2014-12-08 14:38:42 +03:00
										 |  |  | 			// skip missing dependencies...
 | 
					
						
							| 
									
										
										
										
											2014-12-08 17:37:29 +03:00
										 |  |  | 			// NOTE: we need to check for missing again as a feature 
 | 
					
						
							|  |  |  | 			// 		could have been removed due to inapplicability or
 | 
					
						
							|  |  |  | 			// 		being undefined...
 | 
					
						
							|  |  |  | 			deps = _skipMissing(n, deps, missing) | 
					
						
							| 
									
										
										
										
											2014-12-08 14:38:42 +03:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2014-12-05 00:10:01 +03:00
										 |  |  | 			// no conflicts...
 | 
					
						
							|  |  |  | 			if(deps.length == 0){ | 
					
						
							|  |  |  | 				return true | 
					
						
							|  |  |  | 			} | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2014-12-08 17:37:29 +03:00
										 |  |  | 			// dependency exists but in wrong order -- can't fix...
 | 
					
						
							| 
									
										
										
										
											2014-12-05 00:10:01 +03:00
										 |  |  | 			conflicts[n] = deps | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | 			return false | 
					
						
							|  |  |  | 		}) | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | 		// skip duplicate exclusive features...
 | 
					
						
							|  |  |  | 		var exclusive = [] | 
					
						
							|  |  |  | 		var excluded = [] | 
					
						
							|  |  |  | 		lst = lst.filter(function(n){ | 
					
						
							|  |  |  | 			var e = that[n] | 
					
						
							|  |  |  | 			if(e == null || e.exclusive == null ){ | 
					
						
							|  |  |  | 				return true | 
					
						
							|  |  |  | 			} | 
					
						
							|  |  |  | 			// count the number of exclusive features already present...
 | 
					
						
							|  |  |  | 			var res = e.exclusive | 
					
						
							|  |  |  | 				.filter(function(n){ | 
					
						
							|  |  |  | 					if(exclusive.indexOf(n) < 0){ | 
					
						
							|  |  |  | 						exclusive.push(n) | 
					
						
							|  |  |  | 						return false | 
					
						
							|  |  |  | 					} | 
					
						
							|  |  |  | 					return true | 
					
						
							|  |  |  | 				}) | 
					
						
							|  |  |  | 				.length == 0 | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | 			if(!res){ | 
					
						
							|  |  |  | 				excluded.push(n) | 
					
						
							|  |  |  | 			} | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | 			return res | 
					
						
							|  |  |  | 		}) | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | 		return { | 
					
						
							|  |  |  | 			features: lst, | 
					
						
							|  |  |  | 			excluded: excluded, | 
					
						
							| 
									
										
										
										
											2014-12-08 14:38:42 +03:00
										 |  |  | 			missing: missing, | 
					
						
							| 
									
										
										
										
											2014-12-05 00:10:01 +03:00
										 |  |  | 			conflicts: conflicts, | 
					
						
							|  |  |  | 			unapplicable: unapplicable, | 
					
						
							|  |  |  | 		} | 
					
						
							|  |  |  | 	}, | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2015-12-31 22:23:32 +03:00
										 |  |  | 	//
 | 
					
						
							|  |  |  | 	//	.setup(<actions>, [<feature>, ...])
 | 
					
						
							|  |  |  | 	//		-> <actions>
 | 
					
						
							|  |  |  | 	//
 | 
					
						
							|  |  |  | 	//	.setup([<feature>, ...])
 | 
					
						
							|  |  |  | 	//		-> <actions>
 | 
					
						
							|  |  |  | 	//
 | 
					
						
							| 
									
										
										
										
											2014-12-05 00:10:01 +03:00
										 |  |  | 	setup: function(obj, lst){ | 
					
						
							| 
									
										
										
										
											2015-12-31 22:23:32 +03:00
										 |  |  | 		// if no explicit object is given, just the list...
 | 
					
						
							|  |  |  | 		if(lst == null){ | 
					
						
							|  |  |  | 			lst = obj | 
					
						
							|  |  |  | 			obj = null | 
					
						
							|  |  |  | 		} | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | 		obj = obj || actions.Actions() | 
					
						
							| 
									
										
										
										
											2014-12-05 00:10:01 +03:00
										 |  |  | 		lst = lst.constructor !== Array ? [lst] : lst | 
					
						
							|  |  |  | 		var features = this.buildFeatureList(obj, lst) | 
					
						
							|  |  |  | 		lst = features.features | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | 		// check for conflicts...
 | 
					
						
							| 
									
										
										
										
											2014-12-08 14:38:42 +03:00
										 |  |  | 		if(Object.keys(features.conflicts).length != 0 | 
					
						
							|  |  |  | 				|| Object.keys(features.missing).length != 0){ | 
					
						
							|  |  |  | 			var m = features.missing | 
					
						
							| 
									
										
										
										
											2014-12-05 00:10:01 +03:00
										 |  |  | 			var c = features.conflicts | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | 			// build a report...
 | 
					
						
							|  |  |  | 			var report = [] | 
					
						
							| 
									
										
										
										
											2014-12-08 15:04:50 +03:00
										 |  |  | 
 | 
					
						
							|  |  |  | 			// missing deps...
 | 
					
						
							| 
									
										
										
										
											2014-12-08 14:38:42 +03:00
										 |  |  | 			Object.keys(m).forEach(function(k){ | 
					
						
							|  |  |  | 				report.push(k + ': missing but required by:\n          ' + m[k].join(', ')) | 
					
						
							|  |  |  | 			}) | 
					
						
							|  |  |  | 			report.push('\n') | 
					
						
							| 
									
										
										
										
											2014-12-08 15:04:50 +03:00
										 |  |  | 
 | 
					
						
							|  |  |  | 			// conflicts...
 | 
					
						
							| 
									
										
										
										
											2014-12-05 00:10:01 +03:00
										 |  |  | 			Object.keys(c).forEach(function(k){ | 
					
						
							|  |  |  | 				report.push(k + ': must setup after:\n          ' + c[k].join(', ')) | 
					
						
							|  |  |  | 			}) | 
					
						
							| 
									
										
										
										
											2014-12-08 15:04:50 +03:00
										 |  |  | 
 | 
					
						
							|  |  |  | 			// break...
 | 
					
						
							| 
									
										
										
										
											2014-12-06 15:26:20 +03:00
										 |  |  | 			throw 'Feature dependency error:\n    ' + report.join('\n    ')  | 
					
						
							| 
									
										
										
										
											2014-12-05 00:10:01 +03:00
										 |  |  | 		} | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | 		// report excluded features...
 | 
					
						
							| 
									
										
										
										
											2015-12-13 19:31:03 +03:00
										 |  |  | 		if(this.__verbose__ && features.excluded.length > 0){ | 
					
						
							| 
									
										
										
										
											2014-12-05 00:10:01 +03:00
										 |  |  | 			console.warn('Excluded features due to exclusivity conflict:',  | 
					
						
							|  |  |  | 					features.excluded.join(', ')) | 
					
						
							|  |  |  | 		} | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | 		// report unapplicable features...
 | 
					
						
							| 
									
										
										
										
											2015-12-13 19:31:03 +03:00
										 |  |  | 		if(this.__verbose__ && features.unapplicable.length > 0){ | 
					
						
							| 
									
										
										
										
											2014-12-05 00:10:01 +03:00
										 |  |  | 			console.log('Features not applicable in current context:',  | 
					
						
							|  |  |  | 					features.unapplicable.join(', ')) | 
					
						
							|  |  |  | 		} | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | 		// do the setup...
 | 
					
						
							|  |  |  | 		var that = this | 
					
						
							|  |  |  | 		var setup = FeatureProto.setup | 
					
						
							|  |  |  | 		lst.forEach(function(n){ | 
					
						
							|  |  |  | 			// setup...
 | 
					
						
							|  |  |  | 			if(that[n] != null){ | 
					
						
							| 
									
										
										
										
											2015-12-13 19:31:03 +03:00
										 |  |  | 				this.__verbose__ && console.log('Setting up feature:', n) | 
					
						
							| 
									
										
										
										
											2014-12-05 00:10:01 +03:00
										 |  |  | 				setup.call(that[n], obj) | 
					
						
							|  |  |  | 			} | 
					
						
							|  |  |  | 		}) | 
					
						
							| 
									
										
										
										
											2015-11-07 19:44:52 +03:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2015-12-13 19:31:03 +03:00
										 |  |  | 		// XXX should we extend this if it already was in the object???
 | 
					
						
							|  |  |  | 		obj.features = features | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2015-11-07 19:44:52 +03:00
										 |  |  | 		return obj | 
					
						
							| 
									
										
										
										
											2014-12-05 00:10:01 +03:00
										 |  |  | 	}, | 
					
						
							|  |  |  | 	remove: function(obj, lst){ | 
					
						
							|  |  |  | 		lst = lst.constructor !== Array ? [lst] : lst | 
					
						
							|  |  |  | 		var that = this | 
					
						
							|  |  |  | 		lst.forEach(function(n){ | 
					
						
							|  |  |  | 			if(that[n] != null){ | 
					
						
							|  |  |  | 				console.log('Removing feature:', n) | 
					
						
							|  |  |  | 				that[n].remove(obj) | 
					
						
							|  |  |  | 			} | 
					
						
							|  |  |  | 		}) | 
					
						
							|  |  |  | 	}, | 
					
						
							| 
									
										
										
										
											2015-11-05 02:27:26 +03:00
										 |  |  | 
 | 
					
						
							|  |  |  | 	// shorthand for: Feature(<feature-set>, ...)
 | 
					
						
							|  |  |  | 	// XXX should this return this?
 | 
					
						
							|  |  |  | 	Feature: function(){ | 
					
						
							|  |  |  | 		return this.__feature__.apply(null, [this].concat(args2array(arguments))) | 
					
						
							|  |  |  | 	}, | 
					
						
							| 
									
										
										
										
											2014-12-05 00:10:01 +03:00
										 |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | //---------------------------------------------------------------------
 | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | var Features = | 
					
						
							|  |  |  | module.Features = Object.create(FeatureSet) | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | /********************************************************************** | 
					
						
							|  |  |  | * vim:set ts=4 sw=4 :                                                */ | 
					
						
							|  |  |  | return module }) |