diff options
| author | IRONM00N <64110067+IRONM00N@users.noreply.github.com> | 2022-01-23 18:13:05 -0500 | 
|---|---|---|
| committer | IRONM00N <64110067+IRONM00N@users.noreply.github.com> | 2022-01-23 18:13:05 -0500 | 
| commit | a3f8d3884a1deca5eccfb6d990e2a7b42fbbe08a (patch) | |
| tree | 9b0f8ed8a93c22c90512751e3f2f5937e1925760 /src/lib | |
| parent | 5557677f1570eb564a30cfcebb6030235dc84d47 (diff) | |
| download | tanzanite-a3f8d3884a1deca5eccfb6d990e2a7b42fbbe08a.tar.gz tanzanite-a3f8d3884a1deca5eccfb6d990e2a7b42fbbe08a.tar.bz2 tanzanite-a3f8d3884a1deca5eccfb6d990e2a7b42fbbe08a.zip | |
fix discord.js breaking changes, some other stuff
Diffstat (limited to 'src/lib')
45 files changed, 649 insertions, 399 deletions
| diff --git a/src/lib/common/AutoMod.ts b/src/lib/common/AutoMod.ts index 7a30820..4de242a 100644 --- a/src/lib/common/AutoMod.ts +++ b/src/lib/common/AutoMod.ts @@ -1,5 +1,14 @@  import { banResponse, Moderation, type BushButtonInteraction, type BushMessage } from '#lib'; -import { GuildMember, MessageActionRow, MessageButton, MessageEmbed, type TextChannel } from 'discord.js'; +import { +	ActionRow, +	ButtonComponent, +	ButtonStyle, +	ChannelType, +	GuildMember, +	MessageEmbed, +	Permissions, +	type TextChannel +} from 'discord.js';  /**   * Handles auto moderation functionality. @@ -28,7 +37,7 @@ export class AutoMod {  	 * Handles the auto moderation  	 */  	private async handle() { -		if (this.message.channel.type === 'DM' || !this.message.guild) return; +		if (this.message.channel.type === ChannelType.DM || !this.message.guild) return;  		if (!(await this.message.guild.hasFeature('automod'))) return;  		if (this.message.author.bot) return;  		if (this.message.author.isOwner()) return; @@ -110,7 +119,10 @@ export class AutoMod {  		const includes = this.message.content.toLocaleLowerCase().includes;  		if (!includes('@everyone') && !includes('@here')) return;  		// It would be bad if we deleted a message that actually pinged @everyone or @here -		if (this.message.member?.permissionsIn(this.message.channelId).has('MENTION_EVERYONE') || this.message.mentions.everyone) +		if ( +			this.message.member?.permissionsIn(this.message.channelId).has(Permissions.FLAGS.MENTION_EVERYONE) || +			this.message.mentions.everyone +		)  			return;  		if ( @@ -142,9 +154,9 @@ export class AutoMod {  				components:  					Severity.TEMP_MUTE >= 2  						? [ -								new MessageActionRow().addComponents( -									new MessageButton() -										.setStyle('DANGER') +								new ActionRow().addComponents( +									new ButtonComponent() +										.setStyle(ButtonStyle.Danger)  										.setLabel('Ban User')  										.setCustomId(`automod;ban;${this.message.author.id};everyone mention and scam phrase`)  								) @@ -263,9 +275,9 @@ export class AutoMod {  			components:  				highestOffence.severity >= 2  					? [ -							new MessageActionRow().addComponents( -								new MessageButton() -									.setStyle('DANGER') +							new ActionRow().addComponents( +								new ButtonComponent() +									.setStyle(ButtonStyle.Danger)  									.setLabel('Ban User')  									.setCustomId(`automod;ban;${this.message.author.id};${highestOffence.reason}`)  							) @@ -279,7 +291,7 @@ export class AutoMod {  	 * @param interaction The button interaction.  	 */  	public static async handleInteraction(interaction: BushButtonInteraction) { -		if (!interaction.memberPermissions?.has('BAN_MEMBERS')) +		if (!interaction.memberPermissions?.has(Permissions.FLAGS.BAN_MEMBERS))  			return interaction.reply({  				content: `${util.emojis.error} You are missing the **Ban Members** permission.`,  				ephemeral: true diff --git a/src/lib/common/ButtonPaginator.ts b/src/lib/common/ButtonPaginator.ts index 83f4219..9e72551 100644 --- a/src/lib/common/ButtonPaginator.ts +++ b/src/lib/common/ButtonPaginator.ts @@ -1,13 +1,15 @@  import { DeleteButton, type BushMessage, type BushSlashMessage } from '#lib';  import { CommandUtil } from 'discord-akairo';  import { -	MessageActionRow, -	MessageButton, +	ActionRow, +	ActionRowComponent, +	ButtonComponent, +	ButtonStyle, +	ComponentType,  	MessageEmbed,  	type MessageComponentInteraction,  	type MessageEmbedOptions  } from 'discord.js'; -import { MessageButtonStyles } from 'discord.js/typings/enums';  /**   * Sends multiple embeds with controls to switch between them @@ -94,9 +96,15 @@ export class ButtonPaginator {  		})) as BushMessage;  		const collector = this.sentMessage.createMessageComponentCollector({ -			filter: (i) => i.customId.startsWith('paginate_') && i.message?.id === this.sentMessage!.id, -			time: 300000 +			componentType: ComponentType.Button, +			filter: (i) => { +				const ret = i.customId.startsWith('paginate_') && i.message.id === this.sentMessage!.id; +				console.debug(ret); +				return ret; +			}, +			idle: 300000  		}); +		console.debug('got here');  		collector.on('collect', (i) => void this.collect(i));  		collector.on('end', () => void this.end()); @@ -107,8 +115,9 @@ export class ButtonPaginator {  	 * @param interaction The interaction received  	 */  	protected async collect(interaction: MessageComponentInteraction) { +		console.debug(1);  		if (interaction.user.id !== this.message.author.id && !client.config.owners.includes(interaction.user.id)) -			return await interaction?.deferUpdate().catch(() => null); +			return await interaction?.deferUpdate(); /* .catch(() => null); */  		switch (interaction.customId) {  			case 'paginate_beginning': @@ -121,17 +130,16 @@ export class ButtonPaginator {  				break;  			case 'paginate_stop':  				if (this.deleteOnExit) { -					await interaction.deferUpdate().catch(() => null); -					await this.sentMessage!.delete().catch(() => null); +					await interaction.deferUpdate(); /* .catch(() => null); */ +					await this.sentMessage!.delete(); /* .catch(() => null); */  					break;  				} else { -					await interaction -						?.update({ -							content: `${this.text ? `${this.text}\n` : ''}Command closed by user.`, -							embeds: [], -							components: [] -						}) -						.catch(() => null); +					await interaction?.update({ +						content: `${this.text ? `${this.text}\n` : ''}Command closed by user.`, +						embeds: [], +						components: [] +					}); +					/* .catch(() => null); */  					break;  				}  			case 'paginate_next': @@ -150,13 +158,12 @@ export class ButtonPaginator {  	 */  	protected async end() {  		if (this.sentMessage && !CommandUtil.deletedMessages.has(this.sentMessage.id)) -			await this.sentMessage -				.edit({ -					content: this.text, -					embeds: [this.embeds[this.curPage]], -					components: [this.getPaginationRow(true)] -				}) -				.catch(() => null); +			await this.sentMessage.edit({ +				content: this.text, +				embeds: [this.embeds[this.curPage]], +				components: [this.getPaginationRow(true)] +			}); +		/* .catch(() => null); */  	}  	/** @@ -164,52 +171,46 @@ export class ButtonPaginator {  	 * @param interaction The interaction received  	 */  	protected async edit(interaction: MessageComponentInteraction) { -		await interaction -			?.update({ -				content: this.text, -				embeds: [this.embeds[this.curPage]], -				components: [this.getPaginationRow()] -			}) -			.catch(() => null); +		await interaction?.update({ +			content: this.text, +			embeds: [this.embeds[this.curPage]], +			components: [this.getPaginationRow()] +		}); +		/* .catch(() => null); */  	}  	/**  	 * Generates the pagination row based on the class properties  	 * @param disableAll Whether to disable all buttons -	 * @returns The generated {@link MessageActionRow} +	 * @returns The generated {@link ActionRow}  	 */ -	protected getPaginationRow(disableAll = false): MessageActionRow { -		return new MessageActionRow().addComponents( -			new MessageButton({ -				style: MessageButtonStyles.PRIMARY, -				customId: 'paginate_beginning', -				emoji: PaginateEmojis.BEGGING, -				disabled: disableAll || this.curPage === 0 -			}), -			new MessageButton({ -				style: MessageButtonStyles.PRIMARY, -				customId: 'paginate_back', -				emoji: PaginateEmojis.BACK, -				disabled: disableAll || this.curPage === 0 -			}), -			new MessageButton({ -				style: MessageButtonStyles.PRIMARY, -				customId: 'paginate_stop', -				emoji: PaginateEmojis.STOP, -				disabled: disableAll -			}), -			new MessageButton({ -				style: MessageButtonStyles.PRIMARY, -				customId: 'paginate_next', -				emoji: PaginateEmojis.FORWARD, -				disabled: disableAll || this.curPage === this.numPages - 1 -			}), -			new MessageButton({ -				style: MessageButtonStyles.PRIMARY, -				customId: 'paginate_end', -				emoji: PaginateEmojis.END, -				disabled: disableAll || this.curPage === this.numPages - 1 -			}) +	protected getPaginationRow(disableAll = false): ActionRow<ActionRowComponent> { +		return new ActionRow().addComponents( +			new ButtonComponent() +				.setStyle(ButtonStyle.Primary) +				.setCustomId('paginate_beginning') +				.setEmoji(PaginateEmojis.BEGINNING) +				.setDisabled(disableAll || this.curPage === 0), +			new ButtonComponent() +				.setStyle(ButtonStyle.Primary) +				.setCustomId('paginate_back') +				.setEmoji(PaginateEmojis.BACK) +				.setDisabled(disableAll || this.curPage === 0), +			new ButtonComponent() +				.setStyle(ButtonStyle.Primary) +				.setCustomId('paginate_stop') +				.setEmoji(PaginateEmojis.STOP) +				.setDisabled(disableAll), +			new ButtonComponent() +				.setStyle(ButtonStyle.Primary) +				.setCustomId('paginate_next') +				.setEmoji(PaginateEmojis.FORWARD) +				.setDisabled(disableAll || this.curPage === this.embeds.length - 1), +			new ButtonComponent() +				.setStyle(ButtonStyle.Primary) +				.setCustomId('paginate_end') +				.setEmoji(PaginateEmojis.END) +				.setDisabled(disableAll || this.curPage === this.embeds.length - 1)  		);  	} @@ -235,10 +236,10 @@ export class ButtonPaginator {  	}  } -export const enum PaginateEmojis { -	BEGGING = '853667381335162910', -	BACK = '853667410203770881', -	STOP = '853667471110570034', -	FORWARD = '853667492680564747', -	END = '853667514915225640' -} +export const PaginateEmojis = { +	BEGINNING: { id: '853667381335162910', name: 'w_paginate_beginning', animated: false } as const, +	BACK: { id: '853667410203770881', name: 'w_paginate_back', animated: false } as const, +	STOP: { id: '853667471110570034', name: 'w_paginate_stop', animated: false } as const, +	FORWARD: { id: '853667492680564747', name: 'w_paginate_next', animated: false } as const, +	END: { id: '853667514915225640', name: 'w_paginate_end', animated: false } as const +} as const; diff --git a/src/lib/common/ConfirmationPrompt.ts b/src/lib/common/ConfirmationPrompt.ts index a4acf83..97f18b3 100644 --- a/src/lib/common/ConfirmationPrompt.ts +++ b/src/lib/common/ConfirmationPrompt.ts @@ -1,6 +1,5 @@  import { type BushMessage, type BushSlashMessage } from '#lib'; -import { MessageActionRow, MessageButton, type MessageComponentInteraction, type MessageOptions } from 'discord.js'; -import { MessageButtonStyles } from 'discord.js/typings/enums'; +import { ActionRow, ButtonComponent, ButtonStyle, type MessageComponentInteraction, type MessageOptions } from 'discord.js';  /**   * Sends a message with buttons for the user to confirm or cancel the action. @@ -30,19 +29,17 @@ export class ConfirmationPrompt {  	 */  	protected async send(): Promise<boolean> {  		this.messageOptions.components = [ -			new MessageActionRow().addComponents( -				new MessageButton({ -					style: MessageButtonStyles.SUCCESS, -					customId: 'confirmationPrompt_confirm', -					emoji: util.emojis.successFull, -					label: 'Yes' -				}), -				new MessageButton({ -					style: MessageButtonStyles.DANGER, -					customId: 'confirmationPrompt_deny', -					emoji: util.emojis.errorFull, -					label: 'No' -				}) +			new ActionRow().addComponents( +				new ButtonComponent() +					.setStyle(ButtonStyle.Primary) +					.setCustomId('confirmationPrompt_confirm') +					.setEmoji({ id: util.emojis.successFull, name: 'successFull', animated: false }) +					.setLabel('Yes'), +				new ButtonComponent() +					.setStyle(ButtonStyle.Danger) +					.setCustomId('confirmationPrompt_cancel') +					.setEmoji({ id: util.emojis.errorFull, name: 'errorFull', animated: false }) +					.setLabel('No')  			)  		]; diff --git a/src/lib/common/DeleteButton.ts b/src/lib/common/DeleteButton.ts index b666a4f..edc40fe 100644 --- a/src/lib/common/DeleteButton.ts +++ b/src/lib/common/DeleteButton.ts @@ -1,7 +1,6 @@  import { PaginateEmojis, type BushMessage, type BushSlashMessage } from '#lib';  import { CommandUtil } from 'discord-akairo'; -import { MessageActionRow, MessageButton, type MessageComponentInteraction, type MessageOptions } from 'discord.js'; -import { MessageButtonStyles } from 'discord.js/typings/enums'; +import { ActionRow, ButtonComponent, ButtonStyle, MessageComponentInteraction, type MessageOptions } from 'discord.js';  /**   * Sends a message with a button for the user to delete it. @@ -59,13 +58,12 @@ export class DeleteButton {  	 */  	protected updateComponents(edit = false, disable = false): void {  		this.messageOptions.components = [ -			new MessageActionRow().addComponents( -				new MessageButton({ -					style: MessageButtonStyles.PRIMARY, -					customId: 'paginate__stop', -					emoji: PaginateEmojis.STOP, -					disabled: disable -				}) +			new ActionRow().addComponents( +				new ButtonComponent() +					.setStyle(ButtonStyle.Primary) +					.setCustomId('paginate__stop') +					.setEmoji(PaginateEmojis.STOP) +					.setDisabled(disable)  			)  		];  		if (edit) { diff --git a/src/lib/common/util/Moderation.ts b/src/lib/common/util/Moderation.ts index e5cb872..84d9fbf 100644 --- a/src/lib/common/util/Moderation.ts +++ b/src/lib/common/util/Moderation.ts @@ -10,7 +10,7 @@ import {  	type BushUserResolvable,  	type ModLogType  } from '#lib'; -import { MessageEmbed, type Snowflake } from 'discord.js'; +import { MessageEmbed, Permissions, type Snowflake } from 'discord.js';  /**   * A utility class with moderation-related methods. @@ -70,7 +70,11 @@ export class Moderation {  		) {  			return `${util.emojis.error} You cannot ${type} **${victim.user.tag}** because they have higher or equal role hierarchy as I do.`;  		} -		if (checkModerator && victim.permissions.has('MANAGE_MESSAGES') && !(type.startsWith('un') && moderator.id === victim.id)) { +		if ( +			checkModerator && +			victim.permissions.has(Permissions.FLAGS.MANAGE_MESSAGES) && +			!(type.startsWith('un') && moderator.id === victim.id) +		) {  			if (await moderator.guild.hasFeature('modsCanPunishMods')) {  				return true;  			} else { diff --git a/src/lib/extensions/discord-akairo/BushClient.ts b/src/lib/extensions/discord-akairo/BushClient.ts index cb1e50b..01620a8 100644 --- a/src/lib/extensions/discord-akairo/BushClient.ts +++ b/src/lib/extensions/discord-akairo/BushClient.ts @@ -24,6 +24,7 @@ import { patch, type PatchedElements } from '@notenoughupdates/events-intercept'  import * as Sentry from '@sentry/node';  import { AkairoClient, ContextMenuCommandHandler, version as akairoVersion } from 'discord-akairo';  import { +	ActivityType,  	Intents,  	Options,  	Structures, @@ -61,7 +62,7 @@ import { BushConstants } from '../../utils/BushConstants.js';  import { BushLogger } from '../../utils/BushLogger.js';  import { BushButtonInteraction } from '../discord.js/BushButtonInteraction.js';  import { BushCategoryChannel } from '../discord.js/BushCategoryChannel.js'; -import { BushCommandInteraction } from '../discord.js/BushCommandInteraction.js'; +import { BushChatInputCommandInteraction } from '../discord.js/BushChatInputCommandInteraction.js';  import { BushDMChannel } from '../discord.js/BushDMChannel.js';  import { BushGuild } from '../discord.js/BushGuild.js';  import { BushGuildEmoji } from '../discord.js/BushGuildEmoji.js'; @@ -193,7 +194,7 @@ export class BushClient<Ready extends boolean = boolean> extends AkairoClient<Re  				activities: [  					{  						name: 'Beep Boop', -						type: 'WATCHING' +						type: ActivityType.Watching  					}  				],  				status: 'online' @@ -252,7 +253,7 @@ export class BushClient<Ready extends boolean = boolean> extends AkairoClient<Re  			automateCategories: false,  			autoRegisterSlashCommands: true,  			skipBuiltInPostInhibitors: true, -			useSlashPermissions: true, +			useSlashPermissions: false,  			aliasReplacement: /-/g  		});  		this.contextMenuCommandHandler = new ContextMenuCommandHandler(this, { @@ -320,7 +321,7 @@ export class BushClient<Ready extends boolean = boolean> extends AkairoClient<Re  		Structures.extend('VoiceState', () => BushVoiceState);  		Structures.extend('Role', () => BushRole);  		Structures.extend('User', () => BushUser); -		Structures.extend('CommandInteraction', () => BushCommandInteraction); +		Structures.extend('ChatInputCommandInteraction', () => BushChatInputCommandInteraction);  		Structures.extend('ButtonInteraction', () => BushButtonInteraction);  		Structures.extend('SelectMenuInteraction', () => BushSelectMenuInteraction);  	} @@ -440,13 +441,17 @@ export class BushClient<Ready extends boolean = boolean> extends AkairoClient<Re  	 */  	public async start() {  		this.intercept('ready', async (arg, done) => { -			await this.guilds.fetch(); -			const promises = this.guilds.cache.map((guild) => { -				return guild.members.fetch(); -			}); +			console.debug('ready start'); +			console.time('ready'); +			const promises = this.guilds.cache +				.filter((g) => g.large) +				.map((guild) => { +					return guild.members.fetch(); +				});  			await Promise.all(promises);  			this.customReady = true;  			this.taskHandler.startAll(); +			console.timeEnd('ready');  			return done(null, `intercepted ${arg}`);  		}); diff --git a/src/lib/extensions/discord-akairo/BushClientUtil.ts b/src/lib/extensions/discord-akairo/BushClientUtil.ts index 12becd3..79aa4c1 100644 --- a/src/lib/extensions/discord-akairo/BushClientUtil.ts +++ b/src/lib/extensions/discord-akairo/BushClientUtil.ts @@ -26,13 +26,13 @@ import {  	GuildMember,  	Message,  	MessageEmbed, +	Permissions,  	ThreadMember,  	User,  	Util as DiscordUtil,  	type ColorResolvable,  	type CommandInteraction,  	type InteractionReplyOptions, -	type PermissionResolvable,  	type Snowflake,  	type TextChannel,  	type UserResolvable @@ -43,6 +43,8 @@ import { inspect, promisify } from 'util';  import CommandErrorListener from '../../../listeners/commands/commandError.js';  import { Format } from '../../common/util/Format.js'; +export type StripPrivate<T> = { [K in keyof T]: T[K] extends Record<string, any> ? StripPrivate<T[K]> : T[K] }; +  export class BushClientUtil extends ClientUtil {  	/**  	 * The client. @@ -208,7 +210,7 @@ export class BushClientUtil extends ClientUtil {  		if (author)  			embed = embed.setAuthor({  				name: author.username, -				iconURL: author.displayAvatarURL({ dynamic: true }), +				iconURL: author.displayAvatarURL(),  				url: `https://discord.com/users/${author.id}`  			});  		if (color) embed = embed.setColor(color); @@ -425,7 +427,7 @@ export class BushClientUtil extends ClientUtil {  	 * @returns The combined elements or `ifEmpty`.  	 *  	 * @example -	 * const permissions = oxford(['ADMINISTRATOR', 'SEND_MESSAGES', 'MANAGE_MESSAGES'], 'and', 'none'); +	 * const permissions = oxford([Permissions.FLAGS.ADMINISTRATOR, Permissions.FLAGS.SEND_MESSAGES, Permissions.FLAGS.MANAGE_MESSAGES], 'and', 'none');  	 * console.log(permissions); // ADMINISTRATOR, SEND_MESSAGES and MANAGE_MESSAGES  	 */  	public oxford(array: string[], conjunction: string, ifEmpty?: string): string | undefined { @@ -785,10 +787,10 @@ export class BushClientUtil extends ClientUtil {  	 * @param permissions The permissions to check for.  	 * @returns The missing permissions or null if none are missing.  	 */ -	public userGuildPermCheck(message: BushMessage | BushSlashMessage, permissions: PermissionResolvable) { +	public userGuildPermCheck(message: BushMessage | BushSlashMessage, permissions: bigint[]) {  		const missing = message.member?.permissions.missing(permissions) ?? []; -		return missing.length ? missing : null; +		return missing.length ? missing.map((p) => Permissions.FLAGS[p]) : null;  	}  	/** @@ -797,10 +799,10 @@ export class BushClientUtil extends ClientUtil {  	 * @param permissions The permissions to check for.  	 * @returns The missing permissions or null if none are missing.  	 */ -	public clientGuildPermCheck(message: BushMessage | BushSlashMessage, permissions: PermissionResolvable) { +	public clientGuildPermCheck(message: BushMessage | BushSlashMessage, permissions: bigint[]) {  		const missing = message.guild?.me?.permissions.missing(permissions) ?? []; -		return missing.length ? missing : null; +		return missing.length ? missing.map((p) => Permissions.FLAGS[p]) : null;  	}  	/** @@ -811,19 +813,18 @@ export class BushClientUtil extends ClientUtil {  	 * @param checkChannel Whether to check the channel permissions instead of the guild permissions.  	 * @returns The missing permissions or null if none are missing.  	 */ -	public clientSendAndPermCheck( -		message: BushMessage | BushSlashMessage, -		permissions: PermissionResolvable = [], -		checkChannel = false -	) { +	public clientSendAndPermCheck(message: BushMessage | BushSlashMessage, permissions: bigint[] = [], checkChannel = false) {  		const missing = []; -		const sendPerm = message.channel!.isThread() ? 'SEND_MESSAGES' : 'SEND_MESSAGES_IN_THREADS'; +		const sendPerm = message.channel!.isThread() ? Permissions.FLAGS.SEND_MESSAGES : Permissions.FLAGS.SEND_MESSAGES_IN_THREADS;  		if (!message.guild!.me!.permissionsIn(message.channel!.id!).has(sendPerm)) missing.push(sendPerm);  		missing.push(  			...(checkChannel -				? message.guild!.me!.permissionsIn(message.channel!.id!).missing(permissions) +				? message +						.guild!.me!.permissionsIn(message.channel!.id!) +						.missing(permissions) +						.map((p) => Permissions.FLAGS[p])  				: this.clientGuildPermCheck(message, permissions) ?? [])  		); @@ -894,6 +895,12 @@ export class BushClientUtil extends ClientUtil {  		return Object.fromEntries(values);  	} +	public get invite() { +		return `https://discord.com/api/oauth2/authorize?client_id=${client.user!.id}&permissions=${ +			Permissions.ALL +		}&scope=bot%20applications.commands`; +	} +  	/**  	 * A wrapper for the Argument class that adds custom typings.  	 */ diff --git a/src/lib/extensions/discord-akairo/BushCommand.ts b/src/lib/extensions/discord-akairo/BushCommand.ts index fb488be..0456b80 100644 --- a/src/lib/extensions/discord-akairo/BushCommand.ts +++ b/src/lib/extensions/discord-akairo/BushCommand.ts @@ -315,12 +315,12 @@ export interface BaseBushCommandOptions  	/**  	 * Permissions required by the client to run this command.  	 */ -	clientPermissions: PermissionResolvable | PermissionResolvable[] | BushMissingPermissionSupplier; +	clientPermissions: bigint | bigint[] | BushMissingPermissionSupplier;  	/**  	 * Permissions required by the user to run this command.  	 */ -	userPermissions: PermissionResolvable | PermissionResolvable[] | BushMissingPermissionSupplier; +	userPermissions: bigint | bigint[] | BushMissingPermissionSupplier;  	/**  	 * Restrict this argument to owners diff --git a/src/lib/extensions/discord-akairo/BushSlashMessage.ts b/src/lib/extensions/discord-akairo/BushSlashMessage.ts index cefd360..d342ea6 100644 --- a/src/lib/extensions/discord-akairo/BushSlashMessage.ts +++ b/src/lib/extensions/discord-akairo/BushSlashMessage.ts @@ -1,4 +1,5 @@  import { +	BushGuildTextBasedChannel,  	type BushClient,  	type BushCommandUtil,  	type BushGuild, @@ -7,14 +8,14 @@ import {  	type BushUser  } from '#lib';  import { AkairoMessage } from 'discord-akairo'; -import { type CommandInteraction } from 'discord.js'; +import { type ChatInputCommandInteraction, type ContextMenuCommandInteraction } from 'discord.js';  export class BushSlashMessage extends AkairoMessage {  	public declare client: BushClient;  	public declare util: BushCommandUtil<BushSlashMessage>;  	public declare author: BushUser;  	public declare member: BushGuildMember | null; -	public constructor(client: BushClient, interaction: CommandInteraction) { +	public constructor(client: BushClient, interaction: ChatInputCommandInteraction | ContextMenuCommandInteraction) {  		super(client, interaction);  	}  } @@ -22,5 +23,10 @@ export class BushSlashMessage extends AkairoMessage {  export interface BushSlashMessage extends AkairoMessage {  	get channel(): BushTextBasedChannel | null;  	get guild(): BushGuild | null; -	inGuild(): this is this & { guild: BushGuild; member: BushGuildMember }; +	inGuild(): this is BushSlashMessageInGuild & this; +} + +interface BushSlashMessageInGuild { +	guild: BushGuild; +	channel: BushGuildTextBasedChannel;  } diff --git a/src/lib/extensions/discord.js/BushApplicationCommandManager.d.ts b/src/lib/extensions/discord.js/BushApplicationCommandManager.ts index 2aa366d..24a7b22 100644 --- a/src/lib/extensions/discord.js/BushApplicationCommandManager.d.ts +++ b/src/lib/extensions/discord.js/BushApplicationCommandManager.ts @@ -3,10 +3,12 @@ import type {  	BushApplicationCommandPermissionsManager,  	BushApplicationCommandResolvable,  	BushClient, -	BushGuildResolvable +	BushGuildResolvable, +	StripPrivate  } from '#lib';  import type { APIApplicationCommand } from 'discord-api-types';  import { +	ApplicationCommandManager,  	CachedManager,  	type ApplicationCommandData,  	type Collection, @@ -17,11 +19,14 @@ import {  /**   * Manages API methods for application commands and stores their cache.   */ -export class BushApplicationCommandManager< -	ApplicationCommandScope = BushApplicationCommand<{ guild: BushGuildResolvable }>, -	PermissionsOptionsExtras = { guild: BushGuildResolvable }, -	PermissionsGuildType = null -> extends CachedManager<Snowflake, ApplicationCommandScope, BushApplicationCommandResolvable> { +export declare class BushApplicationCommandManager< +		ApplicationCommandScope = BushApplicationCommand<{ guild: BushGuildResolvable }>, +		PermissionsOptionsExtras = { guild: BushGuildResolvable }, +		PermissionsGuildType = null +	> +	extends CachedManager<Snowflake, ApplicationCommandScope, BushApplicationCommandResolvable> +	implements StripPrivate<ApplicationCommandManager<ApplicationCommandScope, PermissionsOptionsExtras, PermissionsGuildType>> +{  	public constructor(client: BushClient, iterable?: Iterable<unknown>);  	/** diff --git a/src/lib/extensions/discord.js/BushApplicationCommandPermissionsManager.d.ts b/src/lib/extensions/discord.js/BushApplicationCommandPermissionsManager.ts index ff32be4..f07bde9 100644 --- a/src/lib/extensions/discord.js/BushApplicationCommandPermissionsManager.d.ts +++ b/src/lib/extensions/discord.js/BushApplicationCommandPermissionsManager.ts @@ -1,6 +1,7 @@  import type { BushClient, BushRoleResolvable, BushUserResolvable } from '#lib';  import type { APIApplicationCommandPermission } from 'discord-api-types';  import { +	ApplicationCommandPermissionType,  	BaseManager,  	type ApplicationCommand,  	type ApplicationCommandManager, @@ -11,12 +12,11 @@ import {  	type GuildApplicationCommandPermissionData,  	type Snowflake  } from 'discord.js'; -import type { ApplicationCommandPermissionTypes } from 'discord.js/typings/enums';  /**   * Manages API methods for permissions of Application Commands.   */ -export class BushApplicationCommandPermissionsManager< +export declare class BushApplicationCommandPermissionsManager<  	BaseOptions,  	FetchSingleOptions,  	FullPermissionsOptions, @@ -179,6 +179,6 @@ export class BushApplicationCommandPermissionsManager<  	private static transformPermissions(  		permissions: ApplicationCommandPermissionData,  		received: true -	): Omit<APIApplicationCommandPermission, 'type'> & { type: keyof ApplicationCommandPermissionTypes }; +	): Omit<APIApplicationCommandPermission, 'type'> & { type: keyof ApplicationCommandPermissionType };  	private static transformPermissions(permissions: ApplicationCommandPermissionData): APIApplicationCommandPermission;  } diff --git a/src/lib/extensions/discord.js/BushBaseGuildEmojiManager.d.ts b/src/lib/extensions/discord.js/BushBaseGuildEmojiManager.ts index 347ff65..66abbc2 100644 --- a/src/lib/extensions/discord.js/BushBaseGuildEmojiManager.d.ts +++ b/src/lib/extensions/discord.js/BushBaseGuildEmojiManager.ts @@ -1,11 +1,14 @@  import type { BushClient, BushEmojiIdentifierResolvable, BushEmojiResolvable, BushGuildEmoji } from '#lib'; -import { CachedManager, type Snowflake } from 'discord.js'; +import { BaseGuildEmojiManager, CachedManager, type Snowflake } from 'discord.js';  import { type RawGuildEmojiData } from 'discord.js/typings/rawDataTypes';  /**   * Holds methods to resolve GuildEmojis and stores their cache.   */ -export class BushBaseGuildEmojiManager extends CachedManager<Snowflake, BushGuildEmoji, BushEmojiResolvable> { +export declare class BushBaseGuildEmojiManager +	extends CachedManager<Snowflake, BushGuildEmoji, BushEmojiResolvable> +	implements BaseGuildEmojiManager +{  	public constructor(client: BushClient, iterable?: Iterable<RawGuildEmojiData>);  	/** diff --git a/src/lib/extensions/discord.js/BushBaseGuildVoiceChannel.d.ts b/src/lib/extensions/discord.js/BushBaseGuildVoiceChannel.ts index 21be206..2ffb2fd 100644 --- a/src/lib/extensions/discord.js/BushBaseGuildVoiceChannel.d.ts +++ b/src/lib/extensions/discord.js/BushBaseGuildVoiceChannel.ts @@ -6,7 +6,7 @@ import { BushGuildMember } from './BushGuildMember';  /**   * Represents a voice-based guild channel on Discord.   */ -export class BushBaseGuildVoiceChannel extends BaseGuildVoiceChannel { +export declare class BushBaseGuildVoiceChannel extends BaseGuildVoiceChannel {  	public readonly members: Collection<Snowflake, BushGuildMember>;  	public guild: BushGuild;  	public readonly parent: BushCategoryChannel | null; diff --git a/src/lib/extensions/discord.js/BushCategoryChannel.ts b/src/lib/extensions/discord.js/BushCategoryChannel.ts index b711a54..ac82bf0 100644 --- a/src/lib/extensions/discord.js/BushCategoryChannel.ts +++ b/src/lib/extensions/discord.js/BushCategoryChannel.ts @@ -1,4 +1,19 @@ -import { BushNonThreadGuildBasedChannel, type BushClient, type BushGuild, type BushGuildMember } from '#lib'; +import { +	BushDMChannel, +	BushGuildBasedChannel, +	BushNewsChannel, +	BushNonThreadGuildBasedChannel, +	BushStageChannel, +	BushStoreChannel, +	BushTextBasedChannel, +	BushTextChannel, +	BushThreadChannel, +	BushVoiceBasedChannel, +	BushVoiceChannel, +	type BushClient, +	type BushGuild, +	type BushGuildMember +} from '#lib';  import { CategoryChannel, type Collection, type Snowflake } from 'discord.js';  import { type RawGuildChannelData } from 'discord.js/typings/rawDataTypes'; @@ -16,3 +31,17 @@ export class BushCategoryChannel extends CategoryChannel {  		super(guild, data, client, immediatePatch);  	}  } + +export interface BushCategoryChannel extends CategoryChannel { +	isText(): this is BushTextChannel; +	isDM(): this is BushDMChannel; +	isVoice(): this is BushVoiceChannel; +	isCategory(): this is BushCategoryChannel; +	isNews(): this is BushNewsChannel; +	// eslint-disable-next-line deprecation/deprecation +	isStore(): this is BushStoreChannel; +	isThread(): this is BushThreadChannel; +	isStage(): this is BushStageChannel; +	isTextBased(): this is BushGuildBasedChannel & BushTextBasedChannel; +	isVoiceBased(): this is BushVoiceBasedChannel; +} diff --git a/src/lib/extensions/discord.js/BushChannel.d.ts b/src/lib/extensions/discord.js/BushChannel.d.ts deleted file mode 100644 index 42443ba..0000000 --- a/src/lib/extensions/discord.js/BushChannel.d.ts +++ /dev/null @@ -1,24 +0,0 @@ -import type { BushClient, BushTextBasedChannel, BushThreadChannel } from '#lib'; -import { Channel, type ChannelMention, type Snowflake } from 'discord.js'; -import type { ChannelTypes } from 'discord.js/typings/enums'; -import type { RawChannelData } from 'discord.js/typings/rawDataTypes'; -import { BushBaseGuildVoiceChannel } from './BushBaseGuildVoiceChannel'; - -/** - * Represents any channel on Discord. - */ -export class BushChannel extends Channel { -	public constructor(client: BushClient, data?: RawChannelData, immediatePatch?: boolean); -	public readonly createdAt: Date; -	public readonly createdTimestamp: number; -	public deleted: boolean; -	public id: Snowflake; -	public readonly partial: false; -	public type: keyof typeof ChannelTypes; -	public delete(): Promise<this>; -	public fetch(force?: boolean): Promise<this>; -	public isText(): this is BushTextBasedChannel; -	public isVoice(): this is BushBaseGuildVoiceChannel; -	public isThread(): this is BushThreadChannel; -	public toString(): ChannelMention; -} diff --git a/src/lib/extensions/discord.js/BushChannel.ts b/src/lib/extensions/discord.js/BushChannel.ts new file mode 100644 index 0000000..50ec723 --- /dev/null +++ b/src/lib/extensions/discord.js/BushChannel.ts @@ -0,0 +1,41 @@ +/* eslint-disable deprecation/deprecation */ +import type { +	BushCategoryChannel, +	BushClient, +	BushDMChannel, +	BushNewsChannel, +	BushStageChannel, +	BushStoreChannel, +	BushTextBasedChannel, +	BushTextChannel, +	BushThreadChannel, +	BushVoiceBasedChannel, +	BushVoiceChannel +} from '#lib'; +import { Channel, ChannelType, type Snowflake } from 'discord.js'; +import type { RawChannelData } from 'discord.js/typings/rawDataTypes'; + +/** + * Represents any channel on Discord. + */ +export declare class BushChannel extends Channel { +	public constructor(client: BushClient, data?: RawChannelData, immediatePatch?: boolean); +	public readonly createdAt: Date; +	public readonly createdTimestamp: number; +	public deleted: boolean; +	public id: Snowflake; +	public readonly partial: false; +	public type: ChannelType; +	public delete(): Promise<this>; +	public fetch(force?: boolean): Promise<this>; +	public isText(): this is BushTextChannel; +	public isDM(): this is BushDMChannel; +	public isVoice(): this is BushVoiceChannel; +	public isCategory(): this is BushCategoryChannel; +	public isNews(): this is BushNewsChannel; +	public isStore(): this is BushStoreChannel; +	public isThread(): this is BushThreadChannel; +	public isStage(): this is BushStageChannel; +	public isTextBased(): this is BushTextBasedChannel; +	public isVoiceBased(): this is BushVoiceBasedChannel; +} diff --git a/src/lib/extensions/discord.js/BushChannelManager.d.ts b/src/lib/extensions/discord.js/BushChannelManager.ts index 514cdd3..ff93209 100644 --- a/src/lib/extensions/discord.js/BushChannelManager.d.ts +++ b/src/lib/extensions/discord.js/BushChannelManager.ts @@ -1,11 +1,14 @@  import type { BushAnyChannel, BushChannelResolvable } from '#lib'; -import { CachedManager, type Client, type FetchChannelOptions, type Snowflake } from 'discord.js'; +import { CachedManager, ChannelManager, type Client, type FetchChannelOptions, type Snowflake } from 'discord.js';  import type { RawChannelData } from 'discord.js/typings/rawDataTypes';  /**   * A manager of channels belonging to a client   */ -export class BushChannelManager extends CachedManager<Snowflake, BushAnyChannel, BushChannelResolvable> { +export declare class BushChannelManager +	extends CachedManager<Snowflake, BushAnyChannel, BushChannelResolvable> +	implements ChannelManager +{  	public constructor(client: Client, iterable: Iterable<RawChannelData>);  	/** diff --git a/src/lib/extensions/discord.js/BushCommandInteraction.ts b/src/lib/extensions/discord.js/BushChatInputCommandInteraction.ts index f4be5ed..56bef21 100644 --- a/src/lib/extensions/discord.js/BushCommandInteraction.ts +++ b/src/lib/extensions/discord.js/BushChatInputCommandInteraction.ts @@ -11,7 +11,7 @@ import type {  	BushUser  } from '#lib';  import type { APIInteractionGuildMember } from '@discordjs/builders/node_modules/discord-api-types'; -import { CommandInteraction, type CacheType, type CacheTypeReducer, type Invite, type Snowflake } from 'discord.js'; +import { ChatInputCommandInteraction, type CacheType, type CacheTypeReducer, type Invite, type Snowflake } from 'discord.js';  import type { RawCommandInteractionData } from 'discord.js/typings/rawDataTypes';  export type BushGuildResolvable = @@ -26,7 +26,7 @@ export type BushGuildResolvable =  /**   * Represents a command interaction.   */ -export class BushCommandInteraction<Cached extends CacheType = CacheType> extends CommandInteraction<Cached> { +export class BushChatInputCommandInteraction<Cached extends CacheType = CacheType> extends ChatInputCommandInteraction<Cached> {  	public declare readonly client: BushClient;  	public declare readonly command: BushApplicationCommand | BushApplicationCommand<{ guild: BushGuildResolvable }> | null;  	public declare readonly channel: CacheTypeReducer< diff --git a/src/lib/extensions/discord.js/BushClientEvents.d.ts b/src/lib/extensions/discord.js/BushClientEvents.ts index b5ad749..02f0017 100644 --- a/src/lib/extensions/discord.js/BushClientEvents.d.ts +++ b/src/lib/extensions/discord.js/BushClientEvents.ts @@ -43,10 +43,7 @@ import type {  export interface BushClientEvents extends AkairoClientEvents {  	applicationCommandCreate: [command: BushApplicationCommand];  	applicationCommandDelete: [command: BushApplicationCommand]; -	applicationCommandUpdate: [ -		oldCommand: BushApplicationCommand | null, -		newCommand: BushApplicationCommand -	]; +	applicationCommandUpdate: [oldCommand: BushApplicationCommand | null, newCommand: BushApplicationCommand];  	channelCreate: [channel: BushNonThreadGuildBasedChannel];  	channelDelete: [channel: BushDMChannel | BushNonThreadGuildBasedChannel];  	channelPinsUpdate: [channel: BushTextBasedChannel, date: Date]; @@ -78,37 +75,18 @@ export interface BushClientEvents extends AkairoClientEvents {  			nonce: string | undefined;  		}  	]; -	guildMemberUpdate: [ -		oldMember: BushGuildMember | PartialBushGuildMember, -		newMember: BushGuildMember -	]; +	guildMemberUpdate: [oldMember: BushGuildMember | PartialBushGuildMember, newMember: BushGuildMember];  	guildUpdate: [oldGuild: BushGuild, newGuild: BushGuild];  	inviteCreate: [invite: Invite];  	inviteDelete: [invite: Invite];  	messageCreate: [message: BushMessage];  	messageDelete: [message: BushMessage | PartialBushMessage]; -	messageReactionRemoveAll: [ -		message: BushMessage | PartialBushMessage, -		reactions: Collection<string, BushMessageReaction> -	]; -	messageReactionRemoveEmoji: [ -		reaction: BushMessageReaction | PartialBushMessageReaction -	]; -	messageDeleteBulk: [ -		messages: Collection<Snowflake, BushMessage | PartialBushMessage> -	]; -	messageReactionAdd: [ -		reaction: BushMessageReaction | PartialBushMessageReaction, -		user: BushUser | PartialBushUser -	]; -	messageReactionRemove: [ -		reaction: BushMessageReaction | PartialBushMessageReaction, -		user: BushUser | PartialBushUser -	]; -	messageUpdate: [ -		oldMessage: BushMessage | PartialBushMessage, -		newMessage: BushMessage | PartialBushMessage -	]; +	messageReactionRemoveAll: [message: BushMessage | PartialBushMessage, reactions: Collection<string, BushMessageReaction>]; +	messageReactionRemoveEmoji: [reaction: BushMessageReaction | PartialBushMessageReaction]; +	messageDeleteBulk: [messages: Collection<Snowflake, BushMessage | PartialBushMessage>]; +	messageReactionAdd: [reaction: BushMessageReaction | PartialBushMessageReaction, user: BushUser | PartialBushUser]; +	messageReactionRemove: [reaction: BushMessageReaction | PartialBushMessageReaction, user: BushUser | PartialBushUser]; +	messageUpdate: [oldMessage: BushMessage | PartialBushMessage, newMessage: BushMessage | PartialBushMessage];  	presenceUpdate: [oldPresence: BushPresence | null, newPresence: BushPresence];  	rateLimit: [rateLimitData: RateLimitData];  	invalidRequestWarning: [invalidRequestWarningData: InvalidRequestWarningData]; @@ -120,14 +98,8 @@ export interface BushClientEvents extends AkairoClientEvents {  	threadCreate: [thread: BushThreadChannel];  	threadDelete: [thread: BushThreadChannel];  	threadListSync: [threads: Collection<Snowflake, BushThreadChannel>]; -	threadMemberUpdate: [ -		oldMember: BushThreadMember, -		newMember: BushThreadMember -	]; -	threadMembersUpdate: [ -		oldMembers: Collection<Snowflake, BushThreadMember>, -		newMembers: Collection<Snowflake, BushThreadMember> -	]; +	threadMemberUpdate: [oldMember: BushThreadMember, newMember: BushThreadMember]; +	threadMembersUpdate: [oldMembers: Collection<Snowflake, BushThreadMember>, newMembers: Collection<Snowflake, BushThreadMember>];  	threadUpdate: [oldThread: BushThreadChannel, newThread: BushThreadChannel];  	typingStart: [typing: Typing];  	userUpdate: [oldUser: BushUser | PartialBushUser, newUser: BushUser]; @@ -139,28 +111,16 @@ export interface BushClientEvents extends AkairoClientEvents {  	shardReconnecting: [shardId: number];  	shardResume: [shardId: number, replayedEvents: number];  	stageInstanceCreate: [stageInstance: BushStageInstance]; -	stageInstanceUpdate: [ -		oldStageInstance: BushStageInstance | null, -		newStageInstance: BushStageInstance -	]; +	stageInstanceUpdate: [oldStageInstance: BushStageInstance | null, newStageInstance: BushStageInstance];  	stageInstanceDelete: [stageInstance: BushStageInstance];  	stickerCreate: [sticker: Sticker];  	stickerDelete: [sticker: Sticker];  	stickerUpdate: [oldSticker: Sticker, newSticker: Sticker];  	guildScheduledEventCreate: [guildScheduledEvent: GuildScheduledEvent]; -	guildScheduledEventUpdate: [ -		oldGuildScheduledEvent: GuildScheduledEvent, -		newGuildScheduledEvent: GuildScheduledEvent -	]; +	guildScheduledEventUpdate: [oldGuildScheduledEvent: GuildScheduledEvent, newGuildScheduledEvent: GuildScheduledEvent];  	guildScheduledEventDelete: [guildScheduledEvent: GuildScheduledEvent]; -	guildScheduledEventUserAdd: [ -		guildScheduledEvent: GuildScheduledEvent, -		user: BushUser -	]; -	guildScheduledEventUserRemove: [ -		guildScheduledEvent: GuildScheduledEvent, -		user: BushUser -	]; +	guildScheduledEventUserAdd: [guildScheduledEvent: GuildScheduledEvent, user: BushUser]; +	guildScheduledEventUserRemove: [guildScheduledEvent: GuildScheduledEvent, user: BushUser];  	/* Custom */  	bushBan: [  		victim: BushGuildMember | BushUser, @@ -318,9 +278,4 @@ export interface BushClientEvents extends AkairoClientEvents {  	];  } -type Setting = -	| GuildSettings -	| 'enabledFeatures' -	| 'blacklistedChannels' -	| 'blacklistedUsers' -	| 'disabledCommands'; +type Setting = GuildSettings | 'enabledFeatures' | 'blacklistedChannels' | 'blacklistedUsers' | 'disabledCommands'; diff --git a/src/lib/extensions/discord.js/BushClientUser.d.ts b/src/lib/extensions/discord.js/BushClientUser.ts index 503413b..a9a47f9 100644 --- a/src/lib/extensions/discord.js/BushClientUser.d.ts +++ b/src/lib/extensions/discord.js/BushClientUser.ts @@ -13,7 +13,7 @@ import { BushUser } from './BushUser';  /**   * Represents the logged in client's Discord user.   */ -export class BushClientUser extends BushUser implements ClientUser { +export declare class BushClientUser extends BushUser implements ClientUser {  	/**  	 * If the bot's {@link ClientApplication.owner Owner} has MFA enabled on their account  	 */ diff --git a/src/lib/extensions/discord.js/BushDMChannel.ts b/src/lib/extensions/discord.js/BushDMChannel.ts index 1af3ca1..363c620 100644 --- a/src/lib/extensions/discord.js/BushDMChannel.ts +++ b/src/lib/extensions/discord.js/BushDMChannel.ts @@ -1,12 +1,19 @@ +/* eslint-disable deprecation/deprecation */  import type { -	BushBaseGuildVoiceChannel, +	BushCategoryChannel,  	BushClient,  	BushMessageManager, +	BushNewsChannel, +	BushStageChannel, +	BushStoreChannel,  	BushTextBasedChannel, +	BushTextChannel,  	BushThreadChannel, -	BushUser +	BushUser, +	BushVoiceBasedChannel, +	BushVoiceChannel  } from '#lib'; -import { DMChannel } from 'discord.js'; +import { DMChannel, type Partialize } from 'discord.js';  import type { RawDMChannelData } from 'discord.js/typings/rawDataTypes';  /** @@ -23,7 +30,18 @@ export class BushDMChannel extends DMChannel {  }  export interface BushDMChannel extends DMChannel { -	isText(): this is BushTextBasedChannel; -	isVoice(): this is BushBaseGuildVoiceChannel; +	isText(): this is BushTextChannel; +	isDM(): this is BushDMChannel; +	isVoice(): this is BushVoiceChannel; +	isCategory(): this is BushCategoryChannel; +	isNews(): this is BushNewsChannel; +	isStore(): this is BushStoreChannel;  	isThread(): this is BushThreadChannel; +	isStage(): this is BushStageChannel; +	isTextBased(): this is BushTextBasedChannel; +	isVoiceBased(): this is BushVoiceBasedChannel; +} + +export interface PartialBushDMChannel extends Partialize<BushDMChannel, null, null, 'lastMessageId'> { +	lastMessageId: undefined;  } diff --git a/src/lib/extensions/discord.js/BushGuild.ts b/src/lib/extensions/discord.js/BushGuild.ts index 33ee3fc..9f114b6 100644 --- a/src/lib/extensions/discord.js/BushGuild.ts +++ b/src/lib/extensions/discord.js/BushGuild.ts @@ -5,6 +5,7 @@ import {  	punishmentEntryRemove,  	type BanResponse,  	type BushClient, +	type BushGuildChannelManager,  	type BushGuildMember,  	type BushGuildMemberManager,  	type BushGuildMemberResolvable, @@ -17,7 +18,7 @@ import {  	type GuildLogType,  	type GuildModel  } from '#lib'; -import { Collection, Guild, Snowflake, type GuildChannelManager, type MessageOptions, type MessagePayload } from 'discord.js'; +import { Collection, Guild, Permissions, Snowflake, type MessageOptions, type MessagePayload } from 'discord.js';  import type { RawGuildData } from 'discord.js/typings/rawDataTypes';  import _ from 'lodash';  import { Moderation } from '../../common/util/Moderation.js'; @@ -33,7 +34,7 @@ export class BushGuild extends Guild {  	public declare readonly client: BushClient;  	public declare readonly me: BushGuildMember | null;  	public declare members: BushGuildMemberManager; -	public declare channels: GuildChannelManager; +	public declare channels: BushGuildChannelManager;  	public constructor(client: BushClient, data: RawGuildData) {  		super(client, data); @@ -133,8 +134,13 @@ export class BushGuild extends Guild {  	 */  	public async sendLogChannel(logType: GuildLogType, message: string | MessagePayload | MessageOptions) {  		const logChannel = await this.getLogChannel(logType); -		if (!logChannel || logChannel.type !== 'GUILD_TEXT') return; -		if (!logChannel.permissionsFor(this.me!.id)?.has(['VIEW_CHANNEL', 'SEND_MESSAGES', 'EMBED_LINKS'])) return; +		if (!logChannel || !logChannel.isTextBased()) return; +		if ( +			!logChannel +				.permissionsFor(this.me!.id) +				?.has([Permissions.FLAGS.VIEW_CHANNEL, Permissions.FLAGS.SEND_MESSAGES, Permissions.FLAGS.EMBED_LINKS]) +		) +			return;  		return await logChannel.send(message).catch(() => null);  	} @@ -156,12 +162,13 @@ export class BushGuild extends Guild {  	 */  	public async bushBan(options: GuildBushBanOptions): Promise<BanResponse> {  		// checks -		if (!this.me!.permissions.has('BAN_MEMBERS')) return banResponse.MISSING_PERMISSIONS; +		if (!this.me!.permissions.has(Permissions.FLAGS.BAN_MEMBERS)) return banResponse.MISSING_PERMISSIONS;  		let caseID: string | undefined = undefined;  		let dmSuccessEvent: boolean | undefined = undefined; -		const user = (await util.resolveNonCachedUser(options.user))!; -		const moderator = client.users.resolve(options.moderator ?? client.user!)!; +		const user = await util.resolveNonCachedUser(options.user); +		const moderator = client.users.resolve(options.moderator ?? client.user!); +		if (!user || !moderator) return banResponse.CANNOT_RESOLVE_USER;  		if ((await this.bans.fetch()).has(user.id)) return banResponse.ALREADY_BANNED; @@ -234,12 +241,13 @@ export class BushGuild extends Guild {  	 */  	public async bushUnban(options: GuildBushUnbanOptions): Promise<UnbanResponse> {  		// checks -		if (!this.me!.permissions.has('BAN_MEMBERS')) return unbanResponse.MISSING_PERMISSIONS; +		if (!this.me!.permissions.has(Permissions.FLAGS.BAN_MEMBERS)) return unbanResponse.MISSING_PERMISSIONS;  		let caseID: string | undefined = undefined;  		let dmSuccessEvent: boolean | undefined = undefined; -		const user = (await util.resolveNonCachedUser(options.user))!; -		const moderator = client.users.resolve(options.moderator ?? client.user!)!; +		const user = await util.resolveNonCachedUser(options.user); +		const moderator = client.users.resolve(options.moderator ?? client.user!); +		if (!user || !moderator) return unbanResponse.CANNOT_RESOLVE_USER;  		const ret = await (async () => {  			const bans = await this.bans.fetch(); @@ -327,11 +335,11 @@ export class BushGuild extends Guild {  					success.set(channel.id, false);  					continue;  				} -				if (!channel.permissionsFor(this.me!.id)?.has(['MANAGE_CHANNELS'])) { +				if (!channel.permissionsFor(this.me!.id)?.has([Permissions.FLAGS.MANAGE_CHANNELS])) {  					errors.set(channel.id, new Error('client no permission'));  					success.set(channel.id, false);  					continue; -				} else if (!channel.permissionsFor(options.moderator)?.has(['MANAGE_CHANNELS'])) { +				} else if (!channel.permissionsFor(moderator)?.has([Permissions.FLAGS.MANAGE_CHANNELS])) {  					errors.set(channel.id, new Error('moderator no permission'));  					success.set(channel.id, false);  					continue; @@ -342,8 +350,12 @@ export class BushGuild extends Guild {  				}`;  				const permissionOverwrites = channel.isThread() ? channel.parent!.permissionOverwrites : channel.permissionOverwrites; -				const perms = { [channel.isThread() ? 'SEND_MESSAGES_IN_THREADS' : 'SEND_MESSAGES']: options.unlock ? null : false }; -				const permsForMe = { [channel.isThread() ? 'SEND_MESSAGES_IN_THREADS' : 'SEND_MESSAGES']: options.unlock ? null : true }; // so I can send messages in the channel +				const perms = { +					[channel.isThread() ? 'SEND_MESSAGES_IN_THREADS' : 'FLAGS.SEND_MESSAGES']: options.unlock ? null : false +				}; +				const permsForMe = { +					[channel.isThread() ? 'SEND_MESSAGES_IN_THREADS' : 'FLAGS.SEND_MESSAGES']: options.unlock ? null : true +				}; // so I can send messages in the channel  				const changePermSuccess = await permissionOverwrites.edit(this.id, perms, { reason }).catch((e) => e);  				if (changePermSuccess instanceof Error) { @@ -355,7 +367,7 @@ export class BushGuild extends Guild {  					await channel.send({  						embeds: [  							{ -								author: { name: moderator.user.tag, iconURL: moderator.displayAvatarURL({ dynamic: true }) }, +								author: { name: moderator.user.tag, iconURL: moderator.displayAvatarURL() },  								title: `This channel has been ${options.unlock ? 'un' : ''}locked`,  								description: options.reason ?? 'No reason provided',  								color: options.unlock ? util.colors.discord.GREEN : util.colors.discord.RED, diff --git a/src/lib/extensions/discord.js/BushGuildApplicationCommandManager.d.ts b/src/lib/extensions/discord.js/BushGuildApplicationCommandManager.ts index 4d76b07..ba9db66 100644 --- a/src/lib/extensions/discord.js/BushGuildApplicationCommandManager.d.ts +++ b/src/lib/extensions/discord.js/BushGuildApplicationCommandManager.ts @@ -12,7 +12,11 @@ import type { RawApplicationCommandData } from 'discord.js/typings/rawDataTypes'  /**   * An extension for guild-specific application commands.   */ -export class BushGuildApplicationCommandManager extends BushApplicationCommandManager<BushApplicationCommand, {}, BushGuild> { +export declare class BushGuildApplicationCommandManager extends BushApplicationCommandManager< +	BushApplicationCommand, +	{}, +	BushGuild +> {  	public constructor(guild: BushGuild, iterable?: Iterable<RawApplicationCommandData>);  	public declare readonly client: BushClient; diff --git a/src/lib/extensions/discord.js/BushGuildBan.d.ts b/src/lib/extensions/discord.js/BushGuildBan.ts index 11875f3..496e798 100644 --- a/src/lib/extensions/discord.js/BushGuildBan.d.ts +++ b/src/lib/extensions/discord.js/BushGuildBan.ts @@ -5,7 +5,7 @@ import type { RawGuildBanData } from 'discord.js/typings/rawDataTypes';  /**   * Represents a ban in a guild on Discord.   */ -export class BushGuildBan extends GuildBan { +export declare class BushGuildBan extends GuildBan {  	public constructor(client: BushClient, data: RawGuildBanData, guild: BushGuild);  	public guild: BushGuild;  	public user: BushUser; diff --git a/src/lib/extensions/discord.js/BushGuildChannel.ts b/src/lib/extensions/discord.js/BushGuildChannel.ts index 6880daf..053507e 100644 --- a/src/lib/extensions/discord.js/BushGuildChannel.ts +++ b/src/lib/extensions/discord.js/BushGuildChannel.ts @@ -1,4 +1,19 @@ -import type { BushClient, BushGuild } from '#lib'; +/* eslint-disable deprecation/deprecation */ +import type { +	BushCategoryChannel, +	BushClient, +	BushDMChannel, +	BushGuild, +	BushGuildBasedChannel, +	BushNewsChannel, +	BushStageChannel, +	BushStoreChannel, +	BushTextBasedChannel, +	BushTextChannel, +	BushThreadChannel, +	BushVoiceBasedChannel, +	BushVoiceChannel +} from '#lib';  import { GuildChannel } from 'discord.js';  import type { RawGuildChannelData } from 'discord.js/typings/rawDataTypes'; @@ -19,3 +34,16 @@ export class BushGuildChannel extends GuildChannel {  		super(guild, data, client, immediatePatch);  	}  } + +export interface BushGuildChannel extends GuildChannel { +	isText(): this is BushTextChannel; +	isDM(): this is BushDMChannel; +	isVoice(): this is BushVoiceChannel; +	isCategory(): this is BushCategoryChannel; +	isNews(): this is BushNewsChannel; +	isStore(): this is BushStoreChannel; +	isThread(): this is BushThreadChannel; +	isStage(): this is BushStageChannel; +	isTextBased(): this is BushGuildBasedChannel & BushTextBasedChannel; +	isVoiceBased(): this is BushVoiceBasedChannel; +} diff --git a/src/lib/extensions/discord.js/BushGuildChannelManager.d.ts b/src/lib/extensions/discord.js/BushGuildChannelManager.ts index 3b07145..dd3885b 100644 --- a/src/lib/extensions/discord.js/BushGuildChannelManager.d.ts +++ b/src/lib/extensions/discord.js/BushGuildChannelManager.ts @@ -1,6 +1,7 @@  import type {  	BushFetchedThreads,  	BushGuild, +	BushGuildBasedChannel,  	BushMappedGuildChannelTypes,  	BushNonThreadGuildBasedChannel,  	BushStoreChannel @@ -10,7 +11,6 @@ import {  	type BaseFetchOptions,  	type ChannelPosition,  	type Collection, -	type GuildBasedChannel,  	type GuildChannelCreateOptions,  	type GuildChannelManager,  	type GuildChannelResolvable, @@ -22,8 +22,8 @@ import type { RawGuildChannelData } from 'discord.js/typings/rawDataTypes';  /**   * Manages API methods for GuildChannels and stores their cache.   */ -export class BushGuildChannelManager -	extends CachedManager<Snowflake, GuildBasedChannel, GuildChannelResolvable> +export declare class BushGuildChannelManager +	extends CachedManager<Snowflake, BushGuildBasedChannel, GuildChannelResolvable>  	implements GuildChannelManager  {  	public constructor(guild: BushGuild, iterable?: Iterable<RawGuildChannelData>); @@ -51,7 +51,7 @@ export class BushGuildChannelManager  	 * @example  	 * // Create a new channel with permission overwrites  	 * guild.channels.create('new-voice', { -	 *   type: 'GUILD_VOICE', +	 *   type: 'GuildVoice',  	 *   permissionOverwrites: [  	 *      {  	 *        id: message.author.id, @@ -62,7 +62,7 @@ export class BushGuildChannelManager  	 * @deprecated See [Self-serve Game Selling Deprecation](https://support-dev.discord.com/hc/en-us/articles/4414590563479) for more information  	 */  	// eslint-disable-next-line deprecation/deprecation -	public create(name: string, options: GuildChannelCreateOptions & { type: 'GUILD_STORE' }): Promise<BushStoreChannel>; +	public create(name: string, options: GuildChannelCreateOptions & { type: 'GuildStore' }): Promise<BushStoreChannel>;  	/**  	 * Creates a new channel in the guild. @@ -112,7 +112,7 @@ export class BushGuildChannelManager  	/**  	 * Obtains all active thread channels in the guild from Discord -	 * @param cache Whether to cache the fetched data +	 * @param {} [cache=true] Whether to cache the fetched data  	 * @example  	 * // Fetch all threads from the guild  	 * message.guild.channels.fetchActiveThreads() diff --git a/src/lib/extensions/discord.js/BushGuildEmojiRoleManager.d.ts b/src/lib/extensions/discord.js/BushGuildEmojiRoleManager.ts index 9253cad..8b069ae 100644 --- a/src/lib/extensions/discord.js/BushGuildEmojiRoleManager.d.ts +++ b/src/lib/extensions/discord.js/BushGuildEmojiRoleManager.ts @@ -1,10 +1,13 @@  import type { BushClient, BushGuild, BushGuildEmoji, BushRole, BushRoleResolvable } from '#lib'; -import { DataManager, type Collection, type Snowflake } from 'discord.js'; +import { DataManager, GuildEmojiRoleManager, type Collection, type Snowflake } from 'discord.js';  /**   * Manages API methods for roles belonging to emojis and stores their cache.   */ -export class BushGuildEmojiRoleManager extends DataManager<Snowflake, BushRole, BushRoleResolvable> { +export declare class BushGuildEmojiRoleManager +	extends DataManager<Snowflake, BushRole, BushRoleResolvable> +	implements GuildEmojiRoleManager +{  	public constructor(emoji: BushGuildEmoji);  	public declare readonly client: BushClient; diff --git a/src/lib/extensions/discord.js/BushGuildManager.d.ts b/src/lib/extensions/discord.js/BushGuildManager.ts index 95719a3..41618e3 100644 --- a/src/lib/extensions/discord.js/BushGuildManager.d.ts +++ b/src/lib/extensions/discord.js/BushGuildManager.ts @@ -1,6 +1,7 @@  import type { BushClient, BushGuild, BushGuildResolvable } from '#lib';  import {  	CachedManager, +	GuildManager,  	type Collection,  	type FetchGuildOptions,  	type FetchGuildsOptions, @@ -13,7 +14,7 @@ import { type RawGuildData } from 'discord.js/typings/rawDataTypes';  /**   * Manages API methods for Guilds and stores their cache.   */ -export class BushGuildManager extends CachedManager<Snowflake, BushGuild, BushGuildResolvable> { +export declare class BushGuildManager extends CachedManager<Snowflake, BushGuild, BushGuildResolvable> implements GuildManager {  	public constructor(client: BushClient, iterable?: Iterable<RawGuildData>);  	/** diff --git a/src/lib/extensions/discord.js/BushGuildMember.ts b/src/lib/extensions/discord.js/BushGuildMember.ts index 54fb3f0..10d8a4c 100644 --- a/src/lib/extensions/discord.js/BushGuildMember.ts +++ b/src/lib/extensions/discord.js/BushGuildMember.ts @@ -11,7 +11,7 @@ import {  	type BushThreadChannelResolvable,  	type BushUser  } from '#lib'; -import { GuildMember, type Partialize, type Role } from 'discord.js'; +import { GuildMember, Permissions, type Partialize, type Role } from 'discord.js';  import type { RawGuildMemberData } from 'discord.js/typings/rawDataTypes';  /* eslint-enable @typescript-eslint/no-unused-vars */ @@ -48,7 +48,8 @@ export class BushGuildMember extends GuildMember {  	public async bushWarn(options: BushPunishmentOptions): Promise<{ result: WarnResponse; caseNum: number | null }> {  		let caseID: string | undefined = undefined;  		let dmSuccessEvent: boolean | undefined = undefined; -		const moderator = (await util.resolveNonCachedUser(options.moderator ?? this.guild.me))!; +		const moderator = await util.resolveNonCachedUser(options.moderator ?? this.guild.me); +		if (!moderator) return { result: warnResponse.CANNOT_RESOLVE_USER, caseNum: null };  		const ret = await (async (): Promise<{ result: WarnResponse; caseNum: number | null }> => {  			// add modlog entry @@ -89,12 +90,13 @@ export class BushGuildMember extends GuildMember {  	 */  	public async bushAddRole(options: AddRoleOptions): Promise<AddRoleResponse> {  		// checks -		if (!this.guild.me!.permissions.has('MANAGE_ROLES')) return addRoleResponse.MISSING_PERMISSIONS; +		if (!this.guild.me!.permissions.has(Permissions.FLAGS.MANAGE_ROLES)) return addRoleResponse.MISSING_PERMISSIONS;  		const ifShouldAddRole = this.#checkIfShouldAddRole(options.role, options.moderator);  		if (ifShouldAddRole !== true) return ifShouldAddRole;  		let caseID: string | undefined = undefined; -		const moderator = (await util.resolveNonCachedUser(options.moderator ?? this.guild.me))!; +		const moderator = await util.resolveNonCachedUser(options.moderator ?? this.guild.me); +		if (!moderator) return addRoleResponse.CANNOT_RESOLVE_USER;  		const ret = await (async () => {  			if (options.addToModlog || options.duration) { @@ -159,12 +161,13 @@ export class BushGuildMember extends GuildMember {  	 */  	public async bushRemoveRole(options: RemoveRoleOptions): Promise<RemoveRoleResponse> {  		// checks -		if (!this.guild.me!.permissions.has('MANAGE_ROLES')) return removeRoleResponse.MISSING_PERMISSIONS; +		if (!this.guild.me!.permissions.has(Permissions.FLAGS.MANAGE_ROLES)) return removeRoleResponse.MISSING_PERMISSIONS;  		const ifShouldAddRole = this.#checkIfShouldAddRole(options.role, options.moderator);  		if (ifShouldAddRole !== true) return ifShouldAddRole;  		let caseID: string | undefined = undefined; -		const moderator = (await util.resolveNonCachedUser(options.moderator ?? this.guild.me))!; +		const moderator = await util.resolveNonCachedUser(options.moderator ?? this.guild.me); +		if (!moderator) return removeRoleResponse.CANNOT_RESOLVE_USER;  		const ret = await (async () => {  			if (options.addToModlog) { @@ -249,7 +252,7 @@ export class BushGuildMember extends GuildMember {  	 */  	public async bushMute(options: BushTimedPunishmentOptions): Promise<MuteResponse> {  		// checks -		if (!this.guild.me!.permissions.has('MANAGE_ROLES')) return muteResponse.MISSING_PERMISSIONS; +		if (!this.guild.me!.permissions.has(Permissions.FLAGS.MANAGE_ROLES)) return muteResponse.MISSING_PERMISSIONS;  		const muteRoleID = await this.guild.getSetting('muteRole');  		if (!muteRoleID) return muteResponse.NO_MUTE_ROLE;  		const muteRole = this.guild.roles.cache.get(muteRoleID); @@ -259,7 +262,8 @@ export class BushGuildMember extends GuildMember {  		let caseID: string | undefined = undefined;  		let dmSuccessEvent: boolean | undefined = undefined; -		const moderator = (await util.resolveNonCachedUser(options.moderator ?? this.guild.me))!; +		const moderator = await util.resolveNonCachedUser(options.moderator ?? this.guild.me); +		if (!moderator) return muteResponse.CANNOT_RESOLVE_USER;  		const ret = await (async () => {  			// add role @@ -334,7 +338,7 @@ export class BushGuildMember extends GuildMember {  	 */  	public async bushUnmute(options: BushPunishmentOptions): Promise<UnmuteResponse> {  		// checks -		if (!this.guild.me!.permissions.has('MANAGE_ROLES')) return unmuteResponse.MISSING_PERMISSIONS; +		if (!this.guild.me!.permissions.has(Permissions.FLAGS.MANAGE_ROLES)) return unmuteResponse.MISSING_PERMISSIONS;  		const muteRoleID = await this.guild.getSetting('muteRole');  		if (!muteRoleID) return unmuteResponse.NO_MUTE_ROLE;  		const muteRole = this.guild.roles.cache.get(muteRoleID); @@ -344,7 +348,8 @@ export class BushGuildMember extends GuildMember {  		let caseID: string | undefined = undefined;  		let dmSuccessEvent: boolean | undefined = undefined; -		const moderator = (await util.resolveNonCachedUser(options.moderator ?? this.guild.me))!; +		const moderator = await util.resolveNonCachedUser(options.moderator ?? this.guild.me); +		if (!moderator) return unmuteResponse.CANNOT_RESOLVE_USER;  		const ret = await (async () => {  			// remove role @@ -416,11 +421,13 @@ export class BushGuildMember extends GuildMember {  	 */  	public async bushKick(options: BushPunishmentOptions): Promise<KickResponse> {  		// checks -		if (!this.guild.me?.permissions.has('KICK_MEMBERS') || !this.kickable) return kickResponse.MISSING_PERMISSIONS; +		if (!this.guild.me?.permissions.has(Permissions.FLAGS.KICK_MEMBERS) || !this.kickable) +			return kickResponse.MISSING_PERMISSIONS;  		let caseID: string | undefined = undefined;  		let dmSuccessEvent: boolean | undefined = undefined; -		const moderator = (await util.resolveNonCachedUser(options.moderator ?? this.guild.me))!; +		const moderator = await util.resolveNonCachedUser(options.moderator ?? this.guild.me); +		if (!moderator) return kickResponse.CANNOT_RESOLVE_USER;  		const ret = await (async () => {  			// dm user  			const dmSuccess = options.silent ? null : await this.bushPunishDM('kicked', options.reason); @@ -467,11 +474,12 @@ export class BushGuildMember extends GuildMember {  	 */  	public async bushBan(options: BushBanOptions): Promise<Exclude<BanResponse, typeof banResponse['ALREADY_BANNED']>> {  		// checks -		if (!this.guild.me!.permissions.has('BAN_MEMBERS') || !this.bannable) return banResponse.MISSING_PERMISSIONS; +		if (!this.guild.me!.permissions.has(Permissions.FLAGS.BAN_MEMBERS) || !this.bannable) return banResponse.MISSING_PERMISSIONS;  		let caseID: string | undefined = undefined;  		let dmSuccessEvent: boolean | undefined = undefined; -		const moderator = (await util.resolveNonCachedUser(options.moderator ?? this.guild.me))!; +		const moderator = await util.resolveNonCachedUser(options.moderator ?? this.guild.me); +		if (!moderator) return banResponse.CANNOT_RESOLVE_USER;  		// ignore result, they should still be banned even if their mute cannot be removed  		await this.bushUnmute({ @@ -542,16 +550,16 @@ export class BushGuildMember extends GuildMember {  	 * @param options Options for blocking the user.  	 */  	public async bushBlock(options: BlockOptions): Promise<BlockResponse> { -		const _channel = this.guild.channels.resolve(options.channel); -		if (!_channel || (!_channel.isText() && !_channel.isThread())) return blockResponse.INVALID_CHANNEL; -		const channel = _channel as BushGuildTextBasedChannel; +		const channel = this.guild.channels.resolve(options.channel); +		if (!channel || (!channel.isTextBased() && !channel.isThread())) return blockResponse.INVALID_CHANNEL;  		// checks -		if (!channel.permissionsFor(this.guild.me!)!.has('MANAGE_CHANNELS')) return blockResponse.MISSING_PERMISSIONS; +		if (!channel.permissionsFor(this.guild.me!)!.has(Permissions.FLAGS.MANAGE_CHANNELS)) return blockResponse.MISSING_PERMISSIONS;  		let caseID: string | undefined = undefined;  		let dmSuccessEvent: boolean | undefined = undefined; -		const moderator = (await util.resolveNonCachedUser(options.moderator ?? this.guild.me))!; +		const moderator = await util.resolveNonCachedUser(options.moderator ?? this.guild.me); +		if (!moderator) return blockResponse.CANNOT_RESOLVE_USER;  		const ret = await (async () => {  			// change channel permissions @@ -635,11 +643,13 @@ export class BushGuildMember extends GuildMember {  		const channel = _channel as BushGuildTextBasedChannel;  		// checks -		if (!channel.permissionsFor(this.guild.me!)!.has('MANAGE_CHANNELS')) return unblockResponse.MISSING_PERMISSIONS; +		if (!channel.permissionsFor(this.guild.me!)!.has(Permissions.FLAGS.MANAGE_CHANNELS)) +			return unblockResponse.MISSING_PERMISSIONS;  		let caseID: string | undefined = undefined;  		let dmSuccessEvent: boolean | undefined = undefined; -		const moderator = (await util.resolveNonCachedUser(options.moderator ?? this.guild.me))!; +		const moderator = await util.resolveNonCachedUser(options.moderator ?? this.guild.me); +		if (!moderator) return unblockResponse.CANNOT_RESOLVE_USER;  		const ret = await (async () => {  			// change channel permissions @@ -710,14 +720,15 @@ export class BushGuildMember extends GuildMember {  	 */  	public async bushTimeout(options: BushTimeoutOptions): Promise<TimeoutResponse> {  		// checks -		if (!this.guild.me!.permissions.has('MODERATE_MEMBERS')) return timeoutResponse.MISSING_PERMISSIONS; +		if (!this.guild.me!.permissions.has(Permissions.FLAGS.MODERATE_MEMBERS)) return timeoutResponse.MISSING_PERMISSIONS;  		const twentyEightDays = client.consts.timeUnits.days.value * 28;  		if (options.duration > twentyEightDays) return timeoutResponse.INVALID_DURATION;  		let caseID: string | undefined = undefined;  		let dmSuccessEvent: boolean | undefined = undefined; -		const moderator = (await util.resolveNonCachedUser(options.moderator ?? this.guild.me))!; +		const moderator = await util.resolveNonCachedUser(options.moderator ?? this.guild.me); +		if (!moderator) return timeoutResponse.CANNOT_RESOLVE_USER;  		const ret = await (async () => {  			// timeout @@ -773,11 +784,12 @@ export class BushGuildMember extends GuildMember {  	 */  	public async bushRemoveTimeout(options: BushPunishmentOptions): Promise<RemoveTimeoutResponse> {  		// checks -		if (!this.guild.me!.permissions.has('MODERATE_MEMBERS')) return removeTimeoutResponse.MISSING_PERMISSIONS; +		if (!this.guild.me!.permissions.has(Permissions.FLAGS.MODERATE_MEMBERS)) return removeTimeoutResponse.MISSING_PERMISSIONS;  		let caseID: string | undefined = undefined;  		let dmSuccessEvent: boolean | undefined = undefined; -		const moderator = (await util.resolveNonCachedUser(options.moderator ?? this.guild.me))!; +		const moderator = await util.resolveNonCachedUser(options.moderator ?? this.guild.me); +		if (!moderator) return removeTimeoutResponse.CANNOT_RESOLVE_USER;  		const ret = await (async () => {  			// remove timeout @@ -949,7 +961,8 @@ type ValueOf<T> = T[keyof T];  export const basePunishmentResponse = Object.freeze({  	SUCCESS: 'success',  	MODLOG_ERROR: 'error creating modlog entry', -	ACTION_ERROR: 'error performing action' +	ACTION_ERROR: 'error performing action', +	CANNOT_RESOLVE_USER: 'cannot resolve user'  } as const);  export const dmResponse = Object.freeze({ diff --git a/src/lib/extensions/discord.js/BushGuildMemberManager.d.ts b/src/lib/extensions/discord.js/BushGuildMemberManager.ts index a0e65e7..bb130fc 100644 --- a/src/lib/extensions/discord.js/BushGuildMemberManager.d.ts +++ b/src/lib/extensions/discord.js/BushGuildMemberManager.ts @@ -1,6 +1,7 @@  import type { BushClient, BushGuild, BushGuildMember, BushGuildMemberResolvable, BushUser, BushUserResolvable } from '#lib';  import {  	CachedManager, +	GuildMemberManager,  	type AddGuildMemberOptions,  	type BanOptions,  	type Collection, @@ -17,7 +18,10 @@ import type { RawGuildMemberData } from 'discord.js/typings/rawDataTypes';  /**   * Manages API methods for GuildMembers and stores their cache.   */ -export class BushGuildMemberManager extends CachedManager<Snowflake, BushGuildMember, BushGuildMemberResolvable> { +export declare class BushGuildMemberManager +	extends CachedManager<Snowflake, BushGuildMember, BushGuildMemberResolvable> +	implements GuildMemberManager +{  	public constructor(guild: BushGuild, iterable?: Iterable<RawGuildMemberData>);  	public declare readonly client: BushClient; diff --git a/src/lib/extensions/discord.js/BushMessage.ts b/src/lib/extensions/discord.js/BushMessage.ts index 16c57a2..420f8f5 100644 --- a/src/lib/extensions/discord.js/BushMessage.ts +++ b/src/lib/extensions/discord.js/BushMessage.ts @@ -10,10 +10,10 @@ import type {  	BushUser  } from '#lib';  import { +	ActionRowComponent,  	Message,  	type EmojiIdentifierResolvable,  	type If, -	type MessageActionRowComponent,  	type MessageEditOptions,  	type MessagePayload,  	type Partialize, @@ -55,7 +55,7 @@ export interface BushMessage<Cached extends boolean = boolean> extends Message<C  	react(emoji: EmojiIdentifierResolvable): Promise<BushMessageReaction>;  	removeAttachments(): Promise<BushMessage>;  	reply(options: string | MessagePayload | ReplyMessageOptions): Promise<BushMessage>; -	resolveComponent(customId: string): MessageActionRowComponent | null; +	resolveComponent(customId: string): ActionRowComponent | null;  	startThread(options: StartThreadOptions): Promise<BushThreadChannel>;  	suppressEmbeds(suppress?: boolean): Promise<BushMessage>;  	unpin(): Promise<BushMessage>; diff --git a/src/lib/extensions/discord.js/BushMessageManager.d.ts b/src/lib/extensions/discord.js/BushMessageManager.ts index 84918c0..7ed4199 100644 --- a/src/lib/extensions/discord.js/BushMessageManager.d.ts +++ b/src/lib/extensions/discord.js/BushMessageManager.ts @@ -1,6 +1,7 @@  import { BushMessageResolvable, BushTextBasedChannel, type BushMessage } from '#lib';  import {  	CachedManager, +	MessageManager,  	type BaseFetchOptions,  	type ChannelLogsQueryOptions,  	type Collection, @@ -14,7 +15,10 @@ import type { RawMessageData } from 'discord.js/typings/rawDataTypes';  /**   * Manages API methods for Messages and holds their cache.   */ -export class BushMessageManager extends CachedManager<Snowflake, BushMessage, BushMessageResolvable> { +export declare class BushMessageManager +	extends CachedManager<Snowflake, BushMessage, BushMessageResolvable> +	implements MessageManager +{  	public constructor(channel: BushTextBasedChannel, iterable?: Iterable<RawMessageData>);  	/** @@ -44,7 +48,7 @@ export class BushMessageManager extends CachedManager<Snowflake, BushMessage, Bu  	 * @param message The message to edit  	 * @param options The options to edit the message  	 */ -	public edit(message: BushMessageResolvable, options: MessagePayload | MessageEditOptions): Promise<BushMessage>; +	public edit(message: BushMessageResolvable, options: string | MessagePayload | MessageEditOptions): Promise<BushMessage>;  	/**  	 * Gets a message, or messages, from this channel. @@ -75,7 +79,7 @@ export class BushMessageManager extends CachedManager<Snowflake, BushMessage, Bu  	 * Fetches the pinned messages of this channel and returns a collection of them.  	 * <info>The returned Collection does not contain any reaction data of the messages.  	 * Those need to be fetched separately.</info> -	 * @param cache Whether to cache the message(s) +	 * @param {} [cache=true] Whether to cache the message(s)  	 * @example  	 * // Get pinned messages  	 * channel.messages.fetchPinned() diff --git a/src/lib/extensions/discord.js/BushStoreChannel.ts b/src/lib/extensions/discord.js/BushStoreChannel.ts index cb75076..dbc53e8 100644 --- a/src/lib/extensions/discord.js/BushStoreChannel.ts +++ b/src/lib/extensions/discord.js/BushStoreChannel.ts @@ -1,4 +1,19 @@ -import type { BushCategoryChannel, BushClient, BushGuild, BushGuildMember } from '#lib'; +/* eslint-disable deprecation/deprecation */ +import type { +	BushCategoryChannel, +	BushClient, +	BushDMChannel, +	BushGuild, +	BushGuildBasedChannel, +	BushGuildMember, +	BushNewsChannel, +	BushStageChannel, +	BushTextBasedChannel, +	BushTextChannel, +	BushThreadChannel, +	BushVoiceBasedChannel, +	BushVoiceChannel +} from '#lib';  import { StoreChannel, type Collection, type Snowflake } from 'discord.js';  import type { RawGuildChannelData } from 'discord.js/typings/rawDataTypes'; @@ -6,7 +21,6 @@ import type { RawGuildChannelData } from 'discord.js/typings/rawDataTypes';   * Represents a guild store channel on Discord.   * @deprecated Store channels are deprecated and will be removed from Discord in March 2022. See [Self-serve Game Selling Deprecation](https://support-dev.discord.com/hc/en-us/articles/4414590563479) for more information   */ -// eslint-disable-next-line deprecation/deprecation  export class BushStoreChannel extends StoreChannel {  	public declare guild: BushGuild;  	public declare readonly members: Collection<Snowflake, BushGuildMember>; @@ -16,3 +30,16 @@ export class BushStoreChannel extends StoreChannel {  		super(guild, data, client);  	}  } + +export interface BushStoreChannel extends StoreChannel { +	isText(): this is BushTextChannel; +	isDM(): this is BushDMChannel; +	isVoice(): this is BushVoiceChannel; +	isCategory(): this is BushCategoryChannel; +	isNews(): this is BushNewsChannel; +	isStore(): this is BushStoreChannel; +	isThread(): this is BushThreadChannel; +	isStage(): this is BushStageChannel; +	isTextBased(): this is BushGuildBasedChannel & BushTextBasedChannel; +	isVoiceBased(): this is BushVoiceBasedChannel; +} diff --git a/src/lib/extensions/discord.js/BushTextChannel.ts b/src/lib/extensions/discord.js/BushTextChannel.ts index 45e1200..15be7bd 100644 --- a/src/lib/extensions/discord.js/BushTextChannel.ts +++ b/src/lib/extensions/discord.js/BushTextChannel.ts @@ -1,4 +1,19 @@ -import type { BushGuild, BushMessageManager, BushThreadManager } from '#lib'; +/* eslint-disable deprecation/deprecation */ +import type { +	BushCategoryChannel, +	BushDMChannel, +	BushGuild, +	BushGuildBasedChannel, +	BushMessageManager, +	BushNewsChannel, +	BushStageChannel, +	BushStoreChannel, +	BushTextBasedChannel, +	BushThreadChannel, +	BushThreadManager, +	BushVoiceBasedChannel, +	BushVoiceChannel +} from '#lib';  import { TextChannel, type AllowedThreadTypeForTextChannel } from 'discord.js';  import type { RawGuildChannelData } from 'discord.js/typings/rawDataTypes'; @@ -14,3 +29,16 @@ export class BushTextChannel extends TextChannel {  		super(guild, data);  	}  } + +export interface BushTextChannel extends TextChannel { +	isText(): this is BushTextChannel; +	isDM(): this is BushDMChannel; +	isVoice(): this is BushVoiceChannel; +	isCategory(): this is BushCategoryChannel; +	isNews(): this is BushNewsChannel; +	isStore(): this is BushStoreChannel; +	isThread(): this is BushThreadChannel; +	isStage(): this is BushStageChannel; +	isTextBased(): this is BushGuildBasedChannel & BushTextBasedChannel; +	isVoiceBased(): this is BushVoiceBasedChannel; +} diff --git a/src/lib/extensions/discord.js/BushThreadChannel.ts b/src/lib/extensions/discord.js/BushThreadChannel.ts index 3c8859c..a342dd7 100644 --- a/src/lib/extensions/discord.js/BushThreadChannel.ts +++ b/src/lib/extensions/discord.js/BushThreadChannel.ts @@ -1,13 +1,20 @@ +/* eslint-disable deprecation/deprecation */  import type { -	BushBaseGuildVoiceChannel, +	BushCategoryChannel,  	BushClient, +	BushDMChannel,  	BushGuild, +	BushGuildBasedChannel,  	BushGuildMember,  	BushMessageManager,  	BushNewsChannel, +	BushStageChannel, +	BushStoreChannel,  	BushTextBasedChannel,  	BushTextChannel, -	BushThreadMemberManager +	BushThreadMemberManager, +	BushVoiceBasedChannel, +	BushVoiceChannel  } from '#lib';  import { ThreadChannel, type Collection, type Snowflake } from 'discord.js';  import type { RawThreadChannelData } from 'discord.js/typings/rawDataTypes'; @@ -29,7 +36,14 @@ export class BushThreadChannel extends ThreadChannel {  }  export interface BushThreadChannel extends ThreadChannel { -	isText(): this is BushTextBasedChannel; -	isVoice(): this is BushBaseGuildVoiceChannel; +	isText(): this is BushTextChannel; +	isDM(): this is BushDMChannel; +	isVoice(): this is BushVoiceChannel; +	isCategory(): this is BushCategoryChannel; +	isNews(): this is BushNewsChannel; +	isStore(): this is BushStoreChannel;  	isThread(): this is BushThreadChannel; +	isStage(): this is BushStageChannel; +	isTextBased(): this is BushGuildBasedChannel & BushTextBasedChannel; +	isVoiceBased(): this is BushVoiceBasedChannel;  } diff --git a/src/lib/extensions/discord.js/BushThreadManager.d.ts b/src/lib/extensions/discord.js/BushThreadManager.ts index 6b3340d..b84b98d 100644 --- a/src/lib/extensions/discord.js/BushThreadManager.d.ts +++ b/src/lib/extensions/discord.js/BushThreadManager.ts @@ -5,6 +5,7 @@ import {  	NewsChannel,  	TextChannel,  	ThreadChannel, +	ThreadManager,  	type BaseFetchOptions,  	type FetchArchivedThreadOptions,  	type FetchThreadsOptions, @@ -17,7 +18,10 @@ import type { RawThreadChannelData } from 'discord.js/typings/rawDataTypes';  /**   * Manages API methods for {@link BushThreadChannel} objects and stores their cache.   */ -export class BushThreadManager<AllowedThreadType> extends CachedManager<Snowflake, BushThreadChannel, ThreadChannelResolvable> { +export declare class BushThreadManager<AllowedThreadType> +	extends CachedManager<Snowflake, BushThreadChannel, ThreadChannelResolvable> +	implements ThreadManager<AllowedThreadType> +{  	public constructor(channel: TextChannel | NewsChannel, iterable?: Iterable<RawThreadChannelData>);  	/** @@ -44,7 +48,7 @@ export class BushThreadManager<AllowedThreadType> extends CachedManager<Snowflak  	 *   .create({  	 *      name: 'mod-talk',  	 *      autoArchiveDuration: 60, -	 *      type: 'GUILD_PRIVATE_THREAD', +	 *      type: 'GuildPrivateThread',  	 *      reason: 'Needed a separate thread for moderation',  	 *    })  	 *   .then(threadChannel => console.log(threadChannel)) diff --git a/src/lib/extensions/discord.js/BushThreadMemberManager.d.ts b/src/lib/extensions/discord.js/BushThreadMemberManager.ts index dedf102..d597673 100644 --- a/src/lib/extensions/discord.js/BushThreadMemberManager.d.ts +++ b/src/lib/extensions/discord.js/BushThreadMemberManager.ts @@ -1,11 +1,21 @@ -import type { BushClient, BushThreadChannel, BushThreadMember, BushThreadMemberResolvable } from '#lib'; -import { CachedManager, type BaseFetchOptions, type Collection, type Snowflake, type UserResolvable } from 'discord.js'; +import type { BushClient, BushThreadChannel, BushThreadMember, BushThreadMemberResolvable, BushUserResolvable } from '#lib'; +import { +	CachedManager, +	ThreadMemberManager, +	type BaseFetchOptions, +	type Collection, +	type Snowflake, +	type UserResolvable +} from 'discord.js';  import type { RawThreadMemberData } from 'discord.js/typings/rawDataTypes';  /**   * Manages API methods for GuildMembers and stores their cache.   */ -export class BushThreadMemberManager extends CachedManager<Snowflake, BushThreadMember, BushThreadMemberResolvable> { +export declare class BushThreadMemberManager +	extends CachedManager<Snowflake, BushThreadMember, BushThreadMemberResolvable> +	implements ThreadMemberManager +{  	public constructor(thread: BushThreadChannel, iterable?: Iterable<RawThreadMemberData>);  	public declare readonly client: BushClient; @@ -23,15 +33,10 @@ export class BushThreadMemberManager extends CachedManager<Snowflake, BushThread  	/**  	 * Fetches member(s) for the thread from Discord, requires access to the `GUILD_MEMBERS` gateway intent. -	 * @param member The member to fetch. If `undefined`, all members in the thread are fetched, and will be -	 * cached based on `options.cache`. If boolean, this serves the purpose of `options.cache`. -	 * @param options Additional options for this fetch -	 */ -	public fetch(member?: UserResolvable, options?: BaseFetchOptions): Promise<BushThreadMember>; - -	/** -	 * @deprecated Use `fetch(member, options)` instead. +	 * @param options Additional options for this fetch, when a `boolean` is provided +	 * all members are fetched with `options.cache` set to the boolean value  	 */ +	public fetch(options?: BushThreadMemberFetchOptions): Promise<BushThreadMember>;  	public fetch(cache?: boolean): Promise<Collection<Snowflake, BushThreadMember>>;  	/** @@ -41,3 +46,10 @@ export class BushThreadMemberManager extends CachedManager<Snowflake, BushThread  	 */  	public remove(id: Snowflake | '@me', reason?: string): Promise<Snowflake>;  } + +export interface BushThreadMemberFetchOptions extends BaseFetchOptions { +	/** +	 * The specific user to fetch from the thread +	 */ +	member?: BushUserResolvable; +} diff --git a/src/lib/extensions/discord.js/BushUserManager.d.ts b/src/lib/extensions/discord.js/BushUserManager.ts index 5d814da..ae28f1d 100644 --- a/src/lib/extensions/discord.js/BushUserManager.d.ts +++ b/src/lib/extensions/discord.js/BushUserManager.ts @@ -5,6 +5,7 @@ import {  	MessageOptions,  	MessagePayload,  	UserFlags, +	UserManager,  	type BaseFetchOptions,  	type Snowflake  } from 'discord.js'; @@ -13,7 +14,7 @@ import type { RawUserData } from 'discord.js/typings/rawDataTypes';  /**   * Manages API methods for users and stores their cache.   */ -export class BushUserManager extends CachedManager<Snowflake, BushUser, BushUserResolvable> { +export declare class BushUserManager extends CachedManager<Snowflake, BushUser, BushUserResolvable> implements UserManager {  	private constructor(client: BushClient, iterable?: Iterable<RawUserData>);  	/** @@ -24,14 +25,14 @@ export class BushUserManager extends CachedManager<Snowflake, BushUser, BushUser  	public dmChannel(userId: Snowflake): BushDMChannel | null;  	/** -	 * Creates a {@link DMChannel} between the client and a user. +	 * Creates a {@link BushDMChannel} between the client and a user.  	 * @param user The UserResolvable to identify  	 * @param options Additional options for this fetch  	 */  	public createDM(user: BushUserResolvable, options?: BaseFetchOptions): Promise<BushDMChannel>;  	/** -	 * Deletes a {@link DMChannel} (if one exists) between the client and a user. Resolves with the channel if successful. +	 * Deletes a {@link BushDMChannel} (if one exists) between the client and a user. Resolves with the channel if successful.  	 * @param user The UserResolvable to identify  	 */  	public deleteDM(user: BushUserResolvable): Promise<BushDMChannel>; diff --git a/src/lib/extensions/discord.js/BushVoiceChannel.ts b/src/lib/extensions/discord.js/BushVoiceChannel.ts index 9f246e5..a0ee47d 100644 --- a/src/lib/extensions/discord.js/BushVoiceChannel.ts +++ b/src/lib/extensions/discord.js/BushVoiceChannel.ts @@ -1,4 +1,19 @@ -import type { BushClient, BushGuild, BushGuildMember } from '#lib'; +/* eslint-disable deprecation/deprecation */ +import type { +	BushCategoryChannel, +	BushClient, +	BushDMChannel, +	BushGuild, +	BushGuildBasedChannel, +	BushGuildMember, +	BushNewsChannel, +	BushStageChannel, +	BushStoreChannel, +	BushTextBasedChannel, +	BushTextChannel, +	BushThreadChannel, +	BushVoiceBasedChannel +} from '#lib';  import { VoiceChannel, type Collection, type Snowflake } from 'discord.js';  import type { RawGuildChannelData } from 'discord.js/typings/rawDataTypes'; @@ -13,3 +28,16 @@ export class BushVoiceChannel extends VoiceChannel {  		super(guild, data);  	}  } + +export interface BushVoiceChannel extends VoiceChannel { +	isText(): this is BushTextChannel; +	isDM(): this is BushDMChannel; +	isVoice(): this is BushVoiceChannel; +	isCategory(): this is BushCategoryChannel; +	isNews(): this is BushNewsChannel; +	isStore(): this is BushStoreChannel; +	isThread(): this is BushThreadChannel; +	isStage(): this is BushStageChannel; +	isTextBased(): this is BushGuildBasedChannel & BushTextBasedChannel; +	isVoiceBased(): this is BushVoiceBasedChannel; +} diff --git a/src/lib/extensions/discord.js/other.ts b/src/lib/extensions/discord.js/other.ts index f81e01c..784442d 100644 --- a/src/lib/extensions/discord.js/other.ts +++ b/src/lib/extensions/discord.js/other.ts @@ -1,3 +1,4 @@ +/* eslint-disable deprecation/deprecation */  import type {  	BushApplicationCommand,  	BushCategoryChannel, @@ -15,10 +16,10 @@ import type {  	BushThreadChannel,  	BushThreadMember,  	BushUser, -	BushVoiceChannel +	BushVoiceChannel, +	PartialBushDMChannel  } from '#lib'; -import type { Collection, EnumValueMapped, Message, PartialDMChannel, Snowflake } from 'discord.js'; -import type { ChannelTypes } from 'discord.js/typings/enums'; +import type { ChannelType, Collection, Message, Snowflake } from 'discord.js';  /**   * Data that resolves to give a ThreadMember object. @@ -86,10 +87,9 @@ export type BushGuildChannelResolvable = Snowflake | BushGuildBasedChannel;  export type BushAnyChannel =  	| BushCategoryChannel  	| BushDMChannel -	| PartialDMChannel +	| PartialBushDMChannel  	| BushNewsChannel  	| BushStageChannel -	// eslint-disable-next-line deprecation/deprecation  	| BushStoreChannel  	| BushTextChannel  	| BushThreadChannel @@ -98,7 +98,7 @@ export type BushAnyChannel =  /**   * The channels that are text-based.   */ -export type BushTextBasedChannel = PartialDMChannel | BushThreadChannel | BushDMChannel | BushNewsChannel | BushTextChannel; +export type BushTextBasedChannel = PartialBushDMChannel | BushThreadChannel | BushDMChannel | BushNewsChannel | BushTextChannel;  /**   * The types of channels that are text-based. @@ -123,25 +123,17 @@ export type BushTextChannelResolvable = Snowflake | BushTextChannel;   */  export type BushGuildVoiceChannelResolvable = BushVoiceBasedChannel | Snowflake; -export type BushMappedChannelCategoryTypes = EnumValueMapped< -	typeof ChannelTypes, -	{ -		GUILD_NEWS: BushNewsChannel; -		GUILD_VOICE: BushVoiceChannel; -		GUILD_TEXT: BushTextChannel; -		// eslint-disable-next-line deprecation/deprecation -		GUILD_STORE: BushStoreChannel; -		GUILD_STAGE_VOICE: BushStageChannel; -	} ->; - -export type BushMappedGuildChannelTypes = EnumValueMapped< -	typeof ChannelTypes, -	{ -		GUILD_CATEGORY: BushCategoryChannel; -	} -> & -	BushMappedChannelCategoryTypes; +export interface BushMappedChannelCategoryTypes { +	[ChannelType.GuildNews]: BushNewsChannel; +	[ChannelType.GuildVoice]: BushVoiceChannel; +	[ChannelType.GuildText]: BushTextChannel; +	[ChannelType.GuildStore]: BushStoreChannel; +	[ChannelType.GuildStageVoice]: BushStageChannel; +} + +export type BushMappedGuildChannelTypes = { +	[ChannelType.GuildCategory]: BushCategoryChannel; +} & BushMappedChannelCategoryTypes;  /**   * The data returned from a thread fetch that returns multiple threads. @@ -157,3 +149,15 @@ export interface BushFetchedThreads {  	 */  	hasMore?: boolean;  } + +// for reverse key mapping + +/** + * https://discord.com/developers/docs/interactions/receiving-and-responding#interaction-object-interaction-type + */ +export enum BushInteractionType { +	Ping = 1, +	ApplicationCommand = 2, +	MessageComponent = 3, +	ApplicationCommandAutocomplete = 4 +} diff --git a/src/lib/extensions/global.d.ts b/src/lib/extensions/global.ts index a6f2b5a..a6f2b5a 100644 --- a/src/lib/extensions/global.d.ts +++ b/src/lib/extensions/global.ts diff --git a/src/lib/index.ts b/src/lib/index.ts index 37bc443..79bc553 100644 --- a/src/lib/index.ts +++ b/src/lib/index.ts @@ -30,9 +30,9 @@ export * from './extensions/discord.js/BushButtonInteraction.js';  export * from './extensions/discord.js/BushCategoryChannel.js';  export type { BushChannel } from './extensions/discord.js/BushChannel.js';  export type { BushChannelManager } from './extensions/discord.js/BushChannelManager.js'; +export * from './extensions/discord.js/BushChatInputCommandInteraction.js';  export type { BushClientEvents } from './extensions/discord.js/BushClientEvents.js';  export type { BushClientUser } from './extensions/discord.js/BushClientUser.js'; -export * from './extensions/discord.js/BushCommandInteraction.js';  export * from './extensions/discord.js/BushDMChannel.js';  export * from './extensions/discord.js/BushEmoji.js';  export * from './extensions/discord.js/BushGuild.js'; diff --git a/src/lib/models/Guild.ts b/src/lib/models/Guild.ts index 34f6747..fd90dbf 100644 --- a/src/lib/models/Guild.ts +++ b/src/lib/models/Guild.ts @@ -1,5 +1,4 @@ -import { ExcludeEnum, type Snowflake } from 'discord.js'; -import { ChannelTypes } from 'discord.js/typings/enums'; +import { ChannelType, type Snowflake } from 'discord.js';  import { type Sequelize } from 'sequelize';  import { BadWordDetails } from '../common/AutoMod.js';  import { type BushClient } from '../extensions/discord-akairo/BushClient.js'; @@ -183,7 +182,7 @@ export interface GuildSetting {  	name: string;  	description: string;  	type: GuildSettingType; -	subType: ExcludeEnum<typeof ChannelTypes, 'UNKNOWN'>[] | undefined; +	subType: (keyof typeof ChannelType)[] | undefined;  	configurable: boolean;  }  const asGuildSetting = <T>(et: { [K in keyof T]: GuildSetting }) => et; @@ -200,14 +199,14 @@ export const guildSettingsObj = asGuildSetting({  		name: 'Auto Publish Channels',  		description: 'Channels were every message is automatically published.',  		type: 'channel-array', -		subType: ['GUILD_NEWS'], +		subType: ['GuildNews'],  		configurable: true  	},  	welcomeChannel: {  		name: 'Welcome Channel',  		description: 'The channel where the bot will send join and leave message.',  		type: 'channel', -		subType: ['GUILD_TEXT', 'GUILD_NEWS', 'GUILD_NEWS_THREAD', 'GUILD_PUBLIC_THREAD', 'GUILD_PRIVATE_THREAD'], +		subType: ['GuildText', 'GuildNews', 'GuildNewsThread', 'GuildPublicThread', 'GuildPrivateThread'],  		configurable: true  	},  	muteRole: { @@ -228,7 +227,7 @@ export const guildSettingsObj = asGuildSetting({  		name: 'Lockdown Channels',  		description: 'Channels that are locked down when a mass lockdown is specified.',  		type: 'channel-array', -		subType: ['GUILD_TEXT'], +		subType: ['GuildText'],  		configurable: true  	},  	joinRoles: { @@ -249,7 +248,7 @@ export const guildSettingsObj = asGuildSetting({  		name: 'Log Channels',  		description: 'The channel were logs are sent.',  		type: 'custom', -		subType: ['GUILD_TEXT'], +		subType: ['GuildText'],  		configurable: false  	},  	autoModPhases: { @@ -263,7 +262,7 @@ export const guildSettingsObj = asGuildSetting({  		name: 'No Xp Channels',  		description: 'Channels where users will not earn xp for leveling.',  		type: 'channel-array', -		subType: ['GUILD_TEXT', 'GUILD_NEWS', 'GUILD_NEWS_THREAD', 'GUILD_PUBLIC_THREAD', 'GUILD_PRIVATE_THREAD'], +		subType: ['GuildText', 'GuildNews', 'GuildNewsThread', 'GuildPublicThread', 'GuildPrivateThread'],  		configurable: true  	},  	levelRoles: { @@ -277,7 +276,7 @@ export const guildSettingsObj = asGuildSetting({  		name: 'Level Up Channel',  		description: 'The channel to send level up messages in instead of last channel.',  		type: 'channel', -		subType: ['GUILD_TEXT', 'GUILD_NEWS', 'GUILD_NEWS_THREAD', 'GUILD_PUBLIC_THREAD', 'GUILD_PRIVATE_THREAD'], +		subType: ['GuildText', 'GuildNews', 'GuildNewsThread', 'GuildPublicThread', 'GuildPrivateThread'],  		configurable: true  	}  }); diff --git a/src/lib/utils/AllowedMentions.ts b/src/lib/utils/AllowedMentions.ts index b4b188d..c0c76ee 100644 --- a/src/lib/utils/AllowedMentions.ts +++ b/src/lib/utils/AllowedMentions.ts @@ -7,31 +7,37 @@ export class AllowedMentions {  	public static none(): MessageMentionOptions {  		return { parse: [] };  	} +  	public static all(): MessageMentionOptions {  		return {  			parse: ['everyone', 'roles', 'users']  		};  	} +  	public static users(): MessageMentionOptions {  		return {  			parse: ['users']  		};  	} +  	public static everyone(): MessageMentionOptions {  		return {  			parse: ['everyone']  		};  	} +  	public static roles(): MessageMentionOptions {  		return {  			parse: ['roles']  		};  	} +  	public constructor(users = true, roles = false, everyone = false) {  		this.everyone = everyone;  		this.roles = roles;  		this.users = users;  	} +  	public toObject(): MessageMentionOptions {  		return {  			parse: [ diff --git a/src/lib/utils/BushConstants.ts b/src/lib/utils/BushConstants.ts index 306f580..e8366a1 100644 --- a/src/lib/utils/BushConstants.ts +++ b/src/lib/utils/BushConstants.ts @@ -85,7 +85,7 @@ export class BushConstants {  	} as const);  	public static regex = BushClientUtil.deepFreeze({ -		snowflake: /\d{15,21}/im, +		snowflake: /^\d{15,21}$/im,  		discordEmoji: /<a?:(?<name>[a-zA-Z0-9_]+):(?<id>\d{15,21})>/im,  		//stolen from geek @@ -222,49 +222,48 @@ export class BushConstants {  		},  		otherEmojis: { -			SERVER_BOOSTER_1: '<:serverBooster1:848740052091142145>', -			SERVER_BOOSTER_2: '<:serverBooster2:848740090506510388>', -			SERVER_BOOSTER_3: '<:serverBooster3:848740124992077835>', -			SERVER_BOOSTER_6: '<:serverBooster6:848740155245461514>', -			SERVER_BOOSTER_9: '<:serverBooster9:848740188846030889>', -			SERVER_BOOSTER_12: '<:serverBooster12:848740304365551668>', -			SERVER_BOOSTER_15: '<:serverBooster15:848740354890137680>', -			SERVER_BOOSTER_18: '<:serverBooster18:848740402886606868>', -			SERVER_BOOSTER_24: '<:serverBooster24:848740444628320256>', -			NITRO: '<:nitro:848740498054971432>', -			BOOSTER: '<:booster:848747775020892200>', -			OWNER: '<:owner:848746439311753286>', -			ADMIN: '<:admin:848963914628333598>', -			SUPERUSER: '<:superUser:848947986326224926>', -			DEVELOPER: '<:developer:848954538111139871>', -			BUSH_VERIFIED: '<:verfied:853360152090771497>', -			BOOST_TIER_1: '<:boostitle:853363736679940127>', -			BOOST_TIER_2: '<:boostitle:853363752728789075>', -			BOOST_TIER_3: '<:boostitle:853363769132056627>', -			GUILD_TEXT: '<:text:853375537791893524>', -			GUILD_NEWS: '<:announcements:853375553531674644>', -			GUILD_VOICE: '<:voice:853375566735212584>', -			GUILD_STAGE_VOICE: '<:stage:853375583521210468>', -			GUILD_STORE: '<:store:853375601175691266>', -			GUILD_CATEGORY: '<:category:853375615260819476>', -			THREAD: '<:thread:865033845753249813>' +			ServerBooster1: '<:serverBooster1:848740052091142145>', +			ServerBooster2: '<:serverBooster2:848740090506510388>', +			ServerBooster3: '<:serverBooster3:848740124992077835>', +			ServerBooster6: '<:serverBooster6:848740155245461514>', +			ServerBooster9: '<:serverBooster9:848740188846030889>', +			ServerBooster12: '<:serverBooster12:848740304365551668>', +			ServerBooster15: '<:serverBooster15:848740354890137680>', +			ServerBooster18: '<:serverBooster18:848740402886606868>', +			ServerBooster24: '<:serverBooster24:848740444628320256>', +			Nitro: '<:nitro:848740498054971432>', +			Booster: '<:booster:848747775020892200>', +			Owner: '<:owner:848746439311753286>', +			Admin: '<:admin:848963914628333598>', +			Superuser: '<:superUser:848947986326224926>', +			Developer: '<:developer:848954538111139871>', +			BushVerified: '<:verfied:853360152090771497>', +			BoostTier1: '<:boostitle:853363736679940127>', +			BoostTier2: '<:boostitle:853363752728789075>', +			BoostTier3: '<:boostitle:853363769132056627>', +			ChannelText: '<:text:853375537791893524>', +			ChannelNews: '<:announcements:853375553531674644>', +			ChannelVoice: '<:voice:853375566735212584>', +			ChannelStage: '<:stage:853375583521210468>', +			ChannelStore: '<:store:853375601175691266>', +			ChannelCategory: '<:category:853375615260819476>', +			ChannelThread: '<:thread:865033845753249813>'  		},  		userFlags: { -			DISCORD_EMPLOYEE: '<:discordEmployee:848742947826434079>', -			PARTNERED_SERVER_OWNER: '<:partneredServerOwner:848743051593777152>', -			HYPESQUAD_EVENTS: '<:hypeSquadEvents:848743108283072553>', -			BUGHUNTER_LEVEL_1: '<:bugHunter:848743239850393640>', -			HOUSE_BRAVERY: '<:hypeSquadBravery:848742910563844127>', -			HOUSE_BRILLIANCE: '<:hypeSquadBrilliance:848742840649646101>', -			HOUSE_BALANCE: '<:hypeSquadBalance:848742877537370133>', -			EARLY_SUPPORTER: '<:earlySupporter:848741030102171648>', -			TEAM_USER: 'TEAM_USER', -			SYSTEM: 'SYSTEM', -			BUGHUNTER_LEVEL_2: '<:bugHunterGold:848743283080822794>', +			STAFF: '<:discordEmployee:848742947826434079>', +			PARTNER: '<:partneredServerOwner:848743051593777152>', +			HYPESQUAD: '<:hypeSquadEvents:848743108283072553>', +			BUG_HUNTER_LEVEL_1: '<:bugHunter:848743239850393640>', +			HYPESQUAD_ONLINE_HOUSE_1: '<:hypeSquadBravery:848742910563844127>', +			HYPESQUAD_ONLINE_HOUSE_2: '<:hypeSquadBrilliance:848742840649646101>', +			HYPESQUAD_ONLINE_HOUSE_3: '<:hypeSquadBalance:848742877537370133>', +			PREMIUM_EARLY_SUPPORTER: '<:earlySupporter:848741030102171648>', +			TEAM_PSEUDO_USER: 'TEAM_PSEUDO_USER', +			BUG_HUNTER_LEVEL_2: '<:bugHunterGold:848743283080822794>',  			VERIFIED_BOT: 'VERIFIED_BOT', -			EARLY_VERIFIED_BOT_DEVELOPER: '<:earlyVerifiedBotDeveloper:848741079875846174>', -			DISCORD_CERTIFIED_MODERATOR: '<:discordCertifiedModerator:877224285901582366>', +			VERIFIED_DEVELOPER: '<:earlyVerifiedBotDeveloper:848741079875846174>', +			CERTIFIED_MODERATOR: '<:discordCertifiedModerator:877224285901582366>',  			BOT_HTTP_INTERACTIONS: 'BOT_HTTP_INTERACTIONS'  		}, @@ -365,8 +364,7 @@ export class BushConstants {  			'Giveaway (25m)': ['*', 'Admin Perms', 'Sr. Moderator', 'Moderator'],  			'Giveaway (10m)': ['*', 'Admin Perms', 'Sr. Moderator', 'Moderator'],  			'Giveaway (5m)': ['*', 'Admin Perms', 'Sr. Moderator', 'Moderator'], -			'Giveaway (1m)': ['*', 'Admin Perms', 'Sr. Moderator', 'Moderator'], -			'DJ': ['*', 'Admin Perms', 'Sr. Moderator', 'Moderator'] +			'Giveaway (1m)': ['*', 'Admin Perms', 'Sr. Moderator', 'Moderator']  		}  	} as const); | 
