| 
									
										
										
										
											2016-08-20 23:46:34 +03:00
										 |  |  | /********************************************************************** | 
					
						
							|  |  |  | *  | 
					
						
							|  |  |  | * | 
					
						
							|  |  |  | * | 
					
						
							| 
									
										
										
										
											2020-05-06 20:39:51 +03:00
										 |  |  | * XXX should this redefine its own mixin functionality or use object.js?? | 
					
						
							|  |  |  | * | 
					
						
							| 
									
										
										
										
											2016-08-20 23:46:34 +03:00
										 |  |  | **********************************************************************/ | 
					
						
							| 
									
										
										
										
											2016-08-21 01:52:45 +03:00
										 |  |  | ((typeof define)[0]=='u'?function(f){module.exports=f(require)}:define)( | 
					
						
							| 
									
										
										
										
											2016-08-20 23:46:34 +03:00
										 |  |  | function(require){ var module={} // makes module AMD/node compatible...
 | 
					
						
							|  |  |  | /*********************************************************************/ | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2016-08-21 01:44:50 +03:00
										 |  |  | var object = require('ig-object') | 
					
						
							| 
									
										
										
										
											2016-08-20 23:46:34 +03:00
										 |  |  | 
 | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | /*********************************************************************/ | 
					
						
							|  |  |  | // helpers...
 | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2020-04-23 20:50:21 +03:00
										 |  |  | // XXX doc...
 | 
					
						
							| 
									
										
										
										
											2016-08-20 23:46:34 +03:00
										 |  |  | var doWithRootAction =  | 
					
						
							|  |  |  | module.doWithRootAction =  | 
					
						
							|  |  |  | function(func){ | 
					
						
							| 
									
										
										
										
											2016-08-21 01:27:20 +03:00
										 |  |  | 	return function(){ | 
					
						
							| 
									
										
										
										
											2020-04-17 23:21:39 +03:00
										 |  |  | 		var args = [...arguments] | 
					
						
							| 
									
										
										
										
											2016-08-20 23:46:34 +03:00
										 |  |  | 		var handlers = (this.getHandlerList  | 
					
						
							|  |  |  | 				|| MetaActions.getHandlerList) | 
					
						
							| 
									
										
										
										
											2016-08-21 01:27:20 +03:00
										 |  |  | 			.apply(this, args) | 
					
						
							| 
									
										
										
										
											2020-04-22 04:32:44 +03:00
										 |  |  | 		return func.apply(this, [handlers.pop()].concat(args)) } } | 
					
						
							| 
									
										
										
										
											2016-08-20 23:46:34 +03:00
										 |  |  | 
 | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2020-04-23 20:50:21 +03:00
										 |  |  | 
 | 
					
						
							|  |  |  | //---------------------------------------------------------------------
 | 
					
						
							| 
									
										
										
										
											2020-04-23 17:59:13 +03:00
										 |  |  | // String action parser/runner...
 | 
					
						
							| 
									
										
										
										
											2020-04-23 20:50:21 +03:00
										 |  |  | //
 | 
					
						
							|  |  |  | // Examples:
 | 
					
						
							|  |  |  | // 	'actionName'
 | 
					
						
							|  |  |  | // 	'actionName: attr 123 "string" -- comment...'
 | 
					
						
							|  |  |  | // 	'actionName: ...'
 | 
					
						
							| 
									
										
										
										
											2017-12-17 04:27:04 +03:00
										 |  |  | // 
 | 
					
						
							| 
									
										
										
										
											2020-04-23 20:50:21 +03:00
										 |  |  | //
 | 
					
						
							| 
									
										
										
										
											2017-12-17 04:27:04 +03:00
										 |  |  | // Syntax:
 | 
					
						
							|  |  |  | // 		ALIAS ::= 
 | 
					
						
							|  |  |  | // 			<action-name>
 | 
					
						
							|  |  |  | // 			| <action-name>: <args>
 | 
					
						
							|  |  |  | // 			| <action-name>: <args> <comment>
 | 
					
						
							|  |  |  | // 		<args> ::=
 | 
					
						
							|  |  |  | // 			<arg>
 | 
					
						
							|  |  |  | // 			| <arg> <args>
 | 
					
						
							|  |  |  | // 		<arg> ::=
 | 
					
						
							|  |  |  | // 			Number|String|Array|Object
 | 
					
						
							|  |  |  | // 			IDENTIFIER
 | 
					
						
							|  |  |  | // 			| ...
 | 
					
						
							|  |  |  | // 			| '$[0-9]'
 | 
					
						
							|  |  |  | // 		<comment> ::=
 | 
					
						
							|  |  |  | // 			'--.*$'
 | 
					
						
							|  |  |  | // 			
 | 
					
						
							|  |  |  | // 	Special args:
 | 
					
						
							|  |  |  | // 		IDENTIFIER
 | 
					
						
							|  |  |  | // 				- expanded to context[IDENTIFIER]
 | 
					
						
							| 
									
										
										
										
											2018-02-12 03:44:03 +03:00
										 |  |  | // 		$N		- expanded to an instance of parseStringAction.Argument
 | 
					
						
							| 
									
										
										
										
											2017-12-17 04:27:04 +03:00
										 |  |  | // 		...		- expanded to parseStringAction.ALLARGS (singleton)
 | 
					
						
							|  |  |  | // 			
 | 
					
						
							| 
									
										
										
										
											2017-12-24 06:46:39 +03:00
										 |  |  | // 			
 | 
					
						
							| 
									
										
										
										
											2020-04-24 00:21:02 +03:00
										 |  |  | // Returns:
 | 
					
						
							| 
									
										
										
										
											2017-12-24 06:46:39 +03:00
										 |  |  | //		{
 | 
					
						
							|  |  |  | //			action: action,
 | 
					
						
							|  |  |  | //			arguments: args,
 | 
					
						
							|  |  |  | //			doc: doc,
 | 
					
						
							|  |  |  | //			no_default: no_default,
 | 
					
						
							|  |  |  | //			stop_propagation: false,
 | 
					
						
							|  |  |  | //
 | 
					
						
							|  |  |  | //			code: txt,
 | 
					
						
							|  |  |  | //		}
 | 
					
						
							| 
									
										
										
										
											2017-12-17 04:27:04 +03:00
										 |  |  | // 		
 | 
					
						
							| 
									
										
										
										
											2020-04-23 20:50:21 +03:00
										 |  |  | //
 | 
					
						
							| 
									
										
										
										
											2017-11-19 01:00:12 +03:00
										 |  |  | // NOTE: identifiers are resolved as attributes of the context...
 | 
					
						
							| 
									
										
										
										
											2020-04-23 17:59:13 +03:00
										 |  |  | // NOTE: this is a stateless object...
 | 
					
						
							| 
									
										
										
										
											2017-08-03 21:35:10 +03:00
										 |  |  | // XXX this is the same as ImageGrid's keyboard.parseActionCall(..), reuse	
 | 
					
						
							|  |  |  | // 		in a logical manner...
 | 
					
						
							| 
									
										
										
										
											2020-04-22 04:32:44 +03:00
										 |  |  | 
 | 
					
						
							|  |  |  | // placeholders...
 | 
					
						
							|  |  |  | var __Atom | 
					
						
							|  |  |  | var __Argument | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2017-08-03 21:35:10 +03:00
										 |  |  | var parseStringAction = | 
					
						
							|  |  |  | module.parseStringAction = | 
					
						
							| 
									
										
										
										
											2020-04-22 04:32:44 +03:00
										 |  |  | Object.assign( | 
					
						
							|  |  |  | 	// parser...
 | 
					
						
							|  |  |  | 	function(txt){ | 
					
						
							|  |  |  | 		// split off the doc...
 | 
					
						
							|  |  |  | 		var c = txt.split('--') | 
					
						
							|  |  |  | 		var doc = (c[1] || '').trim() | 
					
						
							|  |  |  | 		// the actual code...
 | 
					
						
							|  |  |  | 		c = c[0].split(':') | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | 		// action and no default flag...
 | 
					
						
							|  |  |  | 		var action = c[0].trim() | 
					
						
							|  |  |  | 		var no_default = action.slice(-1) == '!' | 
					
						
							|  |  |  | 		action = no_default ? action.slice(0, -1) : action | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | 		// parse arguments...
 | 
					
						
							|  |  |  | 		var args = ((c[1] || '') | 
					
						
							|  |  |  | 				.match(RegExp([ | 
					
						
							|  |  |  | 					// strings...
 | 
					
						
							|  |  |  | 					'"[^"]*"', | 
					
						
							|  |  |  | 					"'[^']*'", | 
					
						
							|  |  |  | 					'`[^`]*`', | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | 					// objects...
 | 
					
						
							|  |  |  | 					// XXX hack-ish...
 | 
					
						
							|  |  |  | 					'\\{[^\\}]*\\}', | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | 					// lists...
 | 
					
						
							|  |  |  | 					// XXX hack-ish...
 | 
					
						
							|  |  |  | 					'\\[[^\]]*\]', | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | 					// numbers...
 | 
					
						
							|  |  |  | 					'\\d+\\.\\d+|\\d+', | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | 					// identifiers...
 | 
					
						
							|  |  |  | 					'[a-zA-Z$@#_][a-zA-Z0-9$@#_]*', | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | 					// rest args...
 | 
					
						
							|  |  |  | 					'\\.\\.\\.', | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | 					// null...
 | 
					
						
							|  |  |  | 					'null', | 
					
						
							|  |  |  | 				].join('|'), 'gm'))  | 
					
						
							|  |  |  | 			|| []) | 
					
						
							|  |  |  | 			.map(function(e){ | 
					
						
							|  |  |  | 				// argument placeholder...
 | 
					
						
							|  |  |  | 				return /^\.\.\.$/.test(e) ? | 
					
						
							|  |  |  | 						parseStringAction.ALLARGS | 
					
						
							|  |  |  | 					: /^\$[a-zA-Z0-9$@#_]*$/.test(e) ? | 
					
						
							|  |  |  | 						new parseStringAction.Argument(e.slice(1)) | 
					
						
							|  |  |  | 					// idetifiers...
 | 
					
						
							|  |  |  | 					// NOTE: keep this last as it is the most general...
 | 
					
						
							|  |  |  | 					: /^[a-zA-Z$@#_][a-zA-Z0-9$@#_]*$/.test(e) ? | 
					
						
							|  |  |  | 						new parseStringAction.Identifier(e) | 
					
						
							|  |  |  | 					: JSON.parse(e) }) | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | 		return { | 
					
						
							|  |  |  | 			action: action, | 
					
						
							|  |  |  | 			arguments: args, | 
					
						
							|  |  |  | 			doc: doc, | 
					
						
							|  |  |  | 			no_default: no_default, | 
					
						
							|  |  |  | 			stop_propagation: false, | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | 			code: txt, | 
					
						
							| 
									
										
										
										
											2020-04-23 17:59:13 +03:00
										 |  |  | 		} },  | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | 	// API and utils...
 | 
					
						
							|  |  |  | 	{ | 
					
						
							| 
									
										
										
										
											2020-04-22 04:32:44 +03:00
										 |  |  | 		// atoms...
 | 
					
						
							|  |  |  | 		Atom: (__Atom = object.Constructor('Atom', { | 
					
						
							|  |  |  | 			__init__: function(value){ | 
					
						
							|  |  |  | 				this.value = value }, | 
					
						
							|  |  |  | 			valueOf: function(){  | 
					
						
							|  |  |  | 				return this.value }, | 
					
						
							|  |  |  | 		})), | 
					
						
							|  |  |  | 		Identifier: object.Constructor('Identifier',  | 
					
						
							|  |  |  | 			Object.create(__Atom.prototype)), | 
					
						
							|  |  |  | 		Argument: (__Argument = object.Constructor('Argument',  | 
					
						
							|  |  |  | 			Object.create(__Atom.prototype))), | 
					
						
							|  |  |  | 		ALLARGS: new __Argument('...'), | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | 		// general API...
 | 
					
						
							|  |  |  | 		resolveArgs: function(context, action_args, call_args){ | 
					
						
							|  |  |  | 			var that = this | 
					
						
							|  |  |  | 			var rest | 
					
						
							| 
									
										
										
										
											2020-04-23 17:59:13 +03:00
										 |  |  | 			var args = [...action_args] | 
					
						
							| 
									
										
										
										
											2020-04-22 04:32:44 +03:00
										 |  |  | 				// merge args...
 | 
					
						
							|  |  |  | 				.map(function(arg, i){ | 
					
						
							|  |  |  | 					return arg instanceof that.Argument ? | 
					
						
							|  |  |  | 						(arg === that.ALLARGS ? | 
					
						
							|  |  |  | 							(function(){ | 
					
						
							|  |  |  | 								rest = i | 
					
						
							|  |  |  | 								return arg | 
					
						
							|  |  |  | 							})() | 
					
						
							|  |  |  | 							: call_args[parseInt(arg.value)]) | 
					
						
							|  |  |  | 						// resolve idents...
 | 
					
						
							|  |  |  | 						: arg instanceof that.Identifier ? | 
					
						
							|  |  |  | 							context[arg.value] | 
					
						
							|  |  |  | 						: arg }) | 
					
						
							|  |  |  | 			rest != null | 
					
						
							|  |  |  | 				&& args.splice(rest, 1, ...call_args) | 
					
						
							|  |  |  | 			return args }, | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | 		// XXX should this break if action does not exist???
 | 
					
						
							|  |  |  | 		callAction: function(context, action, ...args){ | 
					
						
							|  |  |  | 			action = typeof(action) == typeof('str') ?  | 
					
						
							|  |  |  | 				this(action)  | 
					
						
							|  |  |  | 				: action | 
					
						
							|  |  |  | 			// XXX should this break if action does not exist???
 | 
					
						
							|  |  |  | 			return context[action.action] instanceof Function ?  | 
					
						
							|  |  |  | 				context[action.action] | 
					
						
							|  |  |  | 					.apply(context, this.resolveArgs(context, action.arguments, args)) | 
					
						
							|  |  |  | 				// action not found or is not callable... (XXX)
 | 
					
						
							|  |  |  | 				: undefined }, | 
					
						
							|  |  |  | 		applyAction: function(context, action, args){ | 
					
						
							|  |  |  | 			return this.callAction(context, action, ...args) }, | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | 		// XXX make this stricter...
 | 
					
						
							|  |  |  | 		isStringAction: function(txt){ | 
					
						
							|  |  |  | 			try{ | 
					
						
							|  |  |  | 				var parsed = typeof(txt) == typeof('str') | 
					
						
							|  |  |  | 					&& (this.parseStringAction || parseStringAction)(txt) | 
					
						
							|  |  |  | 				return parsed  | 
					
						
							|  |  |  | 					&& /[a-zA-Z_][a-zA-Z0-9_]*/.test(parsed.action) | 
					
						
							|  |  |  | 			} catch(e){ | 
					
						
							|  |  |  | 				return false } }, | 
					
						
							| 
									
										
										
										
											2017-12-17 03:58:46 +03:00
										 |  |  | 	}) | 
					
						
							| 
									
										
										
										
											2020-04-22 04:32:44 +03:00
										 |  |  | 
 | 
					
						
							|  |  |  | // shorthand...
 | 
					
						
							| 
									
										
										
										
											2017-08-04 03:58:24 +03:00
										 |  |  | var isStringAction = | 
					
						
							|  |  |  | module.isStringAction = | 
					
						
							| 
									
										
										
										
											2020-04-22 04:32:44 +03:00
										 |  |  | 	parseStringAction.isStringAction | 
					
						
							| 
									
										
										
										
											2017-08-04 03:58:24 +03:00
										 |  |  | 
 | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2016-08-20 23:46:34 +03:00
										 |  |  | 
 | 
					
						
							|  |  |  | /*********************************************************************/ | 
					
						
							| 
									
										
										
										
											2020-04-23 20:50:21 +03:00
										 |  |  | // Action...
 | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | // Return value wrapper...
 | 
					
						
							|  |  |  | // 
 | 
					
						
							|  |  |  | // Wrapping a value in this and returning it from an action will force
 | 
					
						
							|  |  |  | // the action to return the value as-is...
 | 
					
						
							|  |  |  | // This is mainly usefull for specially handled values.
 | 
					
						
							|  |  |  | var ASIS = | 
					
						
							|  |  |  | module.ASIS =  | 
					
						
							|  |  |  | object.Constructor('ASIS', { | 
					
						
							|  |  |  | 	__init__: function(obj){ this.value = obj } }) | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | // undefined wrapper...
 | 
					
						
							|  |  |  | var UNDEFINED = | 
					
						
							|  |  |  | module.UNDEFINED = ASIS(undefined) | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2016-08-20 23:46:34 +03:00
										 |  |  | 
 | 
					
						
							|  |  |  | // Construct an action object...
 | 
					
						
							|  |  |  | //
 | 
					
						
							| 
									
										
										
										
											2017-02-18 06:54:20 +03:00
										 |  |  | // 	Action(<name>, <function>)
 | 
					
						
							|  |  |  | // 	Action(<name>[, <doc>[, <long-doc>]][, <attrs>,] <function>)
 | 
					
						
							| 
									
										
										
										
											2020-04-29 16:33:39 +03:00
										 |  |  | // 	Action(<name>[, [<doc>[, <long-doc>]][, <attrs>,] <function> ])
 | 
					
						
							| 
									
										
										
										
											2017-02-18 06:54:20 +03:00
										 |  |  | // 		-> <action>
 | 
					
						
							|  |  |  | // 	
 | 
					
						
							|  |  |  | //
 | 
					
						
							| 
									
										
										
										
											2016-08-20 23:46:34 +03:00
										 |  |  | // Action function format:
 | 
					
						
							|  |  |  | //
 | 
					
						
							|  |  |  | // 		// pre event code...
 | 
					
						
							|  |  |  | // 		function(..){
 | 
					
						
							|  |  |  | //			... // pre code
 | 
					
						
							|  |  |  | // 		}
 | 
					
						
							|  |  |  | //
 | 
					
						
							|  |  |  | // 		// pre/post event code...
 | 
					
						
							|  |  |  | // 		function(..){
 | 
					
						
							|  |  |  | //			... // pre code
 | 
					
						
							|  |  |  | //			return function(<return>, ..){
 | 
					
						
							|  |  |  | //				... // post code
 | 
					
						
							|  |  |  | //			}
 | 
					
						
							|  |  |  | // 		}
 | 
					
						
							|  |  |  | //
 | 
					
						
							|  |  |  | //
 | 
					
						
							|  |  |  | // An action is essentially a method with several additional features:
 | 
					
						
							|  |  |  | //
 | 
					
						
							|  |  |  | // 	- actions are split into two stages:
 | 
					
						
							|  |  |  | // 		pre: 	the code of the method is executed before the action 
 | 
					
						
							|  |  |  | // 				event is fired
 | 
					
						
							| 
									
										
										
										
											2016-09-15 21:46:57 +03:00
										 |  |  | // 		post:	if the action returns a callback function it will be 
 | 
					
						
							|  |  |  | // 				executed after the event is fired
 | 
					
						
							| 
									
										
										
										
											2016-08-20 23:46:34 +03:00
										 |  |  | // 				NOTE: the signature if the post stage is the same as the
 | 
					
						
							|  |  |  | // 					action's with the added return value as first argument
 | 
					
						
							|  |  |  | // 					(the rest og the arguments are shifted by 1).
 | 
					
						
							|  |  |  | //
 | 
					
						
							|  |  |  | // 	- actions automatically call the shadowed action, the pre stage is
 | 
					
						
							|  |  |  | // 	  executed down-up while the post stage is run in reverse order, 
 | 
					
						
							|  |  |  | // 	  i.e. the pre is going down and the post is going up.
 | 
					
						
							|  |  |  | //
 | 
					
						
							|  |  |  | // 	- actions provide an event-like mechanism to register handlers or 
 | 
					
						
							|  |  |  | // 	  callbacks. These callbacks are local to a specific object and will
 | 
					
						
							|  |  |  | // 	  be fired on action event/call starting from the current action 
 | 
					
						
							|  |  |  | // 	  caller and down the inheritance chain, i.e. all event handlers 
 | 
					
						
							|  |  |  | // 	  registered from the current object and up to the base action set
 | 
					
						
							|  |  |  | // 	  will be fired.
 | 
					
						
							|  |  |  | //
 | 
					
						
							|  |  |  | // 	- an action will return the deepest (root) action's return, if that 
 | 
					
						
							|  |  |  | // 	  return is undefined, then the action set is returned instead.
 | 
					
						
							| 
									
										
										
										
											2017-10-08 18:01:13 +03:00
										 |  |  | // 	  If the root action returns a Promise, then the post phase will be 
 | 
					
						
							|  |  |  | // 	  triggerd AFTER that promise is resolved or rejected, this can be 
 | 
					
						
							|  |  |  | // 	  disabled by setting the 'await' action attribute to false (see:
 | 
					
						
							|  |  |  | // 	  Action.prototype.await for details)
 | 
					
						
							| 
									
										
										
										
											2016-08-20 23:46:34 +03:00
										 |  |  | //
 | 
					
						
							|  |  |  | // 	- action arguments are "threaded" through the action chain down and 
 | 
					
						
							|  |  |  | // 	  root action return value and arguments are threaded back up the 
 | 
					
						
							|  |  |  | // 	  action chain.
 | 
					
						
							|  |  |  | //
 | 
					
						
							|  |  |  | // NOTE: actions once defined do not depend on the inheritance hierarchy, 
 | 
					
						
							|  |  |  | // 		other than the .getHandlerList(..) method. If this method is not 
 | 
					
						
							|  |  |  | // 		found in the inheritance chain (i.e. the link to MetaActions)
 | 
					
						
							|  |  |  | // 		was severed, then the default will be used: 
 | 
					
						
							|  |  |  | // 			MetaActions.getHandlerList(..)
 | 
					
						
							|  |  |  | // 		This makes it possible to redefine the method if needed but 
 | 
					
						
							|  |  |  | // 		prevents the system from breaking when an action set gets 
 | 
					
						
							|  |  |  | // 		disconnected from MetaActions. This can be useful, for example,
 | 
					
						
							|  |  |  | // 		to remove .on(..) / .off(..) handler functionality.
 | 
					
						
							|  |  |  | // 		XXX is this correct??
 | 
					
						
							|  |  |  | // NOTE: by default an action will return 'this', i.e. the action set
 | 
					
						
							|  |  |  | // 		object the action was called from.
 | 
					
						
							| 
									
										
										
										
											2018-09-01 20:01:22 +03:00
										 |  |  | // NOTE: if func.nmae is set to '<action-name>' it will be reset to the 
 | 
					
						
							|  |  |  | // 		action name by Action(..). This is a means for extending functions 
 | 
					
						
							|  |  |  | // 		to get the specific action name.
 | 
					
						
							|  |  |  | // 		Example:
 | 
					
						
							|  |  |  | // 			var getActionName = function(func){
 | 
					
						
							|  |  |  | // 				var f = function(...args){
 | 
					
						
							|  |  |  | // 					return func(f.name, ...args) } 
 | 
					
						
							|  |  |  | // 				// this will force Actions(..) to set a name on f
 | 
					
						
							|  |  |  | //				Object.defineProperty(f, 'name', { value: '<action-name>' })
 | 
					
						
							|  |  |  | // 				return f
 | 
					
						
							|  |  |  | // 			}
 | 
					
						
							|  |  |  | //
 | 
					
						
							|  |  |  | // 			...
 | 
					
						
							|  |  |  | //
 | 
					
						
							|  |  |  | // 			someAction: [
 | 
					
						
							|  |  |  | // 				getActionName(function(name, ...args){
 | 
					
						
							|  |  |  | // 					console.log('Action name:', name)
 | 
					
						
							|  |  |  | // 				})],
 | 
					
						
							|  |  |  | // 			someOtherAction: [
 | 
					
						
							|  |  |  | // 				function(name, ...args){
 | 
					
						
							|  |  |  | // 					// there is no way to know the action name from within
 | 
					
						
							|  |  |  | // 					// and action...
 | 
					
						
							|  |  |  | // 				}],
 | 
					
						
							|  |  |  | //
 | 
					
						
							|  |  |  | // 		But note that the .name is set in definition time and not in 
 | 
					
						
							|  |  |  | // 		call time, so renaming the action in runtime will have no effect 
 | 
					
						
							|  |  |  | // 		on what it will log...
 | 
					
						
							|  |  |  | // 		Also note that using Object.defineProperty(..) is required as 
 | 
					
						
							|  |  |  | // 		chrome ignores changes to function's .name in other cases...
 | 
					
						
							|  |  |  | // 		
 | 
					
						
							| 
									
										
										
										
											2016-08-20 23:46:34 +03:00
										 |  |  | //
 | 
					
						
							|  |  |  | // XXX add more metadata/docs:
 | 
					
						
							|  |  |  | // 		.section
 | 
					
						
							|  |  |  | // 		.category
 | 
					
						
							|  |  |  | // 		...
 | 
					
						
							|  |  |  | // XXX might be a good idea to add an option to return the full results...
 | 
					
						
							|  |  |  | var Action = | 
					
						
							| 
									
										
										
										
											2020-04-23 17:59:13 +03:00
										 |  |  | module.Action =  | 
					
						
							| 
									
										
										
										
											2020-05-06 20:39:51 +03:00
										 |  |  | object.Constructor('Action', Function, { | 
					
						
							| 
									
										
										
										
											2020-04-23 17:59:13 +03:00
										 |  |  | 	// Control how an action handles returned promises...
 | 
					
						
							|  |  |  | 	// 
 | 
					
						
							|  |  |  | 	// Possible values:
 | 
					
						
							|  |  |  | 	// 	true	- if an action returns a promise then trigger the post phase
 | 
					
						
							|  |  |  | 	// 				after that promise is resolved / rejected... (default)
 | 
					
						
							|  |  |  | 	// 	false	- handle promises like any other returned value.
 | 
					
						
							|  |  |  | 	// 	
 | 
					
						
							|  |  |  | 	// 	
 | 
					
						
							|  |  |  | 	// NOTE: .await is only checked in the root action, thus it can not be 
 | 
					
						
							|  |  |  | 	// 		overloaded by extending actions.
 | 
					
						
							|  |  |  | 	// 		This is done intentionally, as the action actually returning a 
 | 
					
						
							|  |  |  | 	// 		value (and defining the signature) is the only one responsible 
 | 
					
						
							|  |  |  | 	// 		for controlling how it's handled.
 | 
					
						
							|  |  |  | 	// 	
 | 
					
						
							|  |  |  | 	// For implmentation see: Action.prototype.chainApply(..)
 | 
					
						
							|  |  |  | 	// 
 | 
					
						
							|  |  |  | 	// XXX should we be able to set this in the context???
 | 
					
						
							|  |  |  | 	// XXX can we use 'await'???
 | 
					
						
							|  |  |  | 	await: true, | 
					
						
							| 
									
										
										
										
											2016-08-20 23:46:34 +03:00
										 |  |  | 
 | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2020-04-23 17:59:13 +03:00
										 |  |  | 	// pre/post stage runners...
 | 
					
						
							|  |  |  | 	//
 | 
					
						
							|  |  |  | 	// 	.pre(context, args)	
 | 
					
						
							|  |  |  | 	// 		-> data
 | 
					
						
							|  |  |  | 	//
 | 
					
						
							|  |  |  | 	// 	.post(context, data)
 | 
					
						
							|  |  |  | 	// 		-> result
 | 
					
						
							|  |  |  | 	// 		
 | 
					
						
							|  |  |  | 	// 		
 | 
					
						
							|  |  |  | 	// Call data format:
 | 
					
						
							|  |  |  | 	// 	{
 | 
					
						
							|  |  |  | 	//		arguments: args,
 | 
					
						
							|  |  |  | 	//
 | 
					
						
							|  |  |  | 	//		wrapper: call_wrapper,
 | 
					
						
							|  |  |  | 	//		handlers: handlers,
 | 
					
						
							|  |  |  | 	//
 | 
					
						
							|  |  |  | 	//		result: res,
 | 
					
						
							|  |  |  | 	// 	}
 | 
					
						
							|  |  |  | 	//
 | 
					
						
							|  |  |  | 	//
 | 
					
						
							|  |  |  | 	// External methods (required):
 | 
					
						
							|  |  |  | 	// 	.getHandlers(..)			resolved from: context, MetaActions
 | 
					
						
							|  |  |  | 	//
 | 
					
						
							|  |  |  | 	//
 | 
					
						
							|  |  |  | 	// External methods (optoinal):
 | 
					
						
							|  |  |  | 	// 	.__actioncall__(..)			resolved from: context
 | 
					
						
							|  |  |  | 	// 	.preActionHandler(..)		resolved from: context, MetaActions
 | 
					
						
							|  |  |  | 	//
 | 
					
						
							|  |  |  | 	//
 | 
					
						
							|  |  |  | 	// Special cases:
 | 
					
						
							|  |  |  | 	// 	- An action is referenced via a different name than is in its .name
 | 
					
						
							|  |  |  | 	// 		this can occur if:
 | 
					
						
							|  |  |  | 	// 			1) an action is renamed but its .name is not
 | 
					
						
							|  |  |  | 	// 			2) an action is created and stored with a different name
 | 
					
						
							|  |  |  | 	// 				var f = new Action('m', function(){ ... })
 | 
					
						
							|  |  |  | 	//
 | 
					
						
							|  |  |  | 	//
 | 
					
						
							|  |  |  | 	// NOTE: All the defaults should be handled by the pre stage, post will 
 | 
					
						
							|  |  |  | 	// 		process data assuming that it is correct.
 | 
					
						
							|  |  |  | 	// NOTE: .post(..) will not wait for returned promises to resolve, use 
 | 
					
						
							|  |  |  | 	// 		.chainApply(..) / ,chainCall(..) instead, or handle .result 
 | 
					
						
							|  |  |  | 	// 		manually...
 | 
					
						
							|  |  |  | 	// 		(see: Action.prototype.chainApply(..))
 | 
					
						
							|  |  |  | 	// XXX revise the structure....
 | 
					
						
							|  |  |  | 	// 		...is it a better idea to define action methods in an object 
 | 
					
						
							|  |  |  | 	// 		and assign that???
 | 
					
						
							|  |  |  | 	pre: function(context, args){ | 
					
						
							|  |  |  | 		var that = this | 
					
						
							|  |  |  | 		args = args || [] | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | 		// prepare for after calls...
 | 
					
						
							|  |  |  | 		// XXX this may pose problems with concurency...
 | 
					
						
							|  |  |  | 		// XXX do not like that this forces exception rethrowing...
 | 
					
						
							|  |  |  | 		// XXX EXPERIMENTAL (after calls)...
 | 
					
						
							|  |  |  | 		context.__action_after_running = [ | 
					
						
							|  |  |  | 			// nested call...
 | 
					
						
							|  |  |  | 			context.__action_after_running, | 
					
						
							|  |  |  | 			// top list...
 | 
					
						
							|  |  |  | 			(context.__action_after_running || [null, []])[1], | 
					
						
							|  |  |  | 		] | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | 		var res = context | 
					
						
							|  |  |  | 		var outer = this.name | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | 		// get the handler list...
 | 
					
						
							|  |  |  | 		var getHandlers = context.getHandlers  | 
					
						
							|  |  |  | 			|| MetaActions.getHandlers | 
					
						
							|  |  |  | 		var handlers = getHandlers.call(context, outer) | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | 		// handle cases where .func is not in handlers...
 | 
					
						
							|  |  |  | 		//
 | 
					
						
							|  |  |  | 		// NOTE: see Special cases in method doc above...
 | 
					
						
							|  |  |  | 		if(handlers.length == 0  | 
					
						
							|  |  |  | 				|| handlers.filter(function(h){  | 
					
						
							|  |  |  | 					return h.pre === that.func }).length == 0){ | 
					
						
							|  |  |  | 			var cur = { | 
					
						
							|  |  |  | 				pre: this.func, | 
					
						
							|  |  |  | 			} | 
					
						
							|  |  |  | 			this.doc | 
					
						
							|  |  |  | 				&& (cur.doc = this.doc) | 
					
						
							|  |  |  | 			this.long_doc | 
					
						
							|  |  |  | 				&& (cur.long_doc = this.long_doc) | 
					
						
							| 
									
										
										
										
											2020-11-23 14:47:00 +03:00
										 |  |  | 			handlers.unshift(cur) } | 
					
						
							| 
									
										
										
										
											2016-08-20 23:46:34 +03:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2020-04-23 17:59:13 +03:00
										 |  |  | 		// special case: see if we need to handle the call without handlers...
 | 
					
						
							|  |  |  | 		var preActionHandler = context.preActionHandler  | 
					
						
							|  |  |  | 			|| MetaActions.preActionHandler | 
					
						
							|  |  |  | 		if(preActionHandler){ | 
					
						
							|  |  |  | 			var res = preActionHandler.call(context, outer, handlers, args) | 
					
						
							|  |  |  | 			if(res !== undefined){ | 
					
						
							|  |  |  | 				return res } } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | 		var call_wrapper = outer != '__actioncall__' ?  | 
					
						
							|  |  |  | 			getHandlers.call(context, '__actioncall__')  | 
					
						
							|  |  |  | 			: [] | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | 		try { | 
					
						
							|  |  |  | 			// wrapper handlers: pre phase...
 | 
					
						
							|  |  |  | 			call_wrapper = call_wrapper | 
					
						
							|  |  |  | 				.map(function(a){ | 
					
						
							|  |  |  | 					if(a.pre){ | 
					
						
							|  |  |  | 						res = a.pre.call(context, outer, args) | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | 						// if a handler returns a function register is as a post
 | 
					
						
							|  |  |  | 						// handler...
 | 
					
						
							|  |  |  | 						if(res  | 
					
						
							|  |  |  | 								&& res !== context  | 
					
						
							|  |  |  | 								&& res instanceof Function){ | 
					
						
							| 
									
										
										
										
											2020-11-23 14:47:00 +03:00
										 |  |  | 							a.post = res } } | 
					
						
							| 
									
										
										
										
											2020-04-23 17:59:13 +03:00
										 |  |  | 					return a }) | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | 			// handlers: pre phase...
 | 
					
						
							|  |  |  | 			handlers | 
					
						
							|  |  |  | 				// NOTE: current action will get included and called by the code 
 | 
					
						
							|  |  |  | 				// 		above and below, so no need to explicitly call func...
 | 
					
						
							|  |  |  | 				// NOTE: pre handlers are called FIFO, i.e. the last defined first... 
 | 
					
						
							|  |  |  | 				.map(function(a){ | 
					
						
							|  |  |  | 					if(a.pre){ | 
					
						
							|  |  |  | 						res = a.pre.apply(context, args) | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | 						// if a handler returns a function register is as a post
 | 
					
						
							|  |  |  | 						// handler...
 | 
					
						
							|  |  |  | 						if(res  | 
					
						
							|  |  |  | 								&& res !== context  | 
					
						
							|  |  |  | 								&& res instanceof Function){ | 
					
						
							|  |  |  | 							a.post = res | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | 							// reset the result...
 | 
					
						
							|  |  |  | 							// NOTE: this is the only difference between this 
 | 
					
						
							|  |  |  | 							// 		and wrapper stages...
 | 
					
						
							| 
									
										
										
										
											2020-11-23 14:47:00 +03:00
										 |  |  | 							res = context } } | 
					
						
							| 
									
										
										
										
											2020-04-23 17:59:13 +03:00
										 |  |  | 					return a }) | 
					
						
							| 
									
										
										
										
											2016-08-20 23:46:34 +03:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2020-04-23 17:59:13 +03:00
										 |  |  | 		// XXX EXPERIMENTAL (after calls)...
 | 
					
						
							|  |  |  | 		} catch(error){ | 
					
						
							|  |  |  | 			// XXX should we unwind this???
 | 
					
						
							|  |  |  | 			delete context.__action_after_running | 
					
						
							| 
									
										
										
										
											2020-11-23 14:47:00 +03:00
										 |  |  | 			throw error } | 
					
						
							| 
									
										
										
										
											2016-08-20 23:46:34 +03:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2020-04-23 17:59:13 +03:00
										 |  |  | 		// return context if nothing specific is returned...
 | 
					
						
							|  |  |  | 		res = res === undefined ? context  | 
					
						
							|  |  |  | 			: res instanceof ASIS ? res.value | 
					
						
							|  |  |  | 			// XXX returning an explicit [undefined]...
 | 
					
						
							|  |  |  | 			//: res instanceof Array
 | 
					
						
							|  |  |  | 			//		&& res.length == 1
 | 
					
						
							|  |  |  | 			//		&& res.indexOf(undefined) == 0 ?
 | 
					
						
							|  |  |  | 			//	undefined
 | 
					
						
							|  |  |  | 			: res | 
					
						
							| 
									
										
										
										
											2016-08-20 23:46:34 +03:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2020-04-23 17:59:13 +03:00
										 |  |  | 		return { | 
					
						
							|  |  |  | 			arguments: args, | 
					
						
							| 
									
										
										
										
											2016-08-20 23:46:34 +03:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2020-04-23 17:59:13 +03:00
										 |  |  | 			wrapper: call_wrapper, | 
					
						
							|  |  |  | 			handlers: handlers, | 
					
						
							| 
									
										
										
										
											2016-08-20 23:46:34 +03:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2020-04-23 17:59:13 +03:00
										 |  |  | 			result: res, | 
					
						
							| 
									
										
										
										
											2020-04-29 21:14:09 +03:00
										 |  |  | 		} }, | 
					
						
							| 
									
										
										
										
											2020-04-23 17:59:13 +03:00
										 |  |  | 	post: function(context, data){ | 
					
						
							|  |  |  | 		var res = data.result | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | 		var args = data.arguments || [] | 
					
						
							|  |  |  | 		// the post handlers get the result as the first argument...
 | 
					
						
							|  |  |  | 		args.splice(0, 0, res) | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | 		var outer = this.name | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | 		try { | 
					
						
							|  |  |  | 			// handlers: post phase...
 | 
					
						
							|  |  |  | 			data.handlers && data.handlers | 
					
						
							|  |  |  | 				// NOTE: post handlers are called LIFO -- last defined last...
 | 
					
						
							|  |  |  | 				.reverse() | 
					
						
							|  |  |  | 				.forEach(function(a){ | 
					
						
							|  |  |  | 					a.post | 
					
						
							| 
									
										
										
										
											2020-11-23 14:47:00 +03:00
										 |  |  | 						&& a.post.apply(context, args) }) | 
					
						
							| 
									
										
										
										
											2020-04-23 17:59:13 +03:00
										 |  |  | 
 | 
					
						
							|  |  |  | 			// wrapper handlers: post phase...
 | 
					
						
							|  |  |  | 			data.wrapper && data.wrapper | 
					
						
							|  |  |  | 				// NOTE: post handlers are called LIFO -- last defined last...
 | 
					
						
							|  |  |  | 				.reverse() | 
					
						
							|  |  |  | 				.forEach(function(a){ | 
					
						
							|  |  |  | 					a.post | 
					
						
							| 
									
										
										
										
											2020-11-23 14:47:00 +03:00
										 |  |  | 						&& a.post.call(context, res, outer, args.slice(1)) }) | 
					
						
							| 
									
										
										
										
											2020-04-23 17:59:13 +03:00
										 |  |  | 
 | 
					
						
							|  |  |  | 		// XXX EXPERIMENTAL (after calls)...
 | 
					
						
							|  |  |  | 		} catch(error){ | 
					
						
							|  |  |  | 			// should we unwind this???
 | 
					
						
							|  |  |  | 			delete context.__action_after_running | 
					
						
							| 
									
										
										
										
											2020-11-23 14:47:00 +03:00
										 |  |  | 			throw error } | 
					
						
							| 
									
										
										
										
											2016-08-20 23:46:34 +03:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2020-04-23 17:59:13 +03:00
										 |  |  | 		// handle after calls...
 | 
					
						
							|  |  |  | 		// XXX EXPERIMENTAL (after calls)...
 | 
					
						
							|  |  |  | 		;(context.__action_after_running || []) | 
					
						
							|  |  |  | 			.slice(2) | 
					
						
							|  |  |  | 			.forEach(function(func){ | 
					
						
							|  |  |  | 				func.call(context) }) | 
					
						
							|  |  |  | 		// top calls...
 | 
					
						
							|  |  |  | 		if(context.__action_after_running){ | 
					
						
							|  |  |  | 			if(context.__action_after_running[0] == null){ | 
					
						
							|  |  |  | 				;(context.__action_after_running[1] || []) | 
					
						
							|  |  |  | 					.forEach(function(func){ | 
					
						
							|  |  |  | 						func.call(context) }) | 
					
						
							|  |  |  | 				delete context.__action_after_running | 
					
						
							|  |  |  | 			// back to prev level...
 | 
					
						
							|  |  |  | 			} else { | 
					
						
							| 
									
										
										
										
											2020-11-23 14:47:00 +03:00
										 |  |  | 				context.__action_after_running = context.__action_after_running[0] } } | 
					
						
							| 
									
										
										
										
											2016-08-20 23:46:34 +03:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2020-04-29 21:14:09 +03:00
										 |  |  | 		return res }, | 
					
						
							| 
									
										
										
										
											2018-12-11 20:07:36 +03:00
										 |  |  | 
 | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2020-04-23 17:59:13 +03:00
										 |  |  | 	// chaining...
 | 
					
						
							|  |  |  | 	// 
 | 
					
						
							|  |  |  | 	// For docs see: MetaActions.chainApply(..) and the base module doc.
 | 
					
						
							|  |  |  | 	chainApply: function(context, inner, args){ | 
					
						
							|  |  |  | 		args = [...(args || [])] | 
					
						
							|  |  |  | 		var outer = this.name | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | 		var data = this.pre(context, args) | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | 		// call the inner action/function if preset....
 | 
					
						
							|  |  |  | 		// NOTE: this is slightly different (see docs) to what happens in 
 | 
					
						
							|  |  |  | 		// 		.pre(..)/.post(..), thus we are doing this separately and 
 | 
					
						
							|  |  |  | 		// 		not reusing existing code...
 | 
					
						
							|  |  |  | 		if(inner){ | 
					
						
							|  |  |  | 			var res = inner instanceof Function ?  | 
					
						
							|  |  |  | 					inner.apply(context, args) | 
					
						
							|  |  |  | 				: inner instanceof Array && inner.length > 0 ?  | 
					
						
							|  |  |  | 					context[inner.pop()].chainApply(context, inner, args) | 
					
						
							|  |  |  | 				: typeof(inner) == typeof('str') ? | 
					
						
							|  |  |  | 					context[inner].chainApply(context, null, args) | 
					
						
							|  |  |  | 				: undefined | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | 			// call the resulting function...
 | 
					
						
							|  |  |  | 			if(res instanceof Function){ | 
					
						
							|  |  |  | 				res.apply(context, [context].concat(args)) | 
					
						
							|  |  |  | 				data.result = context | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | 			// push the inner result into the chain...
 | 
					
						
							|  |  |  | 			} else if(res !== undefined){ | 
					
						
							| 
									
										
										
										
											2020-11-23 14:47:00 +03:00
										 |  |  | 				data.result = res } } | 
					
						
							| 
									
										
										
										
											2016-08-20 23:46:34 +03:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2020-04-23 17:59:13 +03:00
										 |  |  | 		// returned promise -> await for resolve/error...
 | 
					
						
							|  |  |  | 		// XXX should we be able to set this in the context???
 | 
					
						
							|  |  |  | 		if(data.result instanceof Promise | 
					
						
							|  |  |  | 				&& (context.getRootActionAttr || MetaActions.getRootActionAttr) | 
					
						
							|  |  |  | 					.call(context, this.name, 'await') ){ | 
					
						
							|  |  |  | 			var that = this | 
					
						
							|  |  |  | 			return data.result | 
					
						
							|  |  |  | 				.then(function(){ | 
					
						
							|  |  |  | 					return that.post(context, data) }) | 
					
						
							|  |  |  | 				.catch(function(){ | 
					
						
							| 
									
										
										
										
											2020-11-23 14:47:00 +03:00
										 |  |  | 					return that.post(context, data) }) } | 
					
						
							| 
									
										
										
										
											2017-10-08 18:01:13 +03:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2020-04-29 21:14:09 +03:00
										 |  |  | 		return this.post(context, data) }, | 
					
						
							| 
									
										
										
										
											2020-04-23 17:59:13 +03:00
										 |  |  | 	chainCall: function(context, inner){ | 
					
						
							|  |  |  | 		return this.chainApply(context, inner, [...arguments].slice(2)) }, | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | 	// constructor...
 | 
					
						
							|  |  |  | 	//
 | 
					
						
							|  |  |  | 	// 	Action(<name>, <function>)
 | 
					
						
							|  |  |  | 	// 	Action(<name>[, <doc>[, <long-doc>]][, <attrs>,] <function>)
 | 
					
						
							|  |  |  | 	// 	Action(<name>, [ [<doc>[, <long-doc>]][, <attrs>,] <function> ])
 | 
					
						
							|  |  |  | 	// 		-> <action>
 | 
					
						
							|  |  |  | 	//
 | 
					
						
							| 
									
										
										
										
											2020-04-29 07:49:21 +03:00
										 |  |  | 	__new__: function(context, name, doc, ldoc, attrs, func){ | 
					
						
							| 
									
										
										
										
											2020-04-23 17:59:13 +03:00
										 |  |  | 		// prevent action overloading...
 | 
					
						
							|  |  |  | 		// XXX do we need this???
 | 
					
						
							|  |  |  | 		//if(context != null && context[name] != null){
 | 
					
						
							|  |  |  | 		//	throw 'action "'+name+'" already exists.' }
 | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | 		// create the actual instance we will be returning...
 | 
					
						
							|  |  |  | 		var meth = function(){ | 
					
						
							|  |  |  | 			return meth.chainApply(this, null, arguments) } | 
					
						
							|  |  |  | 		meth.__proto__ = this.__proto__ | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | 		// precess args...
 | 
					
						
							|  |  |  | 		var args = doc instanceof Array ?  | 
					
						
							|  |  |  | 			doc  | 
					
						
							|  |  |  | 			: [...arguments] | 
					
						
							| 
									
										
										
										
											2020-04-29 07:49:21 +03:00
										 |  |  | 				.slice(2) | 
					
						
							| 
									
										
										
										
											2020-04-23 17:59:13 +03:00
										 |  |  | 				.filter(function(e){ return e !== undefined }) | 
					
						
							|  |  |  | 		func = args.pop() | 
					
						
							|  |  |  | 		last = args[args.length-1] | 
					
						
							|  |  |  | 		attrs = (last != null && typeof(last) != typeof('str')) ?  | 
					
						
							|  |  |  | 			args.pop()  | 
					
						
							|  |  |  | 			: {} | 
					
						
							|  |  |  | 		doc = typeof(args[0]) == typeof('str') ?  | 
					
						
							|  |  |  | 				args.shift()  | 
					
						
							|  |  |  | 			: func.doc ?  | 
					
						
							|  |  |  | 				func.doc | 
					
						
							|  |  |  | 			: null | 
					
						
							|  |  |  | 		ldoc = typeof(args[0]) == typeof('str') ?  | 
					
						
							|  |  |  | 				args.shift()  | 
					
						
							|  |  |  | 			: func.long_doc ?  | 
					
						
							|  |  |  | 				func.long_doc | 
					
						
							|  |  |  | 			: null | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | 		// populate the action attributes...
 | 
					
						
							|  |  |  | 		//meth.name = name
 | 
					
						
							|  |  |  | 		Object.defineProperty(meth, 'name', { | 
					
						
							|  |  |  | 			value: name, | 
					
						
							|  |  |  | 		}) | 
					
						
							|  |  |  | 		func.doc = meth.doc = doc | 
					
						
							|  |  |  | 		func.long_doc = meth.long_doc = ldoc | 
					
						
							| 
									
										
										
										
											2017-10-08 18:01:13 +03:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2020-04-23 17:59:13 +03:00
										 |  |  | 		meth.func = func | 
					
						
							| 
									
										
										
										
											2017-10-08 18:01:13 +03:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2020-04-23 17:59:13 +03:00
										 |  |  | 		if(func.name == '<action-name>'){ | 
					
						
							|  |  |  | 			Object.defineProperty(func, 'name', { | 
					
						
							|  |  |  | 				value: name, | 
					
						
							| 
									
										
										
										
											2020-11-23 14:47:00 +03:00
										 |  |  | 			}) } | 
					
						
							| 
									
										
										
										
											2016-08-20 23:46:34 +03:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2020-04-23 17:59:13 +03:00
										 |  |  | 		// make introspection be a bit better...
 | 
					
						
							| 
									
										
										
										
											2020-04-23 19:40:43 +03:00
										 |  |  | 		meth.toString = function(){ | 
					
						
							| 
									
										
										
										
											2020-04-23 20:50:21 +03:00
										 |  |  | 			return object.normalizeIndent(func.toString()) } | 
					
						
							| 
									
										
										
										
											2017-10-08 18:01:13 +03:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2020-04-23 17:59:13 +03:00
										 |  |  | 		// setup attrs...
 | 
					
						
							|  |  |  | 		Object.assign(meth, attrs) | 
					
						
							|  |  |  | 		Object.assign(func, attrs) | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2020-04-29 21:14:09 +03:00
										 |  |  | 		return meth }, | 
					
						
							| 
									
										
										
										
											2020-04-23 17:59:13 +03:00
										 |  |  | }) | 
					
						
							| 
									
										
										
										
											2016-08-20 23:46:34 +03:00
										 |  |  | 
 | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2017-08-03 20:44:26 +03:00
										 |  |  | //---------------------------------------------------------------------
 | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2017-08-04 04:14:17 +03:00
										 |  |  | // Action alias constructor...
 | 
					
						
							|  |  |  | // 
 | 
					
						
							|  |  |  | // This is signature compatible with Action(..) with one difference being 
 | 
					
						
							|  |  |  | // that this expects the target to be a string compatible with 
 | 
					
						
							|  |  |  | // .parseStringAction(..)...
 | 
					
						
							|  |  |  | // 
 | 
					
						
							| 
									
										
										
										
											2017-12-17 04:27:04 +03:00
										 |  |  | // This will resolve special alias args:
 | 
					
						
							|  |  |  | // 		name	-> parseStringAction.Identifier(name)	-> this[name]
 | 
					
						
							|  |  |  | // 		$N		-> parseStringAction.Argument(N)		-> arguments[n]
 | 
					
						
							|  |  |  | // 		...		-> parseStringAction.ALLARGS			-> arguments
 | 
					
						
							|  |  |  | // 	
 | 
					
						
							|  |  |  | // 
 | 
					
						
							| 
									
										
										
										
											2017-08-03 21:44:16 +03:00
										 |  |  | // XXX alias parsing is dependant on the action set, move this functionality
 | 
					
						
							|  |  |  | // 		to the ActionSet.alias(..) method/action...
 | 
					
						
							| 
									
										
										
										
											2017-08-03 20:44:26 +03:00
										 |  |  | // XXX handle alias args and pass them to the target...
 | 
					
						
							| 
									
										
										
										
											2018-02-12 03:44:03 +03:00
										 |  |  | // XXX should an alias return a value???
 | 
					
						
							| 
									
										
										
										
											2017-08-03 20:44:26 +03:00
										 |  |  | var Alias = | 
					
						
							|  |  |  | module.Alias = | 
					
						
							| 
									
										
										
										
											2020-05-06 20:39:51 +03:00
										 |  |  | object.Constructor('Alias', Action, { | 
					
						
							| 
									
										
										
										
											2020-04-29 07:49:21 +03:00
										 |  |  | 	__new__: function(context, alias, doc, ldoc, attrs, target){ | 
					
						
							| 
									
										
										
										
											2020-04-23 17:59:13 +03:00
										 |  |  | 		// precess args...
 | 
					
						
							|  |  |  | 		var args = doc instanceof Array ?  | 
					
						
							|  |  |  | 			doc  | 
					
						
							|  |  |  | 			: [...arguments] | 
					
						
							| 
									
										
										
										
											2020-04-29 07:49:21 +03:00
										 |  |  | 				.slice(2) | 
					
						
							| 
									
										
										
										
											2020-04-23 17:59:13 +03:00
										 |  |  | 				.filter(function(e){ return e !== undefined }) | 
					
						
							|  |  |  | 		target = args.pop() | 
					
						
							|  |  |  | 		last = args[args.length-1] | 
					
						
							| 
									
										
										
										
											2020-04-29 07:49:21 +03:00
										 |  |  | 		attrs = (last != null && typeof(last) != typeof('str')) ?  | 
					
						
							|  |  |  | 			args.pop()  | 
					
						
							|  |  |  | 			: {} | 
					
						
							|  |  |  | 		doc = typeof(args[0]) == typeof('str') ?  | 
					
						
							|  |  |  | 			args.shift()  | 
					
						
							|  |  |  | 			: null | 
					
						
							|  |  |  | 		ldoc = typeof(args[0]) == typeof('str') ?  | 
					
						
							|  |  |  | 			args.shift()  | 
					
						
							|  |  |  | 			: null | 
					
						
							| 
									
										
										
										
											2020-04-23 17:59:13 +03:00
										 |  |  | 
 | 
					
						
							|  |  |  | 		attrs.alias = target | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | 		// NOTE: we are not parsing this directly here because the context
 | 
					
						
							|  |  |  | 		// 		may define a different .parseStringAction(..)
 | 
					
						
							| 
									
										
										
										
											2020-04-29 07:49:21 +03:00
										 |  |  | 		var parsed = typeof(target) == typeof('str') ?  | 
					
						
							|  |  |  | 			null  | 
					
						
							|  |  |  | 			: target | 
					
						
							| 
									
										
										
										
											2020-04-23 17:59:13 +03:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2020-04-29 07:49:21 +03:00
										 |  |  | 		doc = (!doc && parsed) ?  | 
					
						
							|  |  |  | 			parsed.doc  | 
					
						
							|  |  |  | 			: doc | 
					
						
							| 
									
										
										
										
											2020-04-23 17:59:13 +03:00
										 |  |  | 
 | 
					
						
							|  |  |  | 		var func = function(){ | 
					
						
							|  |  |  | 			// empty alias...
 | 
					
						
							|  |  |  | 			if(target == ''){ | 
					
						
							| 
									
										
										
										
											2020-04-29 16:45:01 +03:00
										 |  |  | 				return } | 
					
						
							| 
									
										
										
										
											2017-08-03 22:34:02 +03:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2020-04-23 17:59:13 +03:00
										 |  |  | 			var p = parsed  | 
					
						
							|  |  |  | 				|| (this.parseStringAction || parseStringAction)(target) | 
					
						
							| 
									
										
										
										
											2017-08-03 22:05:53 +03:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2020-04-23 17:59:13 +03:00
										 |  |  | 			return p.action in this ? | 
					
						
							| 
									
										
										
										
											2020-04-29 16:45:01 +03:00
										 |  |  | 				(this.parseStringAction || parseStringAction).callAction(this, p, ...arguments) | 
					
						
							| 
									
										
										
										
											2020-04-23 17:59:13 +03:00
										 |  |  | 				// error...
 | 
					
						
							| 
									
										
										
										
											2020-04-29 07:49:21 +03:00
										 |  |  | 				: console.error(`${alias}: Unknown alias target action: ${p.action}`) } | 
					
						
							| 
									
										
										
										
											2020-04-23 17:59:13 +03:00
										 |  |  | 		func.toString = function(){  | 
					
						
							|  |  |  | 			return meth.alias.code || meth.alias } | 
					
						
							| 
									
										
										
										
											2017-08-04 04:09:22 +03:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2020-04-23 17:59:13 +03:00
										 |  |  | 		// make the action...
 | 
					
						
							| 
									
										
										
										
											2020-04-29 07:49:21 +03:00
										 |  |  | 		var meth = object.parentCall(Alias.prototype.__new__, this, context, alias, doc, ldoc, attrs, func) | 
					
						
							| 
									
										
										
										
											2017-08-03 20:44:26 +03:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2020-04-23 17:59:13 +03:00
										 |  |  | 		meth.func.alias = target | 
					
						
							| 
									
										
										
										
											2017-08-03 22:34:02 +03:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2020-04-29 21:14:09 +03:00
										 |  |  | 		return meth }, | 
					
						
							| 
									
										
										
										
											2020-04-23 17:59:13 +03:00
										 |  |  | }) | 
					
						
							| 
									
										
										
										
											2017-08-03 20:44:26 +03:00
										 |  |  | 
 | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2016-08-20 23:46:34 +03:00
										 |  |  | //---------------------------------------------------------------------
 | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | // A base action-set object...
 | 
					
						
							|  |  |  | //
 | 
					
						
							|  |  |  | // This will define a set of action-set specific methods and helpers.
 | 
					
						
							|  |  |  | //
 | 
					
						
							|  |  |  | // XXX .off(...) needs more work...
 | 
					
						
							|  |  |  | // XXX need a mechanism to publish specific attrs...
 | 
					
						
							|  |  |  | var MetaActions = | 
					
						
							|  |  |  | module.MetaActions = { | 
					
						
							|  |  |  | 	// List actions...
 | 
					
						
							|  |  |  | 	//
 | 
					
						
							|  |  |  | 	get actions(){ | 
					
						
							|  |  |  | 		var res = [] | 
					
						
							|  |  |  | 		for(var k in this){ | 
					
						
							|  |  |  | 			// avoid recursion, skip props...
 | 
					
						
							|  |  |  | 			var cur = this | 
					
						
							|  |  |  | 			var prop = Object.getOwnPropertyDescriptor(cur, k) | 
					
						
							|  |  |  | 			while(!prop && cur.__proto__ != null){ | 
					
						
							|  |  |  | 				var cur = cur.__proto__ | 
					
						
							| 
									
										
										
										
											2020-11-23 14:47:00 +03:00
										 |  |  | 				var prop = Object.getOwnPropertyDescriptor(cur, k) } | 
					
						
							| 
									
										
										
										
											2016-08-20 23:46:34 +03:00
										 |  |  | 			if(prop.get != null){ | 
					
						
							| 
									
										
										
										
											2020-11-23 14:47:00 +03:00
										 |  |  | 				continue } | 
					
						
							| 
									
										
										
										
											2016-08-20 23:46:34 +03:00
										 |  |  | 			//if(k == 'actions' || k == 'length'){
 | 
					
						
							|  |  |  | 			//	continue
 | 
					
						
							|  |  |  | 			//}
 | 
					
						
							|  |  |  | 			// get only actions...
 | 
					
						
							|  |  |  | 			if(this[k] instanceof Action){ | 
					
						
							| 
									
										
										
										
											2020-11-23 14:47:00 +03:00
										 |  |  | 				res.push(k) } } | 
					
						
							| 
									
										
										
										
											2020-04-29 21:14:09 +03:00
										 |  |  | 		return res }, | 
					
						
							| 
									
										
										
										
											2016-08-20 23:46:34 +03:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2017-08-07 17:05:30 +03:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2017-08-03 20:44:26 +03:00
										 |  |  | 	// List aliases...
 | 
					
						
							| 
									
										
										
										
											2017-08-07 00:16:31 +03:00
										 |  |  | 	//
 | 
					
						
							| 
									
										
										
										
											2017-08-03 20:44:26 +03:00
										 |  |  | 	get aliases(){ | 
					
						
							|  |  |  | 		var that = this | 
					
						
							|  |  |  | 		return this.actions | 
					
						
							| 
									
										
										
										
											2017-08-07 00:16:31 +03:00
										 |  |  | 			.filter(function(n){  | 
					
						
							| 
									
										
										
										
											2020-04-29 20:56:39 +03:00
										 |  |  | 				return that[n] instanceof Alias }) }, | 
					
						
							|  |  |  | 	get localAliases(){ | 
					
						
							|  |  |  | 		var that = this | 
					
						
							|  |  |  | 		return this.aliases | 
					
						
							|  |  |  | 			.filter(function(n){ | 
					
						
							|  |  |  | 				return that.hasOwnProperty(n) })}, | 
					
						
							| 
									
										
										
										
											2017-08-03 20:44:26 +03:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2017-08-03 21:35:10 +03:00
										 |  |  | 	// XXX move this to the right spot...
 | 
					
						
							|  |  |  | 	parseStringAction: parseStringAction, | 
					
						
							| 
									
										
										
										
											2017-08-04 03:58:24 +03:00
										 |  |  | 	isStringAction: isStringAction, | 
					
						
							| 
									
										
										
										
											2017-08-03 21:35:10 +03:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2018-02-10 13:56:32 +03:00
										 |  |  | 	// XXX EXPERIMENTAL...
 | 
					
						
							|  |  |  | 	call: function(action, ...args){ | 
					
						
							| 
									
										
										
										
											2018-02-10 14:01:41 +03:00
										 |  |  | 		return action instanceof Function ? | 
					
						
							|  |  |  | 				action.apply(this, args) | 
					
						
							|  |  |  | 			: this[action] ? | 
					
						
							|  |  |  | 				this[action].apply(this, args) | 
					
						
							| 
									
										
										
										
											2018-02-10 13:56:32 +03:00
										 |  |  | 			: this.parseStringAction.applyAction(this, action, args) }, | 
					
						
							|  |  |  | 	apply: function(action, args){ | 
					
						
							|  |  |  | 		return this.call(action, ...args)}, | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2017-08-04 03:58:24 +03:00
										 |  |  | 	// Set/remove action alias...
 | 
					
						
							|  |  |  | 	//
 | 
					
						
							|  |  |  | 	// 	Set alias...
 | 
					
						
							|  |  |  | 	// 	.alias(alias, code)
 | 
					
						
							| 
									
										
										
										
											2017-08-04 09:42:49 +03:00
										 |  |  | 	// 	.alias(alias[, doc[, long-doc]][, attrs,] code)
 | 
					
						
							|  |  |  | 	// 	.alias(alias, [ [doc[, long-doc]][, attrs,] code ])
 | 
					
						
							| 
									
										
										
										
											2017-08-04 03:58:24 +03:00
										 |  |  | 	// 		-> action-set
 | 
					
						
							|  |  |  | 	//
 | 
					
						
							|  |  |  | 	// 	Remove alias...
 | 
					
						
							|  |  |  | 	// 	.alias(alias, null)
 | 
					
						
							| 
									
										
										
										
											2017-08-04 09:42:49 +03:00
										 |  |  | 	// 	.alias(alias, false)
 | 
					
						
							| 
									
										
										
										
											2017-08-04 03:58:24 +03:00
										 |  |  | 	// 		-> action-set
 | 
					
						
							|  |  |  | 	//
 | 
					
						
							|  |  |  | 	// code should be compatible with .parseStringAction(..)
 | 
					
						
							|  |  |  | 	//
 | 
					
						
							|  |  |  | 	// NOTE: this does not check if it will override anything, so it is
 | 
					
						
							|  |  |  | 	// 		possible to override/delete an action/method/attribute with 
 | 
					
						
							|  |  |  | 	// 		this...
 | 
					
						
							|  |  |  | 	//
 | 
					
						
							| 
									
										
										
										
											2017-08-03 22:05:53 +03:00
										 |  |  | 	// XXX should this prevent overriding stuff???
 | 
					
						
							|  |  |  | 	// XXX move to a better spot...
 | 
					
						
							| 
									
										
										
										
											2017-12-09 01:38:20 +03:00
										 |  |  | 	alias: Action('alias', function(alias, target){ | 
					
						
							| 
									
										
										
										
											2017-08-04 03:58:24 +03:00
										 |  |  | 		// remove alias...
 | 
					
						
							| 
									
										
										
										
											2018-03-30 14:47:54 +03:00
										 |  |  | 		if(arguments.length == 2 | 
					
						
							|  |  |  | 				&& (target === false || target === null)){ | 
					
						
							|  |  |  | 			// delete only aliases...
 | 
					
						
							|  |  |  | 			this[alias] instanceof Alias | 
					
						
							|  |  |  | 				&& (delete this[alias]) | 
					
						
							| 
									
										
										
										
											2017-08-04 03:58:24 +03:00
										 |  |  | 
 | 
					
						
							|  |  |  | 		// set alias...
 | 
					
						
							|  |  |  | 		} else { | 
					
						
							| 
									
										
										
										
											2020-11-23 14:47:00 +03:00
										 |  |  | 			this[alias] = Alias(...arguments) } }), | 
					
						
							| 
									
										
										
										
											2017-08-03 22:05:53 +03:00
										 |  |  | 
 | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2016-12-30 03:19:12 +03:00
										 |  |  | 	// Get action attribute...
 | 
					
						
							|  |  |  | 	//
 | 
					
						
							| 
									
										
										
										
											2017-09-06 20:29:16 +03:00
										 |  |  | 	// Attribute search order (return first matching):
 | 
					
						
							|  |  |  | 	// 	- Local action
 | 
					
						
							| 
									
										
										
										
											2017-12-24 06:46:39 +03:00
										 |  |  | 	// 	- Local action function (.func)
 | 
					
						
							|  |  |  | 	// 	- if an alias look in the target...
 | 
					
						
							| 
									
										
										
										
											2017-09-06 20:29:16 +03:00
										 |  |  | 	// 	- repeat for .__proto__ (until top of MRO)
 | 
					
						
							| 
									
										
										
										
											2020-04-23 17:59:13 +03:00
										 |  |  | 	// 	- repeat for '__actioncall__' special action (XXX EXPERIMENTAL)
 | 
					
						
							| 
									
										
										
										
											2017-09-06 20:29:16 +03:00
										 |  |  | 	//
 | 
					
						
							|  |  |  | 	//
 | 
					
						
							| 
									
										
										
										
											2017-02-19 05:18:29 +03:00
										 |  |  | 	// NOTE: this will get attribute set both on the action object and 
 | 
					
						
							|  |  |  | 	// 		the action function, this covers two usecases:
 | 
					
						
							|  |  |  | 	// 		1) action constructor attributes...
 | 
					
						
							|  |  |  | 	// 			someAction: ['...',
 | 
					
						
							|  |  |  | 	// 				// action attribute...
 | 
					
						
							|  |  |  | 	// 				{attr: 'value'},
 | 
					
						
							|  |  |  | 	// 				function(){ ... }],
 | 
					
						
							|  |  |  | 	// 		2) action modifiers... 
 | 
					
						
							|  |  |  | 	// 			var modifyAction = function(func){
 | 
					
						
							|  |  |  | 	// 				// function attribute...
 | 
					
						
							|  |  |  | 	// 				func.attr = 'value'
 | 
					
						
							|  |  |  | 	// 				return func
 | 
					
						
							|  |  |  | 	// 			}
 | 
					
						
							|  |  |  | 	//			...
 | 
					
						
							|  |  |  | 	// 			someAction: ['...',
 | 
					
						
							|  |  |  | 	// 				modifyAction(function(){ ... })],
 | 
					
						
							| 
									
										
										
										
											2017-09-06 20:29:16 +03:00
										 |  |  | 	//
 | 
					
						
							|  |  |  | 	// XXX document...
 | 
					
						
							| 
									
										
										
										
											2020-04-23 17:59:13 +03:00
										 |  |  | 	// XXX add option to to enable/disable look in .__actioncall__... 
 | 
					
						
							| 
									
										
										
										
											2017-01-02 19:42:56 +03:00
										 |  |  | 	getActionAttr: function(action, attr){ | 
					
						
							| 
									
										
										
										
											2016-12-30 03:19:12 +03:00
										 |  |  | 		var cur = this | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | 		// go up the proto chain...
 | 
					
						
							|  |  |  | 		while(cur.__proto__ != null){ | 
					
						
							| 
									
										
										
										
											2018-03-25 12:23:36 +03:00
										 |  |  | 			var c = cur[action] | 
					
						
							|  |  |  | 			if(c != null){ | 
					
						
							| 
									
										
										
										
											2017-02-19 05:18:29 +03:00
										 |  |  | 				// attribute of action...
 | 
					
						
							| 
									
										
										
										
											2018-03-25 12:23:36 +03:00
										 |  |  | 				if(c[attr] !== undefined){ | 
					
						
							|  |  |  | 					return c[attr] | 
					
						
							| 
									
										
										
										
											2017-02-19 05:18:29 +03:00
										 |  |  | 
 | 
					
						
							|  |  |  | 				// attribute of action function...
 | 
					
						
							| 
									
										
										
										
											2018-03-25 12:23:36 +03:00
										 |  |  | 				} else if(c.func && c.func[attr] !== undefined){ | 
					
						
							|  |  |  | 					return c.func[attr] | 
					
						
							| 
									
										
										
										
											2017-12-24 06:46:39 +03:00
										 |  |  | 
 | 
					
						
							|  |  |  | 				// alias -> look in the target action...
 | 
					
						
							| 
									
										
										
										
											2018-03-25 12:23:36 +03:00
										 |  |  | 				} else if(c instanceof Alias){ | 
					
						
							| 
									
										
										
										
											2017-12-24 06:46:39 +03:00
										 |  |  | 					var res = this.getActionAttr( | 
					
						
							|  |  |  | 						this.parseStringAction(cur[action].alias).action,  | 
					
						
							|  |  |  | 						attr) | 
					
						
							|  |  |  | 					if(res !== undefined){ | 
					
						
							| 
									
										
										
										
											2020-11-23 14:47:00 +03:00
										 |  |  | 						return res } } } | 
					
						
							|  |  |  | 			cur = cur.__proto__ } | 
					
						
							| 
									
										
										
										
											2017-09-06 20:29:16 +03:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2020-04-23 17:59:13 +03:00
										 |  |  | 		// search .__actioncall__ action...
 | 
					
						
							|  |  |  | 		if(cur[action] != null && action != '__actioncall__'){ | 
					
						
							| 
									
										
										
										
											2020-11-23 14:47:00 +03:00
										 |  |  | 			return this.getActionAttr('__actioncall__', attr) } }, | 
					
						
							| 
									
										
										
										
											2016-12-30 03:19:12 +03:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2017-10-08 18:01:13 +03:00
										 |  |  | 	// Get root action attribute value...
 | 
					
						
							|  |  |  | 	//
 | 
					
						
							|  |  |  | 	// This is similar to .getActionAttr(..) but will only chenck the 
 | 
					
						
							|  |  |  | 	// root action for the attribute...
 | 
					
						
							|  |  |  | 	//
 | 
					
						
							|  |  |  | 	// NOTE: if an attr is not explicitly defined in the root action, the
 | 
					
						
							|  |  |  | 	// 		base Action object is checked (Action.prototype.await)...
 | 
					
						
							|  |  |  | 	getRootActionAttr: function(action, attr){ | 
					
						
							|  |  |  | 		var cur = this | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | 		// go up the proto chain...
 | 
					
						
							|  |  |  | 		while(cur.__proto__ != null){ | 
					
						
							|  |  |  | 			if(cur[action] != null){ | 
					
						
							| 
									
										
										
										
											2020-11-23 14:47:00 +03:00
										 |  |  | 				var target = cur } | 
					
						
							|  |  |  | 			cur = cur.__proto__ } | 
					
						
							| 
									
										
										
										
											2017-10-08 18:01:13 +03:00
										 |  |  | 
 | 
					
						
							|  |  |  | 		// attribute of action...
 | 
					
						
							|  |  |  | 		if(target[action][attr] !== undefined){ | 
					
						
							|  |  |  | 			return target[action][attr] | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | 		// attribute of action function...
 | 
					
						
							|  |  |  | 		} else if(target[action].func  | 
					
						
							|  |  |  | 				&& target[action].func[attr] !== undefined){ | 
					
						
							| 
									
										
										
										
											2020-11-23 14:47:00 +03:00
										 |  |  | 			return target[action].func[attr] } }, | 
					
						
							| 
									
										
										
										
											2017-10-08 18:01:13 +03:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2016-08-20 23:46:34 +03:00
										 |  |  | 	// Get action documentation...
 | 
					
						
							|  |  |  | 	//
 | 
					
						
							| 
									
										
										
										
											2017-07-13 02:37:12 +03:00
										 |  |  | 	// Format:
 | 
					
						
							|  |  |  | 	// 	{
 | 
					
						
							|  |  |  | 	// 		action-name: [
 | 
					
						
							|  |  |  | 	// 			doc,
 | 
					
						
							|  |  |  | 	// 			long_doc,
 | 
					
						
							|  |  |  | 	// 			name,
 | 
					
						
							|  |  |  | 	// 		],
 | 
					
						
							|  |  |  | 	// 		...
 | 
					
						
							|  |  |  | 	// 	}
 | 
					
						
							|  |  |  | 	//
 | 
					
						
							|  |  |  | 	// NOTE: oveloading actions will shadow parents doc if they define .doc.
 | 
					
						
							| 
									
										
										
										
											2016-08-20 23:46:34 +03:00
										 |  |  | 	getDoc: function(actions){ | 
					
						
							|  |  |  | 		var res = {} | 
					
						
							|  |  |  | 		var that = this | 
					
						
							|  |  |  | 		actions = actions == null ? this.actions | 
					
						
							| 
									
										
										
										
											2020-04-17 23:21:39 +03:00
										 |  |  | 			: arguments.length > 1 ? [...arguments] | 
					
						
							| 
									
										
										
										
											2016-08-20 23:46:34 +03:00
										 |  |  | 			: typeof(actions) == typeof('str') ? [actions] | 
					
						
							|  |  |  | 			: actions | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | 		// get the first defined set of docs in the inheritance chain...
 | 
					
						
							|  |  |  | 		actions.forEach(function(n){ | 
					
						
							|  |  |  | 			var cur = that | 
					
						
							|  |  |  | 			res[n] = [] | 
					
						
							|  |  |  | 			// go up the proto chain...
 | 
					
						
							|  |  |  | 			while(cur.__proto__ != null){ | 
					
						
							|  |  |  | 				if(cur[n] != null && cur[n].doc != null){ | 
					
						
							|  |  |  | 					res[n] = [ cur[n].doc, cur[n].long_doc, cur[n].name ] | 
					
						
							| 
									
										
										
										
											2020-11-23 14:47:00 +03:00
										 |  |  | 					break } | 
					
						
							|  |  |  | 				cur = cur.__proto__ } }) | 
					
						
							| 
									
										
										
										
											2020-04-29 21:14:09 +03:00
										 |  |  | 		return res }, | 
					
						
							| 
									
										
										
										
											2016-08-20 23:46:34 +03:00
										 |  |  | 
 | 
					
						
							|  |  |  | 	getPath: function(actions){ | 
					
						
							|  |  |  | 		var res = {} | 
					
						
							|  |  |  | 		var that = this | 
					
						
							|  |  |  | 		actions = actions == null ? this.actions | 
					
						
							| 
									
										
										
										
											2020-04-17 23:21:39 +03:00
										 |  |  | 			: arguments.length > 1 ? [...arguments] | 
					
						
							| 
									
										
										
										
											2016-08-20 23:46:34 +03:00
										 |  |  | 			: typeof(actions) == typeof('str') ? [actions] | 
					
						
							|  |  |  | 			: actions | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | 		// get the first defined set of docs in the inheritance chain...
 | 
					
						
							|  |  |  | 		actions.forEach(function(n){ | 
					
						
							|  |  |  | 			var cur = that | 
					
						
							|  |  |  | 			// go up the proto chain...
 | 
					
						
							|  |  |  | 			while(cur.__proto__ != null){ | 
					
						
							|  |  |  | 				if(cur[n] != null && cur[n].doc != null){ | 
					
						
							|  |  |  | 					var doc = cur[n].doc | 
					
						
							|  |  |  | 					var long_doc = cur[n].long_doc | 
					
						
							| 
									
										
										
										
											2020-11-23 14:47:00 +03:00
										 |  |  | 					break } | 
					
						
							|  |  |  | 				cur = cur.__proto__ } | 
					
						
							| 
									
										
										
										
											2016-08-20 23:46:34 +03:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2020-11-23 14:47:00 +03:00
										 |  |  | 			res[(doc && doc.replace(/[\\\/]$/, '/'+n)) || n] = [n, doc, long_doc] }) | 
					
						
							| 
									
										
										
										
											2020-04-29 21:14:09 +03:00
										 |  |  | 		return res }, | 
					
						
							| 
									
										
										
										
											2016-08-20 23:46:34 +03:00
										 |  |  | 
 | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2016-10-27 18:47:17 +03:00
										 |  |  | 	// Toggle handler cache...
 | 
					
						
							|  |  |  | 	//
 | 
					
						
							|  |  |  | 	//	Toggle cache...
 | 
					
						
							|  |  |  | 	//	.toggleHandlerCache()
 | 
					
						
							|  |  |  | 	//
 | 
					
						
							|  |  |  | 	//	Set caching on...
 | 
					
						
							|  |  |  | 	//	.toggleHandlerCache('on')
 | 
					
						
							|  |  |  | 	//	.toggleHandlerCache(true)
 | 
					
						
							|  |  |  | 	//
 | 
					
						
							|  |  |  | 	//	Set caching off...
 | 
					
						
							|  |  |  | 	//	.toggleHandlerCache('off')
 | 
					
						
							|  |  |  | 	//	.toggleHandlerCache(false)
 | 
					
						
							|  |  |  | 	//
 | 
					
						
							|  |  |  | 	//	Reset caching...
 | 
					
						
							|  |  |  | 	//	.toggleHandlerCache('!')
 | 
					
						
							|  |  |  | 	//
 | 
					
						
							|  |  |  | 	//	Get current caching state...
 | 
					
						
							|  |  |  | 	//	.toggleHandlerCache('?')
 | 
					
						
							|  |  |  | 	//
 | 
					
						
							|  |  |  | 	//	Get supported states...
 | 
					
						
							|  |  |  | 	//	.toggleHandlerCache('??')
 | 
					
						
							|  |  |  | 	//		-> ['on', 'off']
 | 
					
						
							|  |  |  | 	//
 | 
					
						
							|  |  |  | 	//
 | 
					
						
							|  |  |  | 	// NOTE: setting the cache on may prevent calling of actions event
 | 
					
						
							|  |  |  | 	// 		handlers of parent action-sets if they are added (via .on(..)
 | 
					
						
							|  |  |  | 	// 		or .one(..), ...) AFTER the current object cloned it's cache.
 | 
					
						
							|  |  |  | 	// 		to avoid this, care must be taken to reset the cache of 
 | 
					
						
							|  |  |  | 	// 		children objects, or not use caching for cases where action
 | 
					
						
							|  |  |  | 	// 		event handlers can be added on the tree in runtime.
 | 
					
						
							|  |  |  | 	//
 | 
					
						
							|  |  |  | 	//
 | 
					
						
							|  |  |  | 	// XXX should we use the toggler object here???
 | 
					
						
							| 
									
										
										
										
											2018-12-11 20:07:36 +03:00
										 |  |  | 	// XXX EXPERIMENTAL (handler cache)...
 | 
					
						
							| 
									
										
										
										
											2016-10-27 18:47:17 +03:00
										 |  |  | 	toggleHandlerCache: function(to){ | 
					
						
							|  |  |  | 		if(to == '?'){ | 
					
						
							|  |  |  | 			return this.__handler_cache ? 'on' : 'off' | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | 		} else if(to == '??'){ | 
					
						
							| 
									
										
										
										
											2020-11-23 14:47:00 +03:00
										 |  |  | 			return ['on', 'off'] } | 
					
						
							| 
									
										
										
										
											2016-10-27 18:47:17 +03:00
										 |  |  | 
 | 
					
						
							|  |  |  | 		to = (to === true || to == 'on') ? true | 
					
						
							|  |  |  | 			: (to === false || to == 'off') ? false | 
					
						
							|  |  |  | 			: to == '!' ? !!this.__handler_cache | 
					
						
							|  |  |  | 			: !this.__handler_cache  | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | 		if(to){ | 
					
						
							|  |  |  | 			// no local cache -> copy from parent...
 | 
					
						
							|  |  |  | 			if(this.__handler_cache  | 
					
						
							|  |  |  | 					&& !Object.hasOwnProperty(this, '__handler_cache')){ | 
					
						
							|  |  |  | 				var parent = this.__handler_cache | 
					
						
							|  |  |  | 				var cache = this.__handler_cache = {} | 
					
						
							|  |  |  | 				for(var a in parent){ | 
					
						
							| 
									
										
										
										
											2020-11-23 14:47:00 +03:00
										 |  |  | 					cache[a] = parent[a] } | 
					
						
							| 
									
										
										
										
											2016-10-27 18:47:17 +03:00
										 |  |  | 
 | 
					
						
							|  |  |  | 			// local cache only...
 | 
					
						
							|  |  |  | 			} else { | 
					
						
							| 
									
										
										
										
											2020-11-23 14:47:00 +03:00
										 |  |  | 				this.__handler_cache = this.__handler_cache || {} } | 
					
						
							| 
									
										
										
										
											2016-10-27 18:47:17 +03:00
										 |  |  | 
 | 
					
						
							|  |  |  | 		} else { | 
					
						
							|  |  |  | 			// NOTE: we do not delete here so as to shadow the parent's 
 | 
					
						
							|  |  |  | 			// 		cache...
 | 
					
						
							| 
									
										
										
										
											2020-11-23 14:47:00 +03:00
										 |  |  | 			this.__handler_cache = false } | 
					
						
							| 
									
										
										
										
											2016-10-27 18:47:17 +03:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2016-12-11 05:13:50 +03:00
										 |  |  | 		// XXX this is not the handler protocol...
 | 
					
						
							| 
									
										
										
										
											2020-04-29 21:14:09 +03:00
										 |  |  | 		return this }, | 
					
						
							| 
									
										
										
										
											2016-10-27 18:47:17 +03:00
										 |  |  | 
 | 
					
						
							|  |  |  | 	// Rest handler cache...
 | 
					
						
							|  |  |  | 	// 	
 | 
					
						
							|  |  |  | 	// 	Reset the full cache...
 | 
					
						
							|  |  |  | 	// 	.resetHandlerCache()
 | 
					
						
							|  |  |  | 	// 		-> this
 | 
					
						
							|  |  |  | 	//
 | 
					
						
							|  |  |  | 	// 	Reset handler cache for action...
 | 
					
						
							|  |  |  | 	// 	.resetHandlerCache(action)
 | 
					
						
							|  |  |  | 	// 		-> this
 | 
					
						
							|  |  |  | 	//
 | 
					
						
							|  |  |  | 	// NOTE: when .toggleHandlerCache('?') is 'off' this has no effect.
 | 
					
						
							|  |  |  | 	//
 | 
					
						
							| 
									
										
										
										
											2018-12-11 20:07:36 +03:00
										 |  |  | 	// XXX EXPERIMENTAL (handler cache)...
 | 
					
						
							| 
									
										
										
										
											2016-10-27 18:47:17 +03:00
										 |  |  | 	resetHandlerCache: function(name){ | 
					
						
							|  |  |  | 		var cache = this.__handler_cache | 
					
						
							|  |  |  | 		if(cache){ | 
					
						
							|  |  |  | 			// full reset...
 | 
					
						
							|  |  |  | 			if(name == null){ | 
					
						
							|  |  |  | 				this.__handler_cache = {} | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | 			// reset action...
 | 
					
						
							|  |  |  | 			} else { | 
					
						
							|  |  |  | 				// no local cache -> copy from parent...
 | 
					
						
							|  |  |  | 				if(!Object.hasOwnProperty(this, '__handler_cache')){ | 
					
						
							|  |  |  | 					var parent = this.__handler_cache | 
					
						
							|  |  |  | 					var cache = this.__handler_cache = {} | 
					
						
							|  |  |  | 					for(var a in parent){ | 
					
						
							| 
									
										
										
										
											2020-11-23 14:47:00 +03:00
										 |  |  | 						cache[a] = parent[a] } } | 
					
						
							| 
									
										
										
										
											2016-10-27 18:47:17 +03:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2020-11-23 14:47:00 +03:00
										 |  |  | 				delete cache[name] } } | 
					
						
							| 
									
										
										
										
											2020-04-29 21:14:09 +03:00
										 |  |  | 		return this }, | 
					
						
							| 
									
										
										
										
											2016-10-27 18:47:17 +03:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2016-08-20 23:46:34 +03:00
										 |  |  | 	// Get action handlers from the inheritance chain...
 | 
					
						
							|  |  |  | 	//
 | 
					
						
							|  |  |  | 	// NOTE: this collects both the event handlers (in order of hierarchy,
 | 
					
						
							|  |  |  | 	// 		then order of definition) and actions (in order of hierarchy)
 | 
					
						
							|  |  |  | 	// NOTE: this is the correct order for 'pre' calling, but is the 
 | 
					
						
							|  |  |  | 	// 		reverse of how the 'post' handlers must be called.
 | 
					
						
							| 
									
										
										
										
											2016-10-27 18:47:17 +03:00
										 |  |  | 	// NOTE: if .toggleHandlerCache('?') is on, this will serch once and
 | 
					
						
							|  |  |  | 	// 		return the cached results on every subsequent call.
 | 
					
						
							| 
									
										
										
										
											2016-08-20 23:46:34 +03:00
										 |  |  | 	//
 | 
					
						
							|  |  |  | 	// For more docs on handler sequencing and definition see: .on(..)
 | 
					
						
							|  |  |  | 	getHandlerList: function(name){ | 
					
						
							| 
									
										
										
										
											2018-12-11 20:07:36 +03:00
										 |  |  | 		// handler cache...
 | 
					
						
							|  |  |  | 		// XXX EXPERIMENTAL (handler cache)...
 | 
					
						
							| 
									
										
										
										
											2016-10-27 18:47:17 +03:00
										 |  |  | 		var cache = this.__handler_cache | 
					
						
							|  |  |  | 		if(cache && cache[name]){ | 
					
						
							| 
									
										
										
										
											2020-11-23 14:47:00 +03:00
										 |  |  | 			return cache[name].slice() } | 
					
						
							| 
									
										
										
										
											2016-10-27 18:47:17 +03:00
										 |  |  | 
 | 
					
						
							|  |  |  | 		// get the handlers...
 | 
					
						
							| 
									
										
										
										
											2016-08-20 23:46:34 +03:00
										 |  |  | 		var handlers = [] | 
					
						
							|  |  |  | 		var cur = this | 
					
						
							|  |  |  | 		while(cur.__proto__ != null){ | 
					
						
							|  |  |  | 			// get action "event" handlers...
 | 
					
						
							|  |  |  | 			if(cur.hasOwnProperty('_action_handlers')  | 
					
						
							|  |  |  | 					&& name in cur._action_handlers){ | 
					
						
							|  |  |  | 				handlers.splice.apply(handlers, | 
					
						
							| 
									
										
										
										
											2020-11-23 14:47:00 +03:00
										 |  |  | 						[handlers.length, 0] | 
					
						
							|  |  |  | 							.concat(cur._action_handlers[name])) } | 
					
						
							| 
									
										
										
										
											2016-08-20 23:46:34 +03:00
										 |  |  | 
 | 
					
						
							|  |  |  | 			// get the overloading action...
 | 
					
						
							|  |  |  | 			// NOTE: this will get all the handlers including the root 
 | 
					
						
							|  |  |  | 			// 		and the current handlers...
 | 
					
						
							| 
									
										
										
										
											2017-09-07 16:53:41 +03:00
										 |  |  | 			// NOTE: if this encounters a matching mormal method/function 
 | 
					
						
							|  |  |  | 			// 		this will not search beyond it.
 | 
					
						
							|  |  |  | 			if(cur.hasOwnProperty(name)){ | 
					
						
							|  |  |  | 				// action -> collect...
 | 
					
						
							|  |  |  | 				if(cur[name] instanceof Action){ | 
					
						
							|  |  |  | 					handlers.push(cur[name].func) | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | 				// function -> terminate chain...
 | 
					
						
							|  |  |  | 				} else if(cur[name] instanceof Function){ | 
					
						
							|  |  |  | 					handlers.push(cur[name]) | 
					
						
							| 
									
										
										
										
											2020-11-23 14:47:00 +03:00
										 |  |  | 					break } } | 
					
						
							| 
									
										
										
										
											2016-08-20 23:46:34 +03:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2020-11-23 14:47:00 +03:00
										 |  |  | 			cur = cur.__proto__ } | 
					
						
							| 
									
										
										
										
											2016-10-27 18:47:17 +03:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2018-12-11 20:07:36 +03:00
										 |  |  | 		// handler cache... 
 | 
					
						
							|  |  |  | 		// XXX EXPERIMENTAL (handler cache)...
 | 
					
						
							| 
									
										
										
										
											2016-10-27 18:47:17 +03:00
										 |  |  | 		if(cache){ | 
					
						
							| 
									
										
										
										
											2020-11-23 14:47:00 +03:00
										 |  |  | 			cache[name] = handlers } | 
					
						
							| 
									
										
										
										
											2016-10-27 18:47:17 +03:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2020-04-29 21:14:09 +03:00
										 |  |  | 		return handlers }, | 
					
						
							| 
									
										
										
										
											2016-08-20 23:46:34 +03:00
										 |  |  | 
 | 
					
						
							|  |  |  | 	// Get structured action handler definitions...
 | 
					
						
							|  |  |  | 	//
 | 
					
						
							|  |  |  | 	// Format:
 | 
					
						
							|  |  |  | 	// 	[
 | 
					
						
							|  |  |  | 	// 		{
 | 
					
						
							|  |  |  | 	// 			// NOTE: only one handler per level can be present, either
 | 
					
						
							|  |  |  | 	// 			//		.pre or .post, this does not mean that one can
 | 
					
						
							|  |  |  | 	// 			//		not define both, just that they are stored separately
 | 
					
						
							|  |  |  | 	// 			pre|post: <handler>,
 | 
					
						
							| 
									
										
										
										
											2017-08-03 22:34:02 +03:00
										 |  |  | 	//
 | 
					
						
							|  |  |  | 	// 			// XXX
 | 
					
						
							|  |  |  | 	// 			alias: <target>,
 | 
					
						
							| 
									
										
										
										
											2016-08-20 23:46:34 +03:00
										 |  |  | 	// 		},
 | 
					
						
							|  |  |  | 	// 		...
 | 
					
						
							|  |  |  | 	// 	]
 | 
					
						
							|  |  |  | 	//
 | 
					
						
							|  |  |  | 	// XXX need to get parent action or definition context for each level... 
 | 
					
						
							|  |  |  | 	// XXX is this simpler to use than the original .getHandlerList(..)
 | 
					
						
							|  |  |  | 	// XXX rename this....
 | 
					
						
							|  |  |  | 	getHandlers: function(name){ | 
					
						
							|  |  |  | 		return (this.getHandlerList || MetaActions.getHandlerList).call(this, name) | 
					
						
							|  |  |  | 			.map(function(a){  | 
					
						
							|  |  |  | 				var res = { | 
					
						
							|  |  |  | 					// action doc...
 | 
					
						
							|  |  |  | 					// XXX
 | 
					
						
							|  |  |  | 				} | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | 				if(!a.post_handler){ | 
					
						
							|  |  |  | 					res.pre = a | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | 				} else { | 
					
						
							| 
									
										
										
										
											2020-11-23 14:47:00 +03:00
										 |  |  | 					res.post = a.post_handler } | 
					
						
							| 
									
										
										
										
											2016-08-20 23:46:34 +03:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2018-10-30 22:00:24 +03:00
										 |  |  | 				a.doc | 
					
						
							|  |  |  | 					&& (res.doc = a.doc) | 
					
						
							|  |  |  | 				a.long_doc | 
					
						
							|  |  |  | 					&& (res.long_doc = a.long_doc) | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2020-04-23 17:59:13 +03:00
										 |  |  | 				return res }) }, | 
					
						
							| 
									
										
										
										
											2016-08-20 23:46:34 +03:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2016-08-21 01:27:20 +03:00
										 |  |  | 	// Handler for cases when we need to avoid the pre/post handlers...
 | 
					
						
							| 
									
										
										
										
											2016-08-20 23:46:34 +03:00
										 |  |  | 	//
 | 
					
						
							| 
									
										
										
										
											2016-08-21 01:27:20 +03:00
										 |  |  | 	// Returns:
 | 
					
						
							|  |  |  | 	// 	- undefined		- handle the action normally.
 | 
					
						
							|  |  |  | 	// 	- object		- bypass action handlers.
 | 
					
						
							| 
									
										
										
										
											2016-08-20 23:46:34 +03:00
										 |  |  | 	//
 | 
					
						
							| 
									
										
										
										
											2016-08-21 01:27:20 +03:00
										 |  |  | 	// NOTE: the object result must be compatible with Action.pre(..) 
 | 
					
						
							|  |  |  | 	// 		return value...
 | 
					
						
							|  |  |  | 	// NOTE: this is mostly a stub, here for documentation reasons...
 | 
					
						
							| 
									
										
										
										
											2020-04-23 20:50:21 +03:00
										 |  |  | 	// XXX doc / revise...
 | 
					
						
							| 
									
										
										
										
											2016-08-21 01:27:20 +03:00
										 |  |  | 	//preActionHandler: doWithRootAction(
 | 
					
						
							|  |  |  | 	//	function(action, name, handlers, args){ return null }),
 | 
					
						
							| 
									
										
										
										
											2016-08-20 23:46:34 +03:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2016-08-21 01:27:20 +03:00
										 |  |  | 	 | 
					
						
							| 
									
										
										
										
											2016-08-20 23:46:34 +03:00
										 |  |  | 	// Register an action callback...
 | 
					
						
							|  |  |  | 	//
 | 
					
						
							|  |  |  | 	//	Register a post action callback
 | 
					
						
							|  |  |  | 	// 	.on('action', [<tag>, ]<function>)
 | 
					
						
							|  |  |  | 	// 	.on('action.post', [<tag>, ]<function>)
 | 
					
						
							|  |  |  | 	// 		-> <action-set>
 | 
					
						
							|  |  |  | 	//
 | 
					
						
							|  |  |  | 	// 	Register a pre action callback
 | 
					
						
							|  |  |  | 	// 	.on('action.pre', [<tag>, ]<function>)
 | 
					
						
							|  |  |  | 	// 		-> <action-set>
 | 
					
						
							|  |  |  | 	//
 | 
					
						
							|  |  |  | 	// Modes:
 | 
					
						
							|  |  |  | 	// 	'pre'		- the handler is fired before the action is triggered,
 | 
					
						
							|  |  |  | 	// 					and if the handler returns a deferred or a function
 | 
					
						
							|  |  |  | 	// 					then that will get resolved, called resp. after
 | 
					
						
							|  |  |  | 	// 					the action is done.
 | 
					
						
							|  |  |  | 	// 	'post'		- the handler is fired after the action is finished.
 | 
					
						
							|  |  |  | 	// 					this is the default.
 | 
					
						
							|  |  |  | 	//
 | 
					
						
							|  |  |  | 	// Handler Arguments:
 | 
					
						
							|  |  |  | 	// 	'pre'		- the handler will get the same arguments as the main
 | 
					
						
							|  |  |  | 	// 					action when called.
 | 
					
						
							|  |  |  | 	// 	'post'		- the handler will get the action return value followed
 | 
					
						
							|  |  |  | 	// 					by action arguments.
 | 
					
						
							|  |  |  | 	//
 | 
					
						
							|  |  |  | 	// The optional tag marks the handler to enable group removal via 
 | 
					
						
							|  |  |  | 	// .off(..)
 | 
					
						
							|  |  |  | 	//
 | 
					
						
							|  |  |  | 	// NOTE: 'post' mode is the default.
 | 
					
						
							|  |  |  | 	//
 | 
					
						
							|  |  |  | 	// XXX should we have multiple tags per handler???
 | 
					
						
							|  |  |  | 	on: function(actions, b, c){ | 
					
						
							| 
									
										
										
										
											2020-11-23 14:47:00 +03:00
										 |  |  | 		var that = this | 
					
						
							|  |  |  | 		//var handler = typeof(c) == 'function' ? c : b
 | 
					
						
							|  |  |  | 		//var tag = typeof(c) == 'function' ? b : c
 | 
					
						
							|  |  |  | 		var _handler = arguments.length == 3 ? c : b | 
					
						
							|  |  |  | 		var tag = arguments.length == 3 ? b : c | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | 		// alias handler...
 | 
					
						
							|  |  |  | 		// cache the parsed handler...
 | 
					
						
							|  |  |  | 		var parsed | 
					
						
							|  |  |  | 		var handler = typeof(_handler) == 'function' ? | 
					
						
							|  |  |  | 			_handler | 
					
						
							|  |  |  | 			// alias handler...
 | 
					
						
							|  |  |  | 			: function(...args){ | 
					
						
							|  |  |  | 				parsed = parsed  | 
					
						
							|  |  |  | 					|| (this.parseStringAction  | 
					
						
							|  |  |  | 						|| parseStringAction)(_handler) | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | 				return parsed.action in this ? | 
					
						
							|  |  |  | 					(this.parseStringAction  | 
					
						
							|  |  |  | 							|| parseStringAction) | 
					
						
							|  |  |  | 						.callAction(this, parsed, ...arguments) | 
					
						
							|  |  |  | 					// error...
 | 
					
						
							|  |  |  | 					: console.error( | 
					
						
							|  |  |  | 						`.on(..): Unknown handler target action: ${parsed.action}`) } | 
					
						
							| 
									
										
										
										
											2016-08-20 23:46:34 +03:00
										 |  |  | 
 | 
					
						
							|  |  |  | 		// XXX make this split by whitespace...
 | 
					
						
							|  |  |  | 		actions = typeof(actions) == 'string' ? actions.split(/ +/) : actions | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | 		actions.forEach(function(action){ | 
					
						
							|  |  |  | 			// prepare the handler...
 | 
					
						
							|  |  |  | 			var mode = action.split('.') | 
					
						
							|  |  |  | 			action = mode[0] | 
					
						
							|  |  |  | 			mode = mode[1] | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2016-10-27 18:47:17 +03:00
										 |  |  | 			that.resetHandlerCache(action) | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2016-08-20 23:46:34 +03:00
										 |  |  | 			// keep the original handler for future use...
 | 
					
						
							|  |  |  | 			var a_handler = handler | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | 			// a post handler (default)...
 | 
					
						
							|  |  |  | 			if(mode == null || mode == 'post'){ | 
					
						
							|  |  |  | 				var old_handler = a_handler | 
					
						
							|  |  |  | 				a_handler = function(){ return old_handler } | 
					
						
							|  |  |  | 				a_handler.post_handler = old_handler | 
					
						
							|  |  |  | 					// NOTE: this is set so as to identify the handler 
 | 
					
						
							|  |  |  | 					// 		for removal via. .off(..)
 | 
					
						
							|  |  |  | 				a_handler.orig_handler = old_handler.orig_handler || old_handler | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2016-11-05 19:31:26 +03:00
										 |  |  | 				a_handler.orig_handler.event_tag = tag | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2016-08-20 23:46:34 +03:00
										 |  |  | 			// not pre mode...
 | 
					
						
							|  |  |  | 			} else if(mode != 'pre') { | 
					
						
							|  |  |  | 				// XXX
 | 
					
						
							| 
									
										
										
										
											2020-11-23 14:47:00 +03:00
										 |  |  | 				throw new Error('Unknown action mode: '+action+'.'+mode) } | 
					
						
							| 
									
										
										
										
											2016-08-20 23:46:34 +03:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2016-11-05 19:31:26 +03:00
										 |  |  | 			a_handler.event_tag = tag | 
					
						
							| 
									
										
										
										
											2016-08-20 23:46:34 +03:00
										 |  |  | 
 | 
					
						
							|  |  |  | 			// register handlers locally only...
 | 
					
						
							|  |  |  | 			if(!that.hasOwnProperty('_action_handlers')){ | 
					
						
							| 
									
										
										
										
											2020-11-23 14:47:00 +03:00
										 |  |  | 				that._action_handlers = {} } | 
					
						
							| 
									
										
										
										
											2016-08-20 23:46:34 +03:00
										 |  |  | 			if(!(action in that._action_handlers)){ | 
					
						
							| 
									
										
										
										
											2020-11-23 14:47:00 +03:00
										 |  |  | 				that._action_handlers[action] = [] } | 
					
						
							| 
									
										
										
										
											2016-08-20 23:46:34 +03:00
										 |  |  | 			// register a handler only once...
 | 
					
						
							|  |  |  | 			if(that._action_handlers[action].indexOf(a_handler) < 0){ | 
					
						
							|  |  |  | 				// NOTE: last registered is first...
 | 
					
						
							| 
									
										
										
										
											2020-11-23 14:47:00 +03:00
										 |  |  | 				that._action_handlers[action].splice(0, 0, a_handler) } }) | 
					
						
							| 
									
										
										
										
											2016-08-20 23:46:34 +03:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2020-04-29 21:14:09 +03:00
										 |  |  | 		return this }, | 
					
						
							| 
									
										
										
										
											2016-08-20 23:46:34 +03:00
										 |  |  | 
 | 
					
						
							|  |  |  | 	// Remove an action callback...
 | 
					
						
							|  |  |  | 	//
 | 
					
						
							|  |  |  | 	//	Remove all handlers from action:
 | 
					
						
							|  |  |  | 	//	.off('action')
 | 
					
						
							|  |  |  | 	//	.off('action', '*')
 | 
					
						
							|  |  |  | 	//	.off('action', 'all')
 | 
					
						
							|  |  |  | 	// 		-> <action-set>
 | 
					
						
							|  |  |  | 	//
 | 
					
						
							|  |  |  | 	//	Remove specific handler from action:
 | 
					
						
							|  |  |  | 	//	.off('action', <handler>)
 | 
					
						
							|  |  |  | 	// 		-> <action-set>
 | 
					
						
							|  |  |  | 	//
 | 
					
						
							|  |  |  | 	//	Remove handlers from action by tag:
 | 
					
						
							|  |  |  | 	//	.off('action', <tag>)
 | 
					
						
							|  |  |  | 	// 		-> <action-set>
 | 
					
						
							|  |  |  | 	//
 | 
					
						
							|  |  |  | 	// NOTE: the handler passed to .off(..) for removal must be the same
 | 
					
						
							|  |  |  | 	// 		as the handler passed to .on(..) / .one(..)
 | 
					
						
							|  |  |  | 	off: function(actions, handler){ | 
					
						
							|  |  |  | 		if(this.hasOwnProperty('_action_handlers')){ | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | 			actions = actions == '*' ? Object.keys(this._action_handlers) | 
					
						
							|  |  |  | 				: typeof(actions) == 'string' ?  actions.split(' ') | 
					
						
							|  |  |  | 				: actions | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | 			var that = this | 
					
						
							|  |  |  | 			actions.forEach(function(action){ | 
					
						
							|  |  |  | 				var mode = action.split('.') | 
					
						
							|  |  |  | 				action = mode[0] | 
					
						
							|  |  |  | 				mode = mode[1] | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2016-10-27 18:47:17 +03:00
										 |  |  | 				that.resetHandlerCache(action) | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2016-08-20 23:46:34 +03:00
										 |  |  | 				// get the handlers...
 | 
					
						
							|  |  |  | 				var h = that._action_handlers[action] || [] | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | 				// remove explicit handler...
 | 
					
						
							|  |  |  | 				if(typeof(handler) == 'function'){ | 
					
						
							|  |  |  | 					var i = -1 | 
					
						
							|  |  |  | 					if(mode == null || mode == 'post'){ | 
					
						
							|  |  |  | 						// XXX find via e.orig_handler == handler && e.mode == 'post'
 | 
					
						
							|  |  |  | 						h.forEach(function(e, j){ | 
					
						
							|  |  |  | 							// NOTE: we will only get the first match...
 | 
					
						
							|  |  |  | 							if(e.orig_handler === handler && i == -1){ | 
					
						
							| 
									
										
										
										
											2020-11-23 14:47:00 +03:00
										 |  |  | 								i = j } }) | 
					
						
							| 
									
										
										
										
											2016-08-20 23:46:34 +03:00
										 |  |  | 
 | 
					
						
							|  |  |  | 					} else if(mode == 'pre'){ | 
					
						
							| 
									
										
										
										
											2020-11-23 14:47:00 +03:00
										 |  |  | 						i = h.indexOf(handler) } | 
					
						
							| 
									
										
										
										
											2016-08-20 23:46:34 +03:00
										 |  |  | 
 | 
					
						
							|  |  |  | 					// NOTE: unknown modes are skipped...
 | 
					
						
							|  |  |  | 					if(i >= 0){ | 
					
						
							| 
									
										
										
										
											2020-11-23 14:47:00 +03:00
										 |  |  | 						h.splice(i, 1) } | 
					
						
							| 
									
										
										
										
											2016-08-20 23:46:34 +03:00
										 |  |  | 
 | 
					
						
							|  |  |  | 				// remove all handlers...
 | 
					
						
							|  |  |  | 				} else if(handler == null || handler == 'all' || handler == '*'){ | 
					
						
							|  |  |  | 					h.splice(0, h.length) | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | 				// remove handlers by tag...
 | 
					
						
							|  |  |  | 				} else { | 
					
						
							|  |  |  | 					// filter out everything that mathches a tag in-place...
 | 
					
						
							|  |  |  | 					h.splice.apply(h,  | 
					
						
							|  |  |  | 							[0, h.length] | 
					
						
							|  |  |  | 								.concat(h.filter(function(e){  | 
					
						
							| 
									
										
										
										
											2020-11-23 14:47:00 +03:00
										 |  |  | 									return e.event_tag != handler }))) } }) } | 
					
						
							| 
									
										
										
										
											2016-08-20 23:46:34 +03:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2020-04-29 21:14:09 +03:00
										 |  |  | 		return this }, | 
					
						
							| 
									
										
										
										
											2016-08-20 23:46:34 +03:00
										 |  |  | 
 | 
					
						
							|  |  |  | 	// Register an action callback that will only fire once per event...
 | 
					
						
							|  |  |  | 	//
 | 
					
						
							|  |  |  | 	// This is signature compatible with .on(..)
 | 
					
						
							|  |  |  | 	one: function(actions, b, c){ | 
					
						
							|  |  |  | 		var _handler = typeof(c) == 'function' ? c : b | 
					
						
							|  |  |  | 		var tag = typeof(c) == 'function' ? b : c | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | 		actions = typeof(actions) == 'string' ? actions.split(' *') : actions | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | 		var that = this | 
					
						
							|  |  |  | 		actions.forEach(function(action){ | 
					
						
							|  |  |  | 			// NOTE: we are using both 'that' below and 'this', so as
 | 
					
						
							|  |  |  | 			// 		to separate the call context and the bind context,
 | 
					
						
							|  |  |  | 			//		.off(..) must be called at the bind context while
 | 
					
						
							|  |  |  | 			//		the actual action is called from the call context
 | 
					
						
							|  |  |  | 			// NOTE: we are not using the closure _handler below to 
 | 
					
						
							|  |  |  | 			// 		keep the code introspectable, and direct the user
 | 
					
						
							|  |  |  | 			// 		to the original function.
 | 
					
						
							|  |  |  | 			var handler = function(){ | 
					
						
							|  |  |  | 				// remove handler...
 | 
					
						
							|  |  |  | 				that.off(action, handler.orig_handler) | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | 				// call the actual supplied handler function...
 | 
					
						
							| 
									
										
										
										
											2020-11-23 14:47:00 +03:00
										 |  |  | 				return handler.orig_handler.apply(this, arguments) } | 
					
						
							| 
									
										
										
										
											2016-08-20 23:46:34 +03:00
										 |  |  | 			handler.orig_handler = _handler | 
					
						
							| 
									
										
										
										
											2020-11-23 14:47:00 +03:00
										 |  |  | 			that.on(action, tag, handler) }) | 
					
						
							| 
									
										
										
										
											2016-08-20 23:46:34 +03:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2020-04-29 21:14:09 +03:00
										 |  |  | 		return this }, | 
					
						
							| 
									
										
										
										
											2016-08-20 23:46:34 +03:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2018-12-11 20:07:36 +03:00
										 |  |  | 	// XXX EXPERIMENTAL (after calls)...
 | 
					
						
							|  |  |  | 	isActionRunning: function(){ | 
					
						
							|  |  |  | 		return !!this.__action_after_running }, | 
					
						
							|  |  |  | 	// Queue a function after the action is done...
 | 
					
						
							|  |  |  | 	//
 | 
					
						
							|  |  |  | 	// 	.afterAction(func)
 | 
					
						
							|  |  |  | 	// 	.afterAction('top', func)
 | 
					
						
							|  |  |  | 	// 		-> this
 | 
					
						
							|  |  |  | 	//
 | 
					
						
							|  |  |  | 	// 	.afterAction('local', func)
 | 
					
						
							|  |  |  | 	// 		-> this
 | 
					
						
							|  |  |  | 	//
 | 
					
						
							|  |  |  | 	// XXX EXPERIMENTAL (after calls)...
 | 
					
						
							|  |  |  | 	afterAction: function(mode, func){ | 
					
						
							|  |  |  | 		func = mode instanceof Function ? mode : func | 
					
						
							|  |  |  | 		mode = mode instanceof Function ? null : mode | 
					
						
							|  |  |  | 		mode = mode || 'top' | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | 		if(!this.__action_after_running){ | 
					
						
							| 
									
										
										
										
											2020-11-23 14:47:00 +03:00
										 |  |  | 			throw new Error('afterAction: no action is running.') } | 
					
						
							| 
									
										
										
										
											2018-12-11 20:07:36 +03:00
										 |  |  | 
 | 
					
						
							|  |  |  | 		;(mode == 'top' ? | 
					
						
							|  |  |  | 				this.__action_after_running[1] | 
					
						
							|  |  |  | 			: mode == 'local' ? | 
					
						
							|  |  |  | 				this.__action_after_running | 
					
						
							|  |  |  | 			: this.__action_after_running) | 
					
						
							|  |  |  | 			.push(func)  | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2020-04-29 21:14:09 +03:00
										 |  |  | 		return this }, | 
					
						
							| 
									
										
										
										
											2018-12-11 20:07:36 +03:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2016-08-20 23:46:34 +03:00
										 |  |  | 	// Apply/call a function/action "inside" an action...
 | 
					
						
							|  |  |  | 	//
 | 
					
						
							|  |  |  | 	// 	.chainApply(outer, inner)
 | 
					
						
							|  |  |  | 	// 	.chainApply(outer, inner, arguments)
 | 
					
						
							|  |  |  | 	// 		-> result
 | 
					
						
							|  |  |  | 	//
 | 
					
						
							|  |  |  | 	// 	.chainCall(outer, inner)
 | 
					
						
							|  |  |  | 	// 	.chainCall(outer, inner, ..)
 | 
					
						
							|  |  |  | 	// 		-> result
 | 
					
						
							|  |  |  | 	//
 | 
					
						
							|  |  |  | 	//
 | 
					
						
							|  |  |  | 	// The inner action call is completely nested as base of the outer 
 | 
					
						
							|  |  |  | 	// action.
 | 
					
						
							|  |  |  | 	//
 | 
					
						
							|  |  |  | 	//		Outer action		o-------x		o-------x
 | 
					
						
							|  |  |  | 	//									v		^
 | 
					
						
							|  |  |  | 	//		Inner action				o---|---x
 | 
					
						
							|  |  |  | 	//
 | 
					
						
							|  |  |  | 	// The given arguments are passed as-is to both the outer and inner
 | 
					
						
							|  |  |  | 	// actions.
 | 
					
						
							| 
									
										
										
										
											2016-11-07 18:15:53 +03:00
										 |  |  | 	// The inner action return value is passed to the outer action
 | 
					
						
							| 
									
										
										
										
											2016-08-20 23:46:34 +03:00
										 |  |  | 	// .post handlers.
 | 
					
						
							|  |  |  | 	//
 | 
					
						
							| 
									
										
										
										
											2016-11-07 18:15:53 +03:00
										 |  |  | 	// inner return value is handling slightly differs from the base
 | 
					
						
							|  |  |  | 	// action protocol in two respects:
 | 
					
						
							|  |  |  | 	// 	1) to keep the outer return value, inner must return undefined.
 | 
					
						
							|  |  |  | 	// 	2) to guarantee returning the context regardless of outer's return 
 | 
					
						
							|  |  |  | 	// 		value, the inner must return the context (this) explicilty.
 | 
					
						
							|  |  |  | 	//
 | 
					
						
							|  |  |  | 	// NOTE: as a restriction of the action protocol the inner return will
 | 
					
						
							|  |  |  | 	// 		override the return value of outer, but there is no way to 
 | 
					
						
							|  |  |  | 	// 		see that value.
 | 
					
						
							| 
									
										
										
										
											2016-08-20 23:46:34 +03:00
										 |  |  | 	// NOTE: these call the action's .chainApply(..) and .chainCall(..)
 | 
					
						
							|  |  |  | 	// 		methods, thus is not compatible with non-action methods...
 | 
					
						
							| 
									
										
										
										
											2017-10-11 00:56:10 +03:00
										 |  |  | 	// NOTE: .chainCall('action', ..) is equivalent to .action.chainCall(..)
 | 
					
						
							| 
									
										
										
										
											2016-08-20 23:46:34 +03:00
										 |  |  | 	chainApply: function(outer, inner, args){ | 
					
						
							|  |  |  | 		return this[outer].chainApply(this, inner, args) }, | 
					
						
							|  |  |  | 	chainCall: function(outer, inner){ | 
					
						
							| 
									
										
										
										
											2020-04-17 23:21:39 +03:00
										 |  |  | 		return this[outer].chainApply(this, inner, [...arguments].slice(2)) }, | 
					
						
							| 
									
										
										
										
											2016-08-20 23:46:34 +03:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2019-10-10 02:34:50 +03:00
										 |  |  | 
 | 
					
						
							|  |  |  | 	// Call action handlers serted by .sortedActionPriority...
 | 
					
						
							|  |  |  | 	//
 | 
					
						
							|  |  |  | 	// NOTE: this by design ignores the action call results to avoid 
 | 
					
						
							|  |  |  | 	//		actions competing on who will return a value...
 | 
					
						
							|  |  |  | 	// NOTE: if action name does not exist this will do nothing and 
 | 
					
						
							|  |  |  | 	//		return normally (without error)...
 | 
					
						
							|  |  |  | 	// NOTE: this essentially re-implements parts of the .pre(..)/.post(..)
 | 
					
						
							|  |  |  | 	// 		action protocol...
 | 
					
						
							|  |  |  | 	// NOTE: this may not support some legacy action protocol features...
 | 
					
						
							|  |  |  | 	callSortedAction: function(name, ...args){ | 
					
						
							|  |  |  | 		var that = this | 
					
						
							|  |  |  | 		this.getHandlers(name) | 
					
						
							|  |  |  | 			.map(function(h, i){  | 
					
						
							|  |  |  | 				var p = (h.pre || {}).sortedActionPriority | 
					
						
							|  |  |  | 				// normalize priority...
 | 
					
						
							|  |  |  | 				p = p == 'high' ? | 
					
						
							|  |  |  | 						50 | 
					
						
							|  |  |  | 					: p == 'normal' ? | 
					
						
							|  |  |  | 						0 | 
					
						
							|  |  |  | 					: p == 'low' ? | 
					
						
							|  |  |  | 						-50 | 
					
						
							|  |  |  | 					: p | 
					
						
							|  |  |  | 				return [i, p, h] }) | 
					
						
							|  |  |  | 			// sort by .sortedActionPriority ascending...
 | 
					
						
							|  |  |  | 			.sort(function([ia, pa, a], [ib, pb, b]){ | 
					
						
							|  |  |  | 				return (pa != null && pb != null) ? | 
					
						
							|  |  |  | 						pa - pb | 
					
						
							|  |  |  | 					: (pa > 0 || pb < 0) ? | 
					
						
							|  |  |  | 						1 | 
					
						
							|  |  |  | 					: (pb > 0 || pa < 0) ? | 
					
						
							|  |  |  | 						-1 | 
					
						
							|  |  |  | 					: ia - ib }) | 
					
						
							|  |  |  | 			// the list should be ordered descending form highest 
 | 
					
						
							|  |  |  | 			// priority or closeness to root action...
 | 
					
						
							|  |  |  | 			.reverse() | 
					
						
							|  |  |  | 			// call the actions (pre)...
 | 
					
						
							|  |  |  | 			.map(function([i, p, a]){ | 
					
						
							|  |  |  | 				return a.pre ?  | 
					
						
							|  |  |  | 					a.pre.call(that, ...args) | 
					
						
							|  |  |  | 					: a.post }) | 
					
						
							|  |  |  | 			.reverse() | 
					
						
							|  |  |  | 			// call the actions (post)...
 | 
					
						
							|  |  |  | 			// NOTE: we do not care about call results here...
 | 
					
						
							|  |  |  | 			.forEach(function(func){ | 
					
						
							|  |  |  | 				func instanceof Function | 
					
						
							| 
									
										
										
										
											2019-10-10 02:42:59 +03:00
										 |  |  | 					&& func.call(that, ...args) })  | 
					
						
							|  |  |  | 		return this }, | 
					
						
							| 
									
										
										
										
											2019-10-10 02:34:50 +03:00
										 |  |  | 		 | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2017-07-02 19:40:39 +03:00
										 |  |  | 	// Get action/method resolution order...
 | 
					
						
							|  |  |  | 	//
 | 
					
						
							| 
									
										
										
										
											2020-04-29 21:14:09 +03:00
										 |  |  | 	// 	List mixin tags if present, else objects...
 | 
					
						
							| 
									
										
										
										
											2017-07-02 19:40:39 +03:00
										 |  |  | 	// 	.mro()
 | 
					
						
							| 
									
										
										
										
											2020-04-29 21:14:09 +03:00
										 |  |  | 	// 	.mro('tag-object')
 | 
					
						
							|  |  |  | 	// 		-> tags
 | 
					
						
							|  |  |  | 	//
 | 
					
						
							|  |  |  | 	// 	List mixin tags...
 | 
					
						
							| 
									
										
										
										
											2017-07-02 19:40:39 +03:00
										 |  |  | 	// 	.mro('tag')
 | 
					
						
							|  |  |  | 	// 		-> tags
 | 
					
						
							|  |  |  | 	//
 | 
					
						
							|  |  |  | 	// 	List mixin objects...
 | 
					
						
							|  |  |  | 	// 	.mro('object')
 | 
					
						
							|  |  |  | 	// 		-> objects
 | 
					
						
							|  |  |  | 	//
 | 
					
						
							|  |  |  | 	// 	List mixin tag-object pairs...
 | 
					
						
							|  |  |  | 	// 	.mro('item')
 | 
					
						
							|  |  |  | 	// 		-> items
 | 
					
						
							|  |  |  | 	//
 | 
					
						
							| 
									
										
										
										
											2017-07-02 19:50:02 +03:00
										 |  |  | 	// NOTE: this will return the full MRO including Object.prototype
 | 
					
						
							| 
									
										
										
										
											2017-07-02 19:40:39 +03:00
										 |  |  | 	mro: function(target){ | 
					
						
							| 
									
										
										
										
											2020-04-29 21:14:09 +03:00
										 |  |  | 		target = target || 'tag-object' | 
					
						
							| 
									
										
										
										
											2017-07-02 19:40:39 +03:00
										 |  |  | 		var res = [] | 
					
						
							|  |  |  | 		var cur = this | 
					
						
							|  |  |  | 		while(cur != null){ | 
					
						
							| 
									
										
										
										
											2020-04-29 21:14:09 +03:00
										 |  |  | 			res.push(target == 'tag-object' ?  | 
					
						
							|  |  |  | 					cur.__mixin_tag || cur | 
					
						
							|  |  |  | 				: target == 'tag' ?  | 
					
						
							|  |  |  | 					cur.__mixin_tag | 
					
						
							|  |  |  | 				: target == 'object' ?  | 
					
						
							|  |  |  | 					cur | 
					
						
							| 
									
										
										
										
											2017-07-02 19:40:39 +03:00
										 |  |  | 				: [cur.__mixin_tag, cur]) | 
					
						
							|  |  |  | 			// go to next item in chain...
 | 
					
						
							| 
									
										
										
										
											2020-11-23 14:47:00 +03:00
										 |  |  | 			cur = cur.__proto__ } | 
					
						
							| 
									
										
										
										
											2020-04-29 21:14:09 +03:00
										 |  |  | 		return res }, | 
					
						
							| 
									
										
										
										
											2017-07-02 19:40:39 +03:00
										 |  |  | 	 | 
					
						
							| 
									
										
										
										
											2016-08-20 23:46:34 +03:00
										 |  |  | 	// Get mixin object in inheritance chain...
 | 
					
						
							|  |  |  | 	//
 | 
					
						
							| 
									
										
										
										
											2017-07-02 19:50:02 +03:00
										 |  |  | 	// NOTE: from can be either an explicit action object or a tag...
 | 
					
						
							| 
									
										
										
										
											2016-08-20 23:46:34 +03:00
										 |  |  | 	// NOTE: if pre is true this will return the chain item before the 
 | 
					
						
							|  |  |  | 	// 		mixin, this is useful, for example, to remove mixins, see 
 | 
					
						
							|  |  |  | 	// 		.mixout(..) for an example...
 | 
					
						
							|  |  |  | 	getMixin: function(from, pre){ | 
					
						
							| 
									
										
										
										
											2017-07-02 19:40:39 +03:00
										 |  |  | 		var mro = this.mro('object') | 
					
						
							|  |  |  | 		var res = (pre ? mro.slice(1) : mro) | 
					
						
							|  |  |  | 			.filter(function(e){  | 
					
						
							|  |  |  | 				return e.__mixin_tag == from  | 
					
						
							|  |  |  | 					|| e.__mixin_source === from }) | 
					
						
							|  |  |  | 			.shift() | 
					
						
							|  |  |  | 		return pre ? | 
					
						
							|  |  |  | 			mro[mro.indexOf(res)-1] | 
					
						
							| 
									
										
										
										
											2020-04-29 21:14:09 +03:00
										 |  |  | 			: res }, | 
					
						
							| 
									
										
										
										
											2017-07-02 19:40:39 +03:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2016-08-20 23:46:34 +03:00
										 |  |  | 	// Mixin a set of actions into this...
 | 
					
						
							|  |  |  | 	//
 | 
					
						
							| 
									
										
										
										
											2016-08-24 01:55:46 +03:00
										 |  |  | 	// NOTE: if 'all' is set then mixin all the actions available, 
 | 
					
						
							| 
									
										
										
										
											2016-08-20 23:46:34 +03:00
										 |  |  | 	// 		otherwise only mixin local actions...
 | 
					
						
							|  |  |  | 	// NOTE: this will override existing own attributes.
 | 
					
						
							| 
									
										
										
										
											2016-08-24 02:43:51 +03:00
										 |  |  | 	//
 | 
					
						
							|  |  |  | 	// XXX should we include functions by default????
 | 
					
						
							| 
									
										
										
										
											2018-03-05 00:24:18 +03:00
										 |  |  | 	// XXX should .source_tag be set here or in Actions(..)???
 | 
					
						
							| 
									
										
										
										
											2016-11-05 19:31:26 +03:00
										 |  |  | 	inlineMixin: function(from, options){ | 
					
						
							| 
									
										
										
										
											2016-08-20 23:46:34 +03:00
										 |  |  | 		// defaults...
 | 
					
						
							| 
									
										
										
										
											2016-11-05 19:31:26 +03:00
										 |  |  | 		options = options || {} | 
					
						
							|  |  |  | 		var descriptors = options.descriptors || true | 
					
						
							|  |  |  | 		var all_attr_types = options.all_attr_types || false | 
					
						
							|  |  |  | 		var source_tag = options.source_tag | 
					
						
							| 
									
										
										
										
											2016-08-20 23:46:34 +03:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2016-10-27 19:36:39 +03:00
										 |  |  | 		resetHandlerCache = (this.resetHandlerCache || MetaActions.resetHandlerCache) | 
					
						
							|  |  |  | 		resetHandlerCache.call(this) | 
					
						
							| 
									
										
										
										
											2016-10-27 18:47:17 +03:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2016-11-05 19:31:26 +03:00
										 |  |  | 		if(options.all){ | 
					
						
							| 
									
										
										
										
											2016-08-20 23:46:34 +03:00
										 |  |  | 			var keys = [] | 
					
						
							|  |  |  | 			for(var k in from){ | 
					
						
							| 
									
										
										
										
											2020-11-23 14:47:00 +03:00
										 |  |  | 				keys.push(k) } | 
					
						
							| 
									
										
										
										
											2016-08-20 23:46:34 +03:00
										 |  |  | 		} else { | 
					
						
							| 
									
										
										
										
											2020-11-23 14:47:00 +03:00
										 |  |  | 			var keys = Object.keys(from) } | 
					
						
							| 
									
										
										
										
											2016-08-20 23:46:34 +03:00
										 |  |  | 
 | 
					
						
							|  |  |  | 		var that = this | 
					
						
							|  |  |  | 		keys.forEach(function(k){ | 
					
						
							|  |  |  | 			/* | 
					
						
							|  |  |  | 			// XXX is this the right way to go???
 | 
					
						
							|  |  |  | 			// check if we are not overwriting anything...
 | 
					
						
							|  |  |  | 			if(that.hasOwnProperty(k)){ | 
					
						
							|  |  |  | 				console.warn('WARNING:', that,'already has attribute', k, '- skipping...') | 
					
						
							|  |  |  | 				return | 
					
						
							|  |  |  | 			} | 
					
						
							|  |  |  | 			*/ | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | 			// properties....
 | 
					
						
							|  |  |  | 			var prop = Object.getOwnPropertyDescriptor(from, k) | 
					
						
							|  |  |  | 			if(descriptors && prop.get != null){ | 
					
						
							|  |  |  | 				// NOTE: so as to be able to delete this on mixout...
 | 
					
						
							|  |  |  | 				prop.configurable = true | 
					
						
							|  |  |  | 				Object.defineProperty(that, k, prop) | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | 			// actions and other attributes...
 | 
					
						
							|  |  |  | 			} else { | 
					
						
							|  |  |  | 				var attr = from[k] | 
					
						
							| 
									
										
										
										
											2016-08-24 02:43:51 +03:00
										 |  |  | 				if(all_attr_types  | 
					
						
							| 
									
										
										
										
											2017-09-08 00:46:16 +03:00
										 |  |  | 						|| attr instanceof Function | 
					
						
							| 
									
										
										
										
											2016-08-24 02:43:51 +03:00
										 |  |  | 						|| attr instanceof Action){ | 
					
						
							| 
									
										
										
										
											2020-11-23 14:47:00 +03:00
										 |  |  | 					that[k] = attr } | 
					
						
							| 
									
										
										
										
											2017-09-08 00:46:16 +03:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2016-11-05 19:31:26 +03:00
										 |  |  | 				// source tag actions...
 | 
					
						
							| 
									
										
										
										
											2018-03-05 00:24:18 +03:00
										 |  |  | 				// XXX should this set action and method .source_tag or only action???
 | 
					
						
							|  |  |  | 				//if(source_tag && attr instanceof Action){
 | 
					
						
							|  |  |  | 				if(source_tag && (attr instanceof Action || attr instanceof Function)){ | 
					
						
							| 
									
										
										
										
											2016-11-07 06:36:41 +03:00
										 |  |  | 					// existing tag...
 | 
					
						
							| 
									
										
										
										
											2018-03-05 00:24:18 +03:00
										 |  |  | 					if(that[k].source_tag == source_tag  | 
					
						
							|  |  |  | 							|| (that[k].func || {}).source_tag == source_tag){ | 
					
						
							| 
									
										
										
										
											2016-11-07 06:36:41 +03:00
										 |  |  | 						return | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | 					// new tag...
 | 
					
						
							| 
									
										
										
										
											2016-11-05 19:31:26 +03:00
										 |  |  | 					// XXX not sure if this is the right way to go...
 | 
					
						
							| 
									
										
										
										
											2018-03-05 00:24:18 +03:00
										 |  |  | 					} else if(that[k].source_tag  | 
					
						
							|  |  |  | 							|| (that[k].func || {}).source_tag){ | 
					
						
							| 
									
										
										
										
											2020-05-06 20:39:51 +03:00
										 |  |  | 						console.warn('Actions: about to overwrite source tag...\n' | 
					
						
							| 
									
										
										
										
											2018-03-05 00:24:18 +03:00
										 |  |  | 							+'  from: "' | 
					
						
							|  |  |  | 								+(that[k].source_tag  | 
					
						
							|  |  |  | 									|| (that[k].func || {}).source_tag)+'"\n' | 
					
						
							| 
									
										
										
										
											2016-11-05 19:31:26 +03:00
										 |  |  | 							+'  to: "'+source_tag+'"\n' | 
					
						
							| 
									
										
										
										
											2020-11-23 14:47:00 +03:00
										 |  |  | 							+'  on:', that[k]) } | 
					
						
							| 
									
										
										
										
											2016-11-07 06:36:41 +03:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2018-03-05 00:24:18 +03:00
										 |  |  | 					if(that[k].func){ | 
					
						
							| 
									
										
										
										
											2020-11-23 14:47:00 +03:00
										 |  |  | 						that[k].func.source_tag = source_tag } | 
					
						
							|  |  |  | 					that[k].source_tag = source_tag } } }) | 
					
						
							| 
									
										
										
										
											2016-08-20 23:46:34 +03:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2020-04-29 21:14:09 +03:00
										 |  |  | 		return this }, | 
					
						
							| 
									
										
										
										
											2016-08-20 23:46:34 +03:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2016-08-24 00:48:40 +03:00
										 |  |  | 	// Same as .inlineMixin(..) but isolates a mixin in a seporate object
 | 
					
						
							| 
									
										
										
										
											2016-08-20 23:46:34 +03:00
										 |  |  | 	// in the inheritance chain...
 | 
					
						
							|  |  |  | 	//
 | 
					
						
							| 
									
										
										
										
											2016-11-05 19:31:26 +03:00
										 |  |  | 	mixin: function(from, options){ | 
					
						
							| 
									
										
										
										
											2017-07-02 20:25:40 +03:00
										 |  |  | 		options = options || {} | 
					
						
							|  |  |  | 		options.source_tag = options.source_tag || from.__mixin_tag | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2016-08-20 23:46:34 +03:00
										 |  |  | 		var proto = Object.create(this.__proto__) | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | 		// mixinto an empty object
 | 
					
						
							| 
									
										
										
										
											2016-11-05 19:31:26 +03:00
										 |  |  | 		proto.inlineMixin(from, options) | 
					
						
							| 
									
										
										
										
											2016-08-20 23:46:34 +03:00
										 |  |  | 
 | 
					
						
							|  |  |  | 		// mark the mixin for simpler removal...
 | 
					
						
							|  |  |  | 		proto.__mixin_source = from | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2017-07-02 19:13:22 +03:00
										 |  |  | 		// add source tag to proto...
 | 
					
						
							|  |  |  | 		if(options && options.source_tag){ | 
					
						
							| 
									
										
										
										
											2020-11-23 14:47:00 +03:00
										 |  |  | 			proto.__mixin_tag = options.source_tag } | 
					
						
							| 
									
										
										
										
											2017-07-02 19:08:03 +03:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2016-08-20 23:46:34 +03:00
										 |  |  | 		this.__proto__ = proto | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2020-04-29 21:14:09 +03:00
										 |  |  | 		return this }, | 
					
						
							| 
									
										
										
										
											2016-08-20 23:46:34 +03:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2017-07-02 20:25:40 +03:00
										 |  |  | 	// Mixin from after target in the mro...
 | 
					
						
							|  |  |  | 	//
 | 
					
						
							|  |  |  | 	// NOTE: target must be .getMixin(..) compatible...
 | 
					
						
							|  |  |  | 	mixinAfter: function(target, from, options){ | 
					
						
							|  |  |  | 		this | 
					
						
							|  |  |  | 			.getMixin(target) | 
					
						
							|  |  |  | 			.mixin(from, options) | 
					
						
							| 
									
										
										
										
											2020-04-29 21:14:09 +03:00
										 |  |  | 		return this }, | 
					
						
							| 
									
										
										
										
											2017-07-02 20:25:40 +03:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2016-08-20 23:46:34 +03:00
										 |  |  | 	// Mixin a set of local actions into an object...
 | 
					
						
							|  |  |  | 	//
 | 
					
						
							|  |  |  | 	// XXX this will not work on non-actions...
 | 
					
						
							| 
									
										
										
										
											2016-11-05 19:31:26 +03:00
										 |  |  | 	mixinTo: function(to, options){ | 
					
						
							| 
									
										
										
										
											2017-12-09 01:38:20 +03:00
										 |  |  | 		return this.mixin.call(to, this, options) }, | 
					
						
							| 
									
										
										
										
											2016-08-20 23:46:34 +03:00
										 |  |  | 
 | 
					
						
							|  |  |  | 	// Remove mixed in actions from this...
 | 
					
						
							|  |  |  | 	//
 | 
					
						
							|  |  |  | 	// NOTE: this will only remove local actions, inherited actions will
 | 
					
						
							|  |  |  | 	// 		not be affected...
 | 
					
						
							|  |  |  | 	// NOTE: this will not affect event handlers, they should be removed
 | 
					
						
							|  |  |  | 	// 		manually if needed...
 | 
					
						
							| 
									
										
										
										
											2016-11-05 19:31:26 +03:00
										 |  |  | 	inlineMixout: function(from, options){ | 
					
						
							| 
									
										
										
										
											2016-08-20 23:46:34 +03:00
										 |  |  | 		// defaults...
 | 
					
						
							| 
									
										
										
										
											2016-11-05 19:31:26 +03:00
										 |  |  | 		options = options || {} | 
					
						
							|  |  |  | 		var descriptors = options.descriptors || true | 
					
						
							|  |  |  | 		var all_attr_types = options.all_attr_types || false | 
					
						
							| 
									
										
										
										
											2016-08-20 23:46:34 +03:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2016-10-27 18:54:17 +03:00
										 |  |  | 		(this.resetHandlerCache || MetaActions.resetHandlerCache).call(this) | 
					
						
							| 
									
										
										
										
											2016-10-27 18:47:17 +03:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2016-11-05 19:31:26 +03:00
										 |  |  | 		if(options.all){ | 
					
						
							| 
									
										
										
										
											2016-08-20 23:46:34 +03:00
										 |  |  | 			var keys = [] | 
					
						
							|  |  |  | 			for(var k in from){ | 
					
						
							|  |  |  | 				keys.push(k) | 
					
						
							|  |  |  | 			} | 
					
						
							|  |  |  | 		} else { | 
					
						
							| 
									
										
										
										
											2020-11-23 14:47:00 +03:00
										 |  |  | 			var keys = Object.keys(from) } | 
					
						
							| 
									
										
										
										
											2016-08-20 23:46:34 +03:00
										 |  |  | 
 | 
					
						
							|  |  |  | 		var locals = Object.keys(this) | 
					
						
							|  |  |  | 		var that = this | 
					
						
							|  |  |  | 		keys.forEach(function(k){ | 
					
						
							|  |  |  | 			var prop = Object.getOwnPropertyDescriptor(from, k) | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | 			// descriptor...
 | 
					
						
							|  |  |  | 			if(descriptors && prop.get != null){ | 
					
						
							|  |  |  | 				if(prop.get === Object.getOwnPropertyDescriptor(that, k).get){ | 
					
						
							| 
									
										
										
										
											2020-11-23 14:47:00 +03:00
										 |  |  | 					delete that[k] } | 
					
						
							| 
									
										
										
										
											2016-08-20 23:46:34 +03:00
										 |  |  | 
 | 
					
						
							|  |  |  | 			// actions and other attrs...
 | 
					
						
							|  |  |  | 			} else { | 
					
						
							|  |  |  | 				var attr = from[k] | 
					
						
							|  |  |  | 				if((all_attr_types || attr instanceof Action)  | 
					
						
							|  |  |  | 						// remove only local attrs...
 | 
					
						
							|  |  |  | 						&& locals.indexOf(k) >= 0){ | 
					
						
							| 
									
										
										
										
											2020-11-23 14:47:00 +03:00
										 |  |  | 					delete that[k] } } }) | 
					
						
							| 
									
										
										
										
											2016-08-20 23:46:34 +03:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2020-04-29 21:14:09 +03:00
										 |  |  | 		return this }, | 
					
						
							| 
									
										
										
										
											2016-08-20 23:46:34 +03:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2016-10-27 18:47:17 +03:00
										 |  |  | 	// This is similar in effect but different in mechanics to .inlineMixout(..)
 | 
					
						
							| 
									
										
										
										
											2016-08-20 23:46:34 +03:00
										 |  |  | 	//
 | 
					
						
							| 
									
										
										
										
											2017-10-11 00:56:10 +03:00
										 |  |  | 	// This will find and remove a mixin object from the inheritance chain.
 | 
					
						
							| 
									
										
										
										
											2016-08-20 23:46:34 +03:00
										 |  |  | 	//
 | 
					
						
							|  |  |  | 	// NOTE: this will remove only the first occurance of a mixin.
 | 
					
						
							|  |  |  | 	mixout: function(from){ | 
					
						
							|  |  |  | 		var o = this.getMixin(from, true) | 
					
						
							| 
									
										
										
										
											2017-07-02 20:25:40 +03:00
										 |  |  | 		var target = null | 
					
						
							| 
									
										
										
										
											2016-08-20 23:46:34 +03:00
										 |  |  | 		// pop the mixin off the chain...
 | 
					
						
							|  |  |  | 		if(o != null){ | 
					
						
							| 
									
										
										
										
											2017-07-02 20:25:40 +03:00
										 |  |  | 			target = o.__proto__ | 
					
						
							| 
									
										
										
										
											2016-08-20 23:46:34 +03:00
										 |  |  | 			o.__proto__ = o.__proto__.__proto__ | 
					
						
							| 
									
										
										
										
											2020-11-23 14:47:00 +03:00
										 |  |  | 			this.resetHandlerCache() } | 
					
						
							| 
									
										
										
										
											2020-04-29 21:14:09 +03:00
										 |  |  | 		return target }, | 
					
						
							| 
									
										
										
										
											2016-08-20 23:46:34 +03:00
										 |  |  | 
 | 
					
						
							|  |  |  | 	// Remove a set of local mixed in actions from object...
 | 
					
						
							|  |  |  | 	//
 | 
					
						
							| 
									
										
										
										
											2016-11-05 19:31:26 +03:00
										 |  |  | 	mixoutFrom: function(to, options){ | 
					
						
							| 
									
										
										
										
											2017-08-03 22:05:53 +03:00
										 |  |  | 		return this.mixout.call(to, this, options) }, | 
					
						
							| 
									
										
										
										
											2016-08-20 23:46:34 +03:00
										 |  |  | 
 | 
					
						
							|  |  |  | 	// Create a child object...
 | 
					
						
							|  |  |  | 	//
 | 
					
						
							|  |  |  | 	// NOTE: this will create a .config in the instance that inherits from
 | 
					
						
							|  |  |  | 	// 		this.config...
 | 
					
						
							|  |  |  | 	// NOTE: this will not copy/clone any data.
 | 
					
						
							|  |  |  | 	//
 | 
					
						
							|  |  |  | 	// XXX is this correct???
 | 
					
						
							|  |  |  | 	// XXX should this be an action???
 | 
					
						
							|  |  |  | 	clone: function(full){ | 
					
						
							|  |  |  | 		var o = Object.create(this) | 
					
						
							|  |  |  | 		if(this.config){ | 
					
						
							|  |  |  | 			if(full){ | 
					
						
							|  |  |  | 				o.config = JSON.parse(JSON.stringify(this.config)) | 
					
						
							|  |  |  | 			} else { | 
					
						
							| 
									
										
										
										
											2020-11-23 14:47:00 +03:00
										 |  |  | 				o.config = Object.create(this.config) } } | 
					
						
							| 
									
										
										
										
											2020-04-29 21:14:09 +03:00
										 |  |  | 		return o }, | 
					
						
							| 
									
										
										
										
											2016-08-20 23:46:34 +03:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2017-01-25 23:31:19 +03:00
										 |  |  | 	getHandlerSourceTags: function(name){ | 
					
						
							| 
									
										
										
										
											2017-01-25 23:26:58 +03:00
										 |  |  | 		return this.getHandlers(name) | 
					
						
							|  |  |  | 			.map(function(a){ | 
					
						
							|  |  |  | 				return a.pre ? (a.pre.source_tag || a.pre.event_tag) | 
					
						
							|  |  |  | 					: a.post ? (a.post.source_tag || a.post.event_tag) | 
					
						
							| 
									
										
										
										
											2020-11-23 14:47:00 +03:00
										 |  |  | 					: null }) | 
					
						
							| 
									
										
										
										
											2017-01-27 04:22:54 +03:00
										 |  |  | 			.unique() }, | 
					
						
							| 
									
										
										
										
											2017-01-25 23:26:58 +03:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2017-05-13 23:38:25 +03:00
										 |  |  | 
 | 
					
						
							|  |  |  | 	// Run a function in the context of the action set...
 | 
					
						
							|  |  |  | 	//
 | 
					
						
							| 
									
										
										
										
											2017-05-13 23:47:07 +03:00
										 |  |  | 	// This will return 'this' if func returns undefined, otherwise func
 | 
					
						
							|  |  |  | 	// return value is returned.
 | 
					
						
							|  |  |  | 	//
 | 
					
						
							| 
									
										
										
										
											2017-05-13 23:38:25 +03:00
										 |  |  | 	// This is here simply as a utility function, to enable running code 
 | 
					
						
							|  |  |  | 	// in a concatinative manner without interruption...
 | 
					
						
							|  |  |  | 	run: function(func){ | 
					
						
							| 
									
										
										
										
											2020-04-29 21:14:09 +03:00
										 |  |  | 		var res = func ?  | 
					
						
							|  |  |  | 			func.call(this)  | 
					
						
							|  |  |  | 			: undefined | 
					
						
							|  |  |  | 		return res === undefined ?  | 
					
						
							|  |  |  | 			this  | 
					
						
							|  |  |  | 			: res }, | 
					
						
							| 
									
										
										
										
											2017-05-13 23:38:25 +03:00
										 |  |  | 
 | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2016-08-20 23:46:34 +03:00
										 |  |  | 	// doc generators...
 | 
					
						
							|  |  |  | 	//
 | 
					
						
							|  |  |  | 	// XXX would be nice to make these prop of the action itself but I 
 | 
					
						
							|  |  |  | 	// 		do not see a way to do this properly yet -- we can't get to 
 | 
					
						
							|  |  |  | 	// 		the action context from the action dynamically...
 | 
					
						
							| 
									
										
										
										
											2018-10-30 22:00:24 +03:00
										 |  |  | 	// XXX add doc per action...
 | 
					
						
							| 
									
										
										
										
											2016-08-20 23:46:34 +03:00
										 |  |  | 	getHandlerDocStr: function(name){ | 
					
						
							|  |  |  | 		var lst = this.getHandlers(name) | 
					
						
							|  |  |  | 		var str = '' | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2018-10-30 22:00:24 +03:00
										 |  |  | 		var getTags = function(handler, p){ | 
					
						
							|  |  |  | 			return (handler.event_tag ?  | 
					
						
							| 
									
										
										
										
											2020-04-23 20:50:21 +03:00
										 |  |  | 					object.normalizeIndent('// Event tag: ' + handler.event_tag) + p  | 
					
						
							| 
									
										
										
										
											2018-10-30 22:00:24 +03:00
										 |  |  | 					: '') | 
					
						
							|  |  |  | 				+ (handler.source_tag ?  | 
					
						
							| 
									
										
										
										
											2020-04-23 20:50:21 +03:00
										 |  |  | 					object.normalizeIndent('// Source tag: ' + handler.source_tag) + p  | 
					
						
							| 
									
										
										
										
											2018-10-30 22:00:24 +03:00
										 |  |  | 					: '') } | 
					
						
							|  |  |  | 		var getDoc = function(cur, p){ | 
					
						
							|  |  |  | 			return (cur.doc ?  | 
					
						
							|  |  |  | 					'// --- .doc ---'+p | 
					
						
							| 
									
										
										
										
											2020-04-23 20:50:21 +03:00
										 |  |  | 					+'// '+ object.normalizeIndent(cur.doc).replace(/\n/g, p+'// ') +p  | 
					
						
							| 
									
										
										
										
											2018-10-30 22:00:24 +03:00
										 |  |  | 					: '') | 
					
						
							|  |  |  | 				+ (cur.long_doc ?  | 
					
						
							|  |  |  | 					'// --- .long_doc ---'+p | 
					
						
							| 
									
										
										
										
											2020-04-23 20:50:21 +03:00
										 |  |  | 					+'// '+ object.normalizeIndent(cur.long_doc).replace(/\n/g, p+'// ') + p  | 
					
						
							| 
									
										
										
										
											2018-10-30 22:00:24 +03:00
										 |  |  | 					: '') } | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2016-08-20 23:46:34 +03:00
										 |  |  | 		var handler = function(p){ | 
					
						
							|  |  |  | 			if(lst.length == 0){ | 
					
						
							|  |  |  | 				//str += p + '---'
 | 
					
						
							| 
									
										
										
										
											2020-11-23 14:47:00 +03:00
										 |  |  | 				return } | 
					
						
							| 
									
										
										
										
											2016-08-20 23:46:34 +03:00
										 |  |  | 
 | 
					
						
							|  |  |  | 			// indicate root action...
 | 
					
						
							|  |  |  | 			p = lst.length == 1 ? p+'| ' : p+' ' | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | 			var cur = lst.shift() | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | 			if(cur.pre){ | 
					
						
							|  |  |  | 				str += p  | 
					
						
							| 
									
										
										
										
											2018-10-30 22:00:24 +03:00
										 |  |  | 					+ getTags(cur.pre, p) | 
					
						
							|  |  |  | 					+ getDoc(cur, p) | 
					
						
							| 
									
										
										
										
											2016-11-05 19:31:26 +03:00
										 |  |  | 					// code...
 | 
					
						
							| 
									
										
										
										
											2020-04-23 20:50:21 +03:00
										 |  |  | 					+ object.normalizeIndent(cur.pre.toString()).replace(/\n/g, p) | 
					
						
							| 
									
										
										
										
											2020-11-23 14:47:00 +03:00
										 |  |  | 					+ p } | 
					
						
							| 
									
										
										
										
											2016-08-20 23:46:34 +03:00
										 |  |  | 
 | 
					
						
							|  |  |  | 			handler(p + '  |') | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | 			str += p | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | 			if(cur.post){ | 
					
						
							|  |  |  | 				str += p + p  | 
					
						
							| 
									
										
										
										
											2018-10-30 22:00:24 +03:00
										 |  |  | 					+ getTags(cur.post, p) | 
					
						
							|  |  |  | 					+ getDoc(cur, p) | 
					
						
							| 
									
										
										
										
											2016-11-05 19:31:26 +03:00
										 |  |  | 					// code...
 | 
					
						
							| 
									
										
										
										
											2020-11-23 14:47:00 +03:00
										 |  |  | 					+ object.normalizeIndent(cur.post.toString()).replace(/\n/g, p) } } | 
					
						
							| 
									
										
										
										
											2016-08-20 23:46:34 +03:00
										 |  |  | 
 | 
					
						
							|  |  |  | 		handler('\n|') | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2020-04-29 21:14:09 +03:00
										 |  |  | 		return str }, | 
					
						
							| 
									
										
										
										
											2016-08-20 23:46:34 +03:00
										 |  |  | 	getHandlerDocHTML: function(name){ | 
					
						
							|  |  |  | 		var lst = this.getHandlers(name) | 
					
						
							|  |  |  | 		var res = $('<div class="action">') | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | 		var handler = function(p){ | 
					
						
							|  |  |  | 			if(lst.length == 0){ | 
					
						
							|  |  |  | 				return | 
					
						
							|  |  |  | 			} | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | 			var cur = lst.shift() | 
					
						
							|  |  |  | 			p = $('<div class="level">') | 
					
						
							|  |  |  | 				.appendTo(p) | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | 			if(cur.pre){ | 
					
						
							|  |  |  | 				p.append($('<pre>').html( | 
					
						
							| 
									
										
										
										
											2016-11-05 19:42:49 +03:00
										 |  |  | 					// meta...
 | 
					
						
							|  |  |  | 					(cur.pre.event_tag ?  | 
					
						
							| 
									
										
										
										
											2020-04-23 20:50:21 +03:00
										 |  |  | 						object.normalizeIndent('// Event tag: ' + cur.pre.event_tag) + p : '') | 
					
						
							| 
									
										
										
										
											2016-11-05 19:42:49 +03:00
										 |  |  | 					+ (cur.pre.source_tag ?  | 
					
						
							| 
									
										
										
										
											2020-04-23 20:50:21 +03:00
										 |  |  | 						object.normalizeIndent('// Source tag: ' + cur.pre.source_tag) + p : '') | 
					
						
							| 
									
										
										
										
											2016-11-05 19:42:49 +03:00
										 |  |  | 					// code...
 | 
					
						
							| 
									
										
										
										
											2020-04-23 20:50:21 +03:00
										 |  |  | 					+ object.normalizeIndent(cur.pre.toString()) | 
					
						
							| 
									
										
										
										
											2020-11-23 14:47:00 +03:00
										 |  |  | 						.replace(/return/g, '<b>return</b>'))) } | 
					
						
							| 
									
										
										
										
											2016-08-20 23:46:34 +03:00
										 |  |  | 
 | 
					
						
							|  |  |  | 			handler(p) | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | 			if(cur.post){ | 
					
						
							|  |  |  | 				p.append($('<pre>').html( | 
					
						
							| 
									
										
										
										
											2016-11-05 19:42:49 +03:00
										 |  |  | 					// meta...
 | 
					
						
							|  |  |  | 					(cur.post.event_tag ?  | 
					
						
							| 
									
										
										
										
											2020-04-23 20:50:21 +03:00
										 |  |  | 						object.normalizeIndent('// Event source tag: ' + cur.post.event_tag) + p : '') | 
					
						
							| 
									
										
										
										
											2016-11-05 19:42:49 +03:00
										 |  |  | 					+ (cur.post.source_tag ?  | 
					
						
							| 
									
										
										
										
											2020-04-23 20:50:21 +03:00
										 |  |  | 						object.normalizeIndent('// Source tag: ' + cur.post.source_tag) + p : '') | 
					
						
							| 
									
										
										
										
											2016-11-05 19:42:49 +03:00
										 |  |  | 					// code...
 | 
					
						
							| 
									
										
										
										
											2020-11-23 14:47:00 +03:00
										 |  |  | 					+ object.normalizeIndent(cur.post.toString()))) } } | 
					
						
							| 
									
										
										
										
											2016-08-20 23:46:34 +03:00
										 |  |  | 
 | 
					
						
							|  |  |  | 		handler(res) | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2020-04-29 21:14:09 +03:00
										 |  |  | 		return res }, | 
					
						
							| 
									
										
										
										
											2016-08-20 23:46:34 +03:00
										 |  |  | 
 | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | 	// This will create a .config in instances...
 | 
					
						
							|  |  |  | 	// NOTE: this makes Actions compatible with lib/object.js...
 | 
					
						
							|  |  |  | 	__init__: function(){ | 
					
						
							| 
									
										
										
										
											2020-04-29 21:14:09 +03:00
										 |  |  | 		this.__proto__.config  | 
					
						
							|  |  |  | 			&& !Object.hasOwnProperty(this, 'config') | 
					
						
							|  |  |  | 			&& (this.config = Object.create(this.__proto__.config)) }, | 
					
						
							| 
									
										
										
										
											2016-08-20 23:46:34 +03:00
										 |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | var ActionSet = | 
					
						
							|  |  |  | module.ActionSet = | 
					
						
							| 
									
										
										
										
											2019-07-16 23:53:17 +03:00
										 |  |  | object.Constructor('ActionSet', MetaActions) | 
					
						
							| 
									
										
										
										
											2016-08-20 23:46:34 +03:00
										 |  |  | 
 | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2020-04-23 17:59:13 +03:00
										 |  |  | // An action set constructor...
 | 
					
						
							| 
									
										
										
										
											2016-08-20 23:46:34 +03:00
										 |  |  | //
 | 
					
						
							|  |  |  | //	Actions(<object>)
 | 
					
						
							|  |  |  | //	Actions(<prototype>, <object>)
 | 
					
						
							|  |  |  | //		-> actions
 | 
					
						
							|  |  |  | //
 | 
					
						
							|  |  |  | // This will pre-process an object to setup the action mechanics.
 | 
					
						
							|  |  |  | //
 | 
					
						
							| 
									
										
										
										
											2016-09-27 06:13:57 +03:00
										 |  |  | // If the 'this' and prototype both contain a .config attribute then this
 | 
					
						
							| 
									
										
										
										
											2016-08-20 23:46:34 +03:00
										 |  |  | // will make set <actions>.config.__proto__ = <prototype>.config 
 | 
					
						
							|  |  |  | //
 | 
					
						
							|  |  |  | //
 | 
					
						
							|  |  |  | // The action format:
 | 
					
						
							|  |  |  | // 	{
 | 
					
						
							|  |  |  | // 		// full format...
 | 
					
						
							|  |  |  | // 		<name> : [
 | 
					
						
							|  |  |  | // 			<doc>,
 | 
					
						
							|  |  |  | // 			<long-doc>,
 | 
					
						
							| 
									
										
										
										
											2016-12-29 22:41:03 +03:00
										 |  |  | // 			<attrs>,
 | 
					
						
							| 
									
										
										
										
											2017-08-04 04:14:17 +03:00
										 |  |  | // 			<function> | <alias-code>
 | 
					
						
							| 
									
										
										
										
											2016-08-20 23:46:34 +03:00
										 |  |  | // 		],
 | 
					
						
							|  |  |  | //
 | 
					
						
							|  |  |  | // 		// short doc only...
 | 
					
						
							|  |  |  | // 		<name> : [
 | 
					
						
							|  |  |  | // 			<doc>,
 | 
					
						
							| 
									
										
										
										
											2017-08-04 04:14:17 +03:00
										 |  |  | // 			<function> | <alias-code>
 | 
					
						
							| 
									
										
										
										
											2016-08-20 23:46:34 +03:00
										 |  |  | // 		],
 | 
					
						
							|  |  |  | //
 | 
					
						
							|  |  |  | // 		// only the code...
 | 
					
						
							|  |  |  | // 		<name> : [
 | 
					
						
							| 
									
										
										
										
											2017-08-04 04:14:17 +03:00
										 |  |  | // 			<function> | <alias-code>
 | 
					
						
							| 
									
										
										
										
											2016-08-20 23:46:34 +03:00
										 |  |  | // 		],
 | 
					
						
							|  |  |  | // 		...
 | 
					
						
							|  |  |  | // 	}
 | 
					
						
							|  |  |  | //
 | 
					
						
							|  |  |  | //
 | 
					
						
							|  |  |  | // NOTE: the action function is always last.
 | 
					
						
							| 
									
										
										
										
											2016-12-29 22:41:03 +03:00
										 |  |  | // NOTE: <attrs> if given must be right before the function and must not
 | 
					
						
							|  |  |  | // 		be a string...
 | 
					
						
							| 
									
										
										
										
											2016-08-20 23:46:34 +03:00
										 |  |  | // NOTE: if <prototype> is not given, MetaActions will be used as default.
 | 
					
						
							|  |  |  | //
 | 
					
						
							|  |  |  | // For more documentation see: Action(..).
 | 
					
						
							|  |  |  | //
 | 
					
						
							|  |  |  | // XXX add doc, ldoc, tags and save them to each action...
 | 
					
						
							|  |  |  | // XXX is .config processing correct here???
 | 
					
						
							| 
									
										
										
										
											2017-09-07 16:53:41 +03:00
										 |  |  | // XXX do we need to handle methods in a special way???
 | 
					
						
							| 
									
										
										
										
											2018-03-05 00:24:18 +03:00
										 |  |  | // XXX should this set the .source_tag???
 | 
					
						
							| 
									
										
										
										
											2016-08-20 23:46:34 +03:00
										 |  |  | var Actions = | 
					
						
							|  |  |  | module.Actions = | 
					
						
							|  |  |  | function Actions(a, b){ | 
					
						
							|  |  |  | 	var obj = b == null ? a : b | 
					
						
							|  |  |  | 	var proto = b == null ? b : a | 
					
						
							|  |  |  | 	obj = obj || new ActionSet() | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2017-08-04 03:58:24 +03:00
										 |  |  | 	if(proto != null){ | 
					
						
							|  |  |  | 		obj.__proto__ = proto | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | 		// XXX is this the right way to go???
 | 
					
						
							|  |  |  | 		if(obj.config != null && proto.config != null){ | 
					
						
							| 
									
										
										
										
											2020-11-23 14:47:00 +03:00
										 |  |  | 			obj.config.__proto__ = proto.config } } | 
					
						
							| 
									
										
										
										
											2017-08-04 03:58:24 +03:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2016-08-20 23:46:34 +03:00
										 |  |  | 	// NOTE: this is intentionally done only for own attributes...
 | 
					
						
							|  |  |  | 	Object.keys(obj).forEach(function(k){ | 
					
						
							|  |  |  | 		// NOTE: we are not getting the attrs directly (vars = obj[k])
 | 
					
						
							|  |  |  | 		// 		as that will trigger the getters on an object that is
 | 
					
						
							|  |  |  | 		// 		not in a consistent state...
 | 
					
						
							|  |  |  | 		// NOTE: this will skip all the getters and setters, they will 
 | 
					
						
							|  |  |  | 		// 		be included as-is...
 | 
					
						
							|  |  |  | 		var arg = Object.getOwnPropertyDescriptor(obj, k).value | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2017-09-07 16:53:41 +03:00
										 |  |  | 		// action/alias...
 | 
					
						
							|  |  |  | 		if(arg instanceof Array  | 
					
						
							|  |  |  | 				&& (arg[arg.length-1] instanceof Function | 
					
						
							| 
									
										
										
										
											2017-08-04 03:58:24 +03:00
										 |  |  | 					|| (typeof(arg[arg.length-1]) == typeof('str') | 
					
						
							| 
									
										
										
										
											2017-08-10 03:26:50 +03:00
										 |  |  | 						&& (arg[arg.length-1] == '' | 
					
						
							| 
									
										
										
										
											2017-09-07 16:53:41 +03:00
										 |  |  | 							// XXX should this be stricter???
 | 
					
						
							| 
									
										
										
										
											2017-08-10 03:26:50 +03:00
										 |  |  | 							|| (obj.isStringAction || isStringAction)(arg[arg.length-1])))) ){ | 
					
						
							| 
									
										
										
										
											2017-09-07 16:53:41 +03:00
										 |  |  | 			obj[k] = arg[arg.length-1] instanceof Function ? | 
					
						
							|  |  |  | 				(new Action(k, arg)) | 
					
						
							| 
									
										
										
										
											2020-11-23 14:47:00 +03:00
										 |  |  | 				: (new Alias(k, arg)) } }) | 
					
						
							| 
									
										
										
										
											2016-08-20 23:46:34 +03:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2020-04-29 21:14:09 +03:00
										 |  |  | 	return obj } | 
					
						
							| 
									
										
										
										
											2016-08-20 23:46:34 +03:00
										 |  |  | 
 | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2020-04-24 00:21:02 +03:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2016-08-24 01:55:46 +03:00
										 |  |  | /*********************************************************************/ | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2016-09-27 06:19:55 +03:00
										 |  |  | // NOTE: this can only mix actions sets and MetaActions, i.e. only the 
 | 
					
						
							|  |  |  | // 		actions, properties and .config will get handled...
 | 
					
						
							|  |  |  | // NOTE: MetaActions is a special case, if given it will be used as the
 | 
					
						
							|  |  |  | // 		prototype for the root object in the created chain...
 | 
					
						
							|  |  |  | // 		...MetaActions order in the list has no effect.
 | 
					
						
							| 
									
										
										
										
											2016-09-27 15:56:50 +03:00
										 |  |  | //
 | 
					
						
							|  |  |  | // XXX what the mix order should be?
 | 
					
						
							|  |  |  | // 		base, extending, surface		- order of application (current)
 | 
					
						
							|  |  |  | // 		surface, extending, base		- python-like
 | 
					
						
							| 
									
										
										
										
											2016-08-24 01:55:46 +03:00
										 |  |  | var mix = | 
					
						
							|  |  |  | module.mix =  | 
					
						
							|  |  |  | function(){ | 
					
						
							| 
									
										
										
										
											2020-04-23 17:59:13 +03:00
										 |  |  | 	var args = [...arguments] | 
					
						
							| 
									
										
										
										
											2016-08-24 01:55:46 +03:00
										 |  |  | 	var res = {} | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2016-09-27 06:13:57 +03:00
										 |  |  | 	// special case: if MetaActions is in the args then inherit the root
 | 
					
						
							|  |  |  | 	// 		object from it...
 | 
					
						
							|  |  |  | 	if(args.indexOf(MetaActions) >= 0){ | 
					
						
							|  |  |  | 		args.splice(args.indexOf(MetaActions), 1) | 
					
						
							| 
									
										
										
										
											2020-11-23 14:47:00 +03:00
										 |  |  | 		res.__proto__ = MetaActions } | 
					
						
							| 
									
										
										
										
											2016-09-27 06:13:57 +03:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2016-08-24 01:55:46 +03:00
										 |  |  | 	var mixin = MetaActions.inlineMixin | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | 	args.forEach(function(p){ | 
					
						
							|  |  |  | 		res = Object.create(mixin.call(res, p)) | 
					
						
							| 
									
										
										
										
											2016-09-27 04:02:38 +03:00
										 |  |  | 
 | 
					
						
							|  |  |  | 		// merge config...
 | 
					
						
							|  |  |  | 		if(p.config){ | 
					
						
							|  |  |  | 			var config = res.config = res.config || Object.create({}) | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | 			Object.keys(p.config).forEach(function(k){ | 
					
						
							| 
									
										
										
										
											2020-11-23 14:47:00 +03:00
										 |  |  | 				res.config.__proto__[k] =  | 
					
						
							|  |  |  | 					JSON.parse(JSON.stringify(p.config[k])) }) } }) | 
					
						
							| 
									
										
										
										
											2016-08-24 01:55:46 +03:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2020-04-29 21:14:09 +03:00
										 |  |  | 	return res } | 
					
						
							| 
									
										
										
										
											2016-08-24 01:55:46 +03:00
										 |  |  | 
 | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2020-04-24 00:21:02 +03:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2016-08-20 23:46:34 +03:00
										 |  |  | /*********************************************************************/ | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | var test = | 
					
						
							|  |  |  | module.test = | 
					
						
							|  |  |  | function test(){ | 
					
						
							|  |  |  | 	// NOTE: this is needed only to add action methods to TestActions...
 | 
					
						
							|  |  |  | 	var BaseActions = new ActionSet() | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | 	var TestActions =  | 
					
						
							|  |  |  | 	module.TestActions =  | 
					
						
							|  |  |  | 	Actions(BaseActions, { | 
					
						
							|  |  |  | 		testActionGen1: ['baisc test action...', | 
					
						
							|  |  |  | 			'some extra info', | 
					
						
							|  |  |  | 			function(){ | 
					
						
							|  |  |  | 				console.log('  test 1!') | 
					
						
							|  |  |  | 				return function(){ | 
					
						
							| 
									
										
										
										
											2020-11-23 14:47:00 +03:00
										 |  |  | 					console.log('  test 2!') } }], | 
					
						
							| 
									
										
										
										
											2016-08-20 23:46:34 +03:00
										 |  |  | 
 | 
					
						
							|  |  |  | 		testActionGen2: ['baisc 2\'nd gen test action...', | 
					
						
							|  |  |  | 			// no extra info...
 | 
					
						
							|  |  |  | 			function(){ | 
					
						
							|  |  |  | 				console.log('  test gen 2!') | 
					
						
							| 
									
										
										
										
											2020-11-23 14:47:00 +03:00
										 |  |  | 				this.testActionGen1() }], | 
					
						
							| 
									
										
										
										
											2016-08-20 23:46:34 +03:00
										 |  |  | 	}) | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | 	var TestActions2 =  | 
					
						
							|  |  |  | 	module.TestActions2 =  | 
					
						
							|  |  |  | 	Actions(TestActions, { | 
					
						
							|  |  |  | 		// NOTE: this looks like an action and feels like an action but 
 | 
					
						
							|  |  |  | 		// 		actually this is a callback as an action with this name 
 | 
					
						
							|  |  |  | 		// 		already exists...
 | 
					
						
							|  |  |  | 		testActionGen1: [ | 
					
						
							|  |  |  | 			function(){ | 
					
						
							|  |  |  | 				console.log('  pre callback!') | 
					
						
							|  |  |  | 				return function(){ | 
					
						
							| 
									
										
										
										
											2020-11-23 14:47:00 +03:00
										 |  |  | 					console.log('  post callback!') } }], | 
					
						
							| 
									
										
										
										
											2016-08-20 23:46:34 +03:00
										 |  |  | 
 | 
					
						
							|  |  |  | 		testAction2: ['this is an action', | 
					
						
							|  |  |  | 			function(){ | 
					
						
							| 
									
										
										
										
											2020-11-23 14:47:00 +03:00
										 |  |  | 				console.log('testAction2 args:', arguments) }], | 
					
						
							| 
									
										
										
										
											2016-08-20 23:46:34 +03:00
										 |  |  | 
 | 
					
						
							|  |  |  | 	}) | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | 	// XXX the main question here is that there is no way to know if a 
 | 
					
						
							|  |  |  | 	// 		particular action is going to be a root action or an action
 | 
					
						
							|  |  |  | 	// 		callback because we do not know if the action in the parent 
 | 
					
						
							|  |  |  | 	// 		will be available at mix time or not, and the two models 
 | 
					
						
							|  |  |  | 	// 		are different...
 | 
					
						
							|  |  |  | 	// 		XXX one way to do this is to make all code a callback and 
 | 
					
						
							|  |  |  | 	// 			just use the root as an event trigger...
 | 
					
						
							|  |  |  | 	//
 | 
					
						
							|  |  |  | 	// 			...but this effectively means we are implementing 
 | 
					
						
							|  |  |  | 	// 			inheritance ourselves as the traditional name resolution
 | 
					
						
							|  |  |  | 	// 			will no longer be used, and as in the case we implement
 | 
					
						
							|  |  |  | 	// 			MRO why not go the whole way and implement multiple 
 | 
					
						
							|  |  |  | 	// 			inheritance in the first place...
 | 
					
						
							|  |  |  | 	//
 | 
					
						
							|  |  |  | 	// 			...let's try and avoid this...
 | 
					
						
							|  |  |  | 	/* | 
					
						
							|  |  |  | 	var TestActionMixin = | 
					
						
							|  |  |  | 	module.TestActionMixin =  | 
					
						
							|  |  |  | 	ActionMixin({ | 
					
						
							|  |  |  | 		// XXX
 | 
					
						
							|  |  |  | 	}) | 
					
						
							|  |  |  | 	*/ | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | 	console.log('TestActions.testActionGen1()') | 
					
						
							|  |  |  | 	TestActions.testActionGen1() | 
					
						
							|  |  |  | 	console.log('TestActions.testActionGen2()') | 
					
						
							|  |  |  | 	TestActions.testActionGen2() | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | 		 | 
					
						
							|  |  |  | 	// both of these should cet a callback...
 | 
					
						
							|  |  |  | 	console.log('TestActions2.testActionGen1()') | 
					
						
							|  |  |  | 	TestActions2.testActionGen1() | 
					
						
							|  |  |  | 	console.log('TestActions2.testActionGen2()') | 
					
						
							|  |  |  | 	TestActions2.testActionGen2() | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | 	// and an event-like handler...
 | 
					
						
							|  |  |  | 	TestActions2.on('testActionGen1.post',  | 
					
						
							|  |  |  | 			function(){ console.log('  post handler! (first defined)') }) | 
					
						
							|  |  |  | 	TestActions2.on('testActionGen1',  | 
					
						
							|  |  |  | 			function(){ console.log('  post handler! (last defined)') }) | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | 	console.log('TestActions2.testActionGen1()') | 
					
						
							|  |  |  | 	TestActions2.testActionGen1() | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | 	TestActions2.on('testActionGen2.pre',  | 
					
						
							|  |  |  | 			function(){ console.log('  pre handler! (first defined)') }) | 
					
						
							|  |  |  | 	TestActions2.on('testActionGen2.pre',  | 
					
						
							|  |  |  | 			function(){ console.log('  pre handler! (last defined)') }) | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | 	console.log('TestActions2.testActionGen2()') | 
					
						
							|  |  |  | 	TestActions2.testActionGen2() | 
					
						
							|  |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | /********************************************************************** | 
					
						
							|  |  |  | * vim:set ts=4 sw=4 :                                                */ | 
					
						
							|  |  |  | return module }) |