diff --git a/TODO b/TODO new file mode 100644 index 0000000..75cfb41 --- /dev/null +++ b/TODO @@ -0,0 +1,3 @@ +Role all +Server rules +A way to add buttons to verify / role menu \ No newline at end of file diff --git a/src/Unfinished/all.ts b/src/Unfinished/all.ts new file mode 100644 index 0000000..05eb603 --- /dev/null +++ b/src/Unfinished/all.ts @@ -0,0 +1,158 @@ +import Discord, { CommandInteraction, GuildMember, MessageActionRow, MessageButton, MessageSelectMenu } from "discord.js"; +import { SlashCommandSubcommandBuilder } from "@discordjs/builders"; +import { WrappedCheck } from "jshaiku"; +import EmojiEmbed from "../utils/generateEmojiEmbed.js"; +import getEmojiByName from "../utils/getEmojiByName.js"; +import addPlural from "../utils/plurals.js"; +import client from "../utils/client.js"; + +const command = (builder: SlashCommandSubcommandBuilder) => + builder // TODO: DON'T RELEASE THIS + .setName("all") + .setDescription("Gives or removes a role from everyone") + +class Filter { + name: string; + data: object; + checkFunction: (member) => boolean; + inverted: boolean = false; + constructor(name: (data) => string | string, data: object, check: (member) => boolean) { + if (typeof name === "function") { this.name = name(data); + } else { this.name = name; } + this.data = data; + this.checkFunction = check; + } + flip() { this.inverted = true; return this; } + check(member) { + if (this.inverted) return !this.checkFunction(member) + else return this.checkFunction(member) + } +} + +const filterList = { + member: { + render: "Member", + has: { + render: "has", + role: (role) => ( new Filter((data) => `Member has role <@&${data.role}>`, {role: role, type: Discord.Role, render: "role"}, (member) => { return member.roles.cache.has(role)})) + }, + joined: { + render: "joined", + before: (date) => ( new Filter((data) => `Joined server before `, {date: date, type: Date, render: "before"}, (member) => { + return member.joinedTimestamp < date.getTime() + })) + }, + nickname: { + render: "Nickname", + set: () => ( new Filter((data) => `Member has a nickname set"`, {render: "set"}, (member) => { return member.nickname !== null})), + includes: (name) => ( new Filter((data) => `Nickname includes "${name}"`, {nickname: name, type: String, render: "includes"}, (member) => { + return member.displayName.includes(name)})), + startsWith: (name) => ( new Filter((data) => `Nickname starts with "${name}"`, {nickname: name, type: String, render: "starts with"}, (member) => { + return member.displayName.startsWith(name)})), + endsWith: (name) => ( new Filter((data) => `Nickname ends with "${name}"`, {nickname: name, type: String, render: "ends with"}, (member) => { + return member.displayName.endsWith(name)})) + } + }, + account: { + render: "Account", + created: { + render: "created", + before: (date) => ( new Filter((data) => `Account created before `, {date: date, type: Date, render: "before"}, (member) => { + return member.user.createdTimestamp < date.getTime() + })) + }, + is: { + render: "is", + human: () => ( new Filter((data) => `Member is a human`, {human: true, render: "human"}, (member) => { return !member.bot })), + }, + username: { + render: "Username", + includes: (name) => ( new Filter((data) => `Nickname includes "${name}"`, {nickname: name, type: String, render: "includes"}, (member) => { + return member.user.name.includes(name)})), + startsWith: (name) => ( new Filter((data) => `Nickname starts with "${name}"`, {nickname: name, type: String, render: "starts with"}, (member) => { + return member.user.name.startsWith(name)})), + endsWith: (name) => ( new Filter((data) => `Nickname ends with "${name}"`, {nickname: name, type: String, render: "ends with"}, (member) => { + return member.user.name.endsWith(name)})) + } + } +} + +const callback = async (interaction: CommandInteraction) => { + await interaction.reply({embeds: [new EmojiEmbed() + .setTitle("Role all") + .setDescription("Loading...") + .setStatus("Danger") + .setEmoji("NUCLEUS.LOADING") + ], ephemeral: true, fetchReply: true}) + let filters: Filter[] = [ + filterList.member.has.role("959901346000154674"), + filterList.member.nickname.startsWith("Pinea"), + filterList.member.joined.before(new Date(2022, 1)).flip() + ] + let all = true; + while (true) { + let count = 0; + let affected = [] + let members = interaction.guild.members.cache + if (all) { + members.forEach(member => { + let applies = true; + filters.forEach(filter => { if (!filter.check(member)) { applies = false } }) + if (applies) { affected.push(member) } + }) + } else { + members.forEach(member => { + let applies = false; + filters.forEach(filter => { if (filter.check(member)) { applies = true } }) + if (applies) { affected.push(member) } + }) + } + await interaction.editReply({embeds: [new EmojiEmbed() + .setTitle("Role all") + .setDescription((all ? "All of the following must be true:" : "Any of the following must be true") + "\n" + + filters.map((f) => { + count ++; + return (count == 1 ? getEmojiByName("ICONS.FILTER") : (all ? "**and** " : "**or** ")) + + (f.inverted ? "**not** " : "") + `${f.name}` + }).join("\n") + "\n\n" + `This will affect ${addPlural(affected.length, "member")}`) + .setEmoji("GUILD.ROLES.CREATE") + .setStatus("Success") + ], components: [ + new MessageActionRow().addComponents([new MessageSelectMenu().setOptions(filters.map((f, index) => ({ + label: (f.inverted ? "(Not) " : "") + f.name, + value: index.toString() + }))).setMinValues(1).setMaxValues(filters.length).setCustomId("select").setPlaceholder("Remove a filter")]), + new MessageActionRow().addComponents([ + new MessageButton() + .setLabel("Apply") + .setStyle("PRIMARY") + .setCustomId("apply") + .setEmoji(client.emojis.cache.get(getEmojiByName("CONTROL.TICK", "id"))) + .setDisabled(affected.length === 0), + new MessageButton() + .setLabel("Add filter") + .setStyle("PRIMARY") + .setCustomId("add") + .setEmoji(client.emojis.cache.get(getEmojiByName("ICONS.FILTER", "id"))) + .setDisabled(filters.length >= 25) + ]) + ]}) + break + } +} + +const check = async (interaction: CommandInteraction, defaultCheck: WrappedCheck) => { + let member = (interaction.member as GuildMember) + let me = (interaction.guild.me as GuildMember) + if (!me.permissions.has("MANAGE_ROLES")) throw "I do not have the Manage roles permission"; + // Allow the owner to role anyone + if (member.id == interaction.guild.ownerId) return true + // Check if the user has manage_roles permission + if (! member.permissions.has("MANAGE_ROLES")) throw "You do not have the Manage roles permission"; + // Allow role + return true; +} + +export { command }; +export { callback }; +export { check }; \ No newline at end of file