2022-04-16 10:44:43 +00:00
|
|
|
'use strict';
|
|
|
|
|
2022-08-14 10:16:51 +00:00
|
|
|
const { Collection } = require('@discordjs/collection');
|
2022-04-16 10:44:43 +00:00
|
|
|
const { Channel } = require('./Channel');
|
|
|
|
const TextBasedChannel = require('./interfaces/TextBasedChannel');
|
|
|
|
const MessageManager = require('../managers/MessageManager');
|
2024-01-11 12:23:00 +00:00
|
|
|
const { Opcodes, Status } = require('../util/Constants');
|
2022-04-16 10:44:43 +00:00
|
|
|
|
|
|
|
/**
|
|
|
|
* Represents a direct message channel between two users.
|
|
|
|
* @extends {Channel}
|
|
|
|
* @implements {TextBasedChannel}
|
|
|
|
*/
|
|
|
|
class DMChannel extends Channel {
|
|
|
|
constructor(client, data) {
|
|
|
|
super(client, data);
|
|
|
|
|
|
|
|
// Override the channel type so partials have a known type
|
|
|
|
this.type = 'DM';
|
|
|
|
|
|
|
|
/**
|
|
|
|
* A manager of the messages belonging to this channel
|
|
|
|
* @type {MessageManager}
|
|
|
|
*/
|
|
|
|
this.messages = new MessageManager(this);
|
|
|
|
}
|
|
|
|
|
|
|
|
_patch(data) {
|
|
|
|
super._patch(data);
|
|
|
|
|
|
|
|
if (data.recipients) {
|
|
|
|
/**
|
|
|
|
* The recipient on the other end of the DM
|
|
|
|
* @type {User}
|
|
|
|
*/
|
|
|
|
this.recipient = this.client.users._add(data.recipients[0]);
|
|
|
|
}
|
|
|
|
|
|
|
|
if ('last_message_id' in data) {
|
|
|
|
/**
|
|
|
|
* The channel's last message id, if one was sent
|
|
|
|
* @type {?Snowflake}
|
|
|
|
*/
|
|
|
|
this.lastMessageId = data.last_message_id;
|
|
|
|
}
|
|
|
|
|
|
|
|
if ('last_pin_timestamp' in data) {
|
|
|
|
/**
|
|
|
|
* The timestamp when the last pinned message was pinned, if there was one
|
|
|
|
* @type {?number}
|
|
|
|
*/
|
|
|
|
this.lastPinTimestamp = new Date(data.last_pin_timestamp).getTime();
|
|
|
|
} else {
|
|
|
|
this.lastPinTimestamp ??= null;
|
|
|
|
}
|
2022-10-27 12:29:45 +00:00
|
|
|
|
|
|
|
if ('is_message_request' in data) {
|
|
|
|
/**
|
|
|
|
* Whether the channel is a message request
|
2024-01-11 12:23:00 +00:00
|
|
|
* @type {?boolean}
|
2022-10-27 12:29:45 +00:00
|
|
|
*/
|
|
|
|
this.messageRequest = data.is_message_request;
|
|
|
|
}
|
|
|
|
|
|
|
|
if ('is_message_request_timestamp' in data) {
|
|
|
|
/**
|
|
|
|
* The timestamp when the message request was created
|
|
|
|
* @type {?number}
|
|
|
|
*/
|
|
|
|
this.messageRequestTimestamp = new Date(data.is_message_request_timestamp).getTime();
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
/**
|
|
|
|
* Accept this DMChannel.
|
|
|
|
* @returns {Promise<DMChannel>}
|
|
|
|
*/
|
|
|
|
async acceptMessageRequest() {
|
|
|
|
if (!this.messageRequest) {
|
|
|
|
throw new Error('NOT_MESSAGE_REQUEST', 'This channel is not a message request');
|
|
|
|
}
|
|
|
|
const c = await this.client.api.channels[this.id].recipients['@me'].put({
|
|
|
|
data: {
|
|
|
|
consent_status: 2,
|
|
|
|
},
|
|
|
|
});
|
|
|
|
this.messageRequest = false;
|
|
|
|
return this.client.channels._add(c);
|
|
|
|
}
|
|
|
|
|
|
|
|
/**
|
|
|
|
* Cancel this DMChannel.
|
|
|
|
* @returns {Promise<DMChannel>}
|
|
|
|
*/
|
|
|
|
async cancelMessageRequest() {
|
|
|
|
if (!this.messageRequest) {
|
|
|
|
throw new Error('NOT_MESSAGE_REQUEST', 'This channel is not a message request');
|
|
|
|
}
|
|
|
|
await this.client.api.channels[this.id].recipients['@me'].delete();
|
|
|
|
return this;
|
2022-04-16 10:44:43 +00:00
|
|
|
}
|
|
|
|
|
|
|
|
/**
|
|
|
|
* Whether this DMChannel is a partial
|
|
|
|
* @type {boolean}
|
|
|
|
* @readonly
|
|
|
|
*/
|
|
|
|
get partial() {
|
|
|
|
return typeof this.lastMessageId === 'undefined';
|
|
|
|
}
|
|
|
|
|
|
|
|
/**
|
|
|
|
* Fetch this DMChannel.
|
|
|
|
* @param {boolean} [force=true] Whether to skip the cache check and request the API
|
|
|
|
* @returns {Promise<DMChannel>}
|
|
|
|
*/
|
|
|
|
fetch(force = true) {
|
|
|
|
return this.recipient.createDM(force);
|
|
|
|
}
|
|
|
|
|
|
|
|
/**
|
|
|
|
* When concatenated with a string, this automatically returns the recipient's mention instead of the
|
|
|
|
* DMChannel object.
|
|
|
|
* @returns {string}
|
|
|
|
* @example
|
|
|
|
* // Logs: Hello from <@123456789012345678>!
|
|
|
|
* console.log(`Hello from ${channel}!`);
|
|
|
|
*/
|
|
|
|
toString() {
|
|
|
|
return this.recipient.toString();
|
|
|
|
}
|
|
|
|
|
2022-04-25 12:28:45 +00:00
|
|
|
/**
|
2024-01-11 12:23:00 +00:00
|
|
|
* Sync VoiceState of this DMChannel.
|
|
|
|
* @returns {undefined}
|
2022-04-25 12:28:45 +00:00
|
|
|
*/
|
2024-01-11 12:23:00 +00:00
|
|
|
sync() {
|
|
|
|
this.client.ws.broadcast({
|
|
|
|
op: Opcodes.DM_UPDATE,
|
|
|
|
d: {
|
|
|
|
channel_id: this.id,
|
2022-10-25 05:37:23 +00:00
|
|
|
},
|
2022-04-25 12:28:45 +00:00
|
|
|
});
|
|
|
|
}
|
2023-10-29 06:28:27 +00:00
|
|
|
|
|
|
|
/**
|
|
|
|
* Ring the user's phone / PC (call)
|
2024-01-11 12:23:00 +00:00
|
|
|
* @returns {Promise<void>}
|
2023-10-29 06:28:27 +00:00
|
|
|
*/
|
|
|
|
ring() {
|
|
|
|
return this.client.api.channels(this.id).call.ring.post({
|
|
|
|
data: {
|
|
|
|
recipients: null,
|
|
|
|
},
|
|
|
|
});
|
|
|
|
}
|
|
|
|
|
2022-08-14 10:16:51 +00:00
|
|
|
/**
|
|
|
|
* The user in this voice-based channel
|
|
|
|
* @type {Collection<Snowflake, User>}
|
|
|
|
* @readonly
|
|
|
|
*/
|
|
|
|
get voiceUsers() {
|
|
|
|
const coll = new Collection();
|
|
|
|
for (const state of this.client.voiceStates.cache.values()) {
|
|
|
|
if (state.channelId === this.id && state.user) {
|
|
|
|
coll.set(state.id, state.user);
|
|
|
|
}
|
|
|
|
}
|
|
|
|
return coll;
|
|
|
|
}
|
2024-01-11 12:23:00 +00:00
|
|
|
|
2022-08-14 10:16:51 +00:00
|
|
|
/**
|
|
|
|
* Get current shard
|
|
|
|
* @type {WebSocketShard}
|
|
|
|
* @readonly
|
|
|
|
*/
|
2022-05-21 13:58:33 +00:00
|
|
|
get shard() {
|
|
|
|
return this.client.ws.shards.first();
|
|
|
|
}
|
2024-01-11 12:23:00 +00:00
|
|
|
|
2022-08-14 10:16:51 +00:00
|
|
|
/**
|
|
|
|
* The voice state adapter for this client that can be used with @discordjs/voice to play audio in DM / Group DM channels.
|
|
|
|
* @type {?Function}
|
|
|
|
* @readonly
|
|
|
|
*/
|
2022-05-21 13:58:33 +00:00
|
|
|
get voiceAdapterCreator() {
|
|
|
|
return methods => {
|
|
|
|
this.client.voice.adapters.set(this.id, methods);
|
|
|
|
return {
|
|
|
|
sendPayload: data => {
|
|
|
|
if (this.shard.status !== Status.READY) return false;
|
|
|
|
this.shard.send(data);
|
|
|
|
return true;
|
|
|
|
},
|
|
|
|
destroy: () => {
|
|
|
|
this.client.voice.adapters.delete(this.id);
|
|
|
|
},
|
|
|
|
};
|
|
|
|
};
|
|
|
|
}
|
2024-01-11 12:23:00 +00:00
|
|
|
|
|
|
|
// These are here only for documentation purposes - they are implemented by TextBasedChannel
|
|
|
|
/* eslint-disable no-empty-function */
|
|
|
|
get lastMessage() {}
|
|
|
|
get lastPinAt() {}
|
|
|
|
send() {}
|
|
|
|
sendTyping() {}
|
|
|
|
createMessageCollector() {}
|
|
|
|
awaitMessages() {}
|
|
|
|
// Doesn't work on DM channels; setRateLimitPerUser() {}
|
|
|
|
// Doesn't work on DM channels; setNSFW() {}
|
2022-04-16 10:44:43 +00:00
|
|
|
}
|
|
|
|
|
2024-01-11 12:23:00 +00:00
|
|
|
TextBasedChannel.applyToClass(DMChannel, true, ['fetchWebhooks', 'createWebhook', 'setRateLimitPerUser', 'setNSFW']);
|
2022-04-16 10:44:43 +00:00
|
|
|
|
|
|
|
module.exports = DMChannel;
|