djs-selfbot-v13 3.1.8 → 3.2.2

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.
Files changed (43) hide show
  1. package/LICENSE +1 -1
  2. package/README.md +18 -8
  3. package/package.json +85 -71
  4. package/src/client/BaseClient.js +1 -1
  5. package/src/client/Client.js +81 -10
  6. package/src/client/actions/GuildMemberRemove.js +0 -1
  7. package/src/client/actions/GuildMemberUpdate.js +0 -1
  8. package/src/client/websocket/WebSocketShard.js +3 -3
  9. package/src/client/websocket/handlers/GUILD_CREATE.js +13 -14
  10. package/src/client/websocket/handlers/GUILD_MEMBER_ADD.js +0 -1
  11. package/src/client/websocket/handlers/MESSAGE_POLL_VOTE_ADD.js +22 -0
  12. package/src/client/websocket/handlers/MESSAGE_POLL_VOTE_REMOVE.js +12 -0
  13. package/src/client/websocket/handlers/READY.js +61 -21
  14. package/src/client/websocket/handlers/RELATIONSHIP_REMOVE.js +1 -1
  15. package/src/client/websocket/handlers/RELATIONSHIP_UPDATE.js +1 -1
  16. package/src/client/websocket/handlers/index.js +2 -0
  17. package/src/errors/Messages.js +1 -0
  18. package/src/index.js +4 -3
  19. package/src/managers/ChannelManager.js +1 -1
  20. package/src/managers/ClientUserSettingManager.js +2 -2
  21. package/src/managers/GuildBanManager.js +46 -0
  22. package/src/managers/GuildChannelManager.js +0 -16
  23. package/src/managers/GuildForumThreadManager.js +3 -3
  24. package/src/managers/GuildManager.js +1 -1
  25. package/src/managers/GuildMemberManager.js +11 -7
  26. package/src/managers/RelationshipManager.js +3 -3
  27. package/src/rest/APIRequest.js +13 -7
  28. package/src/structures/ClientPresence.js +9 -12
  29. package/src/structures/ClientUser.js +0 -2
  30. package/src/structures/Message.js +67 -76
  31. package/src/structures/MessagePayload.js +2 -0
  32. package/src/structures/MessagePoll.js +238 -0
  33. package/src/structures/Modal.js +11 -24
  34. package/src/structures/Presence.js +786 -128
  35. package/src/structures/User.js +35 -1
  36. package/src/structures/interfaces/TextBasedChannel.js +21 -23
  37. package/src/util/Constants.js +17 -4
  38. package/src/util/Options.js +1 -7
  39. package/src/util/Permissions.js +10 -0
  40. package/src/util/Util.js +88 -2
  41. package/typings/enums.d.ts +7 -1
  42. package/typings/index.d.ts +112 -70
  43. package/src/structures/RichPresence.js +0 -702
