2022-03-19 10:37:45 +00:00
|
|
|
'use strict';
|
|
|
|
|
|
|
|
const { TypeError } = require('../errors');
|
|
|
|
|
|
|
|
/**
|
|
|
|
* A resolver for command interaction options.
|
|
|
|
*/
|
|
|
|
class CommandInteractionOptionResolver {
|
|
|
|
constructor(client, options, resolved) {
|
|
|
|
/**
|
|
|
|
* The client that instantiated this.
|
|
|
|
* @name CommandInteractionOptionResolver#client
|
|
|
|
* @type {Client}
|
|
|
|
* @readonly
|
|
|
|
*/
|
|
|
|
Object.defineProperty(this, 'client', { value: client });
|
|
|
|
|
|
|
|
/**
|
|
|
|
* The name of the subcommand group.
|
|
|
|
* @type {?string}
|
|
|
|
* @private
|
|
|
|
*/
|
|
|
|
this._group = null;
|
|
|
|
|
|
|
|
/**
|
|
|
|
* The name of the subcommand.
|
|
|
|
* @type {?string}
|
|
|
|
* @private
|
|
|
|
*/
|
|
|
|
this._subcommand = null;
|
|
|
|
|
|
|
|
/**
|
|
|
|
* The bottom-level options for the interaction.
|
|
|
|
* If there is a subcommand (or subcommand and group), this is the options for the subcommand.
|
|
|
|
* @type {CommandInteractionOption[]}
|
|
|
|
* @private
|
|
|
|
*/
|
|
|
|
this._hoistedOptions = options;
|
|
|
|
|
|
|
|
// Hoist subcommand group if present
|
2022-03-24 10:55:32 +00:00
|
|
|
if (this._hoistedOptions[0]?.type === 'SUB_COMMAND_GROUP') {
|
2022-03-19 10:37:45 +00:00
|
|
|
this._group = this._hoistedOptions[0].name;
|
|
|
|
this._hoistedOptions = this._hoistedOptions[0].options ?? [];
|
|
|
|
}
|
|
|
|
// Hoist subcommand if present
|
2022-03-24 10:55:32 +00:00
|
|
|
if (this._hoistedOptions[0]?.type === 'SUB_COMMAND') {
|
2022-03-19 10:37:45 +00:00
|
|
|
this._subcommand = this._hoistedOptions[0].name;
|
|
|
|
this._hoistedOptions = this._hoistedOptions[0].options ?? [];
|
|
|
|
}
|
|
|
|
|
|
|
|
/**
|
|
|
|
* The interaction options array.
|
|
|
|
* @name CommandInteractionOptionResolver#data
|
|
|
|
* @type {ReadonlyArray<CommandInteractionOption>}
|
|
|
|
* @readonly
|
|
|
|
*/
|
|
|
|
Object.defineProperty(this, 'data', { value: Object.freeze([...options]) });
|
|
|
|
|
|
|
|
/**
|
|
|
|
* The interaction resolved data
|
|
|
|
* @name CommandInteractionOptionResolver#resolved
|
|
|
|
* @type {Readonly<CommandInteractionResolvedData>}
|
|
|
|
*/
|
|
|
|
Object.defineProperty(this, 'resolved', { value: Object.freeze(resolved) });
|
|
|
|
}
|
|
|
|
|
|
|
|
/**
|
|
|
|
* Gets an option by its name.
|
|
|
|
* @param {string} name The name of the option.
|
|
|
|
* @param {boolean} [required=false] Whether to throw an error if the option is not found.
|
|
|
|
* @returns {?CommandInteractionOption} The option, if found.
|
|
|
|
*/
|
|
|
|
get(name, required = false) {
|
|
|
|
const option = this._hoistedOptions.find(opt => opt.name === name);
|
|
|
|
if (!option) {
|
|
|
|
if (required) {
|
|
|
|
throw new TypeError('COMMAND_INTERACTION_OPTION_NOT_FOUND', name);
|
|
|
|
}
|
|
|
|
return null;
|
|
|
|
}
|
|
|
|
return option;
|
|
|
|
}
|
|
|
|
|
|
|
|
/**
|
|
|
|
* Gets an option by name and property and checks its type.
|
|
|
|
* @param {string} name The name of the option.
|
|
|
|
* @param {ApplicationCommandOptionType} type The type of the option.
|
|
|
|
* @param {string[]} properties The properties to check for for `required`.
|
|
|
|
* @param {boolean} required Whether to throw an error if the option is not found.
|
|
|
|
* @returns {?CommandInteractionOption} The option, if found.
|
|
|
|
* @private
|
|
|
|
*/
|
|
|
|
_getTypedOption(name, type, properties, required) {
|
|
|
|
const option = this.get(name, required);
|
|
|
|
if (!option) {
|
|
|
|
return null;
|
|
|
|
} else if (option.type !== type) {
|
|
|
|
throw new TypeError('COMMAND_INTERACTION_OPTION_TYPE', name, option.type, type);
|
|
|
|
} else if (required && properties.every(prop => option[prop] === null || typeof option[prop] === 'undefined')) {
|
|
|
|
throw new TypeError('COMMAND_INTERACTION_OPTION_EMPTY', name, option.type);
|
|
|
|
}
|
|
|
|
return option;
|
|
|
|
}
|
|
|
|
|
|
|
|
/**
|
|
|
|
* Gets the selected subcommand.
|
|
|
|
* @param {boolean} [required=true] Whether to throw an error if there is no subcommand.
|
|
|
|
* @returns {?string} The name of the selected subcommand, or null if not set and not required.
|
|
|
|
*/
|
|
|
|
getSubcommand(required = true) {
|
|
|
|
if (required && !this._subcommand) {
|
|
|
|
throw new TypeError('COMMAND_INTERACTION_OPTION_NO_SUB_COMMAND');
|
|
|
|
}
|
|
|
|
return this._subcommand;
|
|
|
|
}
|
|
|
|
|
|
|
|
/**
|
|
|
|
* Gets the selected subcommand group.
|
2022-03-24 10:55:32 +00:00
|
|
|
* @param {boolean} [required=true] Whether to throw an error if there is no subcommand group.
|
2022-03-19 10:37:45 +00:00
|
|
|
* @returns {?string} The name of the selected subcommand group, or null if not set and not required.
|
|
|
|
*/
|
2022-03-24 10:55:32 +00:00
|
|
|
getSubcommandGroup(required = true) {
|
2022-03-19 10:37:45 +00:00
|
|
|
if (required && !this._group) {
|
|
|
|
throw new TypeError('COMMAND_INTERACTION_OPTION_NO_SUB_COMMAND_GROUP');
|
|
|
|
}
|
|
|
|
return this._group;
|
|
|
|
}
|
|
|
|
|
|
|
|
/**
|
|
|
|
* Gets a boolean option.
|
|
|
|
* @param {string} name The name of the option.
|
|
|
|
* @param {boolean} [required=false] Whether to throw an error if the option is not found.
|
|
|
|
* @returns {?boolean} The value of the option, or null if not set and not required.
|
|
|
|
*/
|
|
|
|
getBoolean(name, required = false) {
|
2022-03-24 10:55:32 +00:00
|
|
|
const option = this._getTypedOption(name, 'BOOLEAN', ['value'], required);
|
2022-03-19 10:37:45 +00:00
|
|
|
return option?.value ?? null;
|
|
|
|
}
|
|
|
|
|
|
|
|
/**
|
|
|
|
* Gets a channel option.
|
|
|
|
* @param {string} name The name of the option.
|
|
|
|
* @param {boolean} [required=false] Whether to throw an error if the option is not found.
|
|
|
|
* @returns {?(GuildChannel|ThreadChannel|APIChannel)}
|
|
|
|
* The value of the option, or null if not set and not required.
|
|
|
|
*/
|
|
|
|
getChannel(name, required = false) {
|
2022-03-24 10:55:32 +00:00
|
|
|
const option = this._getTypedOption(name, 'CHANNEL', ['channel'], required);
|
2022-03-19 10:37:45 +00:00
|
|
|
return option?.channel ?? null;
|
|
|
|
}
|
|
|
|
|
|
|
|
/**
|
|
|
|
* Gets a string option.
|
|
|
|
* @param {string} name The name of the option.
|
|
|
|
* @param {boolean} [required=false] Whether to throw an error if the option is not found.
|
|
|
|
* @returns {?string} The value of the option, or null if not set and not required.
|
|
|
|
*/
|
|
|
|
getString(name, required = false) {
|
2022-03-24 10:55:32 +00:00
|
|
|
const option = this._getTypedOption(name, 'STRING', ['value'], required);
|
2022-03-19 10:37:45 +00:00
|
|
|
return option?.value ?? null;
|
|
|
|
}
|
|
|
|
|
|
|
|
/**
|
|
|
|
* Gets an integer option.
|
|
|
|
* @param {string} name The name of the option.
|
|
|
|
* @param {boolean} [required=false] Whether to throw an error if the option is not found.
|
|
|
|
* @returns {?number} The value of the option, or null if not set and not required.
|
|
|
|
*/
|
|
|
|
getInteger(name, required = false) {
|
2022-03-24 10:55:32 +00:00
|
|
|
const option = this._getTypedOption(name, 'INTEGER', ['value'], required);
|
2022-03-19 10:37:45 +00:00
|
|
|
return option?.value ?? null;
|
|
|
|
}
|
|
|
|
|
|
|
|
/**
|
|
|
|
* Gets a number option.
|
|
|
|
* @param {string} name The name of the option.
|
|
|
|
* @param {boolean} [required=false] Whether to throw an error if the option is not found.
|
|
|
|
* @returns {?number} The value of the option, or null if not set and not required.
|
|
|
|
*/
|
|
|
|
getNumber(name, required = false) {
|
2022-03-24 10:55:32 +00:00
|
|
|
const option = this._getTypedOption(name, 'NUMBER', ['value'], required);
|
2022-03-19 10:37:45 +00:00
|
|
|
return option?.value ?? null;
|
|
|
|
}
|
|
|
|
|
|
|
|
/**
|
|
|
|
* Gets a user option.
|
|
|
|
* @param {string} name The name of the option.
|
|
|
|
* @param {boolean} [required=false] Whether to throw an error if the option is not found.
|
|
|
|
* @returns {?User} The value of the option, or null if not set and not required.
|
|
|
|
*/
|
|
|
|
getUser(name, required = false) {
|
2022-03-24 10:55:32 +00:00
|
|
|
const option = this._getTypedOption(name, 'USER', ['user'], required);
|
2022-03-19 10:37:45 +00:00
|
|
|
return option?.user ?? null;
|
|
|
|
}
|
|
|
|
|
|
|
|
/**
|
|
|
|
* Gets a member option.
|
|
|
|
* @param {string} name The name of the option.
|
2022-03-24 10:55:32 +00:00
|
|
|
* @param {boolean} [required=false] Whether to throw an error if the option is not found.
|
2022-03-19 10:37:45 +00:00
|
|
|
* @returns {?(GuildMember|APIGuildMember)}
|
2022-03-24 10:55:32 +00:00
|
|
|
* The value of the option, or null if not set and not required.
|
2022-03-19 10:37:45 +00:00
|
|
|
*/
|
2022-03-24 10:55:32 +00:00
|
|
|
getMember(name, required = false) {
|
|
|
|
const option = this._getTypedOption(name, 'USER', ['member'], required);
|
2022-03-19 10:37:45 +00:00
|
|
|
return option?.member ?? null;
|
|
|
|
}
|
|
|
|
|
|
|
|
/**
|
|
|
|
* Gets a role option.
|
|
|
|
* @param {string} name The name of the option.
|
|
|
|
* @param {boolean} [required=false] Whether to throw an error if the option is not found.
|
|
|
|
* @returns {?(Role|APIRole)} The value of the option, or null if not set and not required.
|
|
|
|
*/
|
|
|
|
getRole(name, required = false) {
|
2022-03-24 10:55:32 +00:00
|
|
|
const option = this._getTypedOption(name, 'ROLE', ['role'], required);
|
2022-03-19 10:37:45 +00:00
|
|
|
return option?.role ?? null;
|
|
|
|
}
|
|
|
|
|
|
|
|
/**
|
|
|
|
* Gets a mentionable option.
|
|
|
|
* @param {string} name The name of the option.
|
|
|
|
* @param {boolean} [required=false] Whether to throw an error if the option is not found.
|
|
|
|
* @returns {?(User|GuildMember|APIGuildMember|Role|APIRole)}
|
|
|
|
* The value of the option, or null if not set and not required.
|
|
|
|
*/
|
|
|
|
getMentionable(name, required = false) {
|
2022-03-24 10:55:32 +00:00
|
|
|
const option = this._getTypedOption(name, 'MENTIONABLE', ['user', 'member', 'role'], required);
|
2022-03-19 10:37:45 +00:00
|
|
|
return option?.member ?? option?.user ?? option?.role ?? null;
|
|
|
|
}
|
|
|
|
|
|
|
|
/**
|
|
|
|
* Gets a message option.
|
|
|
|
* @param {string} name The name of the option.
|
|
|
|
* @param {boolean} [required=false] Whether to throw an error if the option is not found.
|
|
|
|
* @returns {?(Message|APIMessage)}
|
|
|
|
* The value of the option, or null if not set and not required.
|
|
|
|
*/
|
|
|
|
getMessage(name, required = false) {
|
|
|
|
const option = this._getTypedOption(name, '_MESSAGE', ['message'], required);
|
|
|
|
return option?.message ?? null;
|
|
|
|
}
|
|
|
|
|
|
|
|
/**
|
|
|
|
* Gets the focused option.
|
|
|
|
* @param {boolean} [getFull=false] Whether to get the full option object
|
|
|
|
* @returns {string|number|ApplicationCommandOptionChoice}
|
|
|
|
* The value of the option, or the whole option if getFull is true
|
|
|
|
*/
|
|
|
|
getFocused(getFull = false) {
|
|
|
|
const focusedOption = this._hoistedOptions.find(option => option.focused);
|
|
|
|
if (!focusedOption) throw new TypeError('AUTOCOMPLETE_INTERACTION_OPTION_NO_FOCUSED_OPTION');
|
|
|
|
return getFull ? focusedOption : focusedOption.value;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
module.exports = CommandInteractionOptionResolver;
|