1
2
3
4
5
6
7
8
9
10
11
12
13
14
15
16
17
18
19
20
21
22
23
24
25
26
27
28
29
30
31
32
33
34
35
36
37
38
39
40
41
42
43
44
45
46
47
48
49
50
51
52
53
54
55
56
57
58
59
60
61
62
63
64
65
66
67
68
69
70
71
72
73
74
75
76
77
78
79
80
81
82
83
84
85
86
87
88
89
90
91
92
93
94
95
96
97
98
99
100
101
102
103
104
105
106
107
108
|
import { AllowedMentions, BushCommand, type ArgType, type BushMessage, type BushSlashMessage } from '#lib';
import assert from 'assert';
import { ApplicationCommandOptionType, AutocompleteInteraction, PermissionFlagsBits } from 'discord.js';
import Fuse from 'fuse.js';
assert(Fuse);
export default class DisableCommand extends BushCommand {
private static blacklistedCommands = ['eval', 'disable'];
public constructor() {
super('disable', {
aliases: ['disable', 'enable'],
category: 'config',
description: 'A command to disable and enable commands.',
usage: ['disable|enable <command>'],
examples: ['enable ban', 'disable kick'],
args: [
{
id: 'action',
description: 'Whether to disable or enable the command.',
readableType: "'disable'|'enable",
prompt: 'Would you like to disable or enable a command?',
slashType: ApplicationCommandOptionType.String,
choices: ['disable', 'enable'].map((v) => ({ name: v, value: v })),
only: 'slash'
},
{
id: 'command',
description: 'The command to disable/enable.',
type: util.arg.union('commandAlias', 'command'),
readableType: 'command|commandAlias',
prompt: 'What command would you like to enable/disable?',
retry: '{error} Pick a valid command.',
slashType: ApplicationCommandOptionType.String,
autocomplete: true
},
{
id: 'global',
description: 'Disable the command globally.',
match: 'flag',
flag: '--global',
optional: true,
slashType: false,
only: 'text',
ownerOnly: true
}
],
slash: true,
channel: 'guild',
clientPermissions: (m) => util.clientSendAndPermCheck(m),
userPermissions: [PermissionFlagsBits.ManageGuild]
});
}
public override async exec(
message: BushMessage | BushSlashMessage,
args: { action?: 'enable' | 'disable'; command: ArgType<'commandAlias'> | string; global: boolean }
) {
let action = (args.action ?? message?.util?.parsed?.alias ?? 'toggle') as 'disable' | 'enable' | 'toggle';
const global = args.global && message.author.isOwner();
const commandID =
args.command instanceof BushCommand ? args.command.id : (await util.arg.cast('commandAlias', message, args.command))?.id;
if (!commandID) return await message.util.reply(`${util.emojis.error} Invalid command.`);
if (DisableCommand.blacklistedCommands.includes(commandID))
return message.util.send(`${util.emojis.error} the ${commandID} command cannot be disabled.`);
const disabledCommands = global ? util.getGlobal('disabledCommands') : await message.guild!.getSetting('disabledCommands');
if (action === 'toggle') action = disabledCommands.includes(commandID) ? 'disable' : 'enable';
const newValue = util.addOrRemoveFromArray(action === 'disable' ? 'remove' : 'add', disabledCommands, commandID);
const success = global
? await util.setGlobal('disabledCommands', newValue).catch(() => false)
: await message.guild!.setSetting('disabledCommands', newValue, message.member!).catch(() => false);
if (!success)
return await message.util.reply({
content: `${util.emojis.error} There was an error${global ? ' globally' : ''} **${action.substring(
0,
action.length - 2
)}ing** the **${commandID}** command.`,
allowedMentions: AllowedMentions.none()
});
else
return await message.util.reply({
content: `${util.emojis.success} Successfully **${action.substring(
0,
action.length - 2
)}ed** the **${commandID}** command${global ? ' globally' : ''}.`,
allowedMentions: AllowedMentions.none()
});
}
public override async autocomplete(interaction: AutocompleteInteraction) {
const commands = [...this.handler.modules.keys()];
const fuzzy = new Fuse(commands, {
threshold: 0.5,
isCaseSensitive: false,
findAllMatches: true
}).search(interaction.options.getFocused().toString());
const res = fuzzy.slice(0, fuzzy.length >= 25 ? 25 : undefined).map((v) => ({ name: v.item, value: v.item }));
void interaction.respond(res);
}
}
|