@@ -0,0 +1,238 @@
1
+ 'use strict';
2
+
3
+ const { Collection } = require('@discordjs/collection');
4
+ const { MessagePollLayoutTypes } = require('../util/Constants');
5
+ const Util = require('../util/Util');
6
+
7
+ /**
8
+ * Represents the poll object has a lot of levels and nested structures. It was also designed to support future extensibility, so some fields may appear to be more complex than necessary.
9
+ */
10
+ class MessagePoll {
11
+ /**
12
+ * @param {Object} data Message poll to clone or raw data
13
+ */
14
+ constructor(data = {}) {
15
+ this._patch(data);
16
+ }
17
+
18
+ _patch(data = {}) {
19
+ if (data?.constructor?.name == 'MessagePoll') data = data.toJSON();
20
+ /**
21
+ * The poll media object is a common object that backs both the question and answers. For now, `question` only supports `text`, while `answers` can have an optional `emoji`.
22
+ * @see {@link https://docs.discord.sex/resources/message#poll-media-structure}
23
+ * @typedef {Object} MessagePollMedia
24
+ * @property {?string} text The text of the field (max 300 characters for question, 55 characters for answer)
25
+ * @property {?RawEmoji} emoji The emoji of the field
26
+ */
27
+
28
+ if ('question' in data) {
29
+ /**
30
+ * The question of the poll
31
+ * @type {?MessagePollMedia}
32
+ */
33
+ this.question = this._resolvePollMedia(data.question);
34
+ } else {
35
+ this.question ??= null;
36
+ }
37
+
38
+ if (data.answers?.length) {
39
+ /**
40
+ * The answers available in the poll
41
+ * @type {Collection<number, MessagePollMedia>}
42
+ */
43
+ this.answers = new Collection();
44
+
45
+ data.answers.forEach((obj, index) => {
46
+ this.answers.set(obj?.answer_id || index + 1, this._resolvePollMedia(obj.poll_media));
47
+ });
48
+ } else {
49
+ this.answers ??= new Collection();
50
+ }
51
+
52
+ if ('layout_type' in data) {
53
+ /**
54
+ * The layout type of the poll
55
+ * @type {?MessagePollLayoutTypes}
56
+ */
57
+ this.layoutType = MessagePollLayoutTypes[data.layout_type];
58
+ } else {
59
+ this.layoutType ??= MessagePollLayoutTypes[1]; // Default type
60
+ }
61
+
62
+ if ('allow_multiselect' in data) {
63
+ /**
64
+ * Whether a user can select multiple answers
65
+ * @type {boolean}
66
+ */
67
+ this.allowMultiSelect = !!data.allow_multiselect;
68
+ } else {
69
+ this.allowMultiSelect ??= false;
70
+ }
71
+
72
+ if ('expiry' in data) {
73
+ /**
74
+ * When the poll ends
75
+ * @type {?Date}
76
+ */
77
+ this.expiry = new Date(data.expiry);
78
+ } else {
79
+ this.expiry ??= null;
80
+ }
81
+
82
+ if ('duration' in data) {
83
+ /**
84
+ * Number of hours the poll should be open for (max 32 days, default 1)
85
+ * @type {?Number}
86
+ */
87
+ this.duration = data.duration;
88
+ } else {
89
+ this.duration ??= null;
90
+ }
91
+
92
+ if ('results' in data) {
93
+ /**
94
+ * Poll Results Structure
95
+ * @see {@link https://docs.discord.sex/resources/message#poll-results-structure}
96
+ * @typedef {Object} MessagePollResult
97
+ * @property {boolean} isFinalized Whether the votes have been precisely counted
98
+ * @property {Collection<number, MessagePollResultAnswerCount>} answerCounts The counts for each answer
99
+ */
100
+ /**
101
+ * Poll Answer Count Structure
102
+ * @see {@link https://docs.discord.sex/resources/message#poll-answer-count-structure}
103
+ * @typedef {Object} MessagePollResultAnswerCount
104
+ * @property {MessagePollMedia} answer answer
105
+ * @property {number} count The number of votes for this answer
106
+ * @property {boolean} selfVoted Whether the current user voted for this answer
107
+ */
108
+ /**
109
+ * In a nutshell, this contains the number of votes for each answer.
110
+ * The `results` field may be not present in certain responses where, as an implementation detail,
111
+ * Discord does not fetch the poll results in the backend.
112
+ * This should be treated as "unknown results", as opposed to "no results".
113
+ * You can keep using the results if you have previously received them through other means.
114
+ * Due to the intricacies of counting at scale, while a poll is in progress the results may not
115
+ * be perfectly accurate. They usually are accurate, and shouldn't deviate significantly — it's
116
+ * just difficult to make guarantees. To compensate for this, after a poll is finished there is
117
+ * a background job which performs a final, accurate tally of votes. This tally concludes once
118
+ * `is_finalized` is `true`. Polls that have ended will also always contain results.
119
+ * If `answer_counts` does not contain an entry for a particular answer, then there are no votes
120
+ * for that answer.
121
+ * @type {?MessagePollResult}
122
+ */
123
+ this.results = {
124
+ isFinalized: data.results.is_finalized,
125
+ answerCounts: new Collection(),
126
+ };
127
+ data.results.answer_counts.forEach(obj => {
128
+ this.results.answerCounts.set(obj.id, {
129
+ count: obj.count,
130
+ selfVoted: obj.me_voted,
131
+ answer: this.answers.get(obj.id),
132
+ });
133
+ });
134
+ } else {
135
+ this.results ??= null;
136
+ }
137
+ }
138
+
139
+ _resolvePollMedia(obj) {
140
+ return {
141
+ text: obj.text,
142
+ emoji: Util.resolvePartialEmoji(obj.emoji),
143
+ };
144
+ }
145
+
146
+ /**
147
+ * Convert to JSON
148
+ * @returns {Object}
149
+ */
150
+ toJSON() {
151
+ const data = {
152
+ question: {
153
+ text: this.question.text,
154
+ emoji: this.question.emoji,
155
+ },
156
+ expiry: this.expiry?.toISOString(),
157
+ allow_multiselect: this.allowMultiSelect,
158
+ layout_type: typeof this.layoutType == 'number' ? this.layoutType : MessagePollLayoutTypes[this.layoutType],
159
+ answers: Array.from(this.answers.entries()).map(([id, data]) => ({
160
+ answer_id: id,
161
+ poll_media: {
162
+ text: data.text,
163
+ emoji: data.emoji,
164
+ },
165
+ })),
166
+ duration: this.duration ?? 1,
167
+ };
168
+ if (this.results) {
169
+ data.results = {
170
+ is_finalized: this.results.isFinalized,
171
+ answer_counts: Array.from(this.results.answerCounts.entries()).map(([id, data]) => ({
172
+ id: id,
173
+ count: data.count,
174
+ me_voted: data.selfVoted,
175
+ })),
176
+ };
177
+ }
178
+ return data;
179
+ }
180
+
181
+ /**
182
+ * Set question
183
+ * @param {string} text question
184
+ * @returns {MessagePoll}
185
+ */
186
+ setQuestion(text) {
187
+ this.question = {
188
+ text,
189
+ };
190
+ return this;
191
+ }
192
+
193
+ /**
194
+ * Set answers
195
+ * @param {MessagePollMedia[]} answers answers
196
+ * @returns {MessagePoll}
197
+ */
198
+ setAnswers(answers) {
199
+ this.answers.clear();
200
+ answers.forEach((obj, index) => {
201
+ this.answers.set(index + 1, this._resolvePollMedia(obj));
202
+ });
203
+ return this;
204
+ }
205
+
206
+ /**
207
+ * Add answer
208
+ * @param {MessagePollMedia} answer answer
209
+ * @returns {MessagePoll}
210
+ */
211
+ addAnswer(answer) {
212
+ this.answers.set(this.answers.size + 1, answer);
213
+ return this;
214
+ }
215
+
216
+ /**
217
+ * Set allow multi select
218
+ * @param {boolean} state state
219
+ * @returns {MessagePoll}
220
+ */
221
+ setAllowMultiSelect(state) {
222
+ this.allowMultiSelect = state;
223
+ return this;
224
+ }
225
+
226
+ /**
227
+ * Set duration
228
+ * @param {number} duration duration (hours)
229
+ * @returns {MessagePoll}
230
+ */
231
+ setDuration(duration) {
232
+ // [1, 4, 8, 24, 72, 168, 336];
233
+ this.duration = duration;
234
+ return this;
235
+ }
236
+ }
237
+
238
+ module.exports = MessagePoll;
@@ -1,9 +1,9 @@
1
1
  'use strict';
