/* eslint-disable comma-dangle */ // dotenv for handling environment variables const dotenv = require('dotenv'); dotenv.config(); const isDev = process.env.isDev; const package = require('../package.json'); // filesystem const fs = require('fs'); // Discord.js const Discord = require('discord.js'); const { EmbedBuilder, ActionRowBuilder, ButtonBuilder, ButtonStyle } = Discord; // Various imports from other files const config = require('../data/config.json'); const strings = require('../data/strings.json'); const slashCommandFiles = fs.readdirSync('./slash-commands/').filter(file => file.endsWith('.js')); const dbfn = require('./dbfn.js'); dbfn.createGuildTables().then(res => { console.log(res.status); }).catch(err => { console.error(err); }); const functions = { // Functions for managing and creating Collections collections: { // Create the collection of slash commands slashCommands(client) { if (!client.slashCommands) client.slashCommands = new Discord.Collection(); client.slashCommands.clear(); for (const file of slashCommandFiles) { const slashCommand = require(`../slash-commands/${file}`); if (slashCommand.data != undefined) { client.slashCommands.set(slashCommand.data.name, slashCommand); } } if (isDev) console.log('Slash Commands Collection Built'); } }, builders: { refreshAction() { // Create the button to go in the Action Row const refreshButton = new ButtonBuilder() .setCustomId('refresh') .setLabel('Refresh') .setStyle(ButtonStyle.Primary); // Create the Action Row with the Button in it, to be sent with the Embed const refreshActionRow = new ActionRowBuilder() .addComponents( refreshButton ); return refreshActionRow; }, comparisonEmbed(content, refreshActionRow) { // Create the embed using the content passed to this function const embed = new EmbedBuilder() .setColor(strings.embeds.color) .setTitle('Tree Growth Comparison') .setDescription(content) .setFooter({text: `v${package.version} - ${strings.embeds.footer}`}); const messageContents = { embeds: [embed], components: [refreshActionRow] }; return messageContents; }, helpEmbed(content, private) { const embed = new EmbedBuilder() .setColor(strings.embeds.color) .setTitle(strings.help.title) .setDescription(content) .setFooter({ text: `v${package.version} - ${strings.embeds.footer}` }); const privateBool = private == 'true'; const messageContents = { embeds: [embed], ephemeral: privateBool }; return messageContents; }, errorEmbed(content) { const embed = new EmbedBuilder() .setColor(0xFF0000) .setTitle('Error!') .setDescription("Error: " + content) .setFooter({ text: `v${package.version} - ${strings.embeds.footer}` }); const messageContents = { embeds: [embed], ephemeral: true }; return messageContents; }, embed(content) { const embed = new EmbedBuilder() .setColor(0x8888FF) .setTitle('Information') .setDescription(content) .setFooter({ text: `v${package.version} - ${strings.embeds.footer}` }); const messageContents = { embeds: [embed], ephemeral: true }; return messageContents; } }, rankings: { parse(interaction) { return new Promise ((resolve, reject) => { dbfn.getGuildInfo(interaction.guildId).then(res => { const guildInfo = res.data; if (guildInfo.guildId == "") { reject(strings.error.noGuild); return; } if (guildInfo.leaderboardMessageId != undefined) { interaction.guild.channels.fetch(guildInfo.leaderboardChannelId).then(c => { c.messages.fetch(guildInfo.leaderboardMessageId).then(leaderboardMessage => { if ((leaderboardMessage.embeds.length == 0) || (leaderboardMessage.embeds[0].data.title != 'Tallest Trees' )) { reject("This doesn't appear to be a valid ``/top trees`` message."); return; } let lines = leaderboardMessage.embeds[0].data.description.split('\n'); let leaderboard = { "guildId": interaction.guildId, "entries": [] }; for (let i = 0; i < 10; i++) { // Breakdown each line separating it on each - let breakdown = lines[i].split(' - '); // Check if the first part, the ranking, has these emojis to detect first second and third place if (breakdown[0].includes('🥇')) { breakdown[0] = '``#1 ``' } else if (breakdown[0].includes('🥈')) { breakdown[0] = '``#2 ``' } else if (breakdown[0].includes('🥉')) { breakdown[0] = '``#3 ``' } // Clean off the excess and get just the number from the rank, make sure it's an int not string let trimmedRank = parseInt(breakdown[0].slice(breakdown[0].indexOf('#') + 1, breakdown[0].lastIndexOf('``'))); // Clean off the excess and get just the tree name let trimmedName = breakdown[1].slice(breakdown[1].indexOf('``') + 2); trimmedName = trimmedName.slice(0, trimmedName.indexOf('``')); // Clean off the excess and get just the tree height, make sure it's a 1 decimal float let trimmedHeight = parseFloat(breakdown[2].slice(0, breakdown[2].indexOf('ft'))).toFixed(1); let isMyTree = false; let isMaybeMyTree = false; if (breakdown[2].includes('📍')) isMyTree = true; if (breakdown[1].includes(guildInfo.treeName)) maybeMyTree = true; // "entries": [ { "treeHeight": 12, "treeRank": 34, "treeName": "name" }, ] } leaderboard.entries.push({ treeRank: trimmedRank, treeName: trimmedName, treeHeight: trimmedHeight, hasPin: isMyTree }); } dbfn.uploadLeaderboard(leaderboard).then(res => { console.log(res.status); resolve(res.status); }).catch(err => { console.error(err); reject(err); return; }); }); }); } else { reject("The leaderboardMessageId is undefined somehow"); return; } }).catch(err => { reject(err); return; }); }); }, compare(interaction) { return new Promise((resolve, reject) => { dbfn.getGuildInfo(interaction.guildId).then(res => { const guildInfo = res.data; guildInfo.guildId = interaction.guildId; let treeHeight = parseFloat(guildInfo.treeHeight).toFixed(1); dbfn.getLeaderboard(interaction.guildId).then(res => { const leaderboard = res.data; let replyString = 'Current Tree Height: ' + treeHeight + 'ft\n\n'; leaderboard.reverse().forEach(treeRanking => { let difference = parseFloat(treeRanking.treeHeight).toFixed(1) - treeHeight; let decimal = (treeRanking.treeHeight % 1).toFixed(1); let growthIndicator = ""; if (decimal > 0) { growthIndicator += "[+]"; } const absDifference = parseFloat(Math.abs(difference)).toFixed(1); if (treeRanking.hasPin) { replyString += "This is your tree. "; } else if ((treeRanking.treeHeight == treeHeight) && (treeRanking.treeName == guildInfo.treeName)) { replyString += "This might be your tree. Same height, same name. "; } else { if (difference > 0) { replyString += `#${treeRanking.treeRank} - ${absDifference}ft${growthIndicator} shorter `; } else if (difference < 0) { replyString += `#${treeRanking.treeRank} - ${absDifference}ft${growthIndicator} taller `; } else if (difference == 0) { replyString += `#${treeRanking.treeRank} - Same Height${growthIndicator} `; } } replyString += `[${functions.getWaterTime(treeRanking.treeHeight)} mins]\n`; }); resolve('Here\'s how your tree compares: \n' + replyString); }).catch(err => { console.error(err); }); }).catch(err => { reject(err); return; }); }); } }, tree: { parse(interaction) { let input; return new Promise((resolve, reject) => { dbfn.getGuildInfo(interaction.guildId).then(res => { const guildInfo = res.data; guildInfo.guildId = interaction.guildId; if (guildInfo == undefined) { reject(`The guild entry hasn't been created yet. [${interaction.guildId || interaction.commandGuildId}]`); return; } if (guildInfo.treeMessageId != "Run /setup where your tree is.") { interaction.guild.channels.fetch(guildInfo.treeChannelId).then(c => { c.messages.fetch(guildInfo.treeMessageId).then(m => { if ( (m.embeds.length == 0) || !(m.embeds[0].data.description.includes('Your tree is')) ) { reject("This doesn't appear to be a valid ``/tree`` message."); return; } input = m.embeds[0].data.description; let treeName = m.embeds[0].data.title; let lines = input.split('\n'); guildInfo.treeHeight = parseFloat(lines[0].slice(lines[0].indexOf('is') + 3, lines[0].indexOf('ft'))).toFixed(1); guildInfo.treeName = treeName; dbfn.setTreeInfo(guildInfo).then(res => { resolve("The reference tree message has been saved/updated."); }); }); }) } else { console.error('treeMessageId undefined'); reject("There was an unknown error while setting the tree message."); return; } }).catch(err => { reject(err); console.error(err); return; }); }); } }, refresh(interaction) { functions.rankings.parse(interaction).then(r1 => { functions.tree.parse(interaction).then(r2 => { functions.rankings.compare(interaction).then(res => { const embed = functions.builders.comparisonEmbed(res, functions.builders.refreshAction()) interaction.update(embed); }).catch(err => { console.error(err); }); }).catch(e => { console.error(e); interaction.reply(functions.builders.errorEmbed(e)); }); }).catch(e => { console.error(e); interaction.reply(functions.builders.errorEmbed(e)); }); }, reset(guildId) { return new Promise((resolve, reject) => { dbfn.deleteGuildInfo(guildId).then(res => { resolve(res); }).catch(err => { console.error(err); reject(err); return; }); }); }, getInfo(guildId) { return new Promise((resolve, reject) => { dbfn.getGuildInfo(guildId).then(res => { let guildInfo = res.data; let guildInfoString = ""; guildInfoString += `Tree Message: https://discord.com/channels/${guildId}/${guildInfo.treeChannelId}/${guildInfo.treeMessageId}\n`; guildInfoString += `Rank Message: https://discord.com/channels/${guildId}/${guildInfo.leaderboardChannelId}/${guildInfo.leaderboardMessageId}\n`; resolve(`Here is your servers setup info:\n${guildInfoString}`); }).catch(err => { console.error(err); reject(err); return; }) }); }, getWaterTime(size) { const seconds = Math.floor(Math.pow(size * 0.07 + 5, 1.1)); return (Math.floor((Math.pow(size * 0.07 + 5, 1.1))) / 60).toFixed(2); }, timeToHeight(interaction, destHeight) { return new Promise((resolve, reject) => { dbfn.getGuildInfo(interaction.guildId).then(res => { let guildInfo = res.data; let currentHeight = parseInt(guildInfo.treeHeight); let time = 0; for (let i = currentHeight; i < destHeight; i++) { const waterTime = parseFloat(functions.getWaterTime(i)); console.log("Height: " + i + "Time: " + waterTime); time += waterTime; } resolve(time.toFixed(2)); }).catch(err => { console.error(err); reject(err); return; }) }); } }; module.exports = functions;