/* * Konpeki Discord Bot - Slash Command Definition File * help.js - Uses generated help text file to provide a help command */ const { ActionRowBuilder, ButtonBuilder, ButtonStyle, SlashCommandBuilder } = require('discord.js'); const fs = require('fs'); let helpFile = ''; // Import help text file if (fs.existsSync('./data/help-text.json')) { helpFile = JSON.parse(fs.readFileSync('./data/help-text.json', 'utf8')); } // List of all commands, by category // Command help, listing options. module.exports = { data: new SlashCommandBuilder() .setName('help') .setDescription('Provides information on avalible commands') // Allow choosing the help page to open .addNumberOption(option => option.setName('page') .setDescription('Choose help page to skip to'), ), async execute(interaction) { let pageNumber = interaction.options.getNumber('page') ?? 1; const commandsPerPage = 5; let commandsThisPage = 0; const numberOfCommands = Object.keys(helpFile).length; const pageTotal = String(Math.ceil(numberOfCommands / commandsPerPage)); if (pageNumber > pageTotal) { pageNumber = 1; } let embedPartOne = { color: 0x0099ff, title: 'Help Text', description: `Page ${pageNumber} of ${pageTotal}`, }; if (pageNumber != pageTotal) { commandsThisPage = commandsPerPage; } else { commandsThisPage = numberOfCommands % commandsPerPage; } let embedPartTwo = ''; embedPartTwo += '{"fields": ['; for (let i = 0; i < commandsThisPage; i++) { const currentCommandName = Object.keys(helpFile)[i + (commandsPerPage * (pageNumber - 1))]; embedPartTwo += '{'; embedPartTwo += ' "name": "/' + currentCommandName + '",', embedPartTwo += ' "value": "' + helpFile[currentCommandName].description + '"', embedPartTwo += '},'; } embedPartTwo = embedPartTwo.substring(0, embedPartTwo.length - 1) + ']}'; let buttonList = ''; // Change buttons based on page number. Is there an easier/shorter way to do this? if (pageNumber == 1) { buttonList = new ActionRowBuilder() .addComponents( new ButtonBuilder() .setCustomId('back') .setLabel('Previous') .setStyle(ButtonStyle.Secondary) .setDisabled(true), ) .addComponents( new ButtonBuilder() .setCustomId('page') .setLabel(`${pageNumber}/${pageTotal}`) .setStyle(ButtonStyle.Secondary) .setDisabled(true), ) .addComponents( new ButtonBuilder() .setCustomId('next') .setLabel('Next') .setStyle(ButtonStyle.Primary), ); } else if (pageNumber == pageTotal) { buttonList = new ActionRowBuilder() .addComponents( new ButtonBuilder() .setCustomId('back') .setLabel('Previous') .setStyle(ButtonStyle.Primary), ) .addComponents( new ButtonBuilder() .setCustomId('page') .setLabel(`${pageNumber}/${pageTotal}`) .setStyle(ButtonStyle.Secondary) .setDisabled(true), ) .addComponents( new ButtonBuilder() .setCustomId('next') .setLabel('Next') .setStyle(ButtonStyle.Secondary) .setDisabled(true), ); } else { buttonList = new ActionRowBuilder() .addComponents( new ButtonBuilder() .setCustomId('back') .setLabel('Previous') .setStyle(ButtonStyle.Primary), ) .addComponents( new ButtonBuilder() .setCustomId('page') .setLabel(`${pageNumber}/${pageTotal}`) .setStyle(ButtonStyle.Secondary) .setDisabled(true), ) .addComponents( new ButtonBuilder() .setCustomId('next') .setLabel('Next') .setStyle(ButtonStyle.Primary), ); } // Button code const collector = interaction.channel.createMessageComponentCollector({ time: 60000 }); collector.on('collect', async i => { if (i.customId === 'next') { pageNumber += 1; } if (i.customId === 'back') { pageNumber -= 1; } // Replicate the above code to remake the help model for the new page. Again, there has to be an easier way to do this. embedPartOne = { color: 0x0099ff, title: 'Help Text', description: `Page ${pageNumber} of ${pageTotal}`, }; embedPartTwo = ''; if (pageNumber != pageTotal) { commandsThisPage = commandsPerPage; } else { commandsThisPage = numberOfCommands % commandsPerPage; } embedPartTwo += '{"fields": ['; for (let j = 0; j < commandsThisPage; j++) { const currentCommandName = Object.keys(helpFile)[j + (commandsPerPage * (pageNumber - 1))]; embedPartTwo += '{'; embedPartTwo += ' "name": "/' + currentCommandName + '",', embedPartTwo += ' "value": "' + helpFile[currentCommandName].description + '"', embedPartTwo += '},'; } embedPartTwo = embedPartTwo.substring(0, embedPartTwo.length - 1) + ']}'; // Change buttons based on page number. Is there an easier/shorter way to do this? if (pageNumber == 1) { buttonList = new ActionRowBuilder() .addComponents( new ButtonBuilder() .setCustomId('back') .setLabel('Previous') .setStyle(ButtonStyle.Secondary) .setDisabled(true), ) .addComponents( new ButtonBuilder() .setCustomId('page') .setLabel(`${pageNumber}/${pageTotal}`) .setStyle(ButtonStyle.Secondary) .setDisabled(true), ) .addComponents( new ButtonBuilder() .setCustomId('next') .setLabel('Next') .setStyle(ButtonStyle.Primary), ); } else if (pageNumber == pageTotal) { buttonList = new ActionRowBuilder() .addComponents( new ButtonBuilder() .setCustomId('back') .setLabel('Previous') .setStyle(ButtonStyle.Primary), ) .addComponents( new ButtonBuilder() .setCustomId('page') .setLabel(`${pageNumber}/${pageTotal}`) .setStyle(ButtonStyle.Secondary) .setDisabled(true), ) .addComponents( new ButtonBuilder() .setCustomId('next') .setLabel('Next') .setStyle(ButtonStyle.Secondary) .setDisabled(true), ); } else { buttonList = new ActionRowBuilder() .addComponents( new ButtonBuilder() .setCustomId('back') .setLabel('Previous') .setStyle(ButtonStyle.Primary), ) .addComponents( new ButtonBuilder() .setCustomId('page') .setLabel(`${pageNumber}/${pageTotal}`) .setStyle(ButtonStyle.Secondary) .setDisabled(true), ) .addComponents( new ButtonBuilder() .setCustomId('next') .setLabel('Next') .setStyle(ButtonStyle.Primary), ); } await i.update({ embeds: [Object.assign({}, embedPartOne, JSON.parse(embedPartTwo))], components: [ buttonList ], ephemeral: true }); }); await interaction.reply({ embeds: [Object.assign({}, embedPartOne, JSON.parse(embedPartTwo))], components: [ buttonList ], ephemeral: true }); }, };