2
2
 
3
- const { setTimeout } = require('node:timers');
4
3
  const BaseMessageComponent = require('./BaseMessageComponent');
5
- const { InteractionTypes, Events } = require('../util/Constants');
4
+ const { InteractionTypes } = require('../util/Constants');
6
5
  const SnowflakeUtil = require('../util/SnowflakeUtil');
6
+ const Util = require('../util/Util');
7
7
 
8
8
  /**
9
9
  * Represents a modal (form) to be shown in response to an interaction
@@ -56,6 +56,12 @@ class Modal {
56
56
  */
57
57
  this.channelId = data.channel_id;
58
58
 
59
+ /**
60
+ * Whether this interaction has already been replied to
61
+ * @type {boolean}
62
+ */
63
+ this.replied = false;
64
+
59
65
  Object.defineProperty(this, 'client', {
60
66
  value: client,
61
67
  writable: false,
@@ -110,7 +116,7 @@ class Modal {
110
116
  * })
111
117
  */
112
118
  reply() {
113
- if (!this.applicationId || !this.client || !this.channelId) throw new Error('Modal cannot reply');
119
+ if (!this.applicationId || !this.client || !this.channelId || this.replied) throw new Error('Modal cannot reply');
114
120
  // Get Object
115
121
  const dataFinal = this.toJSON();
116
122
  dataFinal.components = dataFinal.components
@@ -137,27 +143,8 @@ class Modal {
137
143
  this.client.api.interactions.post({
138
144
  data: postData,
139
145
  });
140
- return new Promise((resolve, reject) => {
141
- const timeoutMs = 5_000;
142
- // Waiting for MsgCreate / ModalCreate
143
- const handler = data => {
144
- if (data.nonce !== nonce) return;
145
- clearTimeout(timeout);
146
- this.client.removeListener(Events.MESSAGE_CREATE, handler);
147
- this.client.removeListener(Events.INTERACTION_MODAL_CREATE, handler);
148
- this.client.decrementMaxListeners();
149
- resolve(data);
150
- };
151
- const timeout = setTimeout(() => {
152
- this.client.removeListener(Events.MESSAGE_CREATE, handler);
153
- this.client.removeListener(Events.INTERACTION_MODAL_CREATE, handler);
154
- this.client.decrementMaxListeners();
155
- reject(new Error('INTERACTION_FAILED'));
156
- }, timeoutMs).unref();
157
- this.client.incrementMaxListeners();
158
- this.client.on(Events.MESSAGE_CREATE, handler);
159
- this.client.on(Events.INTERACTION_MODAL_CREATE, handler);
160
- });
146
+ this.replied = true;
147
+ return Util.createPromiseInteraction(this.client, nonce, 5_000, true, this);
161
148
  }
162
149
 
163
150
  // TypeScript