nextgen-events
Version:
The next generation of events handling for javascript! New: abstract away the network!
1,642 lines (1,148 loc) • 66.8 kB
JavaScript
(function(f){if(typeof exports==="object"&&typeof module!=="undefined"){module.exports=f()}else if(typeof define==="function"&&define.amd){define([],f)}else{var g;if(typeof window!=="undefined"){g=window}else if(typeof global!=="undefined"){g=global}else if(typeof self!=="undefined"){g=self}else{g=this}g.NextGenEvents = f()}})(function(){var define,module,exports;return (function(){function r(e,n,t){function o(i,f){if(!n[i]){if(!e[i]){var c="function"==typeof require&&require;if(!f&&c)return c(i,!0);if(u)return u(i,!0);var a=new Error("Cannot find module '"+i+"'");throw a.code="MODULE_NOT_FOUND",a}var p=n[i]={exports:{}};e[i][0].call(p.exports,function(r){var n=e[i][1][r];return o(n||r)},p,p.exports,r,e,n,t)}return n[i].exports}for(var u="function"==typeof require&&require,i=0;i<t.length;i++)o(t[i]);return o}return r})()({1:[function(require,module,exports){
(function (process,global){
/*
Next-Gen Events
Copyright (c) 2015 - 2021 Cédric Ronvel
The MIT License (MIT)
Permission is hereby granted, free of charge, to any person obtaining a copy
of this software and associated documentation files (the "Software"), to deal
in the Software without restriction, including without limitation the rights
to use, copy, modify, merge, publish, distribute, sublicense, and/or sell
copies of the Software, and to permit persons to whom the Software is
furnished to do so, subject to the following conditions:
The above copyright notice and this permission notice shall be included in all
copies or substantial portions of the Software.
THE SOFTWARE IS PROVIDED "AS IS", WITHOUT WARRANTY OF ANY KIND, EXPRESS OR
IMPLIED, INCLUDING BUT NOT LIMITED TO THE WARRANTIES OF MERCHANTABILITY,
FITNESS FOR A PARTICULAR PURPOSE AND NONINFRINGEMENT. IN NO EVENT SHALL THE
AUTHORS OR COPYRIGHT HOLDERS BE LIABLE FOR ANY CLAIM, DAMAGES OR OTHER
LIABILITY, WHETHER IN AN ACTION OF CONTRACT, TORT OR OTHERWISE, ARISING FROM,
OUT OF OR IN CONNECTION WITH THE SOFTWARE OR THE USE OR OTHER DEALINGS IN THE
SOFTWARE.
*/
"use strict" ;
// Some features needs a portable nextTick
const nextTick = process.browser ? window.setImmediate : process.nextTick ;
if ( ! global.__NEXTGEN_EVENTS__ ) {
global.__NEXTGEN_EVENTS__ = {
recursions: 0
} ;
}
var globalData = global.__NEXTGEN_EVENTS__ ;
function NextGenEvents() {}
module.exports = NextGenEvents ;
NextGenEvents.prototype.__prototypeUID__ = 'nextgen-events/NextGenEvents' ;
NextGenEvents.prototype.__prototypeVersion__ = require( '../package.json' ).version ;
/* Basic features, more or less compatible with Node.js */
NextGenEvents.SYNC = -Infinity ;
NextGenEvents.DESYNC = -1 ;
NextGenEvents.defaultMaxListeners = Infinity ;
// Not part of the prototype, because it should not pollute userland's prototype.
// It has an eventEmitter as 'this' anyway (always called using call()).
NextGenEvents.init = function() {
Object.defineProperty( this , '__ngev' , {
configurable: true ,
value: new NextGenEvents.Internal()
} ) ;
} ;
NextGenEvents.Internal = function( from ) {
this.nice = NextGenEvents.SYNC ;
this.interruptible = false ;
this.contexts = {} ;
this.desync = setImmediate ;
this.depth = 0 ;
// States by events
this.states = {} ;
// State groups by events
this.stateGroups = {} ;
// Listeners by events
this.listeners = {
// Special events
error: [] ,
interrupt: [] ,
newListener: [] ,
removeListener: []
} ;
this.hasListenerPriority = false ;
this.maxListeners = NextGenEvents.defaultMaxListeners ;
if ( from ) {
this.nice = from.nice ;
this.interruptible = from.interruptible ;
Object.assign( this.states , from.states ) ,
Object.assign( this.stateGroups , from.stateGroups ) ,
Object.keys( from.listeners ).forEach( eventName => {
this.listeners[ eventName ] = from.listeners[ eventName ].slice() ;
} ) ;
// Copy all contexts
Object.keys( from.contexts ).forEach( contextName => {
var context = from.contexts[ contextName ] ;
this.contexts[ contextName ] = {
nice: context.nice ,
ready: true ,
status: context.status ,
serial: context.serial ,
scopes: {}
} ;
} ) ;
}
} ;
NextGenEvents.initFrom = function( from ) {
if ( ! from.__ngev ) { NextGenEvents.init.call( from ) ; }
Object.defineProperty( this , '__ngev' , {
configurable: true ,
value: new NextGenEvents.Internal( from.__ngev )
} ) ;
} ;
/*
Merge listeners of duplicated event bus:
* listeners that are present locally but not in all foreigner are removed (one of the foreigner has removed it)
* listeners that are not present locally but present in at least one foreigner are copied
Not sure if it will ever go public, it was a very specific use-case (Spellcast).
*/
NextGenEvents.mergeListeners = function( foreigners ) {
if ( ! this.__ngev ) { NextGenEvents.init.call( this ) ; }
// Backup the current listeners...
var oldListeners = this.__ngev.listeners ;
// Reset listeners...
this.__ngev.listeners = {} ;
Object.keys( oldListeners ).forEach( eventName => {
this.__ngev.listeners[ eventName ] = [] ;
} ) ;
foreigners.forEach( foreigner => {
if ( ! foreigner.__ngev ) { NextGenEvents.init.call( foreigner ) ; }
Object.keys( foreigner.__ngev.listeners ).forEach( eventName => {
if ( ! this.__ngev.listeners[ eventName ] ) { this.__ngev.listeners[ eventName ] = [] ; }
} ) ;
} ) ;
// Now we can scan by eventName first
Object.keys( this.__ngev.listeners ).forEach( eventName => {
var i , iMax , blacklist = [] ;
// First pass: find all removed listeners and add them to the blacklist
if ( oldListeners[ eventName ] ) {
oldListeners[ eventName ].forEach( listener => {
for ( i = 0 , iMax = foreigners.length ; i < iMax ; i ++ ) {
if (
! foreigners[ i ].__ngev.listeners[ eventName ] ||
foreigners[ i ].__ngev.listeners[ eventName ].indexOf( listener ) === -1
) {
blacklist.push( listener ) ;
break ;
}
}
} ) ;
}
// Second pass: add all listeners still not present and that are not blacklisted
foreigners.forEach( foreigner => {
foreigner.__ngev.listeners[ eventName ].forEach( listener => {
if ( this.__ngev.listeners[ eventName ].indexOf( listener ) === -1 && blacklist.indexOf( listener ) === -1 ) {
this.__ngev.listeners[ eventName ].push( listener ) ;
}
} ) ;
} ) ;
} ) ;
} ;
// Use it with .bind()
NextGenEvents.filterOutCallback = function( what , currentElement ) { return what !== currentElement ; } ;
// .addListener( eventName , [fn] , [options] )
NextGenEvents.prototype.addListener = function( eventName , fn , options ) {
var listener , newListenerListeners ;
if ( ! this.__ngev ) { NextGenEvents.init.call( this ) ; }
if ( ! this.__ngev.listeners[ eventName ] ) { this.__ngev.listeners[ eventName ] = [] ; }
// Argument management
if ( ! eventName || typeof eventName !== 'string' ) {
throw new TypeError( ".addListener(): argument #0 should be a non-empty string" ) ;
}
if ( typeof fn === 'function' ) {
listener = {} ;
if ( ! options || typeof options !== 'object' ) { options = {} ; }
}
else if ( options === true && fn && typeof fn === 'object' ) {
// We want to use the current object as the listener object (used by Spellcast's serializer)
options = listener = fn ;
fn = undefined ;
}
else {
options = fn ;
if ( ! options || typeof options !== 'object' ) {
throw new TypeError( ".addListener(): a function or an object with a 'fn' property which value is a function should be provided" ) ;
}
fn = undefined ;
listener = {} ;
}
listener.fn = fn || options.fn ;
listener.id = options.id !== undefined ? options.id : listener.fn ;
if ( options.unique ) {
if ( this.__ngev.listeners[ eventName ].find( e => e.id === listener.id ) ) {
// Not unique! Return now!
return ;
}
}
listener.once = !! options.once ;
listener.async = !! options.async ;
listener.eventObject = !! options.eventObject ;
listener.nice = options.nice !== undefined ? Math.floor( options.nice ) : NextGenEvents.SYNC ;
listener.priority = + options.priority || 0 ;
listener.context = options.context && ( typeof options.context === 'string' || typeof options.context === 'object' ) ? options.context : null ;
if ( typeof listener.fn !== 'function' ) {
throw new TypeError( ".addListener(): a function or an object with a 'fn' property which value is a function should be provided" ) ;
}
// Implicit context creation
if ( typeof listener.context === 'string' ) {
listener.context = this.__ngev.contexts[ listener.context ] || this.addListenerContext( listener.context ) ;
}
// Note: 'newListener' and 'removeListener' event return an array of listener, but not the event name.
// So the event's name can be retrieved in the listener itself.
listener.event = eventName ;
if ( this.__ngev.listeners.newListener.length ) {
// Extra care should be taken with the 'newListener' event, we should avoid recursion
// in the case that eventName === 'newListener', but inside a 'newListener' listener,
// .listenerCount() should report correctly
newListenerListeners = this.__ngev.listeners.newListener.slice() ;
this.__ngev.listeners[ eventName ].push( listener ) ;
// Return an array, because one day, .addListener() may support multiple event addition at once,
// e.g.: .addListener( { request: onRequest, close: onClose, error: onError } ) ;
NextGenEvents.emitEvent( {
emitter: this ,
name: 'newListener' ,
args: [ [ listener ] ] ,
listeners: newListenerListeners
} ) ;
if ( this.__ngev.states[ eventName ] ) { NextGenEvents.emitToOneListener( this.__ngev.states[ eventName ] , listener ) ; }
return this ;
}
this.__ngev.listeners[ eventName ].push( listener ) ;
if ( this.__ngev.hasListenerPriority ) {
// order higher priority first
this.__ngev.listeners[ eventName ].sort( ( a , b ) => b.priority - a.priority ) ;
}
if ( this.__ngev.listeners[ eventName ].length === this.__ngev.maxListeners + 1 ) {
process.emitWarning(
"Possible NextGenEvents memory leak detected. " + this.__ngev.listeners[ eventName ].length + ' ' +
eventName + " listeners added. Use emitter.setMaxListeners() to increase limit" ,
{ type: "MaxListenersExceededWarning" }
) ;
}
if ( this.__ngev.states[ eventName ] ) { NextGenEvents.emitToOneListener( this.__ngev.states[ eventName ] , listener ) ; }
return this ;
} ;
NextGenEvents.prototype.on = NextGenEvents.prototype.addListener ;
// Short-hand
// .once( eventName , [fn] , [options] )
NextGenEvents.prototype.once = function( eventName , fn , options ) {
if ( fn && typeof fn === 'object' ) { fn.once = true ; }
else if ( options && typeof options === 'object' ) { options.once = true ; }
else { options = { once: true } ; }
return this.addListener( eventName , fn , options ) ;
} ;
// .waitFor( eventName )
// A Promise-returning .once() variant, only the first arg is returned
NextGenEvents.prototype.waitFor = function( eventName ) {
return new Promise( resolve => {
this.addListener( eventName , ( firstArg ) => resolve( firstArg ) , { once: true } ) ;
} ) ;
} ;
// .waitForAll( eventName )
// A Promise-returning .once() variant, all args are returned as an array
NextGenEvents.prototype.waitForAll = function( eventName ) {
return new Promise( resolve => {
this.addListener( eventName , ( ... args ) => resolve( args ) , { once: true } ) ;
} ) ;
} ;
NextGenEvents.prototype.removeListener = function( eventName , id ) {
if ( ! eventName || typeof eventName !== 'string' ) { throw new TypeError( ".removeListener(): argument #0 should be a non-empty string" ) ; }
if ( ! this.__ngev ) { NextGenEvents.init.call( this ) ; }
var listeners = this.__ngev.listeners[ eventName ] ;
if ( ! listeners || ! listeners.length ) { return this ; }
var i , removedListeners , removeCount = 0 ,
length = listeners.length ,
hasRemoveListener = this.__ngev.listeners.removeListener.length ;
if ( hasRemoveListener ) { removedListeners = [] ; }
// In-place remove (from the listener array)
for ( i = 0 ; i < length ; i ++ ) {
if ( listeners[ i ].id === id ) {
removeCount ++ ;
if ( hasRemoveListener ) { removedListeners.push( listeners[ i ] ) ; }
}
else if ( removeCount ) {
listeners[ i - removeCount ] = listeners[ i ] ;
}
}
// Adjust the length
if ( removeCount ) { listeners.length -= removeCount ; }
if ( hasRemoveListener && removedListeners.length ) {
this.emit( 'removeListener' , removedListeners ) ;
}
return this ;
} ;
NextGenEvents.prototype.off = NextGenEvents.prototype.removeListener ;
NextGenEvents.prototype.removeAllListeners = function( eventName ) {
var removedListeners ;
if ( ! this.__ngev ) { NextGenEvents.init.call( this ) ; }
if ( eventName ) {
// Remove all listeners for a particular event
if ( ! eventName || typeof eventName !== 'string' ) { throw new TypeError( ".removeAllListeners(): argument #0 should be undefined or a non-empty string" ) ; }
if ( ! this.__ngev.listeners[ eventName ] ) { this.__ngev.listeners[ eventName ] = [] ; }
removedListeners = this.__ngev.listeners[ eventName ] ;
this.__ngev.listeners[ eventName ] = [] ;
if ( removedListeners.length && this.__ngev.listeners.removeListener.length ) {
this.emit( 'removeListener' , removedListeners ) ;
}
}
else {
// Remove all listeners for any events
// 'removeListener' listeners cannot be triggered: they are already deleted
this.__ngev.listeners = {
// Special events
error: [] ,
interrupt: [] ,
newListener: [] ,
removeListener: []
} ;
}
return this ;
} ;
NextGenEvents.listenerWrapper = function( listener , event , contextScope , serial , nice ) {
var returnValue , listenerCallback ,
eventMaster = event.master || event ,
interruptible = !! event.master || event.emitter.__ngev.interruptible ;
if ( eventMaster.interrupt ) { return ; }
if ( listener.async ) {
if ( contextScope ) {
contextScope.ready = ! serial ;
}
if ( nice < 0 ) {
if ( globalData.recursions >= -nice ) {
event.emitter.__ngev.desync( NextGenEvents.listenerWrapper.bind( undefined , listener , event , contextScope , serial , NextGenEvents.SYNC ) ) ;
return ;
}
}
else {
setTimeout( NextGenEvents.listenerWrapper.bind( undefined , listener , event , contextScope , serial , NextGenEvents.SYNC ) , nice ) ;
return ;
}
listenerCallback = ( arg ) => {
eventMaster.listenersDone ++ ;
// Async interrupt
if ( arg && interruptible && ! eventMaster.interrupt && event.name !== 'interrupt' ) {
eventMaster.interrupt = arg ;
if ( eventMaster.callback ) { NextGenEvents.emitCallback( event ) ; }
event.emitter.emit( 'interrupt' , eventMaster.interrupt ) ;
}
else if ( eventMaster.listenersDone >= eventMaster.listeners.length && eventMaster.callback ) {
NextGenEvents.emitCallback( event ) ;
}
// Process the queue if serialized
if ( serial ) { NextGenEvents.processScopeQueue( contextScope , true , true ) ; }
} ;
if ( listener.eventObject ) { listener.fn( event , listenerCallback ) ; }
else { returnValue = listener.fn( ... event.args , listenerCallback ) ; }
}
else {
if ( nice < 0 ) {
if ( globalData.recursions >= -nice ) {
event.emitter.__ngev.desync( NextGenEvents.listenerWrapper.bind( undefined , listener , event , contextScope , serial , NextGenEvents.SYNC ) ) ;
return ;
}
}
else {
setTimeout( NextGenEvents.listenerWrapper.bind( undefined , listener , event , contextScope , serial , NextGenEvents.SYNC ) , nice ) ;
return ;
}
if ( listener.eventObject ) { listener.fn( event ) ; }
else { returnValue = listener.fn( ... event.args ) ; }
eventMaster.listenersDone ++ ;
}
// Interrupt if non-falsy return value, if the emitter is interruptible, not already interrupted (emit once),
// and not within an 'interrupt' event.
if ( returnValue && interruptible && ! eventMaster.interrupt && event.name !== 'interrupt' ) {
eventMaster.interrupt = returnValue ;
if ( eventMaster.callback ) { NextGenEvents.emitCallback( event ) ; }
event.emitter.emit( 'interrupt' , eventMaster.interrupt ) ;
}
else if ( eventMaster.listenersDone >= eventMaster.listeners.length && eventMaster.callback ) {
NextGenEvents.emitCallback( event ) ;
}
} ;
// A unique event ID
var nextEventId = 0 ;
/*
emit( [nice] , eventName , [arg1] , [arg2] , [...] , [emitCallback] )
*/
NextGenEvents.prototype.emit = function( ... args ) {
var event = NextGenEvents.createEvent( this , ... args ) ;
return NextGenEvents.emitEvent( event ) ;
} ;
// For performance, do not emit if there is no listener for that event,
// do not even return an Event object, do not throw if the event name is error,
// or whatever the .emit() process could do when there is no listener.
NextGenEvents.prototype.emitIfListener = function( ... args ) {
var eventName = typeof args[ 0 ] === 'number' ? args[ 1 ] : args[ 0 ] ;
if ( ! this.__ngev || ! this.__ngev.listeners[ eventName ] || ! this.__ngev.listeners[ eventName ].length ) { return null ; }
var event = NextGenEvents.createEvent( this , ... args ) ;
return NextGenEvents.emitEvent( event ) ;
} ;
NextGenEvents.prototype.waitForEmit = function( ... args ) {
return new Promise( resolve => {
this.emit( ... args , ( interrupt ) => resolve( interrupt ) ) ;
} ) ;
} ;
// Create an event object
NextGenEvents.createEvent = function( emitter , ... args ) {
var event = {
emitter: emitter ,
interrupt: null ,
master: null , // For grouped-correlated events
sync: true
} ;
// Arguments handling
if ( typeof args[ 0 ] === 'number' ) {
event.nice = Math.floor( args[ 0 ] ) ;
event.name = args[ 1 ] ;
if ( ! event.name || typeof event.name !== 'string' ) {
throw new TypeError( ".emit(): when argument #0 is a number, argument #1 should be a non-empty string" ) ;
}
if ( typeof args[ args.length - 1 ] === 'function' ) {
event.callback = args[ args.length - 1 ] ;
event.args = args.slice( 2 , -1 ) ;
}
else {
event.args = args.slice( 2 ) ;
}
}
else {
//event.nice = emitter.__ngev.nice ;
event.name = args[ 0 ] ;
if ( ! event.name || typeof event.name !== 'string' ) {
throw new TypeError( ".emit(): argument #0 should be an number or a non-empty string" ) ;
}
if ( typeof args[ args.length - 1 ] === 'function' ) {
event.callback = args[ args.length - 1 ] ;
event.args = args.slice( 1 , -1 ) ;
}
else {
event.args = args.slice( 1 ) ;
}
}
return event ;
} ;
/*
At this stage, 'event' should be an object having those properties:
* emitter: the event emitter
* name: the event name
* args: array, the arguments of the event
* nice: (optional) nice value
* callback: (optional) a callback for emit
* listeners: (optional) override the listeners array stored in __ngev
*/
NextGenEvents.emitEvent = function( event ) {
// /!\ Any change here *MUST* be reflected to NextGenEvents.emitIntricatedEvents() /!\
var self = event.emitter ,
i , iMax , count = 0 , state , removedListeners ;
if ( ! self.__ngev ) { NextGenEvents.init.call( self ) ; }
state = self.__ngev.states[ event.name ] ;
// This is a state event, register it now!
if ( state !== undefined ) {
if ( state && event.args.length === state.args.length &&
event.args.every( ( arg , index ) => arg === state.args[ index ] ) ) {
// The emitter is already in this exact state, skip it now!
return ;
}
// Unset all states of that group
self.__ngev.stateGroups[ event.name ].forEach( ( eventName ) => {
self.__ngev.states[ eventName ] = null ;
} ) ;
self.__ngev.states[ event.name ] = event ;
}
if ( ! self.__ngev.listeners[ event.name ] ) { self.__ngev.listeners[ event.name ] = [] ; }
event.id = nextEventId ++ ;
event.listenersDone = 0 ;
if ( event.nice === undefined || event.nice === null ) { event.nice = self.__ngev.nice ; }
// Trouble arise when a listener is removed from another listener, while we are still in the loop.
// So we have to COPY the listener array right now!
if ( ! event.listeners ) { event.listeners = self.__ngev.listeners[ event.name ].slice() ; }
// Increment globalData.recursions
globalData.recursions ++ ;
event.depth = self.__ngev.depth ++ ;
removedListeners = [] ;
try {
// Emit the event to all listeners!
for ( i = 0 , iMax = event.listeners.length ; i < iMax ; i ++ ) {
count ++ ;
NextGenEvents.emitToOneListener( event , event.listeners[ i ] , removedListeners ) ;
}
}
catch ( error ) {
// Catch error, just to decrement globalData.recursions, re-throw after that...
globalData.recursions -- ;
self.__ngev.depth -- ;
throw error ;
}
// Decrement globalData.recursions
globalData.recursions -- ;
if ( ! event.callback ) { self.__ngev.depth -- ; }
// Emit 'removeListener' after calling listeners
if ( removedListeners.length && self.__ngev.listeners.removeListener.length ) {
self.emit( 'removeListener' , removedListeners ) ;
}
// 'error' event is a special case: it should be listened for, or it will throw an error
if ( ! count ) {
if ( event.name === 'error' ) {
if ( event.args[ 0 ] ) { throw event.args[ 0 ] ; }
else { throw Error( "Uncaught, unspecified 'error' event." ) ; }
}
if ( event.callback ) { NextGenEvents.emitCallback( event ) ; }
}
// Leaving sync mode
event.sync = false ;
return event ;
} ;
/*
Spellcast-specific:
Send interruptible events with listener-priority across multiple emitters.
If an event is interrupted, all event are interrupted too.
It has limited feature-support: no state-event, no builtin-event (not even 'error').
*/
NextGenEvents.emitIntricatedEvents = function( array , callback ) {
var i , iMax , count = 0 , removedListeners ;
if ( ! Array.isArray( array ) ) {
throw new TypeError( '.emitCorrelatedEvents() argument should be an array' ) ;
}
var listenerEventRows = [] ,
context = {
nice: NextGenEvents.DESYNC ,
ready: true ,
status: NextGenEvents.CONTEXT_ENABLED ,
serial: true ,
scopes: {}
} ,
master = {
sync: false ,
nice: NextGenEvents.DESYNC ,
context ,
interrupt: null ,
listeners: listenerEventRows , // because we need eventMaster.listeners.length
listenersDone: 0 ,
depth: 0 ,
callback
} ;
array.forEach( eventParams => {
var event = NextGenEvents.createEvent( ... eventParams ) ;
event.master = master ;
if ( ! event.emitter.__ngev ) { NextGenEvents.init.call( event.emitter ) ; }
if ( ! event.emitter.__ngev.listeners[ event.name ] ) { event.emitter.__ngev.listeners[ event.name ] = [] ; }
event.listeners = event.emitter.__ngev.listeners[ event.name ].slice() ;
event.id = nextEventId ++ ;
//event.listenersDone = 0 ;
//event.nice = master.nice ;
event.listeners.forEach( listener => listenerEventRows.push( { event , listener } ) ) ;
} ) ;
// Sort listeners
listenerEventRows.sort( ( a , b ) => b.listener.priority - a.listener.priority ) ;
// Increment globalData.recursions
globalData.recursions ++ ;
removedListeners = [] ;
try {
// Emit the event to all listeners!
for ( i = 0 , iMax = listenerEventRows.length ; i < iMax ; i ++ ) {
count ++ ;
NextGenEvents.emitToOneListener( listenerEventRows[ i ].event , listenerEventRows[ i ].listener , removedListeners ) ;
}
}
catch ( error ) {
// Catch error, just to decrement globalData.recursions, re-throw after that...
globalData.recursions -- ;
throw error ;
}
// Decrement globalData.recursions
globalData.recursions -- ;
if ( ! count && master.callback ) { NextGenEvents.emitCallback( event ) ; }
// Leaving sync mode
master.sync = false ;
} ;
// If removedListeners is not given, then one-time listener emit the 'removeListener' event,
// if given: that's the caller business to do it
NextGenEvents.emitToOneListener = function( event , listener , removedListeners ) {
var self = event.emitter ,
eventMaster = event.master || event ,
context = event.master ? event.master.context : listener.context ,
contextScope , serial , currentNice , emitRemoveListener = false ;
if ( context ) {
// If the listener context is disabled...
if ( context.status === NextGenEvents.CONTEXT_DISABLED ) { return ; }
// The nice value for this listener...
currentNice = Math.max( eventMaster.nice , listener.nice , context.nice ) ;
serial = context.serial ;
contextScope = NextGenEvents.getContextScope( context , eventMaster.depth ) ;
}
else {
currentNice = Math.max( eventMaster.nice , listener.nice ) ;
}
if ( listener.once && self.__ngev.listeners[ event.name ] ) {
// We should remove the current listener RIGHT NOW because of recursive .emit() issues:
// one listener may eventually fire this very same event synchronously during the current loop.
// ALSO another listener may have called .removeAllListeners(), so we first check that there are still
// a listener array for this event.
self.__ngev.listeners[ event.name ] = self.__ngev.listeners[ event.name ].filter(
NextGenEvents.filterOutCallback.bind( undefined , listener )
) ;
if ( removedListeners ) { removedListeners.push( listener ) ; }
else { emitRemoveListener = true ; }
}
if ( context && ( context.status === NextGenEvents.CONTEXT_QUEUED || ! contextScope.ready ) ) {
// Almost all works should be done by .emit(), and little few should be done by .processScopeQueue()
contextScope.queue.push( { event: event , listener: listener , nice: currentNice } ) ;
}
else {
NextGenEvents.listenerWrapper( listener , event , contextScope , serial , currentNice ) ;
}
// Emit 'removeListener' after calling the listener
if ( emitRemoveListener && self.__ngev.listeners.removeListener.length ) {
self.emit( 'removeListener' , [ listener ] ) ;
}
} ;
NextGenEvents.emitCallback = function( event ) {
var callback ;
if ( event.master ) {
callback = event.master.callback ;
delete event.master.callback ;
if ( event.master.sync ) {
nextTick( () => callback( event.master.interrupt , event ) ) ;
}
else {
callback( event.master.interrupt , event ) ;
}
return ;
}
callback = event.callback ;
delete event.callback ;
if ( event.sync && event.emitter.__ngev.nice !== NextGenEvents.SYNC ) {
// Force desync if global nice value is not SYNC
event.emitter.__ngev.desync( () => {
event.emitter.__ngev.depth -- ;
callback( event.interrupt , event ) ;
} ) ;
}
else {
event.emitter.__ngev.depth -- ;
callback( event.interrupt , event ) ;
}
} ;
NextGenEvents.prototype.listeners = function( eventName ) {
if ( ! eventName || typeof eventName !== 'string' ) { throw new TypeError( ".listeners(): argument #0 should be a non-empty string" ) ; }
if ( ! this.__ngev ) { NextGenEvents.init.call( this ) ; }
if ( ! this.__ngev.listeners[ eventName ] ) { this.__ngev.listeners[ eventName ] = [] ; }
// Do not return the array, shallow copy it
return this.__ngev.listeners[ eventName ].slice() ;
} ;
NextGenEvents.listenerCount = function( emitter , eventName ) {
if ( ! emitter || ! ( emitter instanceof NextGenEvents ) ) { throw new TypeError( ".listenerCount(): argument #0 should be an instance of NextGenEvents" ) ; }
return emitter.listenerCount( eventName ) ;
} ;
NextGenEvents.prototype.listenerCount = function( eventName ) {
if ( ! eventName || typeof eventName !== 'string' ) { throw new TypeError( ".listenerCount(): argument #1 should be a non-empty string" ) ; }
if ( ! this.__ngev || ! this.__ngev.listeners[ eventName ] ) { return 0 ; }
return this.__ngev.listeners[ eventName ].length ;
} ;
NextGenEvents.prototype.setNice = function( nice ) {
if ( ! this.__ngev ) { NextGenEvents.init.call( this ) ; }
this.__ngev.nice = Math.floor( + nice || 0 ) ;
} ;
NextGenEvents.prototype.desyncUseNextTick = function( useNextTick ) {
if ( ! this.__ngev ) { NextGenEvents.init.call( this ) ; }
this.__ngev.desync = useNextTick ? nextTick : setImmediate ;
} ;
NextGenEvents.prototype.setInterruptible = function( isInterruptible ) {
if ( ! this.__ngev ) { NextGenEvents.init.call( this ) ; }
this.__ngev.interruptible = !! isInterruptible ;
} ;
NextGenEvents.prototype.setListenerPriority = function( hasListenerPriority ) {
if ( ! this.__ngev ) { NextGenEvents.init.call( this ) ; }
this.__ngev.hasListenerPriority = !! hasListenerPriority ;
} ;
// Make two objects share the same event bus
NextGenEvents.share = function( source , target ) {
if ( ! ( source instanceof NextGenEvents ) || ! ( target instanceof NextGenEvents ) ) {
throw new TypeError( 'NextGenEvents.share() arguments should be instances of NextGenEvents' ) ;
}
if ( ! source.__ngev ) { NextGenEvents.init.call( source ) ; }
Object.defineProperty( target , '__ngev' , {
configurable: true ,
value: source.__ngev
} ) ;
} ;
NextGenEvents.reset = function( emitter ) {
Object.defineProperty( emitter , '__ngev' , {
configurable: true ,
value: null
} ) ;
} ;
NextGenEvents.prototype.getMaxListeners = function() {
if ( ! this.__ngev ) { NextGenEvents.init.call( this ) ; }
return this.__ngev.maxListeners ;
} ;
NextGenEvents.prototype.setMaxListeners = function( n ) {
if ( ! this.__ngev ) { NextGenEvents.init.call( this ) ; }
this.__ngev.maxListeners = typeof n === 'number' && ! Number.isNaN( n ) ? Math.floor( n ) : NextGenEvents.defaultMaxListeners ;
return this ;
} ;
// Sometime useful as a no-op callback...
NextGenEvents.noop = () => undefined ;
/* Next Gen feature: states! */
// .defineStates( exclusiveState1 , [exclusiveState2] , [exclusiveState3] , ... )
NextGenEvents.prototype.defineStates = function( ... states ) {
if ( ! this.__ngev ) { NextGenEvents.init.call( this ) ; }
states.forEach( ( state ) => {
this.__ngev.states[ state ] = null ;
this.__ngev.stateGroups[ state ] = states ;
} ) ;
} ;
NextGenEvents.prototype.hasState = function( state ) {
if ( ! this.__ngev ) { NextGenEvents.init.call( this ) ; }
return !! this.__ngev.states[ state ] ;
} ;
NextGenEvents.prototype.getAllStates = function() {
if ( ! this.__ngev ) { NextGenEvents.init.call( this ) ; }
return Object.keys( this.__ngev.states ).filter( e => this.__ngev.states[ e ] ) ;
} ;
/* Next Gen feature: groups! */
NextGenEvents.groupAddListener = function( emitters , eventName , fn , options ) {
// Manage arguments
if ( typeof fn !== 'function' ) { options = fn ; fn = undefined ; }
if ( ! options || typeof options !== 'object' ) { options = {} ; }
fn = fn || options.fn ;
delete options.fn ;
// Preserve the listener ID, so groupRemoveListener() will work as expected
options.id = options.id || fn ;
emitters.forEach( ( emitter ) => {
emitter.addListener( eventName , fn.bind( undefined , emitter ) , options ) ;
} ) ;
} ;
NextGenEvents.groupOn = NextGenEvents.groupAddListener ;
// Once per emitter
NextGenEvents.groupOnce = function( emitters , eventName , fn , options ) {
if ( fn && typeof fn === 'object' ) { fn.once = true ; }
else if ( options && typeof options === 'object' ) { options.once = true ; }
else { options = { once: true } ; }
return this.groupAddListener( emitters , eventName , fn , options ) ;
} ;
// A Promise-returning .groupOnce() variant, it returns an array with only the first arg for each emitter's event
NextGenEvents.groupWaitFor = function( emitters , eventName ) {
return Promise.all( emitters.map( emitter => emitter.waitFor( eventName ) ) ) ;
} ;
// A Promise-returning .groupOnce() variant, it returns an array of array for each emitter's event
NextGenEvents.groupWaitForAll = function( emitters , eventName ) {
return Promise.all( emitters.map( emitter => emitter.waitForAll( eventName ) ) ) ;
} ;
// Globally once, only one event could be emitted, by the first emitter to emit
NextGenEvents.groupOnceFirst = function( emitters , eventName , fn , options ) {
var fnWrapper , triggered = false ;
// Manage arguments
if ( typeof fn !== 'function' ) { options = fn ; fn = undefined ; }
if ( ! options || typeof options !== 'object' ) { options = {} ; }
fn = fn || options.fn ;
delete options.fn ;
// Preserve the listener ID, so groupRemoveListener() will work as expected
options.id = options.id || fn ;
fnWrapper = ( ... args ) => {
if ( triggered ) { return ; }
triggered = true ;
NextGenEvents.groupRemoveListener( emitters , eventName , options.id ) ;
fn( ... args ) ;
} ;
emitters.forEach( ( emitter ) => {
emitter.once( eventName , fnWrapper.bind( undefined , emitter ) , options ) ;
} ) ;
} ;
// A Promise-returning .groupOnceFirst() variant, only the first arg is returned
NextGenEvents.groupWaitForFirst = function( emitters , eventName ) {
return new Promise( resolve => {
NextGenEvents.groupOnceFirst( emitters , eventName , ( firstArg ) => resolve( firstArg ) ) ;
} ) ;
} ;
// A Promise-returning .groupOnceFirst() variant, all args are returned as an array
NextGenEvents.groupWaitForFirstAll = function( emitters , eventName ) {
return new Promise( resolve => {
NextGenEvents.groupOnceFirst( emitters , eventName , ( ... args ) => resolve( args ) ) ;
} ) ;
} ;
// Globally once, only one event could be emitted, by the last emitter to emit
NextGenEvents.groupOnceLast = function( emitters , eventName , fn , options ) {
var fnWrapper , triggered = false , count = emitters.length ;
// Manage arguments
if ( typeof fn !== 'function' ) { options = fn ; fn = undefined ; }
if ( ! options || typeof options !== 'object' ) { options = {} ; }
fn = fn || options.fn ;
delete options.fn ;
// Preserve the listener ID, so groupRemoveListener() will work as expected
options.id = options.id || fn ;
fnWrapper = ( ... args ) => {
if ( triggered ) { return ; }
if ( -- count ) { return ; }
// So this is the last emitter...
triggered = true ;
// No need to remove listeners: there are already removed anyway
//NextGenEvents.groupRemoveListener( emitters , eventName , options.id ) ;
fn( ... args ) ;
} ;
emitters.forEach( ( emitter ) => {
emitter.once( eventName , fnWrapper.bind( undefined , emitter ) , options ) ;
} ) ;
} ;
// A Promise-returning .groupGlobalWaitFor() variant, only the first arg is returned
NextGenEvents.groupWaitForLast = function( emitters , eventName ) {
return new Promise( resolve => {
NextGenEvents.groupOnceLast( emitters , eventName , ( firstArg ) => resolve( firstArg ) ) ;
} ) ;
} ;
// A Promise-returning .groupGlobalWaitFor() variant, all args are returned as an array
NextGenEvents.groupWaitForLastAll = function( emitters , eventName ) {
return new Promise( resolve => {
NextGenEvents.groupOnceLast( emitters , eventName , ( ... args ) => resolve( args ) ) ;
} ) ;
} ;
NextGenEvents.groupRemoveListener = function( emitters , eventName , id ) {
emitters.forEach( ( emitter ) => {
emitter.removeListener( eventName , id ) ;
} ) ;
} ;
NextGenEvents.groupOff = NextGenEvents.groupRemoveListener ;
NextGenEvents.groupRemoveAllListeners = function( emitters , eventName ) {
emitters.forEach( ( emitter ) => {
emitter.removeAllListeners( eventName ) ;
} ) ;
} ;
NextGenEvents.groupEmit = function( emitters , ... args ) {
var eventName , nice , argStart = 1 , argEnd , count = emitters.length ,
callback , callbackWrapper , callbackTriggered = false ;
if ( typeof args[ args.length - 1 ] === 'function' ) {
argEnd = -1 ;
callback = args[ args.length - 1 ] ;
callbackWrapper = ( interruption ) => {
if ( callbackTriggered ) { return ; }
if ( interruption ) {
callbackTriggered = true ;
callback( interruption ) ;
}
else if ( ! -- count ) {
callbackTriggered = true ;
callback() ;
}
} ;
}
if ( typeof args[ 0 ] === 'number' ) {
argStart = 2 ;
nice = typeof args[ 0 ] ;
}
eventName = args[ argStart - 1 ] ;
args = args.slice( argStart , argEnd ) ;
emitters.forEach( ( emitter ) => {
NextGenEvents.emitEvent( {
emitter: emitter ,
name: eventName ,
args: args ,
nice: nice ,
callback: callbackWrapper
} ) ;
} ) ;
} ;
NextGenEvents.groupWaitForEmit = function( emitters , ... args ) {
return new Promise( resolve => {
NextGenEvents.groupEmit( emitters , ... args , ( interrupt ) => resolve( interrupt ) ) ;
} ) ;
} ;
NextGenEvents.groupDefineStates = function( emitters , ... args ) {
emitters.forEach( ( emitter ) => {
emitter.defineStates( ... args ) ;
} ) ;
} ;
// Bad names, but since they make their way through the API documentation,
// it should be kept for backward compatibility, but they are DEPRECATED.
NextGenEvents.groupGlobalOnce = NextGenEvents.groupOnceFirst ;
NextGenEvents.groupGlobalOnceAll = NextGenEvents.groupOnceLast ;
/* Next Gen feature: contexts! */
NextGenEvents.CONTEXT_ENABLED = 0 ;
NextGenEvents.CONTEXT_DISABLED = 1 ;
NextGenEvents.CONTEXT_QUEUED = 2 ;
NextGenEvents.prototype.addListenerContext = function( contextName , options ) {
if ( ! this.__ngev ) { NextGenEvents.init.call( this ) ; }
if ( ! contextName || typeof contextName !== 'string' ) { throw new TypeError( ".addListenerContext(): argument #0 should be a non-empty string" ) ; }
if ( ! options || typeof options !== 'object' ) { options = {} ; }
var context = this.__ngev.contexts[ contextName ] ;
if ( ! context ) {
context = this.__ngev.contexts[ contextName ] = {
nice: NextGenEvents.SYNC ,
ready: true ,
status: NextGenEvents.CONTEXT_ENABLED ,
serial: false ,
scopes: {}
} ;
}
if ( options.nice !== undefined ) { context.nice = Math.floor( options.nice ) ; }
if ( options.status !== undefined ) { context.status = options.status ; }
if ( options.serial !== undefined ) { context.serial = !! options.serial ; }
return context ;
} ;
NextGenEvents.prototype.getListenerContext = function( contextName ) {
return this.__ngev.contexts[ contextName ] ;
} ;
NextGenEvents.getContextScope = function( context , scopeName ) {
var scope = context.scopes[ scopeName ] ;
if ( ! scope ) {
scope = context.scopes[ scopeName ] = {
ready: true ,
queue: []
} ;
}
return scope ;
} ;
NextGenEvents.prototype.disableListenerContext = function( contextName ) {
if ( ! this.__ngev ) { NextGenEvents.init.call( this ) ; }
if ( ! contextName || typeof contextName !== 'string' ) { throw new TypeError( ".disableListenerContext(): argument #0 should be a non-empty string" ) ; }
if ( ! this.__ngev.contexts[ contextName ] ) { this.addListenerContext( contextName ) ; }
this.__ngev.contexts[ contextName ].status = NextGenEvents.CONTEXT_DISABLED ;
return this ;
} ;
NextGenEvents.prototype.enableListenerContext = function( contextName ) {
if ( ! this.__ngev ) { NextGenEvents.init.call( this ) ; }
if ( ! contextName || typeof contextName !== 'string' ) { throw new TypeError( ".enableListenerContext(): argument #0 should be a non-empty string" ) ; }
if ( ! this.__ngev.contexts[ contextName ] ) { this.addListenerContext( contextName ) ; }
var context = this.__ngev.contexts[ contextName ] ;
context.status = NextGenEvents.CONTEXT_ENABLED ;
Object.values( context.scopes ).forEach( contextScope => {
if ( contextScope.queue.length > 0 ) { NextGenEvents.processScopeQueue( contextScope , context.serial ) ; }
} ) ;
return this ;
} ;
NextGenEvents.prototype.queueListenerContext = function( contextName ) {
if ( ! this.__ngev ) { NextGenEvents.init.call( this ) ; }
if ( ! contextName || typeof contextName !== 'string' ) { throw new TypeError( ".queueListenerContext(): argument #0 should be a non-empty string" ) ; }
if ( ! this.__ngev.contexts[ contextName ] ) { this.addListenerContext( contextName ) ; }
this.__ngev.contexts[ contextName ].status = NextGenEvents.CONTEXT_QUEUED ;
return this ;
} ;
NextGenEvents.prototype.serializeListenerContext = function( contextName , value ) {
if ( ! this.__ngev ) { NextGenEvents.init.call( this ) ; }
if ( ! contextName || typeof contextName !== 'string' ) { throw new TypeError( ".serializeListenerContext(): argument #0 should be a non-empty string" ) ; }
if ( ! this.__ngev.contexts[ contextName ] ) { this.addListenerContext( contextName ) ; }
this.__ngev.contexts[ contextName ].serial = value === undefined ? true : !! value ;
return this ;
} ;
NextGenEvents.prototype.setListenerContextNice = function( contextName , nice ) {
if ( ! this.__ngev ) { NextGenEvents.init.call( this ) ; }
if ( ! contextName || typeof contextName !== 'string' ) { throw new TypeError( ".setListenerContextNice(): argument #0 should be a non-empty string" ) ; }
if ( ! this.__ngev.contexts[ contextName ] ) { this.addListenerContext( contextName ) ; }
this.__ngev.contexts[ contextName ].nice = Math.floor( nice ) ;
return this ;
} ;
NextGenEvents.prototype.destroyListenerContext = function( contextName ) {
var i , length , context , eventName , newListeners , removedListeners = [] ;
if ( ! contextName || typeof contextName !== 'string' ) { throw new TypeError( ".disableListenerContext(): argument #0 should be a non-empty string" ) ; }
if ( ! this.__ngev ) { NextGenEvents.init.call( this ) ; }
context = this.__ngev.contexts[ contextName ] ;
if ( ! context ) { return ; }
for ( eventName in this.__ngev.listeners ) {
newListeners = null ;
length = this.__ngev.listeners[ eventName ].length ;
for ( i = 0 ; i < length ; i ++ ) {
if ( this.__ngev.listeners[ eventName ][ i ].context === context ) {
newListeners = [] ;
removedListeners.push( this.__ngev.listeners[ eventName ][ i ] ) ;
}
else if ( newListeners ) {
newListeners.push( this.__ngev.listeners[ eventName ][ i ] ) ;
}
}
if ( newListeners ) { this.__ngev.listeners[ eventName ] = newListeners ; }
}
delete this.__ngev.contexts[ contextName ] ;
if ( removedListeners.length && this.__ngev.listeners.removeListener.length ) {
this.emit( 'removeListener' , removedListeners ) ;
}
return this ;
} ;
NextGenEvents.processScopeQueue = function( contextScope , serial , isCompletionCallback ) {
var job , event , eventMaster , emitter ;
if ( isCompletionCallback ) { contextScope.ready = true ; }
// Increment recursion
globalData.recursions ++ ;
while ( contextScope.ready && contextScope.queue.length ) {
job = contextScope.queue.shift() ;
event = job.event ;
eventMaster = event.master || event ;
emitter = event.emitter ;
// This event has been interrupted, drop it now!
if ( eventMaster.interrupt ) { continue ; }
NextGenEvents.listenerWrapper( job.listener , event , contextScope , serial , job.nice ) ;
}
// Decrement recursion
globalData.recursions -- ;
} ;
// Backup for the AsyncTryCatch
NextGenEvents.on = NextGenEvents.prototype.on ;
NextGenEvents.once = NextGenEvents.prototype.once ;
NextGenEvents.off = NextGenEvents.prototype.off ;
if ( global.AsyncTryCatch ) {
NextGenEvents.prototype.asyncTryCatchId = global.AsyncTryCatch.NextGenEvents.length ;
global.AsyncTryCatch.NextGenEvents.push( NextGenEvents ) ;
if ( global.AsyncTryCatch.substituted ) {
global.AsyncTryCatch.substitute() ;
}
}
// Load Proxy AT THE END (circular require)
NextGenEvents.Proxy = require( './Proxy.js' ) ;
}).call(this,require('_process'),typeof global !== "undefined" ? global : typeof self !== "undefined" ? self : typeof window !== "undefined" ? window : {})
},{"../package.json":5,"./Proxy.js":2,"_process":4}],2:[function(require,module,exports){
/*
Next-Gen Events
Copyright (c) 2015 - 2021 Cédric Ronvel
The MIT License (MIT)
Permission is hereby granted, free of charge, to any person obtaining a copy
of this software and associated documentation files (the "Software"), to deal
in the Software without restriction, including without limitation the rights
to use, copy, modify, merge, publish, distribute, sublicense, and/or sell
copies of the Software, and to permit persons to whom the Software is
furnished to do so, subject to the following conditions:
The above copyright notice and this permission notice shall be included in all
copies or substantial portions of the Software.
THE SOFTWARE IS PROVIDED "AS IS", WITHOUT WARRANTY OF ANY KIND, EXPRESS OR
IMPLIED, INCLUDING BUT NOT LIMITED TO THE WARRANTIES OF MERCHANTABILITY,
FITNESS FOR A PARTICULAR PURPOSE AND NONINFRINGEMENT. IN NO EVENT SHALL THE
AUTHORS OR COPYRIGHT HOLDERS BE LIABLE FOR ANY CLAIM, DAMAGES OR OTHER
LIABILITY, WHETHER IN AN ACTION OF CONTRACT, TORT OR OTHERWISE, ARISING FROM,
OUT OF OR IN CONNECTION WITH THE SOFTWARE OR THE USE OR OTHER DEALINGS IN THE
SOFTWARE.
*/
"use strict" ;
function Proxy() {
this.localServices = {} ;
this.remoteServices = {} ;
this.nextAckId = 1 ;
}
module.exports = Proxy ;
var NextGenEvents = require( './NextGenEvents.js' ) ;
var MESSAGE_TYPE = 'NextGenEvents/message' ;
function noop() {}
// Backward compatibility
Proxy.create = ( ... args ) => new Proxy( ... args ) ;
// Add a local service accessible remotely
Proxy.prototype.addLocalService = function( id , emitter , options ) {
this.localServices[ id ] = LocalService.create( this , id , emitter , options ) ;
return this.localServices[ id ] ;
} ;
// Add a remote service accessible locally
Proxy.prototype.addRemoteService = function( id ) {
this.remoteServices[ id ] = RemoteService.create( this , id ) ;
return this.remoteServices[ id ] ;
} ;
// Destroy the proxy
Proxy.prototype.destroy = function() {
Object.keys( this.localServices ).forEach( ( id ) => {
this.localServices[ id ].destroy() ;
delete this.localServices[ id ] ;
} ) ;
Object.keys( this.remoteServices ).forEach( ( id ) => {
this.remoteServices[ id ].destroy() ;
delete this.remoteServices[ id ] ;
} ) ;
this.receive = this.send = noop ;
} ;
// Push an event message.
Proxy.prototype.push = function( message ) {
if (
message.__type !== MESSAGE_TYPE ||
! message.service || typeof message.service !== 'string' ||
! message.event || typeof message.event !== 'string' ||
! message.method
) {
return ;
}
switch ( message.method ) {
// Those methods target a remote service
case 'event' :
return this.remoteServices[ message.service ] && this.remoteServices[ message.service ].receiveEvent( message ) ;
case 'ackEmit' :
return this.remoteServices[ message.service ] && this.remoteServices[ message.service ].receiveAckEmit( message ) ;
// Those methods target a local service
case 'emit' :
return this.localServices[ message.service ] && this.localServices[ message.service ].receiveEmit( message ) ;
case 'listen' :
return this.localServices[ message.service ] && this.localServices[ message.service ].receiveListen( message ) ;
case 'ignore' :
return this.localServices[ message.service ] && this.localServices[ message.service ].receiveIgnore( message ) ;
case 'ackEvent' :
return this.localServices[ message.service ] && this.localServices[ message.service ].receiveAckEvent( message ) ;
default :
return ;
}
} ;
// This is the method to receive and decode data from the other side of the communication channel, most of time another proxy.
// In most case, this should be overwritten.
Proxy.prototype.receive = function( raw ) {
this.push( raw ) ;
} ;
// This is the method used to send data to the other side of the communication channel, most of time another proxy.
// This MUST be overwritten in any case.
Proxy.prototype.send = function() {
throw new Error( 'The send() method of the Proxy MUST be extended/overwritten' ) ;
} ;
/* Local Service */
function LocalService( proxy , id , emitter , options ) { return LocalService.create( proxy , id , emitter , options ) ; }
Proxy.LocalService = LocalService ;
LocalService.create = function( proxy , id , emitter , options ) {
var self = Object.create( LocalService.prototype , {
proxy: { value: proxy , enumerable: true } ,
id: { value: id , enumerable: true } ,
emitter: { value: emitter , writable: true , enumerable: true } ,
internalEvents: { value: Object.create( NextGenEvents.prototype ) , writable: true , enumerable: true } ,
events: { value: {} , enumerable: true } ,
canListen: { value: !! options.listen , writable: true , enumerable: true } ,
canEmit: { value: !! options.emit , writable: true , enumerable: true } ,
canAck: { value: !! options.ack , writable: true , enumerable: true } ,
canRpc: { value: !! options.rpc , writable: true , enumerable: true } ,
destroyed: { value: false , writable: true , enumerable: true }
} ) ;
return self ;
} ;
// Destroy a service
LocalService.prototype.destroy = function() {
Object.keys( this.events ).forEach( ( eventName ) => {
this.emitter.off( eventName , this.events[ eventName ] ) ;
delete this.events[ eventName ] ;
} ) ;
this.emitter = null ;
this.destroyed = true ;
} ;
// Remote want to emit on the local service
LocalService.prototype.receiveEmit = function( message ) {
if ( this.destroyed || ! this.canEmit || ( message.ack && ! this.canAck ) ) { return ; }
var event = {
emitter: this.emitter ,
name: message.event ,
args: message.args || []
} ;
if ( message.ack ) {
event.callback = ( interruption ) => {
this.proxy.send( {
__type: MESSAGE_TYPE ,
service: this.id ,
method: 'ackEmit' ,
ack: message.ack ,
event: message.event ,
interruption: interruption
} ) ;
} ;
}
NextGenEvents.emitEvent( event ) ;
} ;
// Remote want to listen to an event of the local service
LocalService.prototype.receiveListen = function( message ) {
if ( this.destroyed || ! this.canListen || ( message.ack && ! this.canAck ) ) { return ; }
if ( message.ack ) {
if ( this.events[ message.event ] ) {
if ( this.events[ message.event ].ack ) { return ; }
// There is already an event, but not featuring ack, remove that listener now
this.emitter.off( message.event , this.events[ message.event ] ) ;
}
this.events[ message.event ] = LocalService.forwardWithAck.bind( this ) ;
this.events[ message.event ].ack = true ;
this.emitter.on( message.event , this