HJ_Server/src/event.ts
2023-12-13 20:51:17 +08:00

322 lines
8.8 KiB
TypeScript

/*
var emitter = EventEmitter.create();
emitter.emit('my_event', args);
emitter.on('my_event', function(args){
......
});
http://nodejs.org/api/events.html
*/
const isUndefined = function (arg) {
return arg === void 0;
};
const isObject = function (arg) {
return typeof arg === "object" && arg !== null;
};
const isFunction = function (arg) {
return typeof arg === "function";
};
function EventEmitter(maxListeners) {
this._events = this._events || {};
this._maxListeners = maxListeners || undefined;
}
// Backwards-compat with node 0.10.x
EventEmitter.EventEmitter = EventEmitter;
EventEmitter.prototype._events = undefined;
EventEmitter.prototype._maxListeners = undefined;
// By default EventEmitters will print a warning if more than 10 listeners are
// added to it. This is a useful default which helps finding memory leaks.
EventEmitter.defaultMaxListeners = 10;
// Obviously not all Emitters should be limited to 10. This function allows
// that to be increased. Set to zero for unlimited.
EventEmitter.prototype.setMaxListeners = function (n) {
if (isNaN(n) || n < 0)
throw TypeError("n must be a positive number");
this._maxListeners = n;
return this;
};
EventEmitter.prototype.emit = function (type) {
var er, handler, len, args, i, listeners;
if (!this._events)
this._events = {};
// If there is no 'error' event listener then throw.
if (type === "error" && !this._events.error) {
er = arguments[1];
if (er instanceof Error) {
throw er; // Unhandled 'error' event
} else {
throw Error("Uncaught, unspecified \"error\" event.");
}
return false;
}
handler = this._events[type];
if (isUndefined(handler))
return false;
if (isFunction(handler)) {
var __caller = handler.__caller || this;
switch (arguments.length) {
// fast cases
case 1:
handler.call(__caller);
break;
case 2:
handler.call(__caller, arguments[1]);
break;
case 3:
handler.call(__caller, arguments[1], arguments[2]);
break;
// slower
default:
len = arguments.length;
args = new Array(len - 1);
for (i = 1; i < len; i++)
args[i - 1] = arguments[i];
handler.apply(__caller, args);
}
} else if (isObject(handler)) {
len = arguments.length;
args = new Array(len - 1);
for (i = 1; i < len; i++)
args[i - 1] = arguments[i];
listeners = handler.slice();
len = listeners.length;
for (i = 0; i < len; i++) {
var __caller = listeners[i].__caller || this;
listeners[i].apply(__caller, args);
}
}
return true;
};
EventEmitter.prototype.addListener = function (type, listener, caller, returnListenter) {
if (!isFunction(listener))
throw TypeError("listener must be a function");
listener.__caller = caller;
if (!this._events)
this._events = {};
// To avoid recursion in the case that type === "newListener"! Before
// adding it to the listeners, first emit "newListener".
if (this._events.newListener)
this.emit("newListener", type,
isFunction(listener.listener) ?
listener.listener : listener);
if (!this._events[type])
// Optimize the case of one listener. Don't need the extra array object.
this._events[type] = listener;
else if (isObject(this._events[type]))
// If we've already got an array, just append.
this._events[type].push(listener);
else
// Adding the second element, need to change to array.
this._events[type] = [this._events[type], listener];
// Check for listener leak
if (isObject(this._events[type]) && !this._events[type].warned) {
var m;
if (!isUndefined(this._maxListeners)) {
m = this._maxListeners;
} else {
m = EventEmitter.defaultMaxListeners;
}
if (m && m > 0 && this._events[type].length > m) {
this._events[type].warned = true;
console.error("(node) warning: possible EventEmitter memory " +
"leak detected. %d listeners added. " +
"Use emitter.setMaxListeners() to increase limit.",
this._events[type].length);
//console.trace();
}
}
if (returnListenter) {
return listener;
} else {
return this;
}
};
EventEmitter.prototype.on = EventEmitter.prototype.addListener;
EventEmitter.prototype.once = function (type, listener, caller) {
if (!isFunction(listener))
throw TypeError("listener must be a function");
var fired = false;
var that = this;
function g() {
that.removeListener(type, g);
if (!fired) {
fired = true;
listener.apply(caller || that, arguments);
}
}
g.listener = listener;
this.on(type, g, caller);
return this;
};
EventEmitter.prototype.onnp = function (type, listener, key) {
if (!isFunction(listener))
throw TypeError("listener must be a function");
if (!this._noRepeatKey) this._noRepeatKey = {};
var _key = key;
if (_key == null) _key = type + "_" + listener.toString();
if (this._noRepeatKey[_key]) {
this.removeListener(type, this._noRepeatKey[_key]);
}
this._noRepeatKey[_key] = listener;
this.on(type, listener);
return this;
};
// emits a 'removeListener' event iff the listener was removed
EventEmitter.prototype.removeListener = EventEmitter.prototype.off = function (type, listener, caller) {
var list, position, length, i;
if (!isFunction(listener)) {
throw TypeError("listener must be a function");
//return this;
}
if (!this._events || !this._events[type])
return this;
list = this._events[type];
length = list.length;
position = -1;
if (list === listener || (isFunction(list.listener) && list.listener === listener)) {
delete this._events[type];
if (this._events.removeListener)
this.emit("removeListener", type, listener);
} else if (isObject(list)) {
for (i = length; i-- > 0;) {
if (list[i] === listener ||
(list[i].listener && list[i].listener === listener) || list[i].__caller == caller) {
position = i;
break;
}
}
if (position < 0)
return this;
if (list.length === 1) {
list.length = 0;
delete this._events[type];
} else {
list.splice(position, 1);
}
if (this._events.removeListener)
this.emit("removeListener", type, listener);
}
return this;
};
EventEmitter.prototype.removeAllListenersExcept = function (type) {
for (var key in this._events) {
if (key === type) continue;
this.removeAllListeners(key);
}
};
EventEmitter.prototype.removeAllListeners = function (type) {
var key, listeners;
if (!this._events)
return this;
// not listening for removeListener, no need to emit
if (!this._events.removeListener) {
if (arguments.length === 0)
this._events = {};
else if (this._events[type])
delete this._events[type];
return this;
}
// emit removeListener for all listeners on all events
if (arguments.length === 0) {
for (key in this._events) {
if (key === "removeListener") continue;
this.removeAllListeners(key);
}
this.removeAllListeners("removeListener");
this._events = {};
return this;
}
listeners = this._events[type];
if (isFunction(listeners)) {
this.removeListener(type, listeners);
} else if (Array.isArray(listeners)) {
// LIFO order
while (listeners.length)
this.removeListener(type, listeners[listeners.length - 1]);
}
delete this._events[type];
this._noRepeatKey = {};
return this;
};
EventEmitter.prototype.listeners = function (type) {
var ret;
if (!this._events || !this._events[type])
ret = [];
else if (isFunction(this._events[type]))
ret = [this._events[type]];
else
ret = this._events[type].slice();
return ret;
};
EventEmitter.prototype.debug = function () {
console.log(this._events);
};
export const MyEvent = EventEmitter as any as {
(maxListeners: number): void;
on(type: any, callback: any): any;
once(type: any, callback: any): any;
off(type: any, callback: any, caller?: any): void;
emit(type: any, ...args: any[]): void;
debug(): any;
removeAllListeners(type?:any):void;
};