feat(VoiceState): add methods for fetching voice state (#10442)

* feat(VoiceState): add methods for fetching voice state

* fix: links to new methods

* chore: remove unused import

* chore: use member id

* chore: requested changes

* chore: '@me' as fetch param

* chore: add ediUserVoiceState return type

* refactor: redirect function calls to VoiceAPI

---------

Co-authored-by: Almeida <almeidx@pm.me>
Co-authored-by: kodiakhq[bot] <49736102+kodiakhq[bot]@users.noreply.github.com>
This commit is contained in:
Naiyar
2024-08-20 15:32:17 +05:30
committed by GitHub
parent 9b707f2b83
commit 9907ff915e
6 changed files with 125 additions and 12 deletions

View File

@@ -67,7 +67,6 @@ import {
type RESTPatchAPIGuildTemplateJSONBody, type RESTPatchAPIGuildTemplateJSONBody,
type RESTPatchAPIGuildTemplateResult, type RESTPatchAPIGuildTemplateResult,
type RESTPatchAPIGuildVoiceStateCurrentMemberJSONBody, type RESTPatchAPIGuildVoiceStateCurrentMemberJSONBody,
type RESTPatchAPIGuildVoiceStateCurrentMemberResult,
type RESTPatchAPIGuildVoiceStateUserJSONBody, type RESTPatchAPIGuildVoiceStateUserJSONBody,
type RESTPatchAPIGuildWelcomeScreenJSONBody, type RESTPatchAPIGuildWelcomeScreenJSONBody,
type RESTPatchAPIGuildWelcomeScreenResult, type RESTPatchAPIGuildWelcomeScreenResult,
@@ -102,6 +101,7 @@ import {
type RESTPutAPIGuildTemplateSyncResult, type RESTPutAPIGuildTemplateSyncResult,
type Snowflake, type Snowflake,
} from 'discord-api-types/v10'; } from 'discord-api-types/v10';
import { VoiceAPI } from './voice';
export class GuildsAPI { export class GuildsAPI {
public constructor(private readonly rest: REST) {} public constructor(private readonly rest: REST) {}
@@ -687,11 +687,12 @@ export class GuildsAPI {
/** /**
* Edits a user's voice state in a guild * Edits a user's voice state in a guild
* *
* @see {@link https://discord.com/developers/docs/resources/guild#modify-user-voice-state} * @see {@link https://discord.com/developers/docs/resources/voice#modify-user-voice-state}
* @param guildId - The id of the guild to edit the current user's voice state in * @param guildId - The id of the guild to edit the current user's voice state in
* @param userId - The id of the user to edit the voice state for * @param userId - The id of the user to edit the voice state for
* @param body - The data for editing the voice state * @param body - The data for editing the voice state
* @param options - The options for editing the voice state * @param options - The options for editing the voice state
* @deprecated Use {@link VoiceAPI.editUserVoiceState} instead
*/ */
public async editUserVoiceState( public async editUserVoiceState(
guildId: Snowflake, guildId: Snowflake,
@@ -699,7 +700,7 @@ export class GuildsAPI {
body: RESTPatchAPIGuildVoiceStateUserJSONBody, body: RESTPatchAPIGuildVoiceStateUserJSONBody,
{ reason, signal }: Pick<RequestData, 'reason' | 'signal'> = {}, { reason, signal }: Pick<RequestData, 'reason' | 'signal'> = {},
) { ) {
await this.rest.patch(Routes.guildVoiceState(guildId, userId), { reason, body, signal }); return new VoiceAPI(this.rest).editUserVoiceState(guildId, userId, body, { reason, signal });
} }
/** /**
@@ -1298,14 +1299,18 @@ export class GuildsAPI {
/** /**
* Sets the voice state for the current user * Sets the voice state for the current user
* *
* @see {@link https://discord.com/developers/docs/resources/guild#modify-current-user-voice-state} * @see {@link https://discord.com/developers/docs/resources/voice#modify-current-user-voice-state}
* @param guildId - The id of the guild * @param guildId - The id of the guild
* @param body - The options for setting the voice state * @param body - The data for setting the voice state
* @param options - The options for setting the voice state
* @deprecated Use {@link VoiceAPI.editVoiceState} instead
*/ */
public async setVoiceState(guildId: Snowflake, body: RESTPatchAPIGuildVoiceStateCurrentMemberJSONBody = {}) { public async setVoiceState(
return this.rest.patch(Routes.guildVoiceState(guildId, '@me'), { guildId: Snowflake,
body, body: RESTPatchAPIGuildVoiceStateCurrentMemberJSONBody = {},
}) as Promise<RESTPatchAPIGuildVoiceStateCurrentMemberResult>; { signal }: Pick<RequestData, 'signal'> = {},
) {
return new VoiceAPI(this.rest).editVoiceState(guildId, body, { signal });
} }
/** /**

View File

@@ -1,7 +1,17 @@
/* eslint-disable jsdoc/check-param-names */ /* eslint-disable jsdoc/check-param-names */
import type { RequestData, REST } from '@discordjs/rest'; import type { RequestData, REST } from '@discordjs/rest';
import { Routes, type RESTGetAPIVoiceRegionsResult } from 'discord-api-types/v10'; import {
Routes,
type Snowflake,
type RESTGetAPIVoiceRegionsResult,
type RESTGetAPIGuildVoiceStateUserResult,
type RESTGetAPIGuildVoiceStateCurrentMemberResult,
type RESTPatchAPIGuildVoiceStateUserJSONBody,
type RESTPatchAPIGuildVoiceStateCurrentMemberResult,
type RESTPatchAPIGuildVoiceStateCurrentMemberJSONBody,
type RESTPatchAPIGuildVoiceStateUserResult,
} from 'discord-api-types/v10';
export class VoiceAPI { export class VoiceAPI {
public constructor(private readonly rest: REST) {} public constructor(private readonly rest: REST) {}
@@ -15,4 +25,69 @@ export class VoiceAPI {
public async getVoiceRegions({ signal }: Pick<RequestData, 'signal'> = {}) { public async getVoiceRegions({ signal }: Pick<RequestData, 'signal'> = {}) {
return this.rest.get(Routes.voiceRegions(), { signal }) as Promise<RESTGetAPIVoiceRegionsResult>; return this.rest.get(Routes.voiceRegions(), { signal }) as Promise<RESTGetAPIVoiceRegionsResult>;
} }
/**
* Fetches voice state of a user by their id
*
* @see {@link https://discord.com/developers/docs/resources/voice#get-user-voice-state}
* @param options - The options for fetching user voice state
*/
public async getUserVoiceState(guildId: Snowflake, userId: Snowflake, { signal }: Pick<RequestData, 'signal'> = {}) {
return this.rest.get(Routes.guildVoiceState(guildId, userId), {
signal,
}) as Promise<RESTGetAPIGuildVoiceStateUserResult>;
}
/**
* Fetches the current user's voice state
*
* @see {@link https://discord.com/developers/docs/resources/voice#get-current-user-voice-state}
* @param options - The options for fetching user voice state
*/
public async getVoiceState(guildId: Snowflake, { signal }: Pick<RequestData, 'signal'> = {}) {
return this.rest.get(Routes.guildVoiceState(guildId, '@me'), {
signal,
}) as Promise<RESTGetAPIGuildVoiceStateCurrentMemberResult>;
}
/**
* Edits a user's voice state in a guild
*
* @see {@link https://discord.com/developers/docs/resources/voice#modify-user-voice-state}
* @param guildId - The id of the guild to edit the current user's voice state in
* @param userId - The id of the user to edit the voice state for
* @param body - The data for editing the voice state
* @param options - The options for editing the voice state
*/
public async editUserVoiceState(
guildId: Snowflake,
userId: Snowflake,
body: RESTPatchAPIGuildVoiceStateUserJSONBody,
{ reason, signal }: Pick<RequestData, 'reason' | 'signal'> = {},
) {
return this.rest.patch(Routes.guildVoiceState(guildId, userId), {
reason,
body,
signal,
}) as Promise<RESTPatchAPIGuildVoiceStateUserResult>;
}
/**
* Edits the voice state for the current user
*
* @see {@link https://discord.com/developers/docs/resources/voice#modify-current-user-voice-state}
* @param guildId - The id of the guild
* @param body - The data for editing the voice state
* @param options - The options for editing the voice state
*/
public async editVoiceState(
guildId: Snowflake,
body: RESTPatchAPIGuildVoiceStateCurrentMemberJSONBody = {},
{ signal }: Pick<RequestData, 'signal'> = {},
) {
return this.rest.patch(Routes.guildVoiceState(guildId, '@me'), {
body,
signal,
}) as Promise<RESTPatchAPIGuildVoiceStateCurrentMemberResult>;
}
} }

View File

@@ -1,5 +1,6 @@
'use strict'; 'use strict';
const { Routes } = require('discord-api-types/v10');
const CachedManager = require('./CachedManager'); const CachedManager = require('./CachedManager');
const VoiceState = require('../structures/VoiceState'); const VoiceState = require('../structures/VoiceState');
@@ -32,6 +33,27 @@ class VoiceStateManager extends CachedManager {
if (cache) this.cache.set(data.user_id, entry); if (cache) this.cache.set(data.user_id, entry);
return entry; return entry;
} }
/**
* Obtains a user's voice state from discord or from the cache if it's already available.
* @param {GuildMemberResolvable|'@me'} member The member whose voice state is to be fetched
* @param {BaseFetchOptions} [options] Additional options for this fetch
* @returns {Promise<VoiceState>}
* @example
* // Fetch a member's voice state
* guild.voiceStates.fetch("66564597481480192")
* .then(console.log)
* .catch(console.error);
*/
async fetch(member, { cache = true, force = false } = {}) {
const id = member === '@me' ? member : this.guild.members.resolveId(member);
if (!force) {
const existing = this.cache.get(id === '@me' ? this.client.user.id : id);
if (existing) return existing;
}
const data = await this.client.rest.get(Routes.guildVoiceState(this.guild.id, id));
return this._add(data, cache);
}
} }
module.exports = VoiceStateManager; module.exports = VoiceStateManager;

View File

@@ -250,6 +250,15 @@ class VoiceState extends Base {
return this; return this;
} }
/**
* Fetches this voice state.
* @param {boolean} [force=true] Whether to skip the cache check and request the API
* @returns {Promise<VoiceState>}
*/
fetch(force = true) {
return this.guild.voiceStates.fetch(this.id, { force });
}
/** /**
* Toggles the request to speak in the channel. * Toggles the request to speak in the channel.
* Only applicable for stage channels and for the client's own voice state. * Only applicable for stage channels and for the client's own voice state.

View File

@@ -3625,6 +3625,7 @@ export class VoiceState extends Base {
public setRequestToSpeak(request?: boolean): Promise<this>; public setRequestToSpeak(request?: boolean): Promise<this>;
public setSuppressed(suppressed?: boolean): Promise<this>; public setSuppressed(suppressed?: boolean): Promise<this>;
public edit(options: VoiceStateEditOptions): Promise<this>; public edit(options: VoiceStateEditOptions): Promise<this>;
public fetch(force?: boolean): Promise<VoiceState>;
} }
// tslint:disable-next-line no-empty-interface // tslint:disable-next-line no-empty-interface
@@ -4627,6 +4628,7 @@ export class UserManager extends CachedManager<Snowflake, User, UserResolvable>
export class VoiceStateManager extends CachedManager<Snowflake, VoiceState, typeof VoiceState> { export class VoiceStateManager extends CachedManager<Snowflake, VoiceState, typeof VoiceState> {
private constructor(guild: Guild, iterable?: Iterable<RawVoiceStateData>); private constructor(guild: Guild, iterable?: Iterable<RawVoiceStateData>);
public guild: Guild; public guild: Guild;
public fetch(member: GuildMemberResolvable | '@me', options?: BaseFetchOptions): Promise<VoiceState>;
} }
//#endregion //#endregion

View File

@@ -47,6 +47,7 @@ import {
APIUnavailableGuild, APIUnavailableGuild,
APIUser, APIUser,
APIVoiceRegion, APIVoiceRegion,
APIVoiceState,
APIWebhook, APIWebhook,
GatewayActivity, GatewayActivity,
GatewayActivityAssets, GatewayActivityAssets,
@@ -62,7 +63,6 @@ import {
GatewayPresenceUpdate, GatewayPresenceUpdate,
GatewayReadyDispatchData, GatewayReadyDispatchData,
GatewayTypingStartDispatchData, GatewayTypingStartDispatchData,
GatewayVoiceState,
RESTAPIPartialCurrentUserGuild, RESTAPIPartialCurrentUserGuild,
RESTGetAPIWebhookWithTokenResult, RESTGetAPIWebhookWithTokenResult,
RESTPatchAPIChannelMessageJSONBody, RESTPatchAPIChannelMessageJSONBody,
@@ -195,7 +195,7 @@ export type RawUserData =
export type RawVoiceRegionData = APIVoiceRegion; export type RawVoiceRegionData = APIVoiceRegion;
export type RawVoiceStateData = GatewayVoiceState | Omit<GatewayVoiceState, 'guild_id'>; export type RawVoiceStateData = APIVoiceState | Omit<APIVoiceState, 'guild_id'>;
export type RawWebhookData = export type RawWebhookData =
| APIWebhook | APIWebhook