const Argument = require('../argument/')
/**
* A class used for the awaiting of a criteria-matching message
*/
class Await {
/**
* Create an Await
* @class
* @param {Object} data The await data
* @prop {Object} [data.options={}] The options for the await
* @prop {Array<ArgData|String>} [data.options.args=[]] The arguments for the await (Can be an argument data object or just the name of the argument)
* @prop {CheckFunction} [data.options.check=() => true] The condition to be met for the await to trigger
* @prop {Number} [data.options.timeout=15000] How long until the await expires
* @prop {Boolean} [data.options.oneTime=false] Whether a non-triggering message cancels the await
* @prop {Boolean} [data.options.refreshOnUse=false] Whether the timeout for the await refreshes after a use
* @prop {CancelFunction} [data.options.onCancelFunction] A function to run once the await expires or is cancelled
* @prop {String} [data.options.user] The ID of the user to await the message from (By default, it's the user who triggered the command)
* @prop {String} [data.options.channel] The ID of the channel to await the message (By default, it's the channel the command was called in)
* @prop {Boolean} [data.options.shouldShift=false] Whether the command handler should shift out the first space-separated keyword for argument parsing or not
* @prop {Boolean} [data.options.requirePrefix=false] Whether the await requires the bot prefix to be triggered
* @prop {AwaitAction} data.action The await action
*/
constructor ({ options = {}, action }) {
const {
args = [],
check = () => true,
timeout = 15000,
oneTime = false,
refreshOnUse = false,
onCancelFunction,
user,
channel,
shouldShift = false,
requirePrefix = false
} = options
/**
* The options for the await
* @type {Object}
* @prop {Argument[]} args The arguments for the await
* @prop {CheckFunction} check The condition to be met for the await to trigger
* @prop {Number} timeout How long until the await expires
* @prop {Boolean} oneTime Whether a non-triggering message cancels the await
* @prop {Boolean} refreshOnUse Whether the timeout for the await refreshes after a use
* @prop {CancelFunction} onCancelFunction A function to run once the await expires or is cancelled
* @prop {String} user The ID of the user to await the message from (By default, it's the user who triggered the command)
* @prop {String} channel The channel to await the message (By default, it's the channel the command was called in)
* @prop {Boolean} shouldShift Whether the command handler should shift out the first space-separated keyword for argument parsing or not
* @prop {Boolean} requirePrefix Whether the await requires the bot prefix to be triggered
*/
this.options = {
args: args.map((arg) => new Argument(arg)),
check,
timeout,
oneTime,
refreshOnUse,
onCancelFunction,
user,
channel,
shouldShift,
requirePrefix
}
/**
* The await action
* @type {AwaitAction}
*/
this.action = action
}
/**
* Clear the await from its storage
* @returns {Await} The await that was cleared
*/
clear () {
if (!this._timer) throw Error('You have not started the timer yet!')
clearTimeout(this._timer)
this._awaitMap.delete(this._id)
if (this.options.onCancelFunction) this.options.onCancelFunction(this)
return this
}
/**
* Refresh the delete timer for the await
* @returns {Await} The await
*/
refresh () {
if (!this._timer) throw Error('You have not started the timer yet!')
clearTimeout(this._timer)
/**
* The delete timer for the await
* @private
* @type {Timeout}
*/
this._timer = setTimeout(this.clear, this.options.timeout)
return this
}
/**
* Start the delete timer for the await
* @param {Object} data Further await data
* @prop {String} data.id The ID of the await in its parent map
* @prop {Map<String, Await>} data.awaitMap The map that the await is stored in
* @prop {Eris.Message} data.triggerResponse The response to the command that initiated the await
* @returns {Await} The await
*/
startTimer ({ id, awaitMap, triggerResponse }) {
/**
* The ID of the await.
* @private
* @type {String}
*/
this._id = id
/**
* The map that the await is stored in
* @private
* @type {Map<String, Await>}
*/
this._awaitMap = awaitMap
/**
* The response to the command that initiated the await
* @type {Eris.Message}
*/
this.triggerResponse = triggerResponse
/**
* The delete timer for the await
* @private
* @type {Timeout}
*/
this._timer = setTimeout(() => this.clear(), this.options.timeout)
return this
}
}
module.exports = Await
/**
* The condition to be met for the await to trigger
* @callback CheckFunction
* @param {Eris.Message} msg The message object, but if a prefix is required, the message content will not include the prefix
* @param {String} content The message content (processed by command handler (DO NOT USE `msg.content`))
* @returns {Boolean} Whether the message meets the await requirements
*/
/**
* The code that runs once an await is cancelled
* @callback CancelFunction
* @param {Await} wait The await
*/
/**
* The await action.
* @callback AwaitAction
* @param {AwaitData} data Data passed from the handler
* @returns {CommandResults[]|CommandResults|String[]|String} Data to respond with
*/
/**
* The object passed to a command action
* @typedef {Object} AwaitData
* @prop {Agent} AwaitData.agent The agent managing the bot
* @prop {Eris.Client} AwaitData.client The Eris client
* @prop {Map<String, Command>} AwaitData.commands The list of bot commands
* @prop {Map<String, Replacer>} AwaitData.replacers The list of bot replacers
* @prop {Eris.Message} AwaitData.msg The message sent by the user
* @prop {String[]} AwaitData.args The arguments supplied by the user
* @prop {Object} AwaitData.attachments User-supplied data that is passed to commands
* @prop {Eris.Message} AwaitData.triggerResponse The bot's response to the message that initiated the await
*/