discord-sb.js 1.0.0
This diff represents the content of publicly available package versions that have been released to one of the supported registries. The information contained in this diff is provided for informational purposes only and reflects changes between package versions as they appear in their respective public registries.
- package/LICENSE +674 -0
- package/README.md +2 -0
- package/package.json +91 -0
- package/src/WebSocket.js +39 -0
- package/src/client/BaseClient.js +86 -0
- package/src/client/Client.js +978 -0
- package/src/client/WebhookClient.js +61 -0
- package/src/client/actions/Action.js +116 -0
- package/src/client/actions/ActionsManager.js +80 -0
- package/src/client/actions/ApplicationCommandPermissionsUpdate.js +34 -0
- package/src/client/actions/AutoModerationActionExecution.js +27 -0
- package/src/client/actions/AutoModerationRuleCreate.js +28 -0
- package/src/client/actions/AutoModerationRuleDelete.js +32 -0
- package/src/client/actions/AutoModerationRuleUpdate.js +30 -0
- package/src/client/actions/ChannelCreate.js +23 -0
- package/src/client/actions/ChannelDelete.js +39 -0
- package/src/client/actions/ChannelUpdate.js +43 -0
- package/src/client/actions/GuildAuditLogEntryCreate.js +29 -0
- package/src/client/actions/GuildBanAdd.js +20 -0
- package/src/client/actions/GuildBanRemove.js +25 -0
- package/src/client/actions/GuildChannelsPositionUpdate.js +21 -0
- package/src/client/actions/GuildDelete.js +65 -0
- package/src/client/actions/GuildEmojiCreate.js +20 -0
- package/src/client/actions/GuildEmojiDelete.js +21 -0
- package/src/client/actions/GuildEmojiUpdate.js +20 -0
- package/src/client/actions/GuildEmojisUpdate.js +34 -0
- package/src/client/actions/GuildIntegrationsUpdate.js +19 -0
- package/src/client/actions/GuildMemberRemove.js +33 -0
- package/src/client/actions/GuildMemberUpdate.js +44 -0
- package/src/client/actions/GuildRoleCreate.js +25 -0
- package/src/client/actions/GuildRoleDelete.js +31 -0
- package/src/client/actions/GuildRoleUpdate.js +39 -0
- package/src/client/actions/GuildRolesPositionUpdate.js +21 -0
- package/src/client/actions/GuildScheduledEventCreate.js +27 -0
- package/src/client/actions/GuildScheduledEventDelete.js +31 -0
- package/src/client/actions/GuildScheduledEventUpdate.js +30 -0
- package/src/client/actions/GuildScheduledEventUserAdd.js +32 -0
- package/src/client/actions/GuildScheduledEventUserRemove.js +32 -0
- package/src/client/actions/GuildStickerCreate.js +20 -0
- package/src/client/actions/GuildStickerDelete.js +21 -0
- package/src/client/actions/GuildStickerUpdate.js +20 -0
- package/src/client/actions/GuildStickersUpdate.js +34 -0
- package/src/client/actions/GuildUpdate.js +33 -0
- package/src/client/actions/InviteCreate.js +28 -0
- package/src/client/actions/InviteDelete.js +30 -0
- package/src/client/actions/MessageCreate.js +50 -0
- package/src/client/actions/MessageDelete.js +32 -0
- package/src/client/actions/MessageDeleteBulk.js +46 -0
- package/src/client/actions/MessagePollVoteAdd.js +33 -0
- package/src/client/actions/MessagePollVoteRemove.js +33 -0
- package/src/client/actions/MessageReactionAdd.js +68 -0
- package/src/client/actions/MessageReactionRemove.js +50 -0
- package/src/client/actions/MessageReactionRemoveAll.js +33 -0
- package/src/client/actions/MessageReactionRemoveEmoji.js +28 -0
- package/src/client/actions/MessageUpdate.js +26 -0
- package/src/client/actions/PresenceUpdate.js +50 -0
- package/src/client/actions/StageInstanceCreate.js +28 -0
- package/src/client/actions/StageInstanceDelete.js +33 -0
- package/src/client/actions/StageInstanceUpdate.js +30 -0
- package/src/client/actions/ThreadCreate.js +24 -0
- package/src/client/actions/ThreadDelete.js +32 -0
- package/src/client/actions/ThreadListSync.js +59 -0
- package/src/client/actions/ThreadMemberUpdate.js +30 -0
- package/src/client/actions/ThreadMembersUpdate.js +34 -0
- package/src/client/actions/TypingStart.js +29 -0
- package/src/client/actions/UserUpdate.js +35 -0
- package/src/client/actions/VoiceStateUpdate.js +50 -0
- package/src/client/actions/WebhooksUpdate.js +20 -0
- package/src/client/voice/ClientVoiceManager.js +151 -0
- package/src/client/voice/VoiceConnection.js +1249 -0
- package/src/client/voice/dispatcher/AnnexBDispatcher.js +120 -0
- package/src/client/voice/dispatcher/AudioDispatcher.js +145 -0
- package/src/client/voice/dispatcher/BaseDispatcher.js +459 -0
- package/src/client/voice/dispatcher/VPxDispatcher.js +54 -0
- package/src/client/voice/dispatcher/VideoDispatcher.js +68 -0
- package/src/client/voice/networking/VoiceUDPClient.js +173 -0
- package/src/client/voice/networking/VoiceWebSocket.js +286 -0
- package/src/client/voice/player/MediaPlayer.js +321 -0
- package/src/client/voice/player/processing/AnnexBNalSplitter.js +244 -0
- package/src/client/voice/player/processing/IvfSplitter.js +106 -0
- package/src/client/voice/player/processing/PCMInsertSilence.js +37 -0
- package/src/client/voice/receiver/PacketHandler.js +260 -0
- package/src/client/voice/receiver/Receiver.js +96 -0
- package/src/client/voice/receiver/Recorder.js +173 -0
- package/src/client/voice/util/Function.js +116 -0
- package/src/client/voice/util/PlayInterface.js +122 -0
- package/src/client/voice/util/Secretbox.js +64 -0
- package/src/client/voice/util/Silence.js +16 -0
- package/src/client/voice/util/Socket.js +62 -0
- package/src/client/voice/util/VolumeInterface.js +104 -0
- package/src/client/websocket/WebSocketManager.js +392 -0
- package/src/client/websocket/WebSocketShard.js +907 -0
- package/src/client/websocket/handlers/APPLICATION_COMMAND_CREATE.js +18 -0
- package/src/client/websocket/handlers/APPLICATION_COMMAND_DELETE.js +20 -0
- package/src/client/websocket/handlers/APPLICATION_COMMAND_PERMISSIONS_UPDATE.js +5 -0
- package/src/client/websocket/handlers/APPLICATION_COMMAND_UPDATE.js +20 -0
- package/src/client/websocket/handlers/AUTO_MODERATION_ACTION_EXECUTION.js +5 -0
- package/src/client/websocket/handlers/AUTO_MODERATION_RULE_CREATE.js +5 -0
- package/src/client/websocket/handlers/AUTO_MODERATION_RULE_DELETE.js +5 -0
- package/src/client/websocket/handlers/AUTO_MODERATION_RULE_UPDATE.js +5 -0
- package/src/client/websocket/handlers/CALL_CREATE.js +14 -0
- package/src/client/websocket/handlers/CALL_DELETE.js +11 -0
- package/src/client/websocket/handlers/CALL_UPDATE.js +11 -0
- package/src/client/websocket/handlers/CHANNEL_CREATE.js +5 -0
- package/src/client/websocket/handlers/CHANNEL_DELETE.js +5 -0
- package/src/client/websocket/handlers/CHANNEL_PINS_UPDATE.js +22 -0
- package/src/client/websocket/handlers/CHANNEL_RECIPIENT_ADD.js +19 -0
- package/src/client/websocket/handlers/CHANNEL_RECIPIENT_REMOVE.js +16 -0
- package/src/client/websocket/handlers/CHANNEL_UPDATE.js +16 -0
- package/src/client/websocket/handlers/GUILD_AUDIT_LOG_ENTRY_CREATE.js +5 -0
- package/src/client/websocket/handlers/GUILD_BAN_ADD.js +5 -0
- package/src/client/websocket/handlers/GUILD_BAN_REMOVE.js +5 -0
- package/src/client/websocket/handlers/GUILD_CREATE.js +52 -0
- package/src/client/websocket/handlers/GUILD_DELETE.js +5 -0
- package/src/client/websocket/handlers/GUILD_EMOJIS_UPDATE.js +5 -0
- package/src/client/websocket/handlers/GUILD_INTEGRATIONS_UPDATE.js +5 -0
- package/src/client/websocket/handlers/GUILD_MEMBERS_CHUNK.js +39 -0
- package/src/client/websocket/handlers/GUILD_MEMBER_ADD.js +20 -0
- package/src/client/websocket/handlers/GUILD_MEMBER_REMOVE.js +5 -0
- package/src/client/websocket/handlers/GUILD_MEMBER_UPDATE.js +5 -0
- package/src/client/websocket/handlers/GUILD_ROLE_CREATE.js +5 -0
- package/src/client/websocket/handlers/GUILD_ROLE_DELETE.js +5 -0
- package/src/client/websocket/handlers/GUILD_ROLE_UPDATE.js +5 -0
- package/src/client/websocket/handlers/GUILD_SCHEDULED_EVENT_CREATE.js +5 -0
- package/src/client/websocket/handlers/GUILD_SCHEDULED_EVENT_DELETE.js +5 -0
- package/src/client/websocket/handlers/GUILD_SCHEDULED_EVENT_UPDATE.js +5 -0
- package/src/client/websocket/handlers/GUILD_SCHEDULED_EVENT_USER_ADD.js +5 -0
- package/src/client/websocket/handlers/GUILD_SCHEDULED_EVENT_USER_REMOVE.js +5 -0
- package/src/client/websocket/handlers/GUILD_STICKERS_UPDATE.js +5 -0
- package/src/client/websocket/handlers/GUILD_UPDATE.js +5 -0
- package/src/client/websocket/handlers/INTERACTION_MODAL_CREATE.js +12 -0
- package/src/client/websocket/handlers/INVITE_CREATE.js +5 -0
- package/src/client/websocket/handlers/INVITE_DELETE.js +5 -0
- package/src/client/websocket/handlers/MESSAGE_CREATE.js +5 -0
- package/src/client/websocket/handlers/MESSAGE_DELETE.js +5 -0
- package/src/client/websocket/handlers/MESSAGE_DELETE_BULK.js +5 -0
- package/src/client/websocket/handlers/MESSAGE_POLL_VOTE_ADD.js +5 -0
- package/src/client/websocket/handlers/MESSAGE_POLL_VOTE_REMOVE.js +5 -0
- package/src/client/websocket/handlers/MESSAGE_REACTION_ADD.js +5 -0
- package/src/client/websocket/handlers/MESSAGE_REACTION_REMOVE.js +5 -0
- package/src/client/websocket/handlers/MESSAGE_REACTION_REMOVE_ALL.js +5 -0
- package/src/client/websocket/handlers/MESSAGE_REACTION_REMOVE_EMOJI.js +5 -0
- package/src/client/websocket/handlers/MESSAGE_UPDATE.js +16 -0
- package/src/client/websocket/handlers/PRESENCE_UPDATE.js +5 -0
- package/src/client/websocket/handlers/READY.js +121 -0
- package/src/client/websocket/handlers/RELATIONSHIP_ADD.js +19 -0
- package/src/client/websocket/handlers/RELATIONSHIP_REMOVE.js +17 -0
- package/src/client/websocket/handlers/RELATIONSHIP_UPDATE.js +41 -0
- package/src/client/websocket/handlers/RESUMED.js +14 -0
- package/src/client/websocket/handlers/STAGE_INSTANCE_CREATE.js +5 -0
- package/src/client/websocket/handlers/STAGE_INSTANCE_DELETE.js +5 -0
- package/src/client/websocket/handlers/STAGE_INSTANCE_UPDATE.js +5 -0
- package/src/client/websocket/handlers/THREAD_CREATE.js +5 -0
- package/src/client/websocket/handlers/THREAD_DELETE.js +5 -0
- package/src/client/websocket/handlers/THREAD_LIST_SYNC.js +5 -0
- package/src/client/websocket/handlers/THREAD_MEMBERS_UPDATE.js +5 -0
- package/src/client/websocket/handlers/THREAD_MEMBER_UPDATE.js +5 -0
- package/src/client/websocket/handlers/THREAD_UPDATE.js +16 -0
- package/src/client/websocket/handlers/TYPING_START.js +5 -0
- package/src/client/websocket/handlers/USER_GUILD_SETTINGS_UPDATE.js +6 -0
- package/src/client/websocket/handlers/USER_NOTE_UPDATE.js +5 -0
- package/src/client/websocket/handlers/USER_REQUIRED_ACTION_UPDATE.js +78 -0
- package/src/client/websocket/handlers/USER_SETTINGS_UPDATE.js +5 -0
- package/src/client/websocket/handlers/USER_UPDATE.js +5 -0
- package/src/client/websocket/handlers/VOICE_CHANNEL_EFFECT_SEND.js +16 -0
- package/src/client/websocket/handlers/VOICE_CHANNEL_STATUS_UPDATE.js +12 -0
- package/src/client/websocket/handlers/VOICE_SERVER_UPDATE.js +6 -0
- package/src/client/websocket/handlers/VOICE_STATE_UPDATE.js +5 -0
- package/src/client/websocket/handlers/WEBHOOKS_UPDATE.js +5 -0
- package/src/client/websocket/handlers/index.js +84 -0
- package/src/errors/DJSError.js +61 -0
- package/src/errors/Messages.js +217 -0
- package/src/errors/index.js +4 -0
- package/src/index.js +172 -0
- package/src/managers/ApplicationCommandManager.js +264 -0
- package/src/managers/ApplicationCommandPermissionsManager.js +417 -0
- package/src/managers/AutoModerationRuleManager.js +296 -0
- package/src/managers/BaseGuildEmojiManager.js +80 -0
- package/src/managers/BaseManager.js +19 -0
- package/src/managers/BillingManager.js +66 -0
- package/src/managers/CachedManager.js +71 -0
- package/src/managers/ChannelManager.js +148 -0
- package/src/managers/ClientUserSettingManager.js +372 -0
- package/src/managers/DataManager.js +61 -0
- package/src/managers/DeveloperManager.js +254 -0
- package/src/managers/GuildBanManager.js +250 -0
- package/src/managers/GuildChannelManager.js +488 -0
- package/src/managers/GuildEmojiManager.js +171 -0
- package/src/managers/GuildEmojiRoleManager.js +118 -0
- package/src/managers/GuildForumThreadManager.js +108 -0
- package/src/managers/GuildInviteManager.js +213 -0
- package/src/managers/GuildManager.js +338 -0
- package/src/managers/GuildMemberManager.js +599 -0
- package/src/managers/GuildMemberRoleManager.js +195 -0
- package/src/managers/GuildScheduledEventManager.js +314 -0
- package/src/managers/GuildSettingManager.js +155 -0
- package/src/managers/GuildStickerManager.js +179 -0
- package/src/managers/GuildTextThreadManager.js +98 -0
- package/src/managers/InteractionManager.js +39 -0
- package/src/managers/MessageManager.js +423 -0
- package/src/managers/PermissionOverwriteManager.js +164 -0
- package/src/managers/PresenceManager.js +71 -0
- package/src/managers/QuestManager.js +353 -0
- package/src/managers/ReactionManager.js +67 -0
- package/src/managers/ReactionUserManager.js +73 -0
- package/src/managers/RelationshipManager.js +278 -0
- package/src/managers/RoleManager.js +448 -0
- package/src/managers/SessionManager.js +66 -0
- package/src/managers/StageInstanceManager.js +162 -0
- package/src/managers/ThreadManager.js +175 -0
- package/src/managers/ThreadMemberManager.js +186 -0
- package/src/managers/UserManager.js +136 -0
- package/src/managers/UserNoteManager.js +53 -0
- package/src/managers/VoiceStateManager.js +59 -0
- package/src/rest/APIRequest.js +154 -0
- package/src/rest/APIRouter.js +53 -0
- package/src/rest/DiscordAPIError.js +119 -0
- package/src/rest/HTTPError.js +62 -0
- package/src/rest/RESTManager.js +67 -0
- package/src/rest/RateLimitError.js +55 -0
- package/src/rest/RequestHandler.js +466 -0
- package/src/sharding/Shard.js +444 -0
- package/src/sharding/ShardClientUtil.js +279 -0
- package/src/sharding/ShardingManager.js +319 -0
- package/src/structures/AnonymousGuild.js +98 -0
- package/src/structures/ApplicationCommand.js +593 -0
- package/src/structures/ApplicationRoleConnectionMetadata.js +48 -0
- package/src/structures/AutoModerationActionExecution.js +89 -0
- package/src/structures/AutoModerationRule.js +294 -0
- package/src/structures/AutocompleteInteraction.js +107 -0
- package/src/structures/Base.js +43 -0
- package/src/structures/BaseCommandInteraction.js +211 -0
- package/src/structures/BaseGuild.js +116 -0
- package/src/structures/BaseGuildEmoji.js +56 -0
- package/src/structures/BaseGuildTextChannel.js +191 -0
- package/src/structures/BaseGuildVoiceChannel.js +241 -0
- package/src/structures/BaseMessageComponent.js +181 -0
- package/src/structures/ButtonInteraction.js +11 -0
- package/src/structures/CallState.js +63 -0
- package/src/structures/CategoryChannel.js +85 -0
- package/src/structures/Channel.js +284 -0
- package/src/structures/ClientPresence.js +77 -0
- package/src/structures/ClientUser.js +655 -0
- package/src/structures/CommandInteraction.js +41 -0
- package/src/structures/CommandInteractionOptionResolver.js +276 -0
- package/src/structures/ContainerComponent.js +68 -0
- package/src/structures/ContextMenuInteraction.js +65 -0
- package/src/structures/DMChannel.js +219 -0
- package/src/structures/DirectoryChannel.js +20 -0
- package/src/structures/Emoji.js +148 -0
- package/src/structures/FileComponent.js +49 -0
- package/src/structures/ForumChannel.js +31 -0
- package/src/structures/GroupDMChannel.js +394 -0
- package/src/structures/Guild.js +1791 -0
- package/src/structures/GuildAuditLogs.js +746 -0
- package/src/structures/GuildBan.js +59 -0
- package/src/structures/GuildBoost.js +108 -0
- package/src/structures/GuildChannel.js +470 -0
- package/src/structures/GuildEmoji.js +161 -0
- package/src/structures/GuildMember.js +636 -0
- package/src/structures/GuildPreview.js +191 -0
- package/src/structures/GuildPreviewEmoji.js +27 -0
- package/src/structures/GuildScheduledEvent.js +536 -0
- package/src/structures/GuildTemplate.js +236 -0
- package/src/structures/Integration.js +188 -0
- package/src/structures/IntegrationApplication.js +96 -0
- package/src/structures/Interaction.js +290 -0
- package/src/structures/InteractionCollector.js +248 -0
- package/src/structures/InteractionWebhook.js +43 -0
- package/src/structures/Invite.js +358 -0
- package/src/structures/InviteGuild.js +23 -0
- package/src/structures/InviteStageInstance.js +86 -0
- package/src/structures/MediaChannel.js +11 -0
- package/src/structures/MediaGalleryComponent.js +41 -0
- package/src/structures/MediaGalleryItem.js +47 -0
- package/src/structures/Message.js +1252 -0
- package/src/structures/MessageActionRow.js +105 -0
- package/src/structures/MessageAttachment.js +216 -0
- package/src/structures/MessageButton.js +166 -0
- package/src/structures/MessageCollector.js +146 -0
- package/src/structures/MessageComponentInteraction.js +120 -0
- package/src/structures/MessageContextMenuInteraction.js +20 -0
- package/src/structures/MessageEmbed.js +596 -0
- package/src/structures/MessageMentions.js +273 -0
- package/src/structures/MessagePayload.js +354 -0
- package/src/structures/MessageReaction.js +181 -0
- package/src/structures/MessageSelectMenu.js +141 -0
- package/src/structures/Modal.js +161 -0
- package/src/structures/ModalSubmitFieldsResolver.js +53 -0
- package/src/structures/ModalSubmitInteraction.js +119 -0
- package/src/structures/NewsChannel.js +32 -0
- package/src/structures/OAuth2Guild.js +28 -0
- package/src/structures/PermissionOverwrites.js +198 -0
- package/src/structures/Poll.js +108 -0
- package/src/structures/PollAnswer.js +88 -0
- package/src/structures/Presence.js +1157 -0
- package/src/structures/ReactionCollector.js +229 -0
- package/src/structures/ReactionEmoji.js +31 -0
- package/src/structures/Role.js +590 -0
- package/src/structures/SectionComponent.js +48 -0
- package/src/structures/SelectMenuInteraction.js +21 -0
- package/src/structures/SeparatorComponent.js +48 -0
- package/src/structures/Session.js +81 -0
- package/src/structures/StageChannel.js +104 -0
- package/src/structures/StageInstance.js +208 -0
- package/src/structures/Sticker.js +310 -0
- package/src/structures/StickerPack.js +95 -0
- package/src/structures/StoreChannel.js +56 -0
- package/src/structures/Team.js +118 -0
- package/src/structures/TeamMember.js +80 -0
- package/src/structures/TextChannel.js +33 -0
- package/src/structures/TextDisplayComponent.js +40 -0
- package/src/structures/TextInputComponent.js +132 -0
- package/src/structures/ThreadChannel.js +605 -0
- package/src/structures/ThreadMember.js +105 -0
- package/src/structures/ThreadOnlyChannel.js +249 -0
- package/src/structures/ThumbnailComponent.js +57 -0
- package/src/structures/Typing.js +74 -0
- package/src/structures/UnfurledMediaItem.js +29 -0
- package/src/structures/User.js +640 -0
- package/src/structures/UserContextMenuInteraction.js +29 -0
- package/src/structures/VoiceChannel.js +110 -0
- package/src/structures/VoiceChannelEffect.js +69 -0
- package/src/structures/VoiceRegion.js +53 -0
- package/src/structures/VoiceState.js +354 -0
- package/src/structures/WebEmbed.js +373 -0
- package/src/structures/Webhook.js +478 -0
- package/src/structures/WelcomeChannel.js +60 -0
- package/src/structures/WelcomeScreen.js +48 -0
- package/src/structures/Widget.js +87 -0
- package/src/structures/WidgetMember.js +99 -0
- package/src/structures/interfaces/Application.js +953 -0
- package/src/structures/interfaces/Collector.js +300 -0
- package/src/structures/interfaces/InteractionResponses.js +313 -0
- package/src/structures/interfaces/TextBasedChannel.js +821 -0
- package/src/util/APITypes.js +59 -0
- package/src/util/ActivityFlags.js +44 -0
- package/src/util/ApplicationFlags.js +76 -0
- package/src/util/AttachmentFlags.js +38 -0
- package/src/util/BitField.js +170 -0
- package/src/util/ChannelFlags.js +45 -0
- package/src/util/Constants.js +1914 -0
- package/src/util/DataResolver.js +146 -0
- package/src/util/Formatters.js +228 -0
- package/src/util/GuildMemberFlags.js +43 -0
- package/src/util/Intents.js +74 -0
- package/src/util/InviteFlags.js +34 -0
- package/src/util/LimitedCollection.js +131 -0
- package/src/util/MessageFlags.js +63 -0
- package/src/util/Options.js +358 -0
- package/src/util/Permissions.js +202 -0
- package/src/util/PremiumUsageFlags.js +31 -0
- package/src/util/PurchasedFlags.js +33 -0
- package/src/util/RemoteAuth.js +415 -0
- package/src/util/RoleFlags.js +37 -0
- package/src/util/SnowflakeUtil.js +92 -0
- package/src/util/Speaking.js +33 -0
- package/src/util/Sweepers.js +466 -0
- package/src/util/SystemChannelFlags.js +55 -0
- package/src/util/ThreadMemberFlags.js +30 -0
- package/src/util/UserFlags.js +104 -0
- package/src/util/Util.js +1048 -0
- package/typings/enums.d.ts +439 -0
- package/typings/index.d.ts +8505 -0
- package/typings/index.test-d.ts +0 -0
- package/typings/rawDataTypes.d.ts +403 -0
|
@@ -0,0 +1,1252 @@
|
|
|
1
|
+
'use strict';
|
|
2
|
+
|
|
3
|
+
const process = require('node:process');
|
|
4
|
+
const { Collection } = require('@discordjs/collection');
|
|
5
|
+
const Base = require('./Base');
|
|
6
|
+
const BaseMessageComponent = require('./BaseMessageComponent');
|
|
7
|
+
const MessageAttachment = require('./MessageAttachment');
|
|
8
|
+
const Embed = require('./MessageEmbed');
|
|
9
|
+
const Mentions = require('./MessageMentions');
|
|
10
|
+
const MessagePayload = require('./MessagePayload');
|
|
11
|
+
const { Poll } = require('./Poll');
|
|
12
|
+
const ReactionCollector = require('./ReactionCollector');
|
|
13
|
+
const { Sticker } = require('./Sticker');
|
|
14
|
+
const Application = require('./interfaces/Application');
|
|
15
|
+
const { Error } = require('../errors');
|
|
16
|
+
const ReactionManager = require('../managers/ReactionManager');
|
|
17
|
+
const {
|
|
18
|
+
InteractionTypes,
|
|
19
|
+
MessageTypes,
|
|
20
|
+
SystemMessageTypes,
|
|
21
|
+
MessageComponentTypes,
|
|
22
|
+
MessageReferenceTypes,
|
|
23
|
+
} = require('../util/Constants');
|
|
24
|
+
const MessageFlags = require('../util/MessageFlags');
|
|
25
|
+
const Permissions = require('../util/Permissions');
|
|
26
|
+
const SnowflakeUtil = require('../util/SnowflakeUtil');
|
|
27
|
+
const Util = require('../util/Util');
|
|
28
|
+
|
|
29
|
+
/**
|
|
30
|
+
* @type {WeakSet<Message>}
|
|
31
|
+
* @private
|
|
32
|
+
* @internal
|
|
33
|
+
*/
|
|
34
|
+
const deletedMessages = new WeakSet();
|
|
35
|
+
let deprecationEmittedForDeleted = false;
|
|
36
|
+
|
|
37
|
+
/**
|
|
38
|
+
* Represents a message on Discord.
|
|
39
|
+
* @extends {Base}
|
|
40
|
+
*/
|
|
41
|
+
class Message extends Base {
|
|
42
|
+
constructor(client, data) {
|
|
43
|
+
super(client);
|
|
44
|
+
|
|
45
|
+
/**
|
|
46
|
+
* The id of the channel the message was sent in
|
|
47
|
+
* @type {Snowflake}
|
|
48
|
+
*/
|
|
49
|
+
this.channelId = data.channel_id;
|
|
50
|
+
|
|
51
|
+
/**
|
|
52
|
+
* The id of the guild the message was sent in, if any
|
|
53
|
+
* @type {?Snowflake}
|
|
54
|
+
*/
|
|
55
|
+
this.guildId = data.guild_id ?? this.channel?.guild?.id ?? null;
|
|
56
|
+
|
|
57
|
+
this._patch(data);
|
|
58
|
+
}
|
|
59
|
+
|
|
60
|
+
_patch(data) {
|
|
61
|
+
/**
|
|
62
|
+
* The message's id
|
|
63
|
+
* @type {Snowflake}
|
|
64
|
+
*/
|
|
65
|
+
this.id = data.id;
|
|
66
|
+
|
|
67
|
+
if ('position' in data) {
|
|
68
|
+
/**
|
|
69
|
+
* A generally increasing integer (there may be gaps or duplicates) that represents
|
|
70
|
+
* the approximate position of the message in a thread.
|
|
71
|
+
* @type {?number}
|
|
72
|
+
*/
|
|
73
|
+
this.position = data.position;
|
|
74
|
+
} else {
|
|
75
|
+
this.position ??= null;
|
|
76
|
+
}
|
|
77
|
+
|
|
78
|
+
/**
|
|
79
|
+
* The timestamp the message was sent at
|
|
80
|
+
* @type {number}
|
|
81
|
+
*/
|
|
82
|
+
this.createdTimestamp = this.id ? SnowflakeUtil.timestampFrom(this.id) : new Date(data.timestamp).getTime();
|
|
83
|
+
|
|
84
|
+
if ('type' in data) {
|
|
85
|
+
/**
|
|
86
|
+
* The type of the message
|
|
87
|
+
* @type {?MessageType}
|
|
88
|
+
*/
|
|
89
|
+
this.type = MessageTypes[data.type];
|
|
90
|
+
|
|
91
|
+
/**
|
|
92
|
+
* Whether or not this message was sent by Discord, not actually a user (e.g. pin notifications)
|
|
93
|
+
* @type {?boolean}
|
|
94
|
+
*/
|
|
95
|
+
this.system = SystemMessageTypes.includes(this.type);
|
|
96
|
+
} else {
|
|
97
|
+
this.system ??= null;
|
|
98
|
+
this.type ??= null;
|
|
99
|
+
}
|
|
100
|
+
|
|
101
|
+
if ('content' in data) {
|
|
102
|
+
/**
|
|
103
|
+
* The content of the message
|
|
104
|
+
* @type {?string}
|
|
105
|
+
*/
|
|
106
|
+
this.content = data.content;
|
|
107
|
+
} else {
|
|
108
|
+
this.content ??= null;
|
|
109
|
+
}
|
|
110
|
+
|
|
111
|
+
if ('author' in data) {
|
|
112
|
+
/**
|
|
113
|
+
* The author of the message
|
|
114
|
+
* @type {?User}
|
|
115
|
+
*/
|
|
116
|
+
this.author = this.client.users._add(data.author, !data.webhook_id);
|
|
117
|
+
} else {
|
|
118
|
+
this.author ??= null;
|
|
119
|
+
}
|
|
120
|
+
|
|
121
|
+
if ('pinned' in data) {
|
|
122
|
+
/**
|
|
123
|
+
* Whether or not this message is pinned
|
|
124
|
+
* @type {?boolean}
|
|
125
|
+
*/
|
|
126
|
+
this.pinned = Boolean(data.pinned);
|
|
127
|
+
} else {
|
|
128
|
+
this.pinned ??= null;
|
|
129
|
+
}
|
|
130
|
+
|
|
131
|
+
if ('tts' in data) {
|
|
132
|
+
/**
|
|
133
|
+
* Whether or not the message was Text-To-Speech
|
|
134
|
+
* @type {?boolean}
|
|
135
|
+
*/
|
|
136
|
+
this.tts = data.tts;
|
|
137
|
+
} else {
|
|
138
|
+
this.tts ??= null;
|
|
139
|
+
}
|
|
140
|
+
|
|
141
|
+
if ('nonce' in data) {
|
|
142
|
+
/**
|
|
143
|
+
* A random number or string used for checking message delivery
|
|
144
|
+
* <warn>This is only received after the message was sent successfully, and
|
|
145
|
+
* lost if re-fetched</warn>
|
|
146
|
+
* @type {?string}
|
|
147
|
+
*/
|
|
148
|
+
this.nonce = data.nonce;
|
|
149
|
+
} else {
|
|
150
|
+
this.nonce ??= null;
|
|
151
|
+
}
|
|
152
|
+
|
|
153
|
+
if ('embeds' in data) {
|
|
154
|
+
/**
|
|
155
|
+
* A list of embeds in the message - e.g. YouTube Player
|
|
156
|
+
* @type {MessageEmbed[]}
|
|
157
|
+
*/
|
|
158
|
+
this.embeds = data.embeds.map(e => new Embed(e, true));
|
|
159
|
+
} else {
|
|
160
|
+
this.embeds = this.embeds?.slice() ?? [];
|
|
161
|
+
}
|
|
162
|
+
|
|
163
|
+
/**
|
|
164
|
+
* @typedef {MessageActionRow|ContainerComponent|FileComponent|MediaGalleryComponent|SectionComponent|SeparatorComponent|TextDisplayComponent} TopLevelComponent
|
|
165
|
+
*/
|
|
166
|
+
|
|
167
|
+
if ('components' in data) {
|
|
168
|
+
/**
|
|
169
|
+
* An array of components in the message
|
|
170
|
+
* @type {TopLevelComponent[]}
|
|
171
|
+
*/
|
|
172
|
+
this.components = data.components.map(c => BaseMessageComponent.create(c, this.client));
|
|
173
|
+
} else {
|
|
174
|
+
this.components = this.components?.slice() ?? [];
|
|
175
|
+
}
|
|
176
|
+
|
|
177
|
+
if ('attachments' in data) {
|
|
178
|
+
/**
|
|
179
|
+
* A collection of attachments in the message - e.g. Pictures - mapped by their ids
|
|
180
|
+
* @type {Collection<Snowflake, MessageAttachment>}
|
|
181
|
+
*/
|
|
182
|
+
this.attachments = new Collection();
|
|
183
|
+
if (data.attachments) {
|
|
184
|
+
for (const attachment of data.attachments) {
|
|
185
|
+
this.attachments.set(attachment.id, new MessageAttachment(attachment.url, attachment.filename, attachment));
|
|
186
|
+
}
|
|
187
|
+
}
|
|
188
|
+
} else {
|
|
189
|
+
this.attachments = new Collection(this.attachments);
|
|
190
|
+
}
|
|
191
|
+
|
|
192
|
+
if ('sticker_items' in data || 'stickers' in data) {
|
|
193
|
+
/**
|
|
194
|
+
* A collection of stickers in the message
|
|
195
|
+
* @type {Collection<Snowflake, Sticker>}
|
|
196
|
+
*/
|
|
197
|
+
this.stickers = new Collection(
|
|
198
|
+
(data.sticker_items ?? data.stickers)?.map(s => [s.id, new Sticker(this.client, s)]),
|
|
199
|
+
);
|
|
200
|
+
} else {
|
|
201
|
+
this.stickers = new Collection(this.stickers);
|
|
202
|
+
}
|
|
203
|
+
|
|
204
|
+
// Discord sends null if the message has not been edited
|
|
205
|
+
if (data.edited_timestamp) {
|
|
206
|
+
/**
|
|
207
|
+
* The timestamp the message was last edited at (if applicable)
|
|
208
|
+
* @type {?number}
|
|
209
|
+
*/
|
|
210
|
+
this.editedTimestamp = data.edited_timestamp ? Date.parse(data.edited_timestamp) : null;
|
|
211
|
+
} else {
|
|
212
|
+
this.editedTimestamp ??= null;
|
|
213
|
+
}
|
|
214
|
+
|
|
215
|
+
if ('reactions' in data) {
|
|
216
|
+
/**
|
|
217
|
+
* A manager of the reactions belonging to this message
|
|
218
|
+
* @type {ReactionManager}
|
|
219
|
+
*/
|
|
220
|
+
this.reactions = new ReactionManager(this);
|
|
221
|
+
if (data.reactions?.length > 0) {
|
|
222
|
+
for (const reaction of data.reactions) {
|
|
223
|
+
this.reactions._add(reaction);
|
|
224
|
+
}
|
|
225
|
+
}
|
|
226
|
+
} else {
|
|
227
|
+
this.reactions ??= new ReactionManager(this);
|
|
228
|
+
}
|
|
229
|
+
|
|
230
|
+
if (!this.mentions) {
|
|
231
|
+
/**
|
|
232
|
+
* All valid mentions that the message contains
|
|
233
|
+
* @type {MessageMentions}
|
|
234
|
+
*/
|
|
235
|
+
this.mentions = new Mentions(
|
|
236
|
+
this,
|
|
237
|
+
data.mentions,
|
|
238
|
+
data.mention_roles,
|
|
239
|
+
data.mention_everyone,
|
|
240
|
+
data.mention_channels,
|
|
241
|
+
data.referenced_message?.author,
|
|
242
|
+
);
|
|
243
|
+
} else {
|
|
244
|
+
this.mentions = new Mentions(
|
|
245
|
+
this,
|
|
246
|
+
data.mentions ?? this.mentions.users,
|
|
247
|
+
data.mention_roles ?? this.mentions.roles,
|
|
248
|
+
data.mention_everyone ?? this.mentions.everyone,
|
|
249
|
+
data.mention_channels ?? this.mentions.crosspostedChannels,
|
|
250
|
+
data.referenced_message?.author ?? this.mentions.repliedUser,
|
|
251
|
+
);
|
|
252
|
+
}
|
|
253
|
+
|
|
254
|
+
if ('webhook_id' in data) {
|
|
255
|
+
/**
|
|
256
|
+
* The id of the webhook that sent the message, if applicable
|
|
257
|
+
* @type {?Snowflake}
|
|
258
|
+
*/
|
|
259
|
+
this.webhookId = data.webhook_id;
|
|
260
|
+
} else {
|
|
261
|
+
this.webhookId ??= null;
|
|
262
|
+
}
|
|
263
|
+
|
|
264
|
+
if (data.poll) {
|
|
265
|
+
/**
|
|
266
|
+
* The poll that was sent with the message
|
|
267
|
+
* @type {?Poll}
|
|
268
|
+
*/
|
|
269
|
+
this.poll = new Poll(this.client, data.poll, this);
|
|
270
|
+
} else {
|
|
271
|
+
this.poll ??= null;
|
|
272
|
+
}
|
|
273
|
+
|
|
274
|
+
if ('application' in data) {
|
|
275
|
+
/**
|
|
276
|
+
* Supplemental application information for group activities
|
|
277
|
+
* @type {?Application}
|
|
278
|
+
*/
|
|
279
|
+
this.groupActivityApplication = new Application(this.client, data.application);
|
|
280
|
+
} else {
|
|
281
|
+
this.groupActivityApplication ??= null;
|
|
282
|
+
}
|
|
283
|
+
|
|
284
|
+
if ('application_id' in data) {
|
|
285
|
+
/**
|
|
286
|
+
* The id of the application of the interaction that sent this message, if any
|
|
287
|
+
* @type {?Snowflake}
|
|
288
|
+
*/
|
|
289
|
+
this.applicationId = data.application_id;
|
|
290
|
+
} else {
|
|
291
|
+
this.applicationId ??= null;
|
|
292
|
+
}
|
|
293
|
+
|
|
294
|
+
if ('activity' in data) {
|
|
295
|
+
/**
|
|
296
|
+
* Group activity
|
|
297
|
+
* @type {?MessageActivity}
|
|
298
|
+
*/
|
|
299
|
+
this.activity = {
|
|
300
|
+
partyId: data.activity.party_id,
|
|
301
|
+
type: data.activity.type,
|
|
302
|
+
};
|
|
303
|
+
} else {
|
|
304
|
+
this.activity ??= null;
|
|
305
|
+
}
|
|
306
|
+
|
|
307
|
+
if ('thread' in data) {
|
|
308
|
+
this.client.channels._add(data.thread, this.guild);
|
|
309
|
+
}
|
|
310
|
+
|
|
311
|
+
if (this.member && data.member) {
|
|
312
|
+
this.member._patch(data.member);
|
|
313
|
+
} else if (data.member && this.guild && this.author) {
|
|
314
|
+
this.guild.members._add(Object.assign(data.member, { user: this.author }));
|
|
315
|
+
}
|
|
316
|
+
|
|
317
|
+
if ('flags' in data) {
|
|
318
|
+
/**
|
|
319
|
+
* Flags that are applied to the message
|
|
320
|
+
* @type {Readonly<MessageFlags>}
|
|
321
|
+
*/
|
|
322
|
+
this.flags = new MessageFlags(data.flags).freeze();
|
|
323
|
+
} else {
|
|
324
|
+
this.flags = new MessageFlags(this.flags).freeze();
|
|
325
|
+
}
|
|
326
|
+
|
|
327
|
+
/**
|
|
328
|
+
* Reference data sent in a message that contains ids identifying the referenced message.
|
|
329
|
+
* This can be present in the following types of message:
|
|
330
|
+
* * Crossposted messages (IS_CROSSPOST {@link MessageFlags.FLAGS message flag})
|
|
331
|
+
* * CHANNEL_FOLLOW_ADD
|
|
332
|
+
* * CHANNEL_PINNED_MESSAGE
|
|
333
|
+
* * REPLY
|
|
334
|
+
* * THREAD_STARTER_MESSAGE
|
|
335
|
+
* @see {@link https://discord.com/developers/docs/resources/channel#message-types}
|
|
336
|
+
* @typedef {Object} MessageReference
|
|
337
|
+
* @property {Snowflake} channelId The channel's id the message was referenced
|
|
338
|
+
* @property {?Snowflake} guildId The guild's id the message was referenced
|
|
339
|
+
* @property {?Snowflake} messageId The message's id that was referenced
|
|
340
|
+
* @property {MessageReferenceType} type The type of the message reference
|
|
341
|
+
*/
|
|
342
|
+
|
|
343
|
+
if ('message_reference' in data) {
|
|
344
|
+
/**
|
|
345
|
+
* Message reference data
|
|
346
|
+
* @type {?MessageReference}
|
|
347
|
+
*/
|
|
348
|
+
this.reference = {
|
|
349
|
+
channelId: data.message_reference.channel_id,
|
|
350
|
+
guildId: data.message_reference.guild_id,
|
|
351
|
+
messageId: data.message_reference.message_id,
|
|
352
|
+
type: MessageReferenceTypes[data.message_reference.type ?? 0],
|
|
353
|
+
};
|
|
354
|
+
} else {
|
|
355
|
+
this.reference ??= null;
|
|
356
|
+
}
|
|
357
|
+
|
|
358
|
+
if (data.referenced_message) {
|
|
359
|
+
this.channel?.messages._add({ guild_id: data.message_reference?.guild_id, ...data.referenced_message });
|
|
360
|
+
}
|
|
361
|
+
|
|
362
|
+
/**
|
|
363
|
+
* Partial data of the interaction that a message is a reply to
|
|
364
|
+
* @typedef {Object} MessageInteraction
|
|
365
|
+
* @property {Snowflake} id The interaction's id
|
|
366
|
+
* @property {InteractionType} type The type of the interaction
|
|
367
|
+
* @property {string} commandName The name of the interaction's application command,
|
|
368
|
+
* as well as the subcommand and subcommand group, where applicable
|
|
369
|
+
* @property {User} user The user that invoked the interaction
|
|
370
|
+
*/
|
|
371
|
+
|
|
372
|
+
if (data.interaction) {
|
|
373
|
+
/**
|
|
374
|
+
* Partial data of the interaction that this message is a reply to
|
|
375
|
+
* @type {?MessageInteraction}
|
|
376
|
+
*/
|
|
377
|
+
this.interaction = {
|
|
378
|
+
id: data.interaction.id,
|
|
379
|
+
type: InteractionTypes[data.interaction.type],
|
|
380
|
+
commandName: data.interaction.name,
|
|
381
|
+
user: this.client.users._add(data.interaction.user),
|
|
382
|
+
};
|
|
383
|
+
} else {
|
|
384
|
+
this.interaction ??= null;
|
|
385
|
+
}
|
|
386
|
+
|
|
387
|
+
if (data.message_snapshots) {
|
|
388
|
+
/**
|
|
389
|
+
* The message snapshots associated with the message reference
|
|
390
|
+
* @type {Collection<Snowflake, Message>}
|
|
391
|
+
*/
|
|
392
|
+
this.messageSnapshots = data.message_snapshots.reduce((coll, snapshot) => {
|
|
393
|
+
const channel = this.client.channels.cache.get(this.reference.channelId);
|
|
394
|
+
const snapshotData = {
|
|
395
|
+
...snapshot.message,
|
|
396
|
+
id: this.reference.messageId,
|
|
397
|
+
channel_id: this.reference.channelId,
|
|
398
|
+
guild_id: this.reference.guildId,
|
|
399
|
+
};
|
|
400
|
+
|
|
401
|
+
return coll.set(
|
|
402
|
+
this.reference.messageId,
|
|
403
|
+
channel ? channel.messages._add(snapshotData) : new this.constructor(this.client, snapshotData),
|
|
404
|
+
);
|
|
405
|
+
}, new Collection());
|
|
406
|
+
} else {
|
|
407
|
+
this.messageSnapshots ??= new Collection();
|
|
408
|
+
}
|
|
409
|
+
|
|
410
|
+
/**
|
|
411
|
+
* A call associated with a message
|
|
412
|
+
* @typedef {Object} MessageCall
|
|
413
|
+
* @property {Readonly<?Date>} endedAt The time the call ended
|
|
414
|
+
* @property {?number} endedTimestamp The timestamp the call ended
|
|
415
|
+
* @property {Snowflake[]} participants The ids of the users that participated in the call
|
|
416
|
+
*/
|
|
417
|
+
|
|
418
|
+
if (data.call) {
|
|
419
|
+
/**
|
|
420
|
+
* The call associated with the message
|
|
421
|
+
* @type {?MessageCall}
|
|
422
|
+
*/
|
|
423
|
+
this.call = {
|
|
424
|
+
endedTimestamp: data.call.ended_timestamp ? Date.parse(data.call.ended_timestamp) : null,
|
|
425
|
+
participants: data.call.participants,
|
|
426
|
+
get endedAt() {
|
|
427
|
+
return this.endedTimestamp && new Date(this.endedTimestamp);
|
|
428
|
+
},
|
|
429
|
+
};
|
|
430
|
+
} else {
|
|
431
|
+
this.call ??= null;
|
|
432
|
+
}
|
|
433
|
+
}
|
|
434
|
+
|
|
435
|
+
/**
|
|
436
|
+
* Whether or not the structure has been deleted
|
|
437
|
+
* @type {boolean}
|
|
438
|
+
* @deprecated This will be removed in the next major version, see https://github.com/discordjs/discord.js/issues/7091
|
|
439
|
+
*/
|
|
440
|
+
get deleted() {
|
|
441
|
+
if (!deprecationEmittedForDeleted) {
|
|
442
|
+
deprecationEmittedForDeleted = true;
|
|
443
|
+
process.emitWarning(
|
|
444
|
+
'Message#deleted is deprecated, see https://github.com/discordjs/discord.js/issues/7091.',
|
|
445
|
+
'DeprecationWarning',
|
|
446
|
+
);
|
|
447
|
+
}
|
|
448
|
+
|
|
449
|
+
return deletedMessages.has(this);
|
|
450
|
+
}
|
|
451
|
+
|
|
452
|
+
set deleted(value) {
|
|
453
|
+
if (!deprecationEmittedForDeleted) {
|
|
454
|
+
deprecationEmittedForDeleted = true;
|
|
455
|
+
process.emitWarning(
|
|
456
|
+
'Message#deleted is deprecated, see https://github.com/discordjs/discord.js/issues/7091.',
|
|
457
|
+
'DeprecationWarning',
|
|
458
|
+
);
|
|
459
|
+
}
|
|
460
|
+
|
|
461
|
+
if (value) deletedMessages.add(this);
|
|
462
|
+
else deletedMessages.delete(this);
|
|
463
|
+
}
|
|
464
|
+
|
|
465
|
+
/**
|
|
466
|
+
* The channel that the message was sent in
|
|
467
|
+
* @type {TextBasedChannels}
|
|
468
|
+
* @readonly
|
|
469
|
+
*/
|
|
470
|
+
get channel() {
|
|
471
|
+
return this.client.channels.cache.get(this.channelId) ?? null;
|
|
472
|
+
}
|
|
473
|
+
|
|
474
|
+
/**
|
|
475
|
+
* Whether or not this message is a partial
|
|
476
|
+
* @type {boolean}
|
|
477
|
+
* @readonly
|
|
478
|
+
*/
|
|
479
|
+
get partial() {
|
|
480
|
+
return typeof this.content !== 'string' || !this.author;
|
|
481
|
+
}
|
|
482
|
+
|
|
483
|
+
/**
|
|
484
|
+
* Represents the author of the message as a guild member.
|
|
485
|
+
* Only available if the message comes from a guild where the author is still a member
|
|
486
|
+
* @type {?GuildMember}
|
|
487
|
+
* @readonly
|
|
488
|
+
*/
|
|
489
|
+
get member() {
|
|
490
|
+
return this.guild?.members.resolve(this.author) ?? null;
|
|
491
|
+
}
|
|
492
|
+
|
|
493
|
+
/**
|
|
494
|
+
* The time the message was sent at
|
|
495
|
+
* @type {Date}
|
|
496
|
+
* @readonly
|
|
497
|
+
*/
|
|
498
|
+
get createdAt() {
|
|
499
|
+
return new Date(this.createdTimestamp);
|
|
500
|
+
}
|
|
501
|
+
|
|
502
|
+
/**
|
|
503
|
+
* The time the message was last edited at (if applicable)
|
|
504
|
+
* @type {?Date}
|
|
505
|
+
* @readonly
|
|
506
|
+
*/
|
|
507
|
+
get editedAt() {
|
|
508
|
+
return this.editedTimestamp ? new Date(this.editedTimestamp) : null;
|
|
509
|
+
}
|
|
510
|
+
|
|
511
|
+
/**
|
|
512
|
+
* The guild the message was sent in (if in a guild channel)
|
|
513
|
+
* @type {?Guild}
|
|
514
|
+
* @readonly
|
|
515
|
+
*/
|
|
516
|
+
get guild() {
|
|
517
|
+
return this.client.guilds.cache.get(this.guildId) ?? this.channel?.guild ?? null;
|
|
518
|
+
}
|
|
519
|
+
|
|
520
|
+
/**
|
|
521
|
+
* Whether this message has a thread associated with it
|
|
522
|
+
* @type {boolean}
|
|
523
|
+
* @readonly
|
|
524
|
+
*/
|
|
525
|
+
get hasThread() {
|
|
526
|
+
return this.flags.has(MessageFlags.FLAGS.HAS_THREAD);
|
|
527
|
+
}
|
|
528
|
+
|
|
529
|
+
/**
|
|
530
|
+
* The thread started by this message
|
|
531
|
+
* <info>This property is not suitable for checking whether a message has a thread,
|
|
532
|
+
* use {@link Message#hasThread} instead.</info>
|
|
533
|
+
* @type {?ThreadChannel}
|
|
534
|
+
* @readonly
|
|
535
|
+
*/
|
|
536
|
+
get thread() {
|
|
537
|
+
return this.channel?.threads?.cache.get(this.id) ?? null;
|
|
538
|
+
}
|
|
539
|
+
|
|
540
|
+
/**
|
|
541
|
+
* The URL to jump to this message
|
|
542
|
+
* @type {string}
|
|
543
|
+
* @readonly
|
|
544
|
+
*/
|
|
545
|
+
get url() {
|
|
546
|
+
return `https://discord.com/channels/${this.guildId ?? '@me'}/${this.channelId}/${this.id}`;
|
|
547
|
+
}
|
|
548
|
+
|
|
549
|
+
/**
|
|
550
|
+
* The message contents with all mentions replaced by the equivalent text.
|
|
551
|
+
* If mentions cannot be resolved to a name, the relevant mention in the message content will not be converted.
|
|
552
|
+
* @type {?string}
|
|
553
|
+
* @readonly
|
|
554
|
+
*/
|
|
555
|
+
get cleanContent() {
|
|
556
|
+
// eslint-disable-next-line eqeqeq
|
|
557
|
+
return this.content != null && this.channel ? Util.cleanContent(this.content, this.channel) : null;
|
|
558
|
+
}
|
|
559
|
+
|
|
560
|
+
/**
|
|
561
|
+
* Creates a reaction collector.
|
|
562
|
+
* @param {ReactionCollectorOptions} [options={}] Options to send to the collector
|
|
563
|
+
* @returns {ReactionCollector}
|
|
564
|
+
* @example
|
|
565
|
+
* // Create a reaction collector
|
|
566
|
+
* const filter = (reaction, user) => reaction.emoji.name === '👌' && user.id === 'someId';
|
|
567
|
+
* const collector = message.createReactionCollector({ filter, time: 15_000 });
|
|
568
|
+
* collector.on('collect', r => console.log(`Collected ${r.emoji.name}`));
|
|
569
|
+
* collector.on('end', collected => console.log(`Collected ${collected.size} items`));
|
|
570
|
+
*/
|
|
571
|
+
createReactionCollector(options = {}) {
|
|
572
|
+
return new ReactionCollector(this, options);
|
|
573
|
+
}
|
|
574
|
+
|
|
575
|
+
/**
|
|
576
|
+
* An object containing the same properties as CollectorOptions, but a few more:
|
|
577
|
+
* @typedef {ReactionCollectorOptions} AwaitReactionsOptions
|
|
578
|
+
* @property {string[]} [errors] Stop/end reasons that cause the promise to reject
|
|
579
|
+
*/
|
|
580
|
+
|
|
581
|
+
/**
|
|
582
|
+
* Similar to createReactionCollector but in promise form.
|
|
583
|
+
* Resolves with a collection of reactions that pass the specified filter.
|
|
584
|
+
* @param {AwaitReactionsOptions} [options={}] Optional options to pass to the internal collector
|
|
585
|
+
* @returns {Promise<Collection<string | Snowflake, MessageReaction>>}
|
|
586
|
+
* @example
|
|
587
|
+
* // Create a reaction collector
|
|
588
|
+
* const filter = (reaction, user) => reaction.emoji.name === '👌' && user.id === 'someId'
|
|
589
|
+
* message.awaitReactions({ filter, time: 15_000 })
|
|
590
|
+
* .then(collected => console.log(`Collected ${collected.size} reactions`))
|
|
591
|
+
* .catch(console.error);
|
|
592
|
+
*/
|
|
593
|
+
awaitReactions(options = {}) {
|
|
594
|
+
return new Promise((resolve, reject) => {
|
|
595
|
+
const collector = this.createReactionCollector(options);
|
|
596
|
+
collector.once('end', (reactions, reason) => {
|
|
597
|
+
if (options.errors?.includes(reason)) reject(reactions);
|
|
598
|
+
else resolve(reactions);
|
|
599
|
+
});
|
|
600
|
+
});
|
|
601
|
+
}
|
|
602
|
+
|
|
603
|
+
/**
|
|
604
|
+
* Whether the message is editable by the client user
|
|
605
|
+
* @type {boolean}
|
|
606
|
+
* @readonly
|
|
607
|
+
*/
|
|
608
|
+
get editable() {
|
|
609
|
+
const precheck = Boolean(
|
|
610
|
+
this.author.id === this.client.user.id &&
|
|
611
|
+
!deletedMessages.has(this) &&
|
|
612
|
+
(!this.guild || this.channel?.viewable) &&
|
|
613
|
+
this.reference?.type !== 'FORWARD',
|
|
614
|
+
);
|
|
615
|
+
|
|
616
|
+
// Regardless of permissions thread messages cannot be edited if
|
|
617
|
+
// the thread is archived or the thread is locked and the bot does not have permission to manage threads.
|
|
618
|
+
if (this.channel?.isThread()) {
|
|
619
|
+
if (this.channel.archived) return false;
|
|
620
|
+
if (this.channel.locked) {
|
|
621
|
+
const permissions = this.channel.permissionsFor(this.client.user);
|
|
622
|
+
if (!permissions?.has(Permissions.FLAGS.MANAGE_THREADS, true)) return false;
|
|
623
|
+
}
|
|
624
|
+
}
|
|
625
|
+
|
|
626
|
+
return precheck;
|
|
627
|
+
}
|
|
628
|
+
|
|
629
|
+
/**
|
|
630
|
+
* Whether the message is deletable by the client user
|
|
631
|
+
* @type {boolean}
|
|
632
|
+
* @readonly
|
|
633
|
+
*/
|
|
634
|
+
get deletable() {
|
|
635
|
+
if (deletedMessages.has(this)) {
|
|
636
|
+
return false;
|
|
637
|
+
}
|
|
638
|
+
if (!this.guild) {
|
|
639
|
+
return this.author.id === this.client.user.id;
|
|
640
|
+
}
|
|
641
|
+
// DMChannel does not have viewable property, so check viewable after proved that message is on a guild.
|
|
642
|
+
if (!this.channel?.viewable) {
|
|
643
|
+
return false;
|
|
644
|
+
}
|
|
645
|
+
|
|
646
|
+
const permissions = this.channel?.permissionsFor(this.client.user);
|
|
647
|
+
if (!permissions) return false;
|
|
648
|
+
// This flag allows deleting even if timed out
|
|
649
|
+
if (permissions.has(Permissions.FLAGS.ADMINISTRATOR, false)) return true;
|
|
650
|
+
|
|
651
|
+
return Boolean(
|
|
652
|
+
this.author.id === this.client.user.id ||
|
|
653
|
+
(permissions.has(Permissions.FLAGS.MANAGE_MESSAGES, false) &&
|
|
654
|
+
this.guild.members.me.communicationDisabledUntilTimestamp < Date.now()),
|
|
655
|
+
);
|
|
656
|
+
}
|
|
657
|
+
|
|
658
|
+
/**
|
|
659
|
+
* Whether the message is bulk deletable by the client user
|
|
660
|
+
* @type {boolean}
|
|
661
|
+
* @readonly
|
|
662
|
+
* @example
|
|
663
|
+
* // Filter for bulk deletable messages
|
|
664
|
+
* channel.bulkDelete(messages.filter(message => message.bulkDeletable));
|
|
665
|
+
*/
|
|
666
|
+
get bulkDeletable() {
|
|
667
|
+
return false;
|
|
668
|
+
}
|
|
669
|
+
|
|
670
|
+
/**
|
|
671
|
+
* Whether the message is pinnable by the client user
|
|
672
|
+
* @type {boolean}
|
|
673
|
+
* @readonly
|
|
674
|
+
*/
|
|
675
|
+
get pinnable() {
|
|
676
|
+
const { channel } = this;
|
|
677
|
+
return Boolean(
|
|
678
|
+
!this.system &&
|
|
679
|
+
!deletedMessages.has(this) &&
|
|
680
|
+
(!this.guild ||
|
|
681
|
+
(channel?.viewable &&
|
|
682
|
+
channel?.permissionsFor(this.client.user)?.has(Permissions.FLAGS.MANAGE_MESSAGES, false))),
|
|
683
|
+
);
|
|
684
|
+
}
|
|
685
|
+
|
|
686
|
+
/**
|
|
687
|
+
* Fetches the Message this crosspost/reply/pin-add references, if available to the client
|
|
688
|
+
* @returns {Promise<Message>}
|
|
689
|
+
*/
|
|
690
|
+
async fetchReference() {
|
|
691
|
+
if (!this.reference) throw new Error('MESSAGE_REFERENCE_MISSING');
|
|
692
|
+
const { channelId, messageId } = this.reference;
|
|
693
|
+
if (!messageId) throw new Error('MESSAGE_REFERENCE_MISSING');
|
|
694
|
+
const channel = this.client.channels.resolve(channelId);
|
|
695
|
+
if (!channel) throw new Error('GUILD_CHANNEL_RESOLVE');
|
|
696
|
+
const message = await channel.messages.fetch(messageId);
|
|
697
|
+
return message;
|
|
698
|
+
}
|
|
699
|
+
|
|
700
|
+
/**
|
|
701
|
+
* Whether the message is crosspostable by the client user
|
|
702
|
+
* @type {boolean}
|
|
703
|
+
* @readonly
|
|
704
|
+
*/
|
|
705
|
+
get crosspostable() {
|
|
706
|
+
const bitfield =
|
|
707
|
+
Permissions.FLAGS.SEND_MESSAGES |
|
|
708
|
+
(this.author.id === this.client.user.id ? Permissions.defaultBit : Permissions.FLAGS.MANAGE_MESSAGES);
|
|
709
|
+
const { channel } = this;
|
|
710
|
+
return Boolean(
|
|
711
|
+
channel?.type === 'GUILD_NEWS' &&
|
|
712
|
+
!this.flags.has(MessageFlags.FLAGS.CROSSPOSTED) &&
|
|
713
|
+
this.reference?.type !== 'FORWARD' &&
|
|
714
|
+
this.type === 'DEFAULT' &&
|
|
715
|
+
!this.poll &&
|
|
716
|
+
channel.viewable &&
|
|
717
|
+
channel.permissionsFor(this.client.user)?.has(bitfield, false) &&
|
|
718
|
+
!deletedMessages.has(this),
|
|
719
|
+
);
|
|
720
|
+
}
|
|
721
|
+
|
|
722
|
+
/**
|
|
723
|
+
* Options that can be passed into {@link Message#edit}.
|
|
724
|
+
* @typedef {Object} MessageEditOptions
|
|
725
|
+
* @property {?string} [content] Content to be edited
|
|
726
|
+
* @property {MessageEmbed[]|APIEmbed[]} [embeds] Embeds to be added/edited
|
|
727
|
+
* @property {MessageMentionOptions} [allowedMentions] Which mentions should be parsed from the message content
|
|
728
|
+
* @property {MessageFlags} [flags] Which flags to set for the message. Only `SUPPRESS_EMBEDS` can be edited.
|
|
729
|
+
* @property {MessageAttachment[]} [attachments] An array of attachments to keep,
|
|
730
|
+
* all attachments will be kept if omitted
|
|
731
|
+
* @property {FileOptions[]|BufferResolvable[]|MessageAttachment[]} [files] Files to add to the message
|
|
732
|
+
* @property {MessageActionRow[]|MessageActionRowOptions[]} [components]
|
|
733
|
+
* Action rows containing interactive components for the message (buttons, select menus)
|
|
734
|
+
*/
|
|
735
|
+
|
|
736
|
+
/**
|
|
737
|
+
* Edits the content of the message.
|
|
738
|
+
* @param {string|MessagePayload|MessageEditOptions} options The options to provide
|
|
739
|
+
* @returns {Promise<Message>}
|
|
740
|
+
* @example
|
|
741
|
+
* // Update the content of a message
|
|
742
|
+
* message.edit('This is my new content!')
|
|
743
|
+
* .then(msg => console.log(`Updated the content of a message to ${msg.content}`))
|
|
744
|
+
* .catch(console.error);
|
|
745
|
+
*/
|
|
746
|
+
async edit(options) {
|
|
747
|
+
if (!this.channel) throw new Error('CHANNEL_NOT_CACHED');
|
|
748
|
+
return this.channel.messages.edit(this, options);
|
|
749
|
+
}
|
|
750
|
+
|
|
751
|
+
/**
|
|
752
|
+
* Publishes a message in an announcement channel to all channels following it.
|
|
753
|
+
* @returns {Promise<Message>}
|
|
754
|
+
* @example
|
|
755
|
+
* // Crosspost a message
|
|
756
|
+
* if (message.channel.type === 'GUILD_NEWS') {
|
|
757
|
+
* message.crosspost()
|
|
758
|
+
* .then(() => console.log('Crossposted message'))
|
|
759
|
+
* .catch(console.error);
|
|
760
|
+
* }
|
|
761
|
+
*/
|
|
762
|
+
async crosspost() {
|
|
763
|
+
if (!this.channel) throw new Error('CHANNEL_NOT_CACHED');
|
|
764
|
+
return this.channel.messages.crosspost(this.id);
|
|
765
|
+
}
|
|
766
|
+
|
|
767
|
+
/**
|
|
768
|
+
* Pins this message to the channel's pinned messages.
|
|
769
|
+
* @param {string} [reason] Reason for pinning
|
|
770
|
+
* @returns {Promise<Message>}
|
|
771
|
+
* @example
|
|
772
|
+
* // Pin a message
|
|
773
|
+
* message.pin()
|
|
774
|
+
* .then(console.log)
|
|
775
|
+
* .catch(console.error)
|
|
776
|
+
*/
|
|
777
|
+
async pin(reason) {
|
|
778
|
+
if (!this.channel) throw new Error('CHANNEL_NOT_CACHED');
|
|
779
|
+
await this.channel.messages.pin(this.id, reason);
|
|
780
|
+
return this;
|
|
781
|
+
}
|
|
782
|
+
|
|
783
|
+
/**
|
|
784
|
+
* Unpins this message from the channel's pinned messages.
|
|
785
|
+
* @param {string} [reason] Reason for unpinning
|
|
786
|
+
* @returns {Promise<Message>}
|
|
787
|
+
* @example
|
|
788
|
+
* // Unpin a message
|
|
789
|
+
* message.unpin()
|
|
790
|
+
* .then(console.log)
|
|
791
|
+
* .catch(console.error)
|
|
792
|
+
*/
|
|
793
|
+
async unpin(reason) {
|
|
794
|
+
if (!this.channel) throw new Error('CHANNEL_NOT_CACHED');
|
|
795
|
+
await this.channel.messages.unpin(this.id, reason);
|
|
796
|
+
return this;
|
|
797
|
+
}
|
|
798
|
+
|
|
799
|
+
/**
|
|
800
|
+
* Adds a reaction to the message.
|
|
801
|
+
* @param {EmojiIdentifierResolvable} emoji The emoji to react with
|
|
802
|
+
* @param {boolean} [burst=false] Super Reactions
|
|
803
|
+
* @returns {Promise<MessageReaction>}
|
|
804
|
+
* @example
|
|
805
|
+
* // React to a message with a unicode emoji
|
|
806
|
+
* message.react('🤔')
|
|
807
|
+
* .then(console.log)
|
|
808
|
+
* .catch(console.error);
|
|
809
|
+
* @example
|
|
810
|
+
* // React to a message with a custom emoji
|
|
811
|
+
* message.react(message.guild.emojis.cache.get('123456789012345678'))
|
|
812
|
+
* .then(console.log)
|
|
813
|
+
* .catch(console.error);
|
|
814
|
+
*/
|
|
815
|
+
async react(emoji, burst = false) {
|
|
816
|
+
if (!this.channel) throw new Error('CHANNEL_NOT_CACHED');
|
|
817
|
+
await this.channel.messages.react(this.id, emoji, burst);
|
|
818
|
+
|
|
819
|
+
return this.client.actions.MessageReactionAdd.handle(
|
|
820
|
+
{
|
|
821
|
+
[this.client.actions.injectedUser]: this.client.user,
|
|
822
|
+
[this.client.actions.injectedChannel]: this.channel,
|
|
823
|
+
[this.client.actions.injectedMessage]: this,
|
|
824
|
+
emoji: Util.resolvePartialEmoji(emoji),
|
|
825
|
+
me_burst: burst,
|
|
826
|
+
},
|
|
827
|
+
true,
|
|
828
|
+
).reaction;
|
|
829
|
+
}
|
|
830
|
+
|
|
831
|
+
/**
|
|
832
|
+
* Deletes the message.
|
|
833
|
+
* @returns {Promise<Message>}
|
|
834
|
+
* @example
|
|
835
|
+
* // Delete a message
|
|
836
|
+
* message.delete()
|
|
837
|
+
* .then(msg => console.log(`Deleted message from ${msg.author.username}`))
|
|
838
|
+
* .catch(console.error);
|
|
839
|
+
*/
|
|
840
|
+
async delete() {
|
|
841
|
+
if (!this.channel) throw new Error('CHANNEL_NOT_CACHED');
|
|
842
|
+
await this.channel.messages.delete(this.id);
|
|
843
|
+
return this;
|
|
844
|
+
}
|
|
845
|
+
|
|
846
|
+
/**
|
|
847
|
+
* Options provided when sending a message as an inline reply.
|
|
848
|
+
* @typedef {BaseMessageOptions} ReplyMessageOptions
|
|
849
|
+
* @property {boolean} [failIfNotExists=true] Whether to error if the referenced message
|
|
850
|
+
* does not exist (creates a standard message in this case when false)
|
|
851
|
+
* @property {StickerResolvable[]} [stickers=[]] Stickers to send in the message
|
|
852
|
+
*/
|
|
853
|
+
|
|
854
|
+
/**
|
|
855
|
+
* Send an inline reply to this message.
|
|
856
|
+
* @param {string|MessagePayload|ReplyMessageOptions} options The options to provide
|
|
857
|
+
* @returns {Promise<Message>}
|
|
858
|
+
* @example
|
|
859
|
+
* // Reply to a message
|
|
860
|
+
* message.reply('This is a reply!')
|
|
861
|
+
* .then(() => console.log(`Replied to message "${message.content}"`))
|
|
862
|
+
* .catch(console.error);
|
|
863
|
+
*/
|
|
864
|
+
async reply(options) {
|
|
865
|
+
if (!this.channel) throw new Error('CHANNEL_NOT_CACHED');
|
|
866
|
+
let data;
|
|
867
|
+
|
|
868
|
+
if (options instanceof MessagePayload) {
|
|
869
|
+
data = options;
|
|
870
|
+
} else {
|
|
871
|
+
data = MessagePayload.create(this, options, {
|
|
872
|
+
reply: {
|
|
873
|
+
messageReference: this,
|
|
874
|
+
failIfNotExists: options?.failIfNotExists ?? this.client.options.failIfNotExists,
|
|
875
|
+
},
|
|
876
|
+
});
|
|
877
|
+
}
|
|
878
|
+
return this.channel.send(data);
|
|
879
|
+
}
|
|
880
|
+
|
|
881
|
+
/**
|
|
882
|
+
* Forwards this message
|
|
883
|
+
* @param {TextBasedChannelResolvable} channel The channel to forward this message to.
|
|
884
|
+
* @returns {Promise<Message>}
|
|
885
|
+
*/
|
|
886
|
+
forward(channel) {
|
|
887
|
+
const resolvedChannel = this.client.channels.resolve(channel);
|
|
888
|
+
if (!resolvedChannel) throw new Error('INVALID_TYPE', 'channel', 'TextBasedChannelResolvable');
|
|
889
|
+
return resolvedChannel.send({
|
|
890
|
+
forward: {
|
|
891
|
+
message: this.id,
|
|
892
|
+
channel: this.channelId,
|
|
893
|
+
guild: this.guildId,
|
|
894
|
+
},
|
|
895
|
+
});
|
|
896
|
+
}
|
|
897
|
+
|
|
898
|
+
/**
|
|
899
|
+
* A number that is allowed to be the duration (in minutes) of inactivity after which a thread is automatically
|
|
900
|
+
* archived. This can be:
|
|
901
|
+
* * `60` (1 hour)
|
|
902
|
+
* * `1440` (1 day)
|
|
903
|
+
* * `4320` (3 days)
|
|
904
|
+
* * `10080` (7 days)
|
|
905
|
+
* * `'MAX'` (7 days)
|
|
906
|
+
* <warn>This option is deprecated and will be removed in the next major version.</warn>
|
|
907
|
+
* @typedef {number|string} ThreadAutoArchiveDuration
|
|
908
|
+
*/
|
|
909
|
+
|
|
910
|
+
/**
|
|
911
|
+
* Options for starting a thread on a message.
|
|
912
|
+
* @typedef {Object} StartThreadOptions
|
|
913
|
+
* @property {string} name The name of the new thread
|
|
914
|
+
* @property {ThreadAutoArchiveDuration} [autoArchiveDuration=this.channel.defaultAutoArchiveDuration] The amount of
|
|
915
|
+
* time (in minutes) after which the thread should automatically archive in case of no recent activity
|
|
916
|
+
* @property {string} [reason] Reason for creating the thread
|
|
917
|
+
* @property {number} [rateLimitPerUser] The rate limit per user (slowmode) for the thread in seconds
|
|
918
|
+
*/
|
|
919
|
+
|
|
920
|
+
/**
|
|
921
|
+
* Create a new public thread from this message
|
|
922
|
+
* @see GuildTextThreadManager#create
|
|
923
|
+
* @param {StartThreadOptions} [options] Options for starting a thread on this message
|
|
924
|
+
* @returns {Promise<ThreadChannel>}
|
|
925
|
+
*/
|
|
926
|
+
async startThread(options = {}) {
|
|
927
|
+
if (!this.channel) throw new Error('CHANNEL_NOT_CACHED');
|
|
928
|
+
if (!['GUILD_TEXT', 'GUILD_NEWS'].includes(this.channel.type)) {
|
|
929
|
+
throw new Error('MESSAGE_THREAD_PARENT');
|
|
930
|
+
}
|
|
931
|
+
if (this.hasThread) throw new Error('MESSAGE_EXISTING_THREAD');
|
|
932
|
+
return this.channel.threads.create({ ...options, startMessage: this });
|
|
933
|
+
}
|
|
934
|
+
|
|
935
|
+
/**
|
|
936
|
+
* Submits a poll vote for the current user. Returns a 204 empty response on success.
|
|
937
|
+
* @param {...number[]} ids ID of the answer
|
|
938
|
+
* @returns {Promise<void>}
|
|
939
|
+
* @example
|
|
940
|
+
* // Vote multi choices
|
|
941
|
+
* message.vote(1,2);
|
|
942
|
+
* // Remove vote
|
|
943
|
+
* message.vote();
|
|
944
|
+
*/
|
|
945
|
+
vote(...ids) {
|
|
946
|
+
return this.client.api
|
|
947
|
+
.channels(this.channel.id)
|
|
948
|
+
.polls(this.id)
|
|
949
|
+
.answers['@me'].put({
|
|
950
|
+
data: {
|
|
951
|
+
answer_ids: ids.flat(1).map(value => value.toString()),
|
|
952
|
+
},
|
|
953
|
+
});
|
|
954
|
+
}
|
|
955
|
+
|
|
956
|
+
/**
|
|
957
|
+
* Fetch this message.
|
|
958
|
+
* @param {boolean} [force=true] Whether to skip the cache check and request the API
|
|
959
|
+
* @returns {Promise<Message>}
|
|
960
|
+
*/
|
|
961
|
+
async fetch(force = true) {
|
|
962
|
+
if (!this.channel) throw new Error('CHANNEL_NOT_CACHED');
|
|
963
|
+
return this.channel.messages.fetch(this.id, { force });
|
|
964
|
+
}
|
|
965
|
+
|
|
966
|
+
/**
|
|
967
|
+
* Fetches the webhook used to create this message.
|
|
968
|
+
* @returns {Promise<?Webhook>}
|
|
969
|
+
*/
|
|
970
|
+
async fetchWebhook() {
|
|
971
|
+
if (!this.webhookId) throw new Error('WEBHOOK_MESSAGE');
|
|
972
|
+
if (this.webhookId === this.applicationId) throw new Error('WEBHOOK_APPLICATION');
|
|
973
|
+
return this.client.fetchWebhook(this.webhookId);
|
|
974
|
+
}
|
|
975
|
+
|
|
976
|
+
/**
|
|
977
|
+
* Suppresses or unsuppresses embeds on a message.
|
|
978
|
+
* @param {boolean} [suppress=true] If the embeds should be suppressed or not
|
|
979
|
+
* @returns {Promise<Message>}
|
|
980
|
+
*/
|
|
981
|
+
suppressEmbeds(suppress = true) {
|
|
982
|
+
const flags = new MessageFlags(this.flags.bitfield);
|
|
983
|
+
|
|
984
|
+
if (suppress) {
|
|
985
|
+
flags.add(MessageFlags.FLAGS.SUPPRESS_EMBEDS);
|
|
986
|
+
} else {
|
|
987
|
+
flags.remove(MessageFlags.FLAGS.SUPPRESS_EMBEDS);
|
|
988
|
+
}
|
|
989
|
+
|
|
990
|
+
return this.edit({ flags });
|
|
991
|
+
}
|
|
992
|
+
|
|
993
|
+
/**
|
|
994
|
+
* Removes the attachments from this message.
|
|
995
|
+
* @returns {Promise<Message>}
|
|
996
|
+
*/
|
|
997
|
+
removeAttachments() {
|
|
998
|
+
return this.edit({ attachments: [] });
|
|
999
|
+
}
|
|
1000
|
+
|
|
1001
|
+
/**
|
|
1002
|
+
* Resolves a component by a custom id.
|
|
1003
|
+
* @param {string} customId The custom id to resolve against
|
|
1004
|
+
* @returns {?MessageActionRowComponent}
|
|
1005
|
+
*/
|
|
1006
|
+
resolveComponent(customId) {
|
|
1007
|
+
return (
|
|
1008
|
+
this.components
|
|
1009
|
+
.flatMap(BaseMessageComponent.extractInteractiveComponents)
|
|
1010
|
+
.find(component => (component.customId ?? component.custom_id) === customId) ?? null
|
|
1011
|
+
);
|
|
1012
|
+
}
|
|
1013
|
+
|
|
1014
|
+
/**
|
|
1015
|
+
* Used mainly internally. Whether two messages are identical in properties. If you want to compare messages
|
|
1016
|
+
* without checking all the properties, use `message.id === message2.id`, which is much more efficient. This
|
|
1017
|
+
* method allows you to see if there are differences in content, embeds, attachments, nonce and tts properties.
|
|
1018
|
+
* @param {Message} message The message to compare it to
|
|
1019
|
+
* @param {APIMessage} rawData Raw data passed through the WebSocket about this message
|
|
1020
|
+
* @returns {boolean}
|
|
1021
|
+
*/
|
|
1022
|
+
equals(message, rawData) {
|
|
1023
|
+
if (!message) return false;
|
|
1024
|
+
const embedUpdate = !message.author && !message.attachments;
|
|
1025
|
+
if (embedUpdate) return this.id === message.id && this.embeds.length === message.embeds.length;
|
|
1026
|
+
|
|
1027
|
+
let equal =
|
|
1028
|
+
this.id === message.id &&
|
|
1029
|
+
this.author.id === message.author.id &&
|
|
1030
|
+
this.content === message.content &&
|
|
1031
|
+
this.tts === message.tts &&
|
|
1032
|
+
this.nonce === message.nonce &&
|
|
1033
|
+
this.embeds.length === message.embeds.length &&
|
|
1034
|
+
this.attachments.size === message.attachments.size &&
|
|
1035
|
+
this.attachments.every(attachment => message.attachments.has(attachment.id)) &&
|
|
1036
|
+
this.embeds.every((embed, index) => embed.equals(message.embeds[index]));
|
|
1037
|
+
|
|
1038
|
+
if (equal && rawData) {
|
|
1039
|
+
equal =
|
|
1040
|
+
this.mentions.everyone === message.mentions.everyone &&
|
|
1041
|
+
this.createdTimestamp === new Date(rawData.timestamp).getTime() &&
|
|
1042
|
+
this.editedTimestamp === new Date(rawData.edited_timestamp).getTime();
|
|
1043
|
+
}
|
|
1044
|
+
|
|
1045
|
+
return equal;
|
|
1046
|
+
}
|
|
1047
|
+
|
|
1048
|
+
/**
|
|
1049
|
+
* Whether this message is from a guild.
|
|
1050
|
+
* @returns {boolean}
|
|
1051
|
+
*/
|
|
1052
|
+
inGuild() {
|
|
1053
|
+
return Boolean(this.guildId);
|
|
1054
|
+
}
|
|
1055
|
+
|
|
1056
|
+
/**
|
|
1057
|
+
* When concatenated with a string, this automatically concatenates the message's content instead of the object.
|
|
1058
|
+
* @returns {string}
|
|
1059
|
+
* @example
|
|
1060
|
+
* // Logs: Message: This is a message!
|
|
1061
|
+
* console.log(`Message: ${message}`);
|
|
1062
|
+
*/
|
|
1063
|
+
toString() {
|
|
1064
|
+
return this.content;
|
|
1065
|
+
}
|
|
1066
|
+
|
|
1067
|
+
toJSON() {
|
|
1068
|
+
return super.toJSON({
|
|
1069
|
+
channel: 'channelId',
|
|
1070
|
+
author: 'authorId',
|
|
1071
|
+
groupActivityApplication: 'groupActivityApplicationId',
|
|
1072
|
+
guild: 'guildId',
|
|
1073
|
+
cleanContent: true,
|
|
1074
|
+
member: false,
|
|
1075
|
+
reactions: false,
|
|
1076
|
+
});
|
|
1077
|
+
}
|
|
1078
|
+
|
|
1079
|
+
// TypeScript
|
|
1080
|
+
/**
|
|
1081
|
+
* Check data
|
|
1082
|
+
* @type {boolean}
|
|
1083
|
+
* @readonly
|
|
1084
|
+
*/
|
|
1085
|
+
get isMessage() {
|
|
1086
|
+
return true;
|
|
1087
|
+
}
|
|
1088
|
+
|
|
1089
|
+
/**
|
|
1090
|
+
* Click a specified button in the message based on the button's CustomID
|
|
1091
|
+
* @param {string} buttonid customId of the button to click
|
|
1092
|
+
*
|
|
1093
|
+
* To be compatible with Components V2, the following methods have been removed in this version:
|
|
1094
|
+
* - Clicking by coordinates (using ActionRow)
|
|
1095
|
+
* - Clicking the first button in the message
|
|
1096
|
+
*
|
|
1097
|
+
* Currently, only clicking by CustomID is supported.
|
|
1098
|
+
* @returns {Promise<Message|Modal>}
|
|
1099
|
+
*/
|
|
1100
|
+
clickButton(buttonid) {
|
|
1101
|
+
const button = this.resolveComponent(buttonid);
|
|
1102
|
+
if (!button || button.type !== 'BUTTON') throw new TypeError('BUTTON_NOT_FOUND');
|
|
1103
|
+
if (button.disabled) throw new TypeError('BUTTON_CANNOT_CLICK');
|
|
1104
|
+
const nonce = SnowflakeUtil.generate();
|
|
1105
|
+
const data = {
|
|
1106
|
+
type: InteractionTypes.MESSAGE_COMPONENT,
|
|
1107
|
+
nonce,
|
|
1108
|
+
guild_id: this.guildId,
|
|
1109
|
+
channel_id: this.channelId,
|
|
1110
|
+
message_id: this.id,
|
|
1111
|
+
application_id: this.applicationId ?? this.author.id,
|
|
1112
|
+
session_id: this.client.sessionId,
|
|
1113
|
+
message_flags: this.flags.bitfield,
|
|
1114
|
+
data: {
|
|
1115
|
+
component_type: MessageComponentTypes.BUTTON,
|
|
1116
|
+
custom_id: button.customId,
|
|
1117
|
+
},
|
|
1118
|
+
};
|
|
1119
|
+
this.client.api.interactions.post({
|
|
1120
|
+
data,
|
|
1121
|
+
});
|
|
1122
|
+
return Util.createPromiseInteraction(this.client, nonce, 5_000, true, this);
|
|
1123
|
+
}
|
|
1124
|
+
|
|
1125
|
+
/**
|
|
1126
|
+
* Select specific menu
|
|
1127
|
+
* @param {number|string} menu Target
|
|
1128
|
+
* @param {Array<UserResolvable | RoleResolvable | ChannelResolvable | string>} values Any value
|
|
1129
|
+
* @returns {Promise<Message|Modal>}
|
|
1130
|
+
*/
|
|
1131
|
+
selectMenu(menu, values = []) {
|
|
1132
|
+
let selectMenu = menu;
|
|
1133
|
+
if (/[0-4]/.test(menu)) {
|
|
1134
|
+
selectMenu = this.components[menu]?.components[0];
|
|
1135
|
+
} else if (typeof menu == 'string') {
|
|
1136
|
+
selectMenu = this.components
|
|
1137
|
+
.flatMap(row => row.components)
|
|
1138
|
+
.find(
|
|
1139
|
+
b =>
|
|
1140
|
+
['STRING_SELECT', 'USER_SELECT', 'ROLE_SELECT', 'MENTIONABLE_SELECT', 'CHANNEL_SELECT'].includes(b.type) &&
|
|
1141
|
+
b.customId == menu &&
|
|
1142
|
+
!b.disabled,
|
|
1143
|
+
);
|
|
1144
|
+
}
|
|
1145
|
+
if (values.length < selectMenu.minValues) {
|
|
1146
|
+
throw new RangeError(`[SELECT_MENU_MIN_VALUES] The minimum number of values is ${selectMenu.minValues}`);
|
|
1147
|
+
}
|
|
1148
|
+
if (values.length > selectMenu?.maxValues) {
|
|
1149
|
+
throw new RangeError(`[SELECT_MENU_MAX_VALUES] The maximum number of values is ${selectMenu.maxValues}`);
|
|
1150
|
+
}
|
|
1151
|
+
values = values.map(value => {
|
|
1152
|
+
switch (selectMenu.type) {
|
|
1153
|
+
case 'STRING_SELECT': {
|
|
1154
|
+
return selectMenu.options.find(obj => obj.value === value || obj.label === value).value;
|
|
1155
|
+
}
|
|
1156
|
+
case 'USER_SELECT': {
|
|
1157
|
+
return this.client.users.resolveId(value);
|
|
1158
|
+
}
|
|
1159
|
+
case 'ROLE_SELECT': {
|
|
1160
|
+
return this.guild.roles.resolveId(value);
|
|
1161
|
+
}
|
|
1162
|
+
case 'MENTIONABLE_SELECT': {
|
|
1163
|
+
return this.client.users.resolveId(value) || this.guild.roles.resolveId(value);
|
|
1164
|
+
}
|
|
1165
|
+
case 'CHANNEL_SELECT': {
|
|
1166
|
+
return this.client.channels.resolveId(value);
|
|
1167
|
+
}
|
|
1168
|
+
default: {
|
|
1169
|
+
return value;
|
|
1170
|
+
}
|
|
1171
|
+
}
|
|
1172
|
+
});
|
|
1173
|
+
const nonce = SnowflakeUtil.generate();
|
|
1174
|
+
const data = {
|
|
1175
|
+
type: InteractionTypes.MESSAGE_COMPONENT,
|
|
1176
|
+
guild_id: this.guildId,
|
|
1177
|
+
channel_id: this.channelId,
|
|
1178
|
+
message_id: this.id,
|
|
1179
|
+
application_id: this.applicationId ?? this.author.id,
|
|
1180
|
+
session_id: this.client.sessionId,
|
|
1181
|
+
message_flags: this.flags.bitfield,
|
|
1182
|
+
data: {
|
|
1183
|
+
component_type: MessageComponentTypes[selectMenu.type],
|
|
1184
|
+
custom_id: selectMenu.customId,
|
|
1185
|
+
type: MessageComponentTypes[selectMenu.type],
|
|
1186
|
+
values,
|
|
1187
|
+
},
|
|
1188
|
+
nonce,
|
|
1189
|
+
};
|
|
1190
|
+
this.client.api.interactions.post({
|
|
1191
|
+
data,
|
|
1192
|
+
});
|
|
1193
|
+
return Util.createPromiseInteraction(this.client, nonce, 5_000, true, this);
|
|
1194
|
+
}
|
|
1195
|
+
|
|
1196
|
+
/**
|
|
1197
|
+
* Marks the message as unread.
|
|
1198
|
+
* @returns {Promise<void>}
|
|
1199
|
+
*/
|
|
1200
|
+
markUnread() {
|
|
1201
|
+
return this.client.api.channels[this.channelId].messages[this.id].ack.post({
|
|
1202
|
+
data: {
|
|
1203
|
+
manual: true,
|
|
1204
|
+
mention_count: 1,
|
|
1205
|
+
},
|
|
1206
|
+
});
|
|
1207
|
+
}
|
|
1208
|
+
|
|
1209
|
+
/**
|
|
1210
|
+
* Marks the message as read.
|
|
1211
|
+
* @returns {Promise<void>}
|
|
1212
|
+
*/
|
|
1213
|
+
markRead() {
|
|
1214
|
+
return this.client.api.channels[this.channelId].messages[this.id].ack.post({
|
|
1215
|
+
data: {
|
|
1216
|
+
token: null,
|
|
1217
|
+
},
|
|
1218
|
+
});
|
|
1219
|
+
}
|
|
1220
|
+
|
|
1221
|
+
/**
|
|
1222
|
+
* Report Message
|
|
1223
|
+
* @param {Arrray<number>} breadcrumbs Options for reporting
|
|
1224
|
+
* @param {Object} [elements={}] Metadata
|
|
1225
|
+
* @returns {Promise<{ report_id: Snowflake }>}
|
|
1226
|
+
* @example
|
|
1227
|
+
* // GET https://discord.com/api/v9/reporting/menu/message?variant=4
|
|
1228
|
+
* // Report Category
|
|
1229
|
+
* // - <hidden>MESSAGE_WELCOME (3)</hidden>
|
|
1230
|
+
* // - Something else (28)
|
|
1231
|
+
* // - Hacks, cheats, phishing or malicious links (72)
|
|
1232
|
+
* message.report([3, 28, 72]).then(console.log);
|
|
1233
|
+
* // { "report_id": "1199663489988440124" }
|
|
1234
|
+
*/
|
|
1235
|
+
report(breadcrumbs, elements = {}) {
|
|
1236
|
+
return this.client.api.reporting.message.post({
|
|
1237
|
+
data: {
|
|
1238
|
+
version: '1.0',
|
|
1239
|
+
variant: '4',
|
|
1240
|
+
language: 'en',
|
|
1241
|
+
breadcrumbs,
|
|
1242
|
+
elements,
|
|
1243
|
+
channel_id: this.channelId,
|
|
1244
|
+
message_id: this.id,
|
|
1245
|
+
name: 'message',
|
|
1246
|
+
},
|
|
1247
|
+
});
|
|
1248
|
+
}
|
|
1249
|
+
}
|
|
1250
|
+
|
|
1251
|
+
exports.Message = Message;
|
|
1252
|
+
exports.deletedMessages = deletedMessages;
|