const Argument = require('../argument/')
/**
* Class representing a command
*/
class Command {
/**
* Create a command
* @class
* @param {Object} data The command data
* @prop {String} data.name The command name
* @prop {String} data.desc The command description
* @prop {Object} [data.options={}] The command options
* @prop {Array<ArgData|String>} [data.options.args=[]] The arguments for the command (Can be an argument data object or just the name of the argument)
* @prop {String[]|String} [data.options.aliases=[]] Other names that trigger the command
* @prop {Object} [data.options.guide={}] Guide info displayed when a command guide is built with the agent
* @prop {Number} [data.options.guide.color=16777215] An integer color for the sidebar of the command guide
* @prop {Object[]} [data.options.guide.fields=[]] A list of field objects displayed in the guide
* @prop {Boolean} [data.options.restricted=false] Whether or not this command is restricted to admin only
* @prop {Boolean} [data.options.guildOnly=false] Whether or not this command can only be triggered in guilds as opposed to DMs
* @prop {Number} [data.options.authLevel=0] The minimum auth level required to execute this command
* @prop {CommandAction} data.action The command action
*/
constructor ({ name, desc, options = {}, action }) {
const {
args = [],
aliases = [],
guide = {},
restricted = false,
guildOnly = false,
authLevel
} = options
const {
color = 16777215, // White
fields = []
} = guide
/**
* Argument type prefixes for command info
* @private
* @type {Object}
* @prop {String} number
* @prop {String} user
* @prop {String} channel
*/
this._typePrefixes = {
number: '#',
user: '@',
channel: '[#]'
}
/**
* The command name
* @type {String}
*/
this.name = name.toLowerCase()
/**
* The command description
* @type {String}
*/
this.desc = desc
/**
* The options for the command
* @type {Object}
* @prop {Argument[]} args The arguments for the command
* @prop {String[]} aliases Other names that trigger the command
* @prop {Object} guide Guide info displayed when a command guide is built with the agent
* @prop {Number} guide.color An integer color for the sidebar of the command guide
* @prop {Object[]} guide.fields A list of field objects displayed in the guide
* @prop {Boolean} restricted Whether or not this command is restricted to admin only
* @prop {Boolean} guildOnly Whether or not this command can only be triggered in guilds as opposed to DMs
* @prop {Number} authLevel The minimum auth level required to execute this command
*/
this.options = {
args: args.map((arg) => new Argument(arg)),
aliases: (Array.isArray(aliases) ? aliases : [aliases]).map((a) => a.toLowerCase()),
guide: {
color,
fields
},
restricted,
guildOnly,
authLevel: parseInt(authLevel) || 0
}
/**
* The command action
* @type {CommandAction}
*/
this.action = action
}
/**
* Get the info of this command
* @type {String}
* @example '**name**|**alias** **<mandatory arg> (optional arg) (#number arg)** - *description*'
*/
get info () {
const name = `**${this.name}**` + this.options.aliases.reduce((accum, alias) => `${accum}|**${alias}**`, '')
const args = this.options.args.reduce((accum, arg, index) => {
const lastArg = index === this.options.args.length - 1
const content = (index ? accum : '') + (arg.mand ? '<' : '(') + `${this._typePrefixes[arg.type] || ''}${arg.name}` + (arg.mand ? '>' : ')') + (lastArg ? '' : arg.delim)
return content
}, '')
return `${this.options.restricted ? '~~' : ''}${name}${args ? ` **${args}**` : ''} - *${this.desc}*${this.options.restricted ? '~~' : ''}`
}
}
module.exports = Command
/**
* The command action
* @callback CommandAction
* @param {CommandData} data Data passed from the handler
* @returns {CommandResults[]|CommandResults|String[]|String} Data to respond with
*/
/**
* The object passed to a command action
* @typedef {Object} CommandData
* @prop {Agent} CommandData.agent The agent managing the bot
* @prop {Eris.Client} CommandData.client The Eris client
* @prop {Map<String, Command>} CommandData.commands The list of bot commands
* @prop {Map<String, Replacer>} CommandData.replacers The list of bot replacers
* @prop {Eris.Message} CommandData.msg The message sent by the user
* @prop {Array<String|Number|Eris.Channel|Eris.User>} CommandData.args The arguments supplied by the user
* @prop {Object} CommandData.attachments User-supplied data that is passed to commands
*/