'use strict'; const EventEmitter = require('node:events'); const { setTimeout, clearTimeout } = require('node:timers'); const { Collection } = require('@discordjs/collection'); const { TypeError } = require('../../errors'); const Util = require('../../util/Util'); /** * Filter to be applied to the collector. * @typedef {Function} CollectorFilter * @param {...*} args Any arguments received by the listener * @param {Collection} collection The items collected by this collector * @returns {boolean|Promise} */ /** * Options to be applied to the collector. * @typedef {Object} CollectorOptions * @property {CollectorFilter} [filter] The filter applied to this collector * @property {number} [time] How long to run the collector for in milliseconds * @property {number} [idle] How long to stop the collector after inactivity in milliseconds * @property {boolean} [dispose=false] Whether to dispose data when it's deleted */ /** * Abstract class for defining a new Collector. * @abstract */ class Collector extends EventEmitter { constructor(client, options = {}) { super(); /** * The client that instantiated this Collector * @name Collector#client * @type {Client} * @readonly */ Object.defineProperty(this, 'client', { value: client }); /** * The filter applied to this collector * @type {CollectorFilter} * @returns {boolean|Promise} */ this.filter = options.filter ?? (() => true); /** * The options of this collector * @type {CollectorOptions} */ this.options = options; /** * The items collected by this collector * @type {Collection} */ this.collected = new Collection(); /** * Whether this collector has finished collecting * @type {boolean} */ this.ended = false; /** * Timeout for cleanup * @type {?Timeout} * @private */ this._timeout = null; /** * Timeout for cleanup due to inactivity * @type {?Timeout} * @private */ this._idletimeout = null; if (typeof this.filter !== 'function') { throw new TypeError('INVALID_TYPE', 'options.filter', 'function'); } this.handleCollect = this.handleCollect.bind(this); this.handleDispose = this.handleDispose.bind(this); if (options.time) this._timeout = setTimeout(() => this.stop('time'), options.time).unref(); if (options.idle) this._idletimeout = setTimeout(() => this.stop('idle'), options.idle).unref(); } /** * Call this to handle an event as a collectable element. Accepts any event data as parameters. * @param {...*} args The arguments emitted by the listener * @returns {Promise} * @emits Collector#collect */ async handleCollect(...args) { const collect = await this.collect(...args); if (collect && (await this.filter(...args, this.collected))) { this.collected.set(collect, args[0]); /** * Emitted whenever an element is collected. * @event Collector#collect * @param {...*} args The arguments emitted by the listener */ this.emit('collect', ...args); if (this._idletimeout) { clearTimeout(this._idletimeout); this._idletimeout = setTimeout(() => this.stop('idle'), this.options.idle).unref(); } } this.checkEnd(); } /** * Call this to remove an element from the collection. Accepts any event data as parameters. * @param {...*} args The arguments emitted by the listener * @returns {Promise} * @emits Collector#dispose */ async handleDispose(...args) { if (!this.options.dispose) return; const dispose = this.dispose(...args); if (!dispose || !(await this.filter(...args)) || !this.collected.has(dispose)) return; this.collected.delete(dispose); /** * Emitted whenever an element is disposed of. * @event Collector#dispose * @param {...*} args The arguments emitted by the listener */ this.emit('dispose', ...args); this.checkEnd(); } /** * Returns a promise that resolves with the next collected element; * rejects with collected elements if the collector finishes without receiving a next element * @type {Promise} * @readonly */ get next() { return new Promise((resolve, reject) => { if (this.ended) { reject(this.collected); return; } const cleanup = () => { this.removeListener('collect', onCollect); this.removeListener('end', onEnd); }; const onCollect = item => { cleanup(); resolve(item); }; const onEnd = () => { cleanup(); reject(this.collected); // eslint-disable-line prefer-promise-reject-errors }; this.on('collect', onCollect); this.on('end', onEnd); }); } /** * Stops this collector and emits the `end` event. * @param {string} [reason='user'] The reason this collector is ending * @emits Collector#end */ stop(reason = 'user') { if (this.ended) return; if (this._timeout) { clearTimeout(this._timeout); this._timeout = null; } if (this._idletimeout) { clearTimeout(this._idletimeout); this._idletimeout = null; } this.ended = true; /** * Emitted when the collector is finished collecting. * @event Collector#end * @param {Collection} collected The elements collected by the collector * @param {string} reason The reason the collector ended */ this.emit('end', this.collected, reason); } /** * Options used to reset the timeout and idle timer of a {@link Collector}. * @typedef {Object} CollectorResetTimerOptions * @property {number} [time] How long to run the collector for (in milliseconds) * @property {number} [idle] How long to wait to stop the collector after inactivity (in milliseconds) */ /** * Resets the collector's timeout and idle timer. * @param {CollectorResetTimerOptions} [options] Options for resetting */ resetTimer({ time, idle } = {}) { if (this._timeout) { clearTimeout(this._timeout); this._timeout = setTimeout(() => this.stop('time'), time ?? this.options.time).unref(); } if (this._idletimeout) { clearTimeout(this._idletimeout); this._idletimeout = setTimeout(() => this.stop('idle'), idle ?? this.options.idle).unref(); } } /** * Checks whether the collector should end, and if so, ends it. * @returns {boolean} Whether the collector ended or not */ checkEnd() { const reason = this.endReason; if (reason) this.stop(reason); return Boolean(reason); } /** * Allows collectors to be consumed with for-await-of loops * @see {@link https://developer.mozilla.org/en-US/docs/Web/JavaScript/Reference/Statements/for-await...of} */ async *[Symbol.asyncIterator]() { const queue = []; const onCollect = (...item) => queue.push(item); this.on('collect', onCollect); try { while (queue.length || !this.ended) { if (queue.length) { yield queue.shift(); } else { // eslint-disable-next-line no-await-in-loop await new Promise(resolve => { const tick = () => { this.removeListener('collect', tick); this.removeListener('end', tick); return resolve(); }; this.on('collect', tick); this.on('end', tick); }); } } } finally { this.removeListener('collect', onCollect); } } toJSON() { return Util.flatten(this); } /* eslint-disable no-empty-function */ /** * The reason this collector has ended with, or null if it hasn't ended yet * @type {?string} * @readonly * @abstract */ get endReason() {} /** * Handles incoming events from the `handleCollect` function. Returns null if the event should not * be collected, or returns an object describing the data that should be stored. * @see Collector#handleCollect * @param {...*} args Any args the event listener emits * @returns {?(*|Promise)} Data to insert into collection, if any * @abstract */ collect() {} /** * Handles incoming events from the `handleDispose`. Returns null if the event should not * be disposed, or returns the key that should be removed. * @see Collector#handleDispose * @param {...*} args Any args the event listener emits * @returns {?*} Key to remove from the collection, if any * @abstract */ dispose() {} /* eslint-enable no-empty-function */ } module.exports = Collector;