2022-08-02 23:13:32 +03:00
|
|
|
const { SlashCommand } = require('@eartharoid/dbf');
|
2022-10-11 23:24:09 +03:00
|
|
|
const {
|
|
|
|
ApplicationCommandOptionType,
|
|
|
|
ActionRowBuilder,
|
|
|
|
ButtonBuilder,
|
|
|
|
ButtonStyle,
|
|
|
|
ComponentType,
|
|
|
|
} = require('discord.js');
|
|
|
|
const ExtendedEmbedBuilder = require('../../lib/embed');
|
|
|
|
const { isStaff } = require('../../lib/users');
|
|
|
|
const ms = require('ms');
|
2022-08-02 23:13:32 +03:00
|
|
|
|
|
|
|
module.exports = class ForceCloseSlashCommand extends SlashCommand {
|
|
|
|
constructor(client, options) {
|
2022-10-26 20:58:46 +03:00
|
|
|
const name = 'force-close';
|
2022-08-02 23:13:32 +03:00
|
|
|
super(client, {
|
|
|
|
...options,
|
2022-10-26 20:58:46 +03:00
|
|
|
description: client.i18n.getMessage(null, `commands.slash.${name}.description`),
|
|
|
|
descriptionLocalizations: client.i18n.getAllMessages(`commands.slash.${name}.description`),
|
2022-08-02 23:13:32 +03:00
|
|
|
dmPermission: false,
|
2022-10-26 20:58:46 +03:00
|
|
|
name,
|
|
|
|
nameLocalizations: client.i18n.getAllMessages(`commands.slash.${name}.name`),
|
|
|
|
options: [
|
|
|
|
{
|
|
|
|
autocomplete: true,
|
|
|
|
name: 'category',
|
|
|
|
required: false,
|
|
|
|
type: ApplicationCommandOptionType.Integer,
|
|
|
|
},
|
|
|
|
{
|
|
|
|
name: 'reason',
|
|
|
|
required: false,
|
|
|
|
type: ApplicationCommandOptionType.String,
|
|
|
|
},
|
|
|
|
{
|
|
|
|
autocomplete: true,
|
|
|
|
name: 'ticket',
|
|
|
|
required: false,
|
|
|
|
type: ApplicationCommandOptionType.String,
|
|
|
|
},
|
|
|
|
{
|
|
|
|
name: 'time',
|
|
|
|
required: false,
|
|
|
|
type: ApplicationCommandOptionType.String,
|
|
|
|
},
|
|
|
|
].map(option => {
|
|
|
|
option.descriptionLocalizations = client.i18n.getAllMessages(`commands.slash.${name}.options.${option.name}.description`);
|
|
|
|
option.description = option.descriptionLocalizations['en-GB'];
|
|
|
|
option.nameLocalizations = client.i18n.getAllMessages(`commands.slash.${name}.options.${option.name}.name`);
|
|
|
|
return option;
|
|
|
|
}),
|
2022-08-02 23:13:32 +03:00
|
|
|
});
|
|
|
|
}
|
|
|
|
|
2022-10-11 23:24:09 +03:00
|
|
|
/**
|
|
|
|
* @param {import("discord.js").ChatInputCommandInteraction} interaction
|
|
|
|
*/
|
|
|
|
async run(interaction) {
|
|
|
|
/** @type {import("client")} */
|
|
|
|
const client = this.client;
|
|
|
|
|
|
|
|
await interaction.deferReply();
|
|
|
|
|
|
|
|
const settings = await client.prisma.guild.findUnique({ where: { id: interaction.guild.id } });
|
2023-01-14 00:25:04 +02:00
|
|
|
const getMessage = client.i18n.getLocale(settings.locale);
|
2022-10-11 23:24:09 +03:00
|
|
|
let ticket;
|
|
|
|
|
2022-10-26 16:13:45 +03:00
|
|
|
if (!(await isStaff(interaction.guild, interaction.user.id))) { // if user is not staff
|
2022-10-11 23:24:09 +03:00
|
|
|
return await interaction.editReply({
|
|
|
|
embeds: [
|
|
|
|
new ExtendedEmbedBuilder({
|
|
|
|
iconURL: interaction.guild.iconURL(),
|
|
|
|
text: settings.footer,
|
|
|
|
})
|
|
|
|
.setColor(settings.errorColour)
|
|
|
|
.setTitle(getMessage('commands.slash.force-close.not_staff.title'))
|
|
|
|
.setDescription(getMessage('commands.slash.force-close.not_staff.description')),
|
|
|
|
],
|
|
|
|
});
|
|
|
|
}
|
|
|
|
|
|
|
|
if (interaction.options.getString('time', false)) { // if time option is passed
|
|
|
|
const time = ms(interaction.options.getString('time', false));
|
|
|
|
|
|
|
|
if (!time) {
|
|
|
|
return await interaction.editReply({
|
|
|
|
embeds: [
|
|
|
|
new ExtendedEmbedBuilder({
|
|
|
|
iconURL: interaction.guild.iconURL(),
|
|
|
|
text: settings.footer,
|
|
|
|
})
|
|
|
|
.setColor(settings.errorColour)
|
|
|
|
.setTitle(getMessage('commands.slash.close.invalid_time.title'))
|
|
|
|
.setDescription(getMessage('commands.slash.close.invalid_time.description', { input: interaction.options.getString('time', false) })),
|
|
|
|
],
|
|
|
|
});
|
|
|
|
}
|
|
|
|
|
|
|
|
const tickets = await client.prisma.ticket.findMany({
|
|
|
|
where: {
|
|
|
|
lastMessageAt: { lte: new Date(Date.now() - time) },
|
|
|
|
open: true,
|
|
|
|
},
|
|
|
|
});
|
|
|
|
|
|
|
|
if (tickets.length === 0) {
|
|
|
|
return await interaction.editReply({
|
|
|
|
embeds: [
|
|
|
|
new ExtendedEmbedBuilder({
|
|
|
|
iconURL: interaction.guild.iconURL(),
|
|
|
|
text: settings.footer,
|
|
|
|
})
|
|
|
|
.setColor(settings.errorColour)
|
|
|
|
.setTitle(getMessage('commands.slash.force-close.no_tickets.title'))
|
|
|
|
.setDescription(getMessage('commands.slash.force-close.no_tickets.description', { time: ms(time, { long: true }) })),
|
|
|
|
],
|
|
|
|
});
|
|
|
|
}
|
|
|
|
|
|
|
|
let confirmed = false;
|
|
|
|
const collectorTime = ms('15s');
|
|
|
|
const confirmationM = await interaction.editReply({
|
|
|
|
components: [
|
|
|
|
new ActionRowBuilder()
|
|
|
|
.addComponents([
|
|
|
|
new ButtonBuilder()
|
|
|
|
.setCustomId(JSON.stringify({
|
|
|
|
action: 'custom',
|
|
|
|
id: 'close',
|
|
|
|
}))
|
|
|
|
.setStyle(ButtonStyle.Danger)
|
|
|
|
.setEmoji(getMessage('buttons.close.emoji'))
|
|
|
|
.setLabel(getMessage('buttons.close.text')),
|
|
|
|
new ButtonBuilder()
|
|
|
|
.setCustomId(JSON.stringify({
|
|
|
|
action: 'custom',
|
|
|
|
id: 'cancel',
|
|
|
|
}))
|
|
|
|
.setStyle(ButtonStyle.Secondary)
|
|
|
|
.setEmoji(getMessage('buttons.cancel.emoji'))
|
|
|
|
.setLabel(getMessage('buttons.cancel.text')),
|
|
|
|
]),
|
|
|
|
],
|
|
|
|
embeds: [
|
|
|
|
new ExtendedEmbedBuilder({
|
|
|
|
iconURL: interaction.guild.iconURL(),
|
|
|
|
text: getMessage('misc.expires_in', { time: ms(collectorTime, { long: true }) }),
|
|
|
|
})
|
|
|
|
.setColor(settings.primaryColour)
|
|
|
|
.setTitle(getMessage('commands.slash.force-close.confirm_multiple.title'))
|
|
|
|
.setDescription(getMessage('commands.slash.force-close.confirm_multiple.description', {
|
|
|
|
count: tickets.length,
|
|
|
|
tickets: tickets.map(t => `> <#${t.id}>`).join('\n'),
|
|
|
|
time: ms(time, { long: true }),
|
|
|
|
})),
|
|
|
|
],
|
|
|
|
});
|
|
|
|
|
|
|
|
|
|
|
|
confirmationM.awaitMessageComponent({
|
|
|
|
componentType: ComponentType.Button,
|
|
|
|
filter: i => {
|
|
|
|
i.deferUpdate();
|
|
|
|
return i.user.id === interaction.user.id;
|
|
|
|
},
|
|
|
|
time: collectorTime,
|
|
|
|
})
|
|
|
|
.then(i => {
|
|
|
|
if (JSON.parse(i.customId).id === 'close') {
|
|
|
|
confirmed = true;
|
|
|
|
// TODO: i.editReply
|
|
|
|
} else {
|
|
|
|
// TODO: cancelled
|
|
|
|
}
|
|
|
|
})
|
|
|
|
.catch(() => interaction.editReply({
|
|
|
|
components: [],
|
|
|
|
embeds: [
|
|
|
|
new ExtendedEmbedBuilder({
|
|
|
|
iconURL: interaction.guild.iconURL(),
|
|
|
|
text: settings.footer,
|
|
|
|
})
|
|
|
|
.setColor(settings.errorColour)
|
|
|
|
.setTitle(getMessage('misc.expired.title'))
|
|
|
|
.setDescription(getMessage('misc.expired.description', { time: ms(time, { long: true }) })),
|
|
|
|
],
|
|
|
|
}));
|
|
|
|
|
|
|
|
if (!confirmed) return;
|
|
|
|
|
|
|
|
// TODO: tickets: for each, close (check reason)
|
|
|
|
} else if (interaction.options.getString('ticket', false)) { // if ticket option is passed
|
|
|
|
ticket = await client.prisma.ticket.findUnique({
|
|
|
|
include: { category: true },
|
|
|
|
where: { id: interaction.options.getString('ticket', false) },
|
|
|
|
});
|
|
|
|
|
|
|
|
if (!ticket) {
|
|
|
|
return await interaction.editReply({
|
|
|
|
embeds: [
|
|
|
|
new ExtendedEmbedBuilder({
|
|
|
|
iconURL: interaction.guild.iconURL(),
|
|
|
|
text: settings.footer,
|
|
|
|
})
|
|
|
|
.setColor(settings.errorColour)
|
|
|
|
.setTitle(getMessage('misc.invalid_ticket.title'))
|
|
|
|
.setDescription(getMessage('misc.invalid_ticket.description')),
|
|
|
|
],
|
|
|
|
});
|
|
|
|
}
|
|
|
|
} else {
|
|
|
|
ticket = await client.prisma.ticket.findUnique({
|
|
|
|
include: { category: true },
|
|
|
|
where: { id: interaction.channel.id },
|
|
|
|
});
|
|
|
|
|
|
|
|
if (!ticket) {
|
|
|
|
return await interaction.editReply({
|
|
|
|
embeds: [
|
|
|
|
new ExtendedEmbedBuilder({
|
|
|
|
iconURL: interaction.guild.iconURL(),
|
|
|
|
text: settings.footer,
|
|
|
|
})
|
|
|
|
.setColor(settings.errorColour)
|
|
|
|
.setTitle(getMessage('misc.not_ticket.title'))
|
|
|
|
.setDescription(getMessage('misc.not_ticket.description')),
|
|
|
|
],
|
|
|
|
});
|
|
|
|
}
|
|
|
|
}
|
|
|
|
}
|
2022-08-02 23:13:32 +03:00
|
|
|
};
|