2020-11-12 17:35:41 +03:00
|
|
|
/**********************************************************************
|
|
|
|
|
*
|
|
|
|
|
*
|
|
|
|
|
*
|
2020-11-15 00:24:22 +03:00
|
|
|
* XXX is types/events the right place for this???
|
2020-11-13 13:37:10 +03:00
|
|
|
* XXX should we have .pre/.post events???
|
2020-11-15 00:24:22 +03:00
|
|
|
* XXX should we propogate event handling to parent/overloaded events???
|
2020-11-13 13:37:10 +03:00
|
|
|
*
|
2020-11-12 17:35:41 +03:00
|
|
|
**********************************************/ /* c8 ignore next 2 */
|
|
|
|
|
((typeof define)[0]=='u'?function(f){module.exports=f(require)}:define)
|
|
|
|
|
(function(require){ var module={} // make module AMD/node compatible...
|
|
|
|
|
/*********************************************************************/
|
|
|
|
|
|
|
|
|
|
var object = require('ig-object')
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
/*********************************************************************/
|
|
|
|
|
|
2020-11-15 00:24:22 +03:00
|
|
|
var bareEventMethod =
|
|
|
|
|
module.bareEventMethod =
|
|
|
|
|
function(name, func, options={}){
|
2020-11-12 17:35:41 +03:00
|
|
|
var hidden
|
|
|
|
|
var method
|
|
|
|
|
|
2020-11-15 00:24:22 +03:00
|
|
|
options = func && typeof(func) != 'function' ?
|
2020-11-12 17:35:41 +03:00
|
|
|
func
|
|
|
|
|
: options
|
|
|
|
|
|
|
|
|
|
return object.mixinFlat(
|
2020-11-15 00:24:22 +03:00
|
|
|
method = function(...args){
|
2020-11-12 17:35:41 +03:00
|
|
|
var handlers =
|
|
|
|
|
// hidden...
|
|
|
|
|
options.handlerLocation == 'hidden' ?
|
2020-11-15 00:24:22 +03:00
|
|
|
(hidden || [])
|
2020-11-12 17:35:41 +03:00
|
|
|
// function...
|
|
|
|
|
: options.handlerLocation == 'method' ?
|
2020-11-15 00:24:22 +03:00
|
|
|
(method.__event_handlers__ || [])
|
2020-11-12 17:35:41 +03:00
|
|
|
// context (default)...
|
2020-11-15 00:24:22 +03:00
|
|
|
: ((this.__event_handlers__ || {})[name] || [])
|
2020-11-12 17:35:41 +03:00
|
|
|
|
2020-11-15 00:24:22 +03:00
|
|
|
// NOTE: this will stop event handling if one of the handlers
|
|
|
|
|
// explicitly returns false...
|
2020-11-12 17:35:41 +03:00
|
|
|
var handle = function(){
|
2020-11-15 00:24:22 +03:00
|
|
|
return handlers
|
|
|
|
|
.reduce(function(res, handler){
|
|
|
|
|
return res === true
|
|
|
|
|
&& handler(...args) !== false }, true) }
|
2020-11-12 17:35:41 +03:00
|
|
|
var res
|
|
|
|
|
func ?
|
|
|
|
|
(res = func.call(this, handle, ...args))
|
|
|
|
|
: handle(...args)
|
|
|
|
|
|
|
|
|
|
return res },
|
|
|
|
|
{
|
2020-11-15 00:24:22 +03:00
|
|
|
__event__: 'bare',
|
2020-11-12 17:35:41 +03:00
|
|
|
get __event_handler_location__(){
|
|
|
|
|
return ['hidden', 'method'].includes(options.handlerLocation) ?
|
|
|
|
|
options.handlerLocation
|
|
|
|
|
: 'context' },
|
|
|
|
|
__event_handler_add__: function(context, func){
|
|
|
|
|
var handlers =
|
|
|
|
|
// hidden...
|
|
|
|
|
options.handlerLocation == 'hidden' ?
|
|
|
|
|
(hidden = hidden || [])
|
|
|
|
|
// function...
|
|
|
|
|
: options.handlerLocation == 'method' ?
|
|
|
|
|
(method.__event_handlers__ = method.__event_handlers__ || [])
|
|
|
|
|
// context (default)...
|
2020-11-15 00:24:22 +03:00
|
|
|
: ((context.__event_handlers__ = context.__event_handlers__ || {})[name] =
|
|
|
|
|
context.__event_handlers__[name] || [])
|
2020-11-12 17:35:41 +03:00
|
|
|
// add handler...
|
2020-11-15 00:24:22 +03:00
|
|
|
handlers.push(func)
|
2020-11-12 17:35:41 +03:00
|
|
|
return this },
|
2020-11-15 00:24:22 +03:00
|
|
|
// XXX should this support the 'all' key -- remove all handlers???
|
2020-11-12 17:35:41 +03:00
|
|
|
__event_handler_remove__: function(context, func){
|
|
|
|
|
var handlers =
|
|
|
|
|
(options.handlerLocation == 'hidden' ?
|
|
|
|
|
hidden
|
|
|
|
|
: options.handlerLocation == 'method' ?
|
|
|
|
|
method.__event_handlers__
|
|
|
|
|
: (context.__event_handlers__ || {})[name]) || []
|
|
|
|
|
handlers.splice(handlers.indexOf(func), 1)
|
|
|
|
|
return this },
|
|
|
|
|
toString: function(){
|
|
|
|
|
return func.toString()
|
|
|
|
|
.replace(/^(function[^(]*\()[^,)]*, ?/, '$1') },
|
|
|
|
|
}) }
|
|
|
|
|
|
|
|
|
|
|
2020-11-15 00:24:22 +03:00
|
|
|
// Extends bareEventMethod(..) adding ability to bind events via the
|
|
|
|
|
// resulting method directly...
|
2020-11-12 17:35:41 +03:00
|
|
|
//
|
|
|
|
|
// eventMethod(name, func)
|
|
|
|
|
// -> method
|
|
|
|
|
// eventMethod(name, func)
|
|
|
|
|
// -> method
|
|
|
|
|
//
|
|
|
|
|
//
|
|
|
|
|
// Bind handler...
|
|
|
|
|
// method(handler)
|
|
|
|
|
// -> this
|
|
|
|
|
//
|
|
|
|
|
// Unbind handler...
|
|
|
|
|
// method(handler, false)
|
|
|
|
|
// -> this
|
|
|
|
|
//
|
|
|
|
|
// Trigger handlers...
|
|
|
|
|
// method(...args)
|
|
|
|
|
// -> this
|
|
|
|
|
//
|
|
|
|
|
//
|
|
|
|
|
// func(handle, ...args)
|
|
|
|
|
//
|
|
|
|
|
//
|
2020-11-15 00:24:22 +03:00
|
|
|
var eventMethod =
|
|
|
|
|
module.eventMethod =
|
|
|
|
|
function(name, func, options={}){
|
2020-11-12 17:35:41 +03:00
|
|
|
var method
|
|
|
|
|
options = typeof(func) != 'function' ?
|
|
|
|
|
func
|
|
|
|
|
: options
|
|
|
|
|
|
|
|
|
|
return Object.assign(
|
|
|
|
|
method = bareEventMethod(name,
|
|
|
|
|
function(handle, ...args){
|
|
|
|
|
// add handler...
|
|
|
|
|
// XXX handle handler tags...
|
|
|
|
|
if(typeof(args[0]) == 'function'){
|
|
|
|
|
method.__event_handler_add__(this, args[0])
|
|
|
|
|
|
|
|
|
|
// call the action...
|
|
|
|
|
} else {
|
2020-11-15 00:24:22 +03:00
|
|
|
func
|
|
|
|
|
&& func.call(this, handle, ...args) }
|
2020-11-12 17:35:41 +03:00
|
|
|
|
|
|
|
|
return this },
|
|
|
|
|
options),
|
|
|
|
|
{
|
2020-11-15 00:24:22 +03:00
|
|
|
__event__: 'full',
|
2020-11-12 17:35:41 +03:00
|
|
|
// NOTE: this is a copy of bareEventMethod's .toString() as we
|
|
|
|
|
// still need to base the doc on the user's func...
|
|
|
|
|
toString: function(){
|
|
|
|
|
return func.toString()
|
|
|
|
|
.replace(/^(function[^(]*\()[^,)]*, ?/, '$1') },
|
|
|
|
|
}) }
|
|
|
|
|
|
|
|
|
|
|
2020-11-15 00:24:22 +03:00
|
|
|
// XXX might be nice to add support to pre/post handlers...
|
|
|
|
|
// XXX still not sure about the builtin-local event control flow...
|
|
|
|
|
var EventHandlerMixin =
|
|
|
|
|
module.EventHandlerMixin = {
|
2020-11-13 13:37:10 +03:00
|
|
|
__event_handlers__: null,
|
|
|
|
|
|
2020-11-15 00:24:22 +03:00
|
|
|
// XXX do we need to be able to force global handler???
|
2020-11-12 17:35:41 +03:00
|
|
|
on: function(evt, func){
|
|
|
|
|
// event...
|
2020-11-13 13:37:10 +03:00
|
|
|
if(evt in this
|
|
|
|
|
&& this[evt].__event_handler_add__){
|
|
|
|
|
this[evt].__event_handler_add__(this, func)
|
2020-11-12 17:35:41 +03:00
|
|
|
// non-event...
|
|
|
|
|
} else {
|
|
|
|
|
;((this.__event_handlers__ = this.__event_handlers__ || {})[evt] =
|
|
|
|
|
this.__event_handlers__[evt] || [])
|
|
|
|
|
.push(func) }
|
|
|
|
|
return this },
|
2020-11-15 00:24:22 +03:00
|
|
|
// XXX do we need .off(evt, 'all')
|
2020-11-13 13:37:10 +03:00
|
|
|
off: function(evt, func){
|
|
|
|
|
// event...
|
|
|
|
|
if(evt in this
|
|
|
|
|
&& this[evt].__event_handler_add__){
|
|
|
|
|
this[evt].__event_handler_remove__(this, func)
|
|
|
|
|
// non-event...
|
|
|
|
|
} else {
|
2020-11-15 00:24:22 +03:00
|
|
|
var handlers = this.__event_handlers__
|
|
|
|
|
&& (this.__event_handlers__[evt] || [])
|
|
|
|
|
handlers
|
|
|
|
|
&& handlers.splice(handlers.indexOf(func), 1) }
|
2020-11-13 13:37:10 +03:00
|
|
|
return this },
|
2020-11-15 00:24:22 +03:00
|
|
|
// XXX add support for stopping handler execution...
|
2020-11-13 13:37:10 +03:00
|
|
|
trigger: function(evt, ...args){
|
2020-11-15 00:24:22 +03:00
|
|
|
// local handler...
|
|
|
|
|
evt in this
|
|
|
|
|
&& this[evt](...args)
|
|
|
|
|
// global events...
|
|
|
|
|
this.__event_handlers__
|
|
|
|
|
&& (this.__event_handlers__[evt] || [])
|
|
|
|
|
.forEach(function(h){ h(evt, ...args) })
|
|
|
|
|
return this },
|
2020-11-12 17:35:41 +03:00
|
|
|
}
|
|
|
|
|
|
|
|
|
|
|
2020-11-15 00:24:22 +03:00
|
|
|
// NOTE: this can't be added via Object.assign(..), use object.mixinFlat(..)
|
|
|
|
|
// instead...
|
|
|
|
|
var EventDocMixin =
|
|
|
|
|
module.EventDocMixin = {
|
|
|
|
|
get eventfull(){
|
|
|
|
|
return object.deepKeys(this)
|
|
|
|
|
.filter(function(n){
|
|
|
|
|
// avoid triggering props...
|
|
|
|
|
return !object.values(this, n, function(){ return object.STOP }, true)[0].get
|
|
|
|
|
&& (this[n] || {}).__event__ == 'bare'}.bind(this)) },
|
|
|
|
|
get events(){
|
|
|
|
|
return object.deepKeys(this)
|
|
|
|
|
.filter(function(n){
|
|
|
|
|
// avoid triggering props...
|
|
|
|
|
return !object.values(this, n, function(){ return object.STOP }, true)[0].get
|
|
|
|
|
&& (this[n] || {}).__event__ == 'full' }.bind(this)) },
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
var EventMixin =
|
|
|
|
|
module.EventMixin =
|
|
|
|
|
object.mixinFlat(
|
|
|
|
|
EventHandlerMixin,
|
|
|
|
|
EventDocMixin)
|
|
|
|
|
|
|
|
|
|
|
2020-11-12 17:35:41 +03:00
|
|
|
|
|
|
|
|
|
|
|
|
|
/**********************************************************************
|
|
|
|
|
* vim:set ts=4 sw=4 : */ return module })
|