mirror of
				https://github.com/flynx/pWiki.git
				synced 2025-11-04 04:50:09 +00:00 
			
		
		
		
	
		
			
				
	
	
		
			1139 lines
		
	
	
		
			34 KiB
		
	
	
	
		
			JavaScript
		
	
	
		
			Executable File
		
	
	
	
	
			
		
		
	
	
			1139 lines
		
	
	
		
			34 KiB
		
	
	
	
		
			JavaScript
		
	
	
		
			Executable File
		
	
	
	
	
/**********************************************************************
 | 
						|
* 
 | 
						|
*
 | 
						|
*
 | 
						|
**********************************************************************/
 | 
						|
((typeof define)[0]=='u'?function(f){module.exports=f(require)}:define)(
 | 
						|
function(require){ var module={} // makes module AMD/node compatible...
 | 
						|
/*********************************************************************/
 | 
						|
 | 
						|
var object = require('ig-object')
 | 
						|
var actions = module.actions = require('ig-actions')
 | 
						|
 | 
						|
 | 
						|
/*********************************************************************/
 | 
						|
 | 
						|
// XXX use object.Error as base when ready...
 | 
						|
var FeatureLinearizationError
 | 
						|
module.FeatureLinearizationError = 
 | 
						|
object.Constructor('FeatureLinearizationError', Error, {
 | 
						|
	get name(){
 | 
						|
		return this.constructor.name },
 | 
						|
	toString: function(){
 | 
						|
		return 'Failed to linearise' },
 | 
						|
	__init__: function(data){
 | 
						|
		this.data = data },
 | 
						|
})
 | 
						|
 | 
						|
 | 
						|
 | 
						|
//---------------------------------------------------------------------
 | 
						|
// Base feature...
 | 
						|
//
 | 
						|
// 	Feature(obj)
 | 
						|
// 		-> feature
 | 
						|
//
 | 
						|
// 	Feature(feature-set, obj)
 | 
						|
// 		-> feature
 | 
						|
//
 | 
						|
// 	Feature(tag, obj)
 | 
						|
// 		-> feature
 | 
						|
//
 | 
						|
// 	Feature(tag, suggested)
 | 
						|
// 		-> feature
 | 
						|
//
 | 
						|
// 	Feature(tag, actions)
 | 
						|
// 		-> feature
 | 
						|
//
 | 
						|
// 	Feature(feature-set, tag, actions)
 | 
						|
// 		-> feature
 | 
						|
//
 | 
						|
//
 | 
						|
// 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.
 | 
						|
// 	.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.
 | 
						|
// 	.depends		- feature dependencies -- tags of features that must 
 | 
						|
// 					  setup before the feature (list | null)
 | 
						|
// 					  NOTE: a feature can depend on an exclusive tag, 
 | 
						|
// 					  		this will remove the need to track which 
 | 
						|
// 					  		specific exclusive tagged feature is loaded...
 | 
						|
// 	.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
 | 
						|
// 					  NOTE: the final .config is an empty object with
 | 
						|
// 					  		.__proto__ set to the merged configuration
 | 
						|
// 					  		data...
 | 
						|
// 	.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(..)
 | 
						|
//
 | 
						|
//
 | 
						|
// Feature applicability:
 | 
						|
// 	If feature.isApplicable(..) returns false then the feature will not be
 | 
						|
// 	considered on setup...
 | 
						|
//
 | 
						|
//
 | 
						|
var Feature =
 | 
						|
module.Feature =
 | 
						|
object.Constructor('Feature', {
 | 
						|
	//__featureset__: Features,
 | 
						|
	__featureset__: null,
 | 
						|
 | 
						|
	__verbose: null,
 | 
						|
	get __verbose__(){
 | 
						|
		return this.__verbose == null 
 | 
						|
			&& (this.__featureset__ || {}).__verbose__ },
 | 
						|
	set __verbose__(value){
 | 
						|
		this.__verbose = value },
 | 
						|
 | 
						|
	// Attributes...
 | 
						|
	tag: null,
 | 
						|
 | 
						|
	//title: null,
 | 
						|
	//doc: null,
 | 
						|
	//priority: null,
 | 
						|
	//exclusive: null,
 | 
						|
	//suggested: null,
 | 
						|
	//depends: null,
 | 
						|
	//actions: null,
 | 
						|
	//config: null,
 | 
						|
	//handlers: null,
 | 
						|
 | 
						|
 | 
						|
	isApplicable: function(actions){ return true },
 | 
						|
 | 
						|
 | 
						|
	// API...
 | 
						|
	getPriority: function(human_readable){
 | 
						|
		var res = this.priority || 0
 | 
						|
		res = res == 'high' ? 99
 | 
						|
			: res == 'low' ? -99
 | 
						|
			: res == 'medium' ? 0
 | 
						|
			: res == 'normal' ? 0
 | 
						|
			: res
 | 
						|
		return human_readable ?
 | 
						|
				(res == 99 ? 'high'
 | 
						|
					: res == 0 ? 'normal'
 | 
						|
					: res == -99 ? 'low'
 | 
						|
					: res)
 | 
						|
			: res },
 | 
						|
 | 
						|
	// XXX HANDLERS this could install the handlers in two locations:
 | 
						|
	// 		- the actions object...
 | 
						|
	// 		- mixin if available...
 | 
						|
	// 		- base object (currently implemented)
 | 
						|
	// 		...the handlers should theoreticly be stored neither in the 
 | 
						|
	// 		instance nor in the mixin but rather in the action-set itself 
 | 
						|
	// 		on feature creation... (???)
 | 
						|
	// 		...feels like user handlers and feature handlers should be 
 | 
						|
	// 		isolated...
 | 
						|
	// 		XXX setting handlers on the .__proto__ breaks...
 | 
						|
	setup: function(actions){
 | 
						|
		var that = this
 | 
						|
 | 
						|
		// mixin actions...
 | 
						|
		// NOTE: this will only mixin functions and actions...
 | 
						|
		if(this.actions != null){
 | 
						|
			this.tag ? 
 | 
						|
				// XXX HANDLERS
 | 
						|
				actions.mixin(this.actions, {source_tag: this.tag, action_handlers: true}) 
 | 
						|
				: actions.mixin(this.actions, {action_handlers: true}) }
 | 
						|
				//actions.mixin(this.actions, {source_tag: this.tag}) 
 | 
						|
				//: actions.mixin(this.actions) }
 | 
						|
 | 
						|
		/*/ XXX HANDLERS this is not needed if handlers are local to actions...
 | 
						|
		// install handlers...
 | 
						|
		if(this.handlers != null){
 | 
						|
			this.handlers.forEach(function([a, h]){
 | 
						|
				//actions.__proto__.on(a, that.tag, h) }) }
 | 
						|
				actions.on(a, that.tag, h) }) }
 | 
						|
		//*/
 | 
						|
 | 
						|
		// merge config...
 | 
						|
		// NOTE: this will merge the actual config in .config.__proto__
 | 
						|
		// 		keeping the .config clean for the user to lay with...
 | 
						|
		if(this.config != null 
 | 
						|
				|| (this.actions != null 
 | 
						|
					&& this.actions.config != null)){
 | 
						|
			// sanity check -- warn of config shadowing...
 | 
						|
			// XXX do we need this???
 | 
						|
			if(this.__verbose__ 
 | 
						|
					&& this.config && (this.actions || {}).config){
 | 
						|
				console.warn('Feature config shadowed: '
 | 
						|
					+'both .config (used) and .actions.config (ignored) are defined for:', 
 | 
						|
					this.tag, 
 | 
						|
					this) }
 | 
						|
 | 
						|
			var config = this.config = this.config || this.actions.config
 | 
						|
 | 
						|
			if(actions.config == null){
 | 
						|
				actions.config = Object.create({}) }
 | 
						|
			Object.keys(config)
 | 
						|
				.forEach(function(n){
 | 
						|
					// NOTE: this will overwrite existing values...
 | 
						|
					actions.config.__proto__[n] = config[n] }) }
 | 
						|
 | 
						|
		// custom setup...
 | 
						|
		// XXX is this the correct way???
 | 
						|
		this.hasOwnProperty('setup') 
 | 
						|
			&& this.setup !== Feature.prototype.setup
 | 
						|
			&& this.setup(actions)
 | 
						|
 | 
						|
		return this },
 | 
						|
 | 
						|
	// XXX need to revise this...
 | 
						|
	// 		- .mixout(..) is available directly from the object while 
 | 
						|
	// 			.remove(..) is not...
 | 
						|
	// 		- might be a good idea to add a specific lifecycle actions to
 | 
						|
	// 			enable feautures to handle their removal correctly... 
 | 
						|
	remove: function(actions){
 | 
						|
		this.actions != null
 | 
						|
			&& actions.mixout(this.tag || this.actions)
 | 
						|
 | 
						|
		/*/ XXX HANDLERS do we need this if .handlers if local to action...
 | 
						|
		this.handlers != null
 | 
						|
			&& actions.off('*', this.tag)
 | 
						|
		//*/
 | 
						|
 | 
						|
		// XXX revise naming...
 | 
						|
		this.hasOwnProperty('remove') 
 | 
						|
			&& this.setup !== Feature.prototype.remove
 | 
						|
			&& this.remove(actions)
 | 
						|
 | 
						|
		return this },
 | 
						|
 | 
						|
 | 
						|
	// XXX EXPERIMENTAL: if called from a feature-set this will add self
 | 
						|
	// 		to that feature-set...
 | 
						|
	// XXX do we need this to be .__new__(..) and not .__init__(..)
 | 
						|
	__new__: function(context, feature_set, tag, obj){
 | 
						|
		// NOTE: we need to account for context here -- inc length...
 | 
						|
		if(arguments.length == 3){
 | 
						|
			// Feature(<tag>, <obj>)
 | 
						|
			if(typeof(feature_set) == typeof('str')){
 | 
						|
				obj = tag
 | 
						|
				tag = feature_set
 | 
						|
				//feature_set = Features
 | 
						|
				// XXX EXPERIMENTAL...
 | 
						|
				feature_set = context instanceof FeatureSet ?
 | 
						|
					context
 | 
						|
					: (this.__featureset__ || Features)
 | 
						|
 | 
						|
			// Feature(<feature-set>, <obj>)
 | 
						|
			} else {
 | 
						|
				obj = tag
 | 
						|
				tag = null }
 | 
						|
 | 
						|
		// Feature(<obj>)
 | 
						|
		// NOTE: we need to account for context here -- inc length...
 | 
						|
		} else if(arguments.length == 2){
 | 
						|
			obj = feature_set
 | 
						|
			//feature_set = Features
 | 
						|
			// XXX EXPERIMENTAL...
 | 
						|
			feature_set = context instanceof FeatureSet ?
 | 
						|
				context
 | 
						|
				: (this.__featureset__ || Features) }
 | 
						|
 | 
						|
		if(tag != null && obj.tag != null && obj.tag != tag){
 | 
						|
			throw new Error('tag and obj.tag mismatch, either use one or both must match.') }
 | 
						|
 | 
						|
		// actions...
 | 
						|
		if(obj instanceof actions.Action){
 | 
						|
			if(tag == null){
 | 
						|
				throw new Error('need a tag to make a feature out of an action') }
 | 
						|
			obj = {
 | 
						|
				tag: tag,
 | 
						|
				actions: obj,
 | 
						|
			}
 | 
						|
 | 
						|
		// meta-feature...
 | 
						|
		} else if(obj.constructor === Array){
 | 
						|
			if(tag == null){
 | 
						|
				throw new Error('need a tag to make a meta-feature') }
 | 
						|
			obj = {
 | 
						|
				tag: tag,
 | 
						|
				suggested: obj,
 | 
						|
			} }
 | 
						|
 | 
						|
		// XXX HANDLERS setup .handlers...
 | 
						|
		if(obj.handlers){
 | 
						|
			obj.actions = obj.actions || {}
 | 
						|
			// NOTE: obj.actions does not have to be an action so w cheat \
 | 
						|
			// 		a bit here, then copy the mindings...
 | 
						|
			var tmp = Object.create(actions.MetaActions)
 | 
						|
			obj.handlers
 | 
						|
				.forEach(function([a, h]){
 | 
						|
					tmp.on(a, obj.tag, h) }) 
 | 
						|
			Object.assign(obj.actions, tmp) }
 | 
						|
 | 
						|
		// feature-set...
 | 
						|
		if(feature_set){
 | 
						|
			feature_set[obj.tag] = obj }
 | 
						|
 | 
						|
		return obj },
 | 
						|
})
 | 
						|
 | 
						|
 | 
						|
 | 
						|
//---------------------------------------------------------------------
 | 
						|
 | 
						|
var FeatureSet =
 | 
						|
module.FeatureSet = 
 | 
						|
object.Constructor('FeatureSet', {
 | 
						|
	// if true, .setup(..) will report things it's doing... 
 | 
						|
	__verbose__: null,
 | 
						|
 | 
						|
 | 
						|
	__actions__: actions.Actions,
 | 
						|
 | 
						|
	// NOTE: a feature is expected to write a reference to itself to the 
 | 
						|
	// 		feature-set (context)...
 | 
						|
	Feature: Feature,
 | 
						|
 | 
						|
 | 
						|
	// List of registered features...
 | 
						|
	get features(){
 | 
						|
		var that = this
 | 
						|
		return Object.keys(this)
 | 
						|
			.filter(function(e){ 
 | 
						|
				return e != 'features' 
 | 
						|
					&& that[e] instanceof Feature }) },
 | 
						|
 | 
						|
	// build exclusive groups...
 | 
						|
	//
 | 
						|
	// 	Get all exclusive tags...
 | 
						|
	// 	.getExclusive()
 | 
						|
	// 	.getExclusive('*')
 | 
						|
	// 		-> exclusive
 | 
						|
	//
 | 
						|
	// 	Get specific exclusive tags...
 | 
						|
	// 	.getExclusive(tag)
 | 
						|
	// 	.getExclusive([tag, ..])
 | 
						|
	// 		-> exclusive
 | 
						|
	//
 | 
						|
	// If features is given, only consider the features in list.
 | 
						|
	// If rev_exclusive is given, also build a reverse exclusive feature
 | 
						|
	// list.
 | 
						|
	//
 | 
						|
	// output format:
 | 
						|
	// 	{
 | 
						|
	// 		exclusive-tag: [
 | 
						|
	// 			feature-tag,
 | 
						|
	// 			...
 | 
						|
	// 		],
 | 
						|
	// 		...
 | 
						|
	// 	}
 | 
						|
	//
 | 
						|
	getExclusive: function(tag, features, rev_exclusive, isDisabled){
 | 
						|
		tag = tag == null || tag == '*' ? '*'
 | 
						|
			: tag instanceof Array ? tag
 | 
						|
			: [tag]
 | 
						|
 | 
						|
		features = features || this.features
 | 
						|
		rev_exclusive = rev_exclusive || {}
 | 
						|
 | 
						|
		var that = this
 | 
						|
		var exclusive = {}
 | 
						|
		features
 | 
						|
			.filter(function(f){ 
 | 
						|
				return !!that[f].exclusive 
 | 
						|
					&& (!isDisabled || !isDisabled(f)) })
 | 
						|
			.forEach(function(k){
 | 
						|
				var e = that[k].exclusive
 | 
						|
				;((e instanceof Array ? e : [e]) || [])
 | 
						|
					.forEach(function(e){
 | 
						|
						// skip tags not explicitly requested...
 | 
						|
						if(tag != '*' && tag.indexOf(e) < 0){
 | 
						|
							return
 | 
						|
						}
 | 
						|
						exclusive[e] = (exclusive[e] || []).concat([k]) 
 | 
						|
						rev_exclusive[k] = (rev_exclusive[k] || []).concat([e]) }) })
 | 
						|
		return exclusive },
 | 
						|
 | 
						|
	// Build list of features in load order...
 | 
						|
	//
 | 
						|
	// 	.buildFeatureList()
 | 
						|
	// 	.buildFeatureList('*')
 | 
						|
	// 		-> data
 | 
						|
	//
 | 
						|
	// 	.buildFeatureList(feature-tag)
 | 
						|
	// 		-> data
 | 
						|
	//
 | 
						|
	// 	.buildFeatureList([feature-tag, .. ])
 | 
						|
	// 		-> data
 | 
						|
	//
 | 
						|
	// 	.buildFeatureList(.., filter)
 | 
						|
	// 		-> data
 | 
						|
	//
 | 
						|
	//
 | 
						|
	// Requirements:
 | 
						|
	//	- features are pre-sorted by priority, original order is kept 
 | 
						|
	//		where possible
 | 
						|
	//	- a feature is loaded strictly after it's dependencies
 | 
						|
	//	- features depending on inapplicable feature(s) are also 
 | 
						|
	//		inapplicable (recursive up)
 | 
						|
	//	- inapplicable features are not loaded
 | 
						|
	//	- missing dependency -> missing dependency error
 | 
						|
	//	- suggested features (and their dependencies) do not produce 
 | 
						|
	//		dependency errors, unless explicitly included in dependency 
 | 
						|
	//		graph (i.e. explicitly depended on by some other feature)
 | 
						|
	//	- features with the same exclusive tag are grouped into an 
 | 
						|
	//		exclusive set
 | 
						|
	//	- only the first feature in an exclusive set is loaded, the rest
 | 
						|
	//		are *excluded*
 | 
						|
	//	- exclusive tag can be used to reference (alias) the loaded 
 | 
						|
	//		feature in exclusive set (i.e. exclusive tag can be used as 
 | 
						|
	//		a dependency)
 | 
						|
	//
 | 
						|
	// NOTE: an exclusive group name can be used as an alias.
 | 
						|
	// NOTE: if an alias is used and no feature from that exclusive group
 | 
						|
	// 		is explicitly included then the actual loaded feature will 
 | 
						|
	// 		depend on the load order, which in an async world is not 
 | 
						|
	// 		deterministic...
 | 
						|
	//
 | 
						|
	//
 | 
						|
	// Algorithm:
 | 
						|
	// 	- expand features:
 | 
						|
	// 		- handle dependencies (detect loops)
 | 
						|
	// 		- handle suggestions
 | 
						|
	// 		- handle explicitly disabled features (detect loops)
 | 
						|
	// 		- handle exclusive feature groups/aliases (handle conflicts)
 | 
						|
	// 	- sort list of features:
 | 
						|
	// 		- by priority
 | 
						|
	// 		- by dependency (detect loops/errors)
 | 
						|
	//
 | 
						|
	//
 | 
						|
	// Return format:
 | 
						|
	// 	{
 | 
						|
	//		// input feature feature tags...
 | 
						|
	//		input: [ .. ],
 | 
						|
	//
 | 
						|
	//		// output list of feature tags...
 | 
						|
	//		features: [ .. ],
 | 
						|
	//
 | 
						|
	//		// disabled features...
 | 
						|
	//		disabled: [ .. ],
 | 
						|
	//		// exclusive features that got excluded... 
 | 
						|
	//		excluded: [ .. ],
 | 
						|
	//
 | 
						|
	//		// Errors...
 | 
						|
	//		error: null | {
 | 
						|
	//			// fatal/recoverable error indicator...
 | 
						|
	//			fatal: bool,
 | 
						|
	//
 | 
						|
	//			// missing dependencies...
 | 
						|
	//			// NOTE: this includes tags only included by .depends and 
 | 
						|
	//			//		ignores tags from .suggested...
 | 
						|
	//			missing: [ .. ],
 | 
						|
	//
 | 
						|
	//			// exclusive feature conflicts...
 | 
						|
	//			// This will include the explicitly required conflicting
 | 
						|
	//			// exclusive features.
 | 
						|
	//			// NOTE: this is not an error, but indicates that the 
 | 
						|
	//			//		system tried to fix the state by disabling all
 | 
						|
	//			//		but the first feature.
 | 
						|
	//			conflicts: {
 | 
						|
	//				exclusive-tag: [ feature-tag, .. ],
 | 
						|
	//				..
 | 
						|
	//			},
 | 
						|
	//
 | 
						|
	//			// detected dependency loops (if .length > 0 sets fatal)...
 | 
						|
	//			loops: [ .. ],
 | 
						|
	//
 | 
						|
	//			// sorting loop overflow error (if true sets fatal)...
 | 
						|
	//			sort_loop_overflow: bool,
 | 
						|
	//		},
 | 
						|
	//
 | 
						|
	//		// Introspection...
 | 
						|
	//		// index of features and their list of dependencies...
 | 
						|
	//		depends: {
 | 
						|
	//			feature-tag: [ feature-tag, .. ],
 | 
						|
	//			..
 | 
						|
	//		},
 | 
						|
	//		// index of features and list of features depending on them...
 | 
						|
	//		// XXX should this include suggestions or should we do them 
 | 
						|
	//		//		in a separate list...
 | 
						|
	//		depended: { 
 | 
						|
	//			feature-tag: [ feature-tag, .. ],
 | 
						|
	//			..
 | 
						|
	//		},
 | 
						|
	// 	}
 | 
						|
	//
 | 
						|
	// XXX should exclusive conflicts resolve to first (current state)
 | 
						|
	//		feature or last in an exclusive group???
 | 
						|
	// XXX PROBLEM: exclusive feature trees should be resolved accounting 
 | 
						|
	// 		feature applicablility...
 | 
						|
	// 		...in this approach it is impossible...
 | 
						|
	// 		...one way to fix this is to make this interactively check 
 | 
						|
	// 		applicability, i.e. pass a context and check applicablility 
 | 
						|
	// 		when needed...
 | 
						|
	buildFeatureList: function(lst, isDisabled){
 | 
						|
		var all = this.features
 | 
						|
		lst = (lst == null || lst == '*') ? 
 | 
						|
			all 
 | 
						|
			: lst
 | 
						|
		lst = lst instanceof Array ? 
 | 
						|
			lst
 | 
						|
			: [lst] 
 | 
						|
 | 
						|
		//isDisabled = isDisabled || function(){ return false }
 | 
						|
 | 
						|
		var that = this
 | 
						|
 | 
						|
		// Pre-sort exclusive feature by their occurrence in dependency
 | 
						|
		// tree...
 | 
						|
		//
 | 
						|
		// NOTE: there can be a case when an exclusive alias is used but
 | 
						|
		//		no feature in a group is loaded, in this case which 
 | 
						|
		//		feature is actually loaded depends on the load order...
 | 
						|
		var sortExclusive = function(features){
 | 
						|
			var loaded = Object.keys(features)
 | 
						|
			Object.keys(exclusive)
 | 
						|
				.forEach(function(k){
 | 
						|
					exclusive[k] = exclusive[k]
 | 
						|
						.map(function(e, i){ return [e, i] })
 | 
						|
						.sort(function(a, b){
 | 
						|
							var i = loaded.indexOf(a[0])
 | 
						|
							var j = loaded.indexOf(b[0]) 
 | 
						|
							// keep the missing at the end...
 | 
						|
							i = i < 0 ? Infinity : i
 | 
						|
							j = j < 0 ? Infinity : j
 | 
						|
							// NOTE: Infinity - Infinity is NaN, so we need 
 | 
						|
							// 		to guard against it...
 | 
						|
							return i - j || 0 })
 | 
						|
						.map(function(e){ return e[0] }) }) }
 | 
						|
 | 
						|
		// Expand feature references (recursive)...
 | 
						|
		//
 | 
						|
		// NOTE: closures are not used here as we need to pass different
 | 
						|
		// 		stuff into data in different situations...
 | 
						|
		var expand = function(target, lst, store, data, _seen){
 | 
						|
			data = data || {}
 | 
						|
			_seen = _seen || []
 | 
						|
 | 
						|
			// clear disabled...
 | 
						|
			// NOTE: we do as a separate stage to avoid loading a 
 | 
						|
			// 		feature before it is disabled in the same list...
 | 
						|
			lst = data.disabled ?
 | 
						|
				lst
 | 
						|
					.filter(function(n){
 | 
						|
						// feature disabled -> record and skip...
 | 
						|
						if(n[0] == '-'){
 | 
						|
							n = n.slice(1)
 | 
						|
							if(_seen.indexOf(n) >= 0){
 | 
						|
								// NOTE: a disable loop is when a feature tries to disable
 | 
						|
								// 		a feature up in the same chain...
 | 
						|
								// XXX should this break or accumulate???
 | 
						|
								console.warn(`Disable loop detected at "${n}" in chain: ${_seen}`)
 | 
						|
								var loop = _seen.slice(_seen.indexOf(n)).concat([n])
 | 
						|
								data.disable_loops = (data.disable_loops || []).push(loop)
 | 
						|
								return false }
 | 
						|
							// XXX STUB -- need to resolve actual loops and 
 | 
						|
							// 		make the disable global...
 | 
						|
							if(n in store){
 | 
						|
								console.warn('Disabling a feature after it is loaded:', n, _seen) }
 | 
						|
							data.disabled.push(n)
 | 
						|
							return false }
 | 
						|
						// skip already disabled features...
 | 
						|
						if(data.disabled.indexOf(n) >= 0){
 | 
						|
							return false }
 | 
						|
						return true })
 | 
						|
				: lst
 | 
						|
 | 
						|
			// traverse the tree...
 | 
						|
			lst
 | 
						|
				// normalize the list -- remove non-features and resolve aliases...
 | 
						|
				.map(function(n){ 
 | 
						|
					var f = that[n]
 | 
						|
					// exclusive tags...
 | 
						|
					if(f == null && data.exclusive && n in data.exclusive){
 | 
						|
						store[n] = null
 | 
						|
						return false }
 | 
						|
					// feature not defined or is not a feature...
 | 
						|
					if(f == null){
 | 
						|
						data.missing 
 | 
						|
							&& data.missing.indexOf(n) < 0
 | 
						|
							&& data.missing.push(n)
 | 
						|
						return false }
 | 
						|
					return n })
 | 
						|
				.filter(function(e){ return e })
 | 
						|
				// traverse down...
 | 
						|
				.forEach(function(f){
 | 
						|
					// dependency loop detection...
 | 
						|
					if(_seen.indexOf(f) >= 0){
 | 
						|
						var loop = _seen.slice(_seen.indexOf(f)).concat([f])
 | 
						|
						data.loops 
 | 
						|
							&& data.loops.push(loop)
 | 
						|
						return }
 | 
						|
 | 
						|
					// skip already done features...
 | 
						|
					if(f in store){
 | 
						|
						return }
 | 
						|
 | 
						|
					//var feature = store[f] = that[f]
 | 
						|
					var feature = that[f]
 | 
						|
					if(feature){
 | 
						|
						var _lst = []
 | 
						|
 | 
						|
						// merge lists...
 | 
						|
						;(target instanceof Array ? target : [target])
 | 
						|
							.forEach(function(t){
 | 
						|
								_lst = _lst.concat(feature[t] || [])
 | 
						|
							})
 | 
						|
						store[f] = _lst 
 | 
						|
 | 
						|
						// traverse down...
 | 
						|
						expand(target, _lst, store, data, _seen.concat([f])) } })
 | 
						|
 | 
						|
			return store }
 | 
						|
 | 
						|
		// Expand feature dependencies and suggestions recursively...
 | 
						|
		//
 | 
						|
		// NOTE: this relies on the following values being in the closure:
 | 
						|
		// 		loops			- list of loop chains found
 | 
						|
		// 		disable_loops	- disable loops
 | 
						|
		// 							when a feature containing a disable 
 | 
						|
		// 							directive gets disabled as a result
 | 
						|
		// 		disabled		- list of disabled features
 | 
						|
		// 		missing			- list of missing features
 | 
						|
		// 		missing_suggested
 | 
						|
		// 						- list of missing suggested features and
 | 
						|
		// 							suggested feature dependencies
 | 
						|
		// 		exclusive		- exclusive feature index
 | 
						|
		// 		suggests		- index of feature suggestions (full)
 | 
						|
		// 		suggested		- suggested feature dependency index
 | 
						|
		// NOTE: the above containers will get updated as a side-effect.
 | 
						|
		// NOTE: all of the above values are defined near the location 
 | 
						|
		// 		they are first used/initiated...
 | 
						|
		// NOTE: closures are used here purely for simplicity and conciseness
 | 
						|
		// 		as threading data would not add any flexibility but make 
 | 
						|
		// 		the code more complex...
 | 
						|
		var expandFeatures = function(lst, features){
 | 
						|
			features = features || {}
 | 
						|
 | 
						|
			// feature tree...
 | 
						|
			var expand_data = {
 | 
						|
				loops: loops, 
 | 
						|
				disabled: disabled, 
 | 
						|
				disable_loops: disable_loops, 
 | 
						|
				missing: missing,
 | 
						|
				exclusive: exclusive,
 | 
						|
			}
 | 
						|
 | 
						|
			features = expand('depends', lst, features, expand_data)
 | 
						|
 | 
						|
			// suggestion list...
 | 
						|
			//	...this will be used to check if we need to break on missing 
 | 
						|
			//	features, e.g. if a feature is suggested we can silently skip 
 | 
						|
			//	it otherwise err...
 | 
						|
			//
 | 
						|
			// NOTE: this stage does not track suggested feature dependencies...
 | 
						|
			// NOTE: we do not need loop detection active here...
 | 
						|
			var s = expand('suggested', Object.keys(features), {}, 
 | 
						|
				{ 
 | 
						|
					disabled: disabled,
 | 
						|
					missing: missing_suggested,
 | 
						|
				})
 | 
						|
			s = Object.keys(s)
 | 
						|
				.filter(function(f){ 
 | 
						|
					// populate the tree of feature suggestions...
 | 
						|
					suggests[f] = s[f]
 | 
						|
					// filter out what's in features already...
 | 
						|
					return !(f in features) })
 | 
						|
			// get suggestion dependencies...
 | 
						|
			// NOTE: we do not care bout missing here...
 | 
						|
			s = expand('depends', s, {}, 
 | 
						|
				{ 
 | 
						|
					loops: loops, 
 | 
						|
					disabled: disabled, 
 | 
						|
					disable_loops: disable_loops, 
 | 
						|
					exclusive: exclusive,
 | 
						|
					missing: missing_suggested,
 | 
						|
				})
 | 
						|
			Object.keys(s)
 | 
						|
				.forEach(function(f){ 
 | 
						|
					// keep only suggested features -- diff with features...
 | 
						|
					if(f in features){
 | 
						|
						delete s[f]
 | 
						|
 | 
						|
					// mix suggested into features...
 | 
						|
					} else {
 | 
						|
						features[f] = s[f]
 | 
						|
						suggested[f] = (s[f] || []).slice() } })
 | 
						|
 | 
						|
			sortExclusive(features)
 | 
						|
 | 
						|
			return features }
 | 
						|
 | 
						|
 | 
						|
		//--------------------- Globals: filtering / exclusive tags ---
 | 
						|
 | 
						|
		var loops = []
 | 
						|
		var disable_loops = []
 | 
						|
		var disabled = []
 | 
						|
		var missing = []
 | 
						|
		var missing_suggested = []
 | 
						|
		var suggests = {}
 | 
						|
		var suggested = {}
 | 
						|
 | 
						|
		// user filter...
 | 
						|
		// NOTE: we build this out of the full feature list...
 | 
						|
		disabled = disabled
 | 
						|
			.concat(isDisabled ? all.filter(isDisabled) : [])
 | 
						|
 | 
						|
		// build exclusive groups...
 | 
						|
		// XXX need to sort the values to the same order as given features...
 | 
						|
		var rev_exclusive = {}
 | 
						|
		var exclusive = this.getExclusive('*', all, rev_exclusive, isDisabled)
 | 
						|
 | 
						|
 | 
						|
		//-------------------------------- Stage 1: expand features ---
 | 
						|
		var features = expandFeatures(lst)
 | 
						|
 | 
						|
 | 
						|
		//-------------------------------- Exclusive groups/aliases ---
 | 
						|
		// Handle exclusive feature groups and aliases...
 | 
						|
		var conflicts = {}
 | 
						|
		var done = []
 | 
						|
		var to_remove = []
 | 
						|
		Object.keys(features)
 | 
						|
			.forEach(function(f){
 | 
						|
				// alias...
 | 
						|
				while(f in exclusive && done.indexOf(f) < 0){
 | 
						|
					var candidates = (exclusive[f] || [])
 | 
						|
						.filter(function(c){ return c in features })
 | 
						|
 | 
						|
					// resolve alias to non-included feature...
 | 
						|
					if(candidates.length == 0){
 | 
						|
						var target = exclusive[f][0]
 | 
						|
 | 
						|
						// expand target to features...
 | 
						|
						expandFeatures([target], features)
 | 
						|
 | 
						|
					// link alias to existing feature...
 | 
						|
					} else {
 | 
						|
						var target = candidates[0] }
 | 
						|
 | 
						|
					// remove the alias...
 | 
						|
					// NOTE: exclusive tag can match a feature tag, thus
 | 
						|
					// 		we do not want to delete such tags...
 | 
						|
					// NOTE: we are not removing to_remove here as they may
 | 
						|
					// 		get added/expanded back in by other features...
 | 
						|
					!(f in that)
 | 
						|
						&& to_remove.push(f)
 | 
						|
					// replace dependencies...
 | 
						|
					Object.keys(features)
 | 
						|
						.forEach(function(e){
 | 
						|
							var i = features[e] ? features[e].indexOf(f) : -1
 | 
						|
							i >= 0
 | 
						|
								&& features[e].splice(i, 1, target)
 | 
						|
						})
 | 
						|
					f = target
 | 
						|
					done.push(f) }
 | 
						|
				
 | 
						|
				// exclusive feature...
 | 
						|
				if(f in rev_exclusive){
 | 
						|
					// XXX handle multiple groups... (???)
 | 
						|
					var group = rev_exclusive[f]
 | 
						|
					var candidates = (exclusive[group] || [])
 | 
						|
						.filter(function(c){ return c in features })
 | 
						|
 | 
						|
					if(!(group in conflicts) && candidates.length > 1){
 | 
						|
						conflicts[group] = candidates } } })
 | 
						|
		// cleanup...
 | 
						|
		to_remove.forEach(function(f){ delete features[f] })
 | 
						|
		// resolve any exclusivity conflicts found...
 | 
						|
		var excluded = []
 | 
						|
		Object.keys(conflicts)
 | 
						|
			.forEach(function(group){
 | 
						|
				// XXX should this resolve to the last of the first feature???
 | 
						|
				excluded = excluded.concat(conflicts[group].slice(1))})
 | 
						|
		disabled = disabled.concat(excluded)
 | 
						|
 | 
						|
 | 
						|
		//--------------------------------------- Disabled features ---
 | 
						|
		// Handle disabled features and cleanup...
 | 
						|
 | 
						|
		// reverse dependency index...
 | 
						|
		// 	...this is used to clear out orphaned features later and for
 | 
						|
		// 	introspection...
 | 
						|
		var rev_features = {}
 | 
						|
		Object.keys(features)
 | 
						|
			.forEach(function(f){
 | 
						|
				(features[f] || [])
 | 
						|
					.forEach(function(d){ 
 | 
						|
						rev_features[d] = (rev_features[d] || []).concat([f]) }) })
 | 
						|
 | 
						|
		// clear dependency trees containing disabled features...
 | 
						|
		var suggested_clear = []
 | 
						|
		do {
 | 
						|
			var expanded_disabled = false
 | 
						|
			disabled
 | 
						|
				.forEach(function(d){ 
 | 
						|
					// disable all features that depend on a disabled feature...
 | 
						|
					Object.keys(features)
 | 
						|
						.forEach(function(f){ 
 | 
						|
							if(features[f]
 | 
						|
									&& features[f].indexOf(d) >= 0
 | 
						|
									&& disabled.indexOf(f) < 0){
 | 
						|
								expanded_disabled = true
 | 
						|
								disabled.push(f) } })
 | 
						|
 | 
						|
					// delete the feature itself...
 | 
						|
					var s = suggests[d] || []
 | 
						|
					delete suggests[d]
 | 
						|
					delete features[d] 
 | 
						|
 | 
						|
					// clear suggested...
 | 
						|
					s
 | 
						|
						.forEach(function(f){
 | 
						|
							if(disabled.indexOf(f) < 0 
 | 
						|
									// not depended/suggested by any of 
 | 
						|
									// the non-disabled features...
 | 
						|
									&& Object.values(features)
 | 
						|
										.concat(Object.values(suggests))
 | 
						|
											.filter(n => n.indexOf(f) >= 0)
 | 
						|
											.length == 0){
 | 
						|
								expanded_disabled = true
 | 
						|
								disabled.push(f) } }) })
 | 
						|
		} while(expanded_disabled)
 | 
						|
 | 
						|
		// remove orphaned features...
 | 
						|
		// ...an orphan is a feature included by a disabled feature...
 | 
						|
		// NOTE: this should take care of missing features too...
 | 
						|
		Object.keys(rev_features)
 | 
						|
			.filter(function(f){
 | 
						|
				return rev_features[f]
 | 
						|
					// keep non-disabled and existing sources only...
 | 
						|
					.filter(function(e){ 
 | 
						|
						return !(e in features) || disabled.indexOf(e) < 0 })
 | 
						|
					// keep features that have no sources left, i.e. orphans...
 | 
						|
					.length == 0 })
 | 
						|
			.forEach(function(f){
 | 
						|
				console.log('ORPHANED:', f)
 | 
						|
				disabled.push(f)
 | 
						|
				delete features[f] })
 | 
						|
 | 
						|
 | 
						|
		//---------------------------------- Stage 2: sort features ---
 | 
						|
 | 
						|
		// Prepare for sort: expand dependency list in features... 
 | 
						|
		//
 | 
						|
		// NOTE: this will expand lst in-place...
 | 
						|
		// NOTE: we are not checking for loops here -- mainly because
 | 
						|
		// 		the input is expected to be loop-free...
 | 
						|
		var expanddeps = function(lst, cur, seen){
 | 
						|
			seen = seen || []
 | 
						|
			if(features[cur] == null){
 | 
						|
				return }
 | 
						|
			// expand the dep list recursively...
 | 
						|
			// NOTE: this will expand features[cur] in-place while 
 | 
						|
			// 		iterating over it...
 | 
						|
			for(var i=0; i < features[cur].length; i++){
 | 
						|
				var f = features[cur][i]
 | 
						|
				if(seen.indexOf(f) < 0){
 | 
						|
					seen.push(f)
 | 
						|
 | 
						|
					expanddeps(features[cur], f, seen)
 | 
						|
 | 
						|
					features[cur].forEach(function(e){
 | 
						|
						lst.indexOf(e) < 0
 | 
						|
							&& lst.push(e) }) } } }
 | 
						|
		// do the actual expansion...
 | 
						|
		var list = Object.keys(features)
 | 
						|
		list.forEach(function(f){ expanddeps(list, f) })
 | 
						|
 | 
						|
		// sort by priority...
 | 
						|
		//
 | 
						|
		// NOTE: this will attempt to only move features with explicitly 
 | 
						|
		// 		defined priorities and keep the rest in the same order 
 | 
						|
		// 		when possible...
 | 
						|
		list = list
 | 
						|
			// format: 
 | 
						|
			// 	[ <feature>, <index>, <priority> ]
 | 
						|
			.map(function(e, i){ 
 | 
						|
				return [e, i, (that[e] && that[e].getPriority) ? that[e].getPriority() : 0 ] })
 | 
						|
			.sort(function(a, b){ 
 | 
						|
				return a[2] - b[2] || a[1] - b[1] })
 | 
						|
			// cleanup...
 | 
						|
			.map(function(e){ return e[0] })
 | 
						|
			// sort by the order features should be loaded...
 | 
						|
			.reverse()
 | 
						|
 | 
						|
		// sort by dependency...
 | 
						|
		//
 | 
						|
		// NOTE: this requires the list to be ordered from high to low 
 | 
						|
		// 		priority, i.e. the same order they should be loaded in...
 | 
						|
		// NOTE: dependency loops will throw this into and "infinite" loop...
 | 
						|
		var loop_limit = list.length + 1
 | 
						|
		do {
 | 
						|
			var moves = 0
 | 
						|
			if(list.length == 0){
 | 
						|
				break }
 | 
						|
			list
 | 
						|
				.slice()
 | 
						|
				.forEach(function(e){
 | 
						|
					var deps = features[e]
 | 
						|
					if(!deps){
 | 
						|
						return
 | 
						|
					}
 | 
						|
					var from = list.indexOf(e)
 | 
						|
					var to = list
 | 
						|
						.map(function(f, i){ return [f, i] })
 | 
						|
						.slice(from+1)
 | 
						|
						// keep only dependencies...
 | 
						|
						.filter(function(f){ return deps.indexOf(f[0]) >= 0 })
 | 
						|
						.pop()
 | 
						|
					if(to){
 | 
						|
						// place after last dependency...
 | 
						|
						list.splice(to[1]+1, 0, e)
 | 
						|
						list.splice(from, 1)
 | 
						|
						moves++ } })
 | 
						|
			loop_limit--
 | 
						|
		} while(moves > 0 && loop_limit > 0)
 | 
						|
 | 
						|
 | 
						|
		//-------------------------------------------------------------
 | 
						|
 | 
						|
		// remove exclusivity tags that were resolved...
 | 
						|
		var isMissing = function(f){
 | 
						|
			return !(
 | 
						|
				// feature is a resolvable exclusive tag...
 | 
						|
				(exclusive[f] || []).length > 0 
 | 
						|
				// feature was resolved...
 | 
						|
				&& exclusive[f]
 | 
						|
					.filter(function(f){ return list.indexOf(f) >= 0 })
 | 
						|
					.length > 0) }
 | 
						|
		
 | 
						|
		return {
 | 
						|
			input: lst,
 | 
						|
 | 
						|
			features: list,
 | 
						|
 | 
						|
			disabled: disabled,
 | 
						|
			excluded: excluded,
 | 
						|
 | 
						|
			// errors and conflicts...
 | 
						|
			error: (loops.length > 0 
 | 
						|
					|| Object.keys(conflicts).length > 0 
 | 
						|
					|| loop_limit <= 0 
 | 
						|
					|| missing.length > 0
 | 
						|
					|| missing_suggested.length > 0) ?
 | 
						|
				{
 | 
						|
					missing: missing.filter(isMissing),
 | 
						|
					missing_suggested: missing_suggested.filter(isMissing),
 | 
						|
					conflicts: conflicts,
 | 
						|
 | 
						|
					// fatal stuff...
 | 
						|
					fatal: loops.length > 0 || loop_limit <= 0,
 | 
						|
					loops: loops,
 | 
						|
					sort_loop_overflow: loop_limit <= 0,
 | 
						|
				}
 | 
						|
				: null,
 | 
						|
 | 
						|
			// introspection...
 | 
						|
			depends: features,
 | 
						|
			depended: rev_features,
 | 
						|
			suggests: suggests,
 | 
						|
			suggested: suggested,
 | 
						|
			//exclusive: exclusive,
 | 
						|
		}
 | 
						|
	},
 | 
						|
 | 
						|
	// Setup features...
 | 
						|
	//
 | 
						|
	//	Setup features on existing actions object...
 | 
						|
	//	.setup(actions, [feature-tag, ...])
 | 
						|
	//		-> actions
 | 
						|
	//
 | 
						|
	//	Setup features on a new actions object...
 | 
						|
	//	.setup(feature-tag)
 | 
						|
	//	.setup([feature-tag, ...])
 | 
						|
	//		-> actions
 | 
						|
	//
 | 
						|
	//
 | 
						|
	// This will set .features on the object.
 | 
						|
	//
 | 
						|
	// .features format:
 | 
						|
	// 	{
 | 
						|
	// 		// the current feature set object...
 | 
						|
	// 		// XXX not sure about this -- revise...
 | 
						|
	// 		FeatureSet: feature-set,
 | 
						|
	//
 | 
						|
	// 		// list of features not applicable in current context...
 | 
						|
	// 		//
 | 
						|
	// 		// i.e. the features that defined .isApplicable(..) and it 
 | 
						|
	// 		// returned false when called.
 | 
						|
	// 		unapplicable: [ feature-tag, .. ],
 | 
						|
	//
 | 
						|
	// 		// output of .buildFeatureList(..)...
 | 
						|
	// 		...
 | 
						|
	// 	}
 | 
						|
	//
 | 
						|
	// NOTE: this will store the build result in .features of the output 
 | 
						|
	// 		actions object.
 | 
						|
	// NOTE: .features is reset even if a FeatureLinearizationError error
 | 
						|
	// 		is thrown.
 | 
						|
	setup: function(obj, lst){
 | 
						|
		// no explicit object is given...
 | 
						|
		if(lst == null){
 | 
						|
			lst = obj
 | 
						|
			obj = null }
 | 
						|
		obj = obj || (this.__actions__ || actions.Actions)()
 | 
						|
		lst = lst instanceof Array ? lst : [lst]
 | 
						|
 | 
						|
		var unapplicable = []
 | 
						|
		var features = this.buildFeatureList(lst, 
 | 
						|
			(function(n){
 | 
						|
				// if we already tested unapplicable, no need to test again...
 | 
						|
				if(unapplicable.indexOf(n) >= 0){
 | 
						|
					return true }
 | 
						|
				var f = this[n]
 | 
						|
				// check applicability if possible...
 | 
						|
				if(f && f.isApplicable && !f.isApplicable.call(this, obj)){
 | 
						|
					unapplicable.push(n)
 | 
						|
					return true }
 | 
						|
				return false }).bind(this)) 
 | 
						|
		features.unapplicable = unapplicable 
 | 
						|
		// cleanup disabled -- filter out unapplicable and excluded features...
 | 
						|
		// NOTE: this is done mainly for cleaner and simpler reporting 
 | 
						|
		// 		later on...
 | 
						|
		features.disabled = features.disabled
 | 
						|
			.filter(function(n){ 
 | 
						|
				return unapplicable.indexOf(n) < 0 
 | 
						|
					&& features.excluded.indexOf(n) < 0 })
 | 
						|
 | 
						|
		// if we have critical errors and set verbose...
 | 
						|
		var fatal = features.error 
 | 
						|
			&& (features.error.loops.length > 0 || features.error.sort_loop_overflow)
 | 
						|
 | 
						|
		// report stuff...
 | 
						|
		if(this.__verbose__){
 | 
						|
			var error = features.error
 | 
						|
			// report dependency loops...
 | 
						|
			error.loops.length > 0
 | 
						|
				&& error.loops
 | 
						|
					.forEach(function(loop){
 | 
						|
						console.warn('Feature dependency loops detected:\n\t' 
 | 
						|
							+ loop.join('\n\t\t-> ')) })
 | 
						|
			// report conflicts...
 | 
						|
			Object.keys(error.conflicts)
 | 
						|
				.forEach(function(group){
 | 
						|
					console.error('Exclusive "'+ group +'" conflict at:', error.conflicts[group]) })
 | 
						|
			// report loop limit...
 | 
						|
			error.sort_loop_overflow
 | 
						|
				&& console.error('Hit loop limit while sorting dependencies!') }
 | 
						|
 | 
						|
		features.FeatureSet = this
 | 
						|
 | 
						|
		obj.features = features
 | 
						|
 | 
						|
		// fatal error -- can't load...
 | 
						|
		if(fatal){
 | 
						|
			throw new FeatureLinearizationError(features) }
 | 
						|
 | 
						|
		// mixout everything...
 | 
						|
		this.remove(obj, 
 | 
						|
			obj.mro('tag')
 | 
						|
				.filter(function(e){ 
 | 
						|
					return !!e }))
 | 
						|
 | 
						|
		// do the setup...
 | 
						|
		var that = this
 | 
						|
		var setup = Feature.prototype.setup
 | 
						|
		features.features.forEach(function(n){
 | 
						|
			// setup...
 | 
						|
			if(that[n] != null){
 | 
						|
				this.__verbose__ 
 | 
						|
					&& console.log('Setting up feature:', n)
 | 
						|
				setup.call(that[n], obj) } })
 | 
						|
 | 
						|
		return obj },
 | 
						|
 | 
						|
	// XXX revise...
 | 
						|
	// 		...the main problem here is that .mixout(..) is accesible 
 | 
						|
	// 		directly from actions while the feature .remove(..) method
 | 
						|
	// 		is not...
 | 
						|
	// 		...would be nice to expose the API to actions directly or 
 | 
						|
	// 		keep it only in features...
 | 
						|
	remove: function(obj, lst){
 | 
						|
		lst = lst.constructor !== Array ? [lst] : lst
 | 
						|
		var that = this
 | 
						|
		lst.forEach(function(n){
 | 
						|
			if(that[n] != null){
 | 
						|
				this.__verbose__ 
 | 
						|
					&& console.log('Removing feature:', n)
 | 
						|
				that[n].remove(obj) } }) },
 | 
						|
 | 
						|
 | 
						|
	// Generate a Graphviz graph from features...
 | 
						|
	//
 | 
						|
	// XXX experimental...
 | 
						|
	gvGraph: function(lst, dep){
 | 
						|
		lst = lst || this.features
 | 
						|
		dep = dep || this
 | 
						|
 | 
						|
		var graph = ''
 | 
						|
		graph += 'digraph ImageGrid {\n'
 | 
						|
		lst
 | 
						|
			.filter(function(f){ return f in dep })
 | 
						|
			.forEach(function(f){
 | 
						|
				var deps = dep[f].depends || []
 | 
						|
 | 
						|
				deps.length > 0 ?
 | 
						|
					deps.forEach(function(d){
 | 
						|
						graph += `\t"${f}" -> "${d}";\n` })
 | 
						|
					: (graph += `\t"${f}";\n`)
 | 
						|
			})
 | 
						|
		graph += '}'
 | 
						|
 | 
						|
		return graph },
 | 
						|
})
 | 
						|
 | 
						|
 | 
						|
 | 
						|
 | 
						|
/*********************************************************************/
 | 
						|
// default feature set...
 | 
						|
 | 
						|
var Features =
 | 
						|
module.Features = new FeatureSet()
 | 
						|
 | 
						|
 | 
						|
 | 
						|
 | 
						|
/**********************************************************************
 | 
						|
* vim:set ts=4 sw=4 :                               */ return module })
 |