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.
- package/LICENSE +1 -1
- package/README.md +18 -8
- package/package.json +85 -71
- package/src/client/BaseClient.js +1 -1
- package/src/client/Client.js +81 -10
- package/src/client/actions/GuildMemberRemove.js +0 -1
- package/src/client/actions/GuildMemberUpdate.js +0 -1
- package/src/client/websocket/WebSocketShard.js +3 -3
- package/src/client/websocket/handlers/GUILD_CREATE.js +13 -14
- package/src/client/websocket/handlers/GUILD_MEMBER_ADD.js +0 -1
- package/src/client/websocket/handlers/MESSAGE_POLL_VOTE_ADD.js +22 -0
- package/src/client/websocket/handlers/MESSAGE_POLL_VOTE_REMOVE.js +12 -0
- package/src/client/websocket/handlers/READY.js +61 -21
- package/src/client/websocket/handlers/RELATIONSHIP_REMOVE.js +1 -1
- package/src/client/websocket/handlers/RELATIONSHIP_UPDATE.js +1 -1
- package/src/client/websocket/handlers/index.js +2 -0
- package/src/errors/Messages.js +1 -0
- package/src/index.js +4 -3
- package/src/managers/ChannelManager.js +1 -1
- package/src/managers/ClientUserSettingManager.js +2 -2
- package/src/managers/GuildBanManager.js +46 -0
- package/src/managers/GuildChannelManager.js +0 -16
- package/src/managers/GuildForumThreadManager.js +3 -3
- package/src/managers/GuildManager.js +1 -1
- package/src/managers/GuildMemberManager.js +11 -7
- package/src/managers/RelationshipManager.js +3 -3
- package/src/rest/APIRequest.js +13 -7
- package/src/structures/ClientPresence.js +9 -12
- package/src/structures/ClientUser.js +0 -2
- package/src/structures/Message.js +67 -76
- package/src/structures/MessagePayload.js +2 -0
- package/src/structures/MessagePoll.js +238 -0
- package/src/structures/Modal.js +11 -24
- package/src/structures/Presence.js +786 -128
- package/src/structures/User.js +35 -1
- package/src/structures/interfaces/TextBasedChannel.js +21 -23
- package/src/util/Constants.js +17 -4
- package/src/util/Options.js +1 -7
- package/src/util/Permissions.js +10 -0
- package/src/util/Util.js +88 -2
- package/typings/enums.d.ts +7 -1
- package/typings/index.d.ts +112 -70
- 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;
|
package/src/structures/Modal.js
CHANGED
|
@@ -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
|
|
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
|
-
|
|
141
|
-
|
|
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
|