import { Arg, BushCommand, clientSendAndPermCheck, colors, emojis, mappings, oxford, sleep, Time, timestampAndDelta, type CommandMessage, type OptArgType, type SlashMessage } from '#lib'; import { ActivityType, ApplicationCommandOptionType, ApplicationFlagsBitField, EmbedBuilder, escapeMarkdown, PermissionFlagsBits, TeamMemberMembershipState, UserFlags, type APIApplication, type ApplicationFlagsString, type Guild, type GuildMember, type User } from 'discord.js'; export default class UserInfoCommand extends BushCommand { public constructor() { super('userInfo', { aliases: ['user-info', 'user', 'u'], category: 'info', description: 'Gives information about a specified user.', usage: ['user-info [user]'], examples: ['user-info 322862723090219008'], args: [ { id: 'user', description: 'The user you would like to find information about.', type: Arg.union('user', 'snowflake'), readableType: 'user|snowflake', prompt: 'What user would you like to find information about?', retry: '{error} Choose a valid user to find information about.', optional: true, slashType: ApplicationCommandOptionType.User } ], slash: true, clientPermissions: (m) => clientSendAndPermCheck(m, [PermissionFlagsBits.EmbedLinks], true), userPermissions: [] }); } public override async exec(message: CommandMessage | SlashMessage, args: { user: OptArgType<'user' | 'snowflake'> }) { const user = args.user === null ? message.author : typeof args.user === 'object' ? args.user : await this.client.users.fetch(`${args.user}`).catch(() => undefined); if (user === undefined) return message.util.reply(`${emojis.error} Invalid user.`); const member = message.guild ? await message.guild.members.fetch(user.id).catch(() => undefined) : undefined; await user.fetch(true); // gets banner info and accent color const userEmbed = await UserInfoCommand.makeUserInfoEmbed(user, member, message.guild); return await message.util.reply({ embeds: [userEmbed] }); } public static async makeUserInfoEmbed(user: User, member?: GuildMember, guild?: Guild | null) { const emojis = []; const superUsers = user.client.utils.getShared('superUsers'); const userEmbed = new EmbedBuilder() .setTitle(escapeMarkdown(user.tag)) .setThumbnail(user.displayAvatarURL({ size: 2048, extension: 'png' })) .setTimestamp() .setFooter({ text: user.tag }) .setColor(member?.displayColor ?? colors.default); // Flags if (user.client.config.owners.includes(user.id)) emojis.push(mappings.otherEmojis.Developer); if (superUsers.includes(user.id)) emojis.push(mappings.otherEmojis.Superuser); const flags = user.flags?.toArray(); if (flags) { flags.forEach((f) => { if (mappings.userFlags[f] !== undefined) { emojis.push(mappings.userFlags[f]); } else emojis.push(`\`${f}\``); }); } // Since discord bald I just guess if someone has nitro if ( Number(user.discriminator) < 10 || mappings.maybeNitroDiscrims.includes(user.discriminator) || user.displayAvatarURL()?.endsWith('.gif') || user.flags?.has(UserFlags.Partner) || user.flags?.has(UserFlags.Staff) || member?.avatar // server avatar ) { emojis.push(mappings.otherEmojis.Nitro); } if (guild?.ownerId == user.id) emojis.push(mappings.otherEmojis.Owner); else if (member?.permissions.has(PermissionFlagsBits.Administrator)) emojis.push(mappings.otherEmojis.Admin); if (member?.premiumSinceTimestamp) emojis.push(mappings.otherEmojis.Booster); await this.generateGeneralInfoField(userEmbed, user); this.generateServerInfoField(userEmbed, member); this.generatePresenceField(userEmbed, member); this.generateRolesField(userEmbed, member); this.generatePermissionsField(userEmbed, member); await this.generateBotField(userEmbed, user); if (emojis) userEmbed.setDescription( `\u200B${emojis.filter((e) => e).join(' ')}${ userEmbed.data.description?.length ? `\n\n${userEmbed.data.description}` : '' }` ); // zero width space return userEmbed; } public static async generateGeneralInfoField(embed: EmbedBuilder, user: User, title = '» General Information') { // General Info const generalInfo = [ `**Mention:** <@${user.id}>`, `**ID:** ${user.id}`, `**Created:** ${timestampAndDelta(user.createdAt, 'd')}` ]; if (user.accentColor !== null) generalInfo.push(`**Accent Color:** ${user.hexAccentColor}`); if (user.banner) generalInfo.push(`**Banner:** [link](${user.bannerURL({ extension: 'png', size: 4096 })})`); const pronouns = await Promise.race([user.client.utils.getPronounsOf(user), sleep(2 * Time.Second)]); // cut off request after 2 seconds if (pronouns && typeof pronouns === 'string' && pronouns !== 'Unspecified') generalInfo.push(`**Pronouns:** ${pronouns}`); embed.addFields({ name: title, value: generalInfo.join('\n') }); } public static generateServerInfoField(embed: EmbedBuilder, member?: GuildMember | undefined, title = '» Server Information') { if (!member) return; // Server User Info const serverUserInfo = []; if (member.joinedTimestamp) serverUserInfo.push( `**${member.guild!.ownerId == member.user.id ? 'Created Server' : 'Joined'}:** ${timestampAndDelta( member.joinedAt!, 'd' )}` ); if (member.premiumSince) serverUserInfo.push(`**Booster Since:** ${timestampAndDelta(member.premiumSince, 'd')}`); if (member.displayHexColor) serverUserInfo.push(`**Display Color:** ${member.displayHexColor}`); if (member.user.id == '322862723090219008' && member.guild?.id == mappings.guilds.bush) serverUserInfo.push(`**General Deletions:** 1⅓`); if ( (['384620942577369088', '496409778822709251'] as const).includes(member.user.id) && member.guild.id == mappings.guilds.bush ) serverUserInfo.push(`**General Deletions:** ⅓`); if (member?.nickname) serverUserInfo.push(`**Nickname:** ${escapeMarkdown(member?.nickname)}`); if (serverUserInfo.length) embed.addFields({ name: title, value: serverUserInfo.join('\n') }); } public static generatePresenceField(embed: EmbedBuilder, member?: GuildMember | undefined, title = '» Presence') { if (!member || !member.presence) return; if (!member.presence.status && !member.presence.clientStatus && !member.presence.activities) return; // User Presence Info let customStatus = ''; const activitiesNames: string[] = []; if (member.presence.activities) { member.presence.activities.forEach((a) => { if (a.type == ActivityType.Custom && a.state) { const emoji = `${a.emoji ? `${a.emoji.toString()} ` : ''}`; customStatus = `${emoji}${a.state}`; } activitiesNames.push(`\`${a.name}\``); }); } let devices; if (member?.presence.clientStatus) devices = Object.keys(member.presence.clientStatus); const presenceInfo = []; if (member?.presence.status) presenceInfo.push(`**Status:** ${member.presence.status}`); if (devices && devices.length) presenceInfo.push(`**${devices.length - 1 ? 'Devices' : 'Device'}:** ${oxford(devices, 'and', '')}`); if (activitiesNames.length) presenceInfo.push(`**Activit${activitiesNames.length - 1 ? 'ies' : 'y'}:** ${oxford(activitiesNames, 'and', '')}`); if (customStatus && customStatus.length) presenceInfo.push(`**Custom Status:** ${escapeMarkdown(customStatus)}`); embed.addFields({ name: title, value: presenceInfo.join('\n') }); enum statusEmojis { online = '787550449435803658', idle = '787550520956551218', dnd = '787550487633330176', offline = '787550565382750239', invisible = '787550565382750239' } embed.setFooter({ text: member.user.tag, iconURL: member.client.emojis.cache.get(statusEmojis[member?.presence.status])?.url ?? undefined }); } public static generateRolesField(embed: EmbedBuilder, member?: GuildMember | undefined) { if (!member || member.roles.cache.size <= 1) return; // roles const roles = member.roles.cache .filter((role) => role.name !== '@everyone') .sort((role1, role2) => role2.position - role1.position) .map((role) => `${role}`); const joined = roles.join(', '); embed.addFields({ name: `» Role${roles.length - 1 ? 's' : ''} [${roles.length}]`, value: joined.length > 1024 ? 'Too Many Roles to Display' + '...' : joined }); } public static generatePermissionsField( embed: EmbedBuilder, member: GuildMember | undefined, title = '» Important Permissions' ) { if (!member) return; // Important Perms const perms = []; if (member?.permissions.has(PermissionFlagsBits.Administrator) || member.guild?.ownerId == member.user.id) { perms.push('`Administrator`'); } else if (member?.permissions.toArray().length) { member.permissions.toArray().forEach((permission) => { if (mappings.permissions[permission]?.important) { perms.push(`\`${mappings.permissions[permission].name}\``); } }); } if (perms.length) embed.addFields({ name: title, value: perms.join(' ') }); } public static async generateBotField(embed: EmbedBuilder, user: User, title = '» Bot Information') { if (!user.bot) return; const applicationInfo = (await user.client.rest .get(`/applications/${user.id}/rpc`) .catch(() => null)) as APIApplication | null; if (!applicationInfo) return; const flags = new ApplicationFlagsBitField(applicationInfo.flags); const intent = (check: ApplicationFlagsString, warn: ApplicationFlagsString) => { if (flags.has(check)) return emojis.check; if (flags.has(warn)) return emojis.warn; return emojis.cross; }; const botInfo = [ `**Publicity:** ${applicationInfo.bot_public ? 'Public' : 'Private'}`, `**Requires Code Grant:** ${applicationInfo.bot_require_code_grant ? emojis.check : emojis.cross}`, `**Server Members Intent:** ${intent('GatewayGuildMembers', 'GatewayGuildMembersLimited')}`, `**Presence Intent:** ${intent('GatewayPresence', 'GatewayPresenceLimited')}`, `**Message Content Intent:** ${intent('GatewayMessageContent', 'GatewayMessageContentLimited')}` ]; if (applicationInfo.owner || applicationInfo.team) { const teamMembers = applicationInfo.owner ? [applicationInfo.owner] : applicationInfo .team!.members.filter((tm) => tm.membership_state === TeamMemberMembershipState.Accepted) .map((tm) => tm.user); botInfo.push( `**Developer${teamMembers.length > 1 ? 's' : ''}:** ${teamMembers .map((m) => `${m.username}#${m.discriminator}`) .join(', ')}` ); } if (botInfo.length) embed.addFields({ name: title, value: botInfo.join('\n') }); } }