mirror of
https://github.com/discordjs/discord.js.git
synced 2026-03-17 12:03:31 +01:00
feat(Application): application flags (#5147)
Co-authored-by: SpaceEEC <spaceeec@yahoo.com> Co-authored-by: Vlad Frangu <kingdgrizzle@gmail.com>
This commit is contained in:
@@ -10,7 +10,6 @@ const ChannelManager = require('../managers/ChannelManager');
|
|||||||
const GuildManager = require('../managers/GuildManager');
|
const GuildManager = require('../managers/GuildManager');
|
||||||
const UserManager = require('../managers/UserManager');
|
const UserManager = require('../managers/UserManager');
|
||||||
const ShardClientUtil = require('../sharding/ShardClientUtil');
|
const ShardClientUtil = require('../sharding/ShardClientUtil');
|
||||||
const ClientApplication = require('../structures/ClientApplication');
|
|
||||||
const GuildPreview = require('../structures/GuildPreview');
|
const GuildPreview = require('../structures/GuildPreview');
|
||||||
const GuildTemplate = require('../structures/GuildTemplate');
|
const GuildTemplate = require('../structures/GuildTemplate');
|
||||||
const Invite = require('../structures/Invite');
|
const Invite = require('../structures/Invite');
|
||||||
@@ -151,6 +150,12 @@ class Client extends BaseClient {
|
|||||||
*/
|
*/
|
||||||
this.user = null;
|
this.user = null;
|
||||||
|
|
||||||
|
/**
|
||||||
|
* The application of this bot
|
||||||
|
* @type {?ClientApplication}
|
||||||
|
*/
|
||||||
|
this.application = null;
|
||||||
|
|
||||||
/**
|
/**
|
||||||
* Time at which the client was last regarded as being in the `READY` state
|
* Time at which the client was last regarded as being in the `READY` state
|
||||||
* (each time the client disconnects and successfully reconnects, this will be overwritten)
|
* (each time the client disconnects and successfully reconnects, this will be overwritten)
|
||||||
@@ -346,17 +351,6 @@ class Client extends BaseClient {
|
|||||||
return messages;
|
return messages;
|
||||||
}
|
}
|
||||||
|
|
||||||
/**
|
|
||||||
* Obtains the OAuth Application of this bot from Discord.
|
|
||||||
* @returns {Promise<ClientApplication>}
|
|
||||||
*/
|
|
||||||
fetchApplication() {
|
|
||||||
return this.api.oauth2
|
|
||||||
.applications('@me')
|
|
||||||
.get()
|
|
||||||
.then(app => new ClientApplication(this, app));
|
|
||||||
}
|
|
||||||
|
|
||||||
/**
|
/**
|
||||||
* Obtains a guild preview from Discord, available for all guilds the bot is in and all Discoverable guilds.
|
* Obtains a guild preview from Discord, available for all guilds the bot is in and all Discoverable guilds.
|
||||||
* @param {GuildResolvable} guild The guild to fetch the preview for
|
* @param {GuildResolvable} guild The guild to fetch the preview for
|
||||||
@@ -383,24 +377,23 @@ class Client extends BaseClient {
|
|||||||
/**
|
/**
|
||||||
* Generates a link that can be used to invite the bot to a guild.
|
* Generates a link that can be used to invite the bot to a guild.
|
||||||
* @param {InviteGenerationOptions} [options={}] Options for the invite
|
* @param {InviteGenerationOptions} [options={}] Options for the invite
|
||||||
* @returns {Promise<string>}
|
* @returns {string}
|
||||||
* @example
|
* @example
|
||||||
* client.generateInvite({
|
* const link = client.generateInvite({
|
||||||
* permissions: [
|
* permissions: [
|
||||||
* Permissions.FLAGS.SEND_MESSAGES,
|
* Permissions.FLAGS.SEND_MESSAGES,
|
||||||
* Permissions.FLAGS.MANAGE_GUILD,
|
* Permissions.FLAGS.MANAGE_GUILD,
|
||||||
* Permissions.FLAGS.MENTION_EVERYONE,
|
* Permissions.FLAGS.MENTION_EVERYONE,
|
||||||
* ],
|
* ],
|
||||||
* })
|
* });
|
||||||
* .then(link => console.log(`Generated bot invite link: ${link}`))
|
* console.log(`Generated bot invite link: ${link}`);
|
||||||
* .catch(console.error);
|
|
||||||
*/
|
*/
|
||||||
async generateInvite(options = {}) {
|
generateInvite(options = {}) {
|
||||||
if (typeof options !== 'object') throw new TypeError('INVALID_TYPE', 'options', 'object', true);
|
if (typeof options !== 'object') throw new TypeError('INVALID_TYPE', 'options', 'object', true);
|
||||||
|
if (!this.application) throw new Error('CLIENT_NOT_READY', 'generate an invite link');
|
||||||
|
|
||||||
const application = await this.fetchApplication();
|
|
||||||
const query = new URLSearchParams({
|
const query = new URLSearchParams({
|
||||||
client_id: application.id,
|
client_id: this.application.id,
|
||||||
scope: 'bot',
|
scope: 'bot',
|
||||||
});
|
});
|
||||||
|
|
||||||
|
|||||||
@@ -1,5 +1,6 @@
|
|||||||
'use strict';
|
'use strict';
|
||||||
|
|
||||||
|
const ClientApplication = require('../../../structures/ClientApplication');
|
||||||
let ClientUser;
|
let ClientUser;
|
||||||
|
|
||||||
module.exports = (client, { d: data }, shard) => {
|
module.exports = (client, { d: data }, shard) => {
|
||||||
@@ -7,9 +8,8 @@ module.exports = (client, { d: data }, shard) => {
|
|||||||
client.user._patch(data.user);
|
client.user._patch(data.user);
|
||||||
} else {
|
} else {
|
||||||
if (!ClientUser) ClientUser = require('../../../structures/ClientUser');
|
if (!ClientUser) ClientUser = require('../../../structures/ClientUser');
|
||||||
const clientUser = new ClientUser(client, data.user);
|
client.user = new ClientUser(client, data.user);
|
||||||
client.user = clientUser;
|
client.users.cache.set(client.user.id, client.user);
|
||||||
client.users.cache.set(clientUser.id, clientUser);
|
|
||||||
}
|
}
|
||||||
|
|
||||||
for (const guild of data.guilds) {
|
for (const guild of data.guilds) {
|
||||||
@@ -17,5 +17,11 @@ module.exports = (client, { d: data }, shard) => {
|
|||||||
client.guilds.add(guild);
|
client.guilds.add(guild);
|
||||||
}
|
}
|
||||||
|
|
||||||
|
if (client.application) {
|
||||||
|
client.application._patch(data.application);
|
||||||
|
} else {
|
||||||
|
client.application = new ClientApplication(client, data.application);
|
||||||
|
}
|
||||||
|
|
||||||
shard.checkReady();
|
shard.checkReady();
|
||||||
};
|
};
|
||||||
|
|||||||
@@ -6,6 +6,7 @@ const Messages = {
|
|||||||
CLIENT_INVALID_OPTION: (prop, must) => `The ${prop} option must be ${must}`,
|
CLIENT_INVALID_OPTION: (prop, must) => `The ${prop} option must be ${must}`,
|
||||||
CLIENT_INVALID_PROVIDED_SHARDS: 'None of the provided shards were valid.',
|
CLIENT_INVALID_PROVIDED_SHARDS: 'None of the provided shards were valid.',
|
||||||
CLIENT_MISSING_INTENTS: 'Valid intents must be provided for the Client.',
|
CLIENT_MISSING_INTENTS: 'Valid intents must be provided for the Client.',
|
||||||
|
CLIENT_NOT_READY: action => `The client needs to be logged in to ${action}.`,
|
||||||
|
|
||||||
TOKEN_INVALID: 'An invalid token was provided.',
|
TOKEN_INVALID: 'An invalid token was provided.',
|
||||||
TOKEN_MISSING: 'Request to use token, but token was unavailable to the client.',
|
TOKEN_MISSING: 'Request to use token, but token was unavailable to the client.',
|
||||||
|
|||||||
@@ -2,6 +2,7 @@
|
|||||||
|
|
||||||
const Team = require('./Team');
|
const Team = require('./Team');
|
||||||
const Application = require('./interfaces/Application');
|
const Application = require('./interfaces/Application');
|
||||||
|
const ApplicationFlags = require('../util/ApplicationFlags');
|
||||||
|
|
||||||
/**
|
/**
|
||||||
* Represents a Client OAuth2 Application.
|
* Represents a Client OAuth2 Application.
|
||||||
@@ -11,35 +12,64 @@ class ClientApplication extends Application {
|
|||||||
_patch(data) {
|
_patch(data) {
|
||||||
super._patch(data);
|
super._patch(data);
|
||||||
|
|
||||||
|
/**
|
||||||
|
* The flags this application has
|
||||||
|
* @type {ApplicationFlags}
|
||||||
|
*/
|
||||||
|
this.flags = 'flags' in data ? new ApplicationFlags(data.flags) : this.flags;
|
||||||
|
|
||||||
/**
|
/**
|
||||||
* The app's cover image
|
* The app's cover image
|
||||||
* @type {?string}
|
* @type {?string}
|
||||||
*/
|
*/
|
||||||
this.cover = data.cover_image || null;
|
this.cover = data.cover_image ?? this.cover ?? null;
|
||||||
|
|
||||||
/**
|
/**
|
||||||
* The app's RPC origins, if enabled
|
* The app's RPC origins, if enabled
|
||||||
* @type {string[]}
|
* @type {string[]}
|
||||||
*/
|
*/
|
||||||
this.rpcOrigins = data.rpc_origins || [];
|
this.rpcOrigins = data.rpc_origins ?? this.rpcOrigins ?? [];
|
||||||
|
|
||||||
/**
|
/**
|
||||||
* If this app's bot requires a code grant when using the OAuth2 flow
|
* If this app's bot requires a code grant when using the OAuth2 flow
|
||||||
* @type {?boolean}
|
* @type {?boolean}
|
||||||
*/
|
*/
|
||||||
this.botRequireCodeGrant = typeof data.bot_require_code_grant !== 'undefined' ? data.bot_require_code_grant : null;
|
this.botRequireCodeGrant = data.bot_require_code_grant ?? this.botRequireCodeGrant ?? null;
|
||||||
|
|
||||||
/**
|
/**
|
||||||
* If this app's bot is public
|
* If this app's bot is public
|
||||||
* @type {?boolean}
|
* @type {?boolean}
|
||||||
*/
|
*/
|
||||||
this.botPublic = typeof data.bot_public !== 'undefined' ? data.bot_public : null;
|
this.botPublic = data.bot_public ?? this.botPublic ?? null;
|
||||||
|
|
||||||
/**
|
/**
|
||||||
* The owner of this OAuth application
|
* The owner of this OAuth application
|
||||||
* @type {?User|Team}
|
* @type {?User|Team}
|
||||||
*/
|
*/
|
||||||
this.owner = data.team ? new Team(this.client, data.team) : data.owner ? this.client.users.add(data.owner) : null;
|
this.owner = data.team
|
||||||
|
? new Team(this.client, data.team)
|
||||||
|
: data.owner
|
||||||
|
? this.client.users.add(data.owner)
|
||||||
|
: this.owner ?? null;
|
||||||
|
}
|
||||||
|
|
||||||
|
/**
|
||||||
|
* Whether this application is partial
|
||||||
|
* @type {boolean}
|
||||||
|
* @readonly
|
||||||
|
*/
|
||||||
|
get partial() {
|
||||||
|
return !this.name;
|
||||||
|
}
|
||||||
|
|
||||||
|
/**
|
||||||
|
* Obtains this application from Discord.
|
||||||
|
* @returns {Promise<ClientApplication>}
|
||||||
|
*/
|
||||||
|
async fetch() {
|
||||||
|
const app = await this.client.api.oauth2.applications('@me').get();
|
||||||
|
this._patch(app);
|
||||||
|
return this;
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
|
|||||||
@@ -10,15 +10,11 @@ class IntegrationApplication extends Application {
|
|||||||
_patch(data) {
|
_patch(data) {
|
||||||
super._patch(data);
|
super._patch(data);
|
||||||
|
|
||||||
if (typeof data.bot !== 'undefined') {
|
/**
|
||||||
/**
|
* The bot user for this application
|
||||||
* The bot {@link User user} for this application
|
* @type {?User}
|
||||||
* @type {?User}
|
*/
|
||||||
*/
|
this.bot = data.bot ? this.client.users.add(data.bot) : this.bot ?? null;
|
||||||
this.bot = this.client.users.add(data.bot);
|
|
||||||
} else if (!this.bot) {
|
|
||||||
this.bot = null;
|
|
||||||
}
|
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
|
|||||||
@@ -25,21 +25,21 @@ class Application extends Base {
|
|||||||
|
|
||||||
/**
|
/**
|
||||||
* The name of the app
|
* The name of the app
|
||||||
* @type {string}
|
* @type {?string}
|
||||||
*/
|
*/
|
||||||
this.name = data.name;
|
this.name = data.name ?? this.name ?? null;
|
||||||
|
|
||||||
/**
|
/**
|
||||||
* The app's description
|
* The app's description
|
||||||
* @type {string}
|
* @type {?string}
|
||||||
*/
|
*/
|
||||||
this.description = data.description;
|
this.description = data.description ?? this.description ?? null;
|
||||||
|
|
||||||
/**
|
/**
|
||||||
* The app's icon hash
|
* The app's icon hash
|
||||||
* @type {string}
|
* @type {?string}
|
||||||
*/
|
*/
|
||||||
this.icon = data.icon;
|
this.icon = data.icon ?? this.icon ?? null;
|
||||||
}
|
}
|
||||||
|
|
||||||
/**
|
/**
|
||||||
@@ -108,7 +108,7 @@ class Application extends Base {
|
|||||||
/**
|
/**
|
||||||
* When concatenated with a string, this automatically returns the application's name instead of the
|
* When concatenated with a string, this automatically returns the application's name instead of the
|
||||||
* Oauth2Application object.
|
* Oauth2Application object.
|
||||||
* @returns {string}
|
* @returns {?string}
|
||||||
* @example
|
* @example
|
||||||
* // Logs: Application name: My App
|
* // Logs: Application name: My App
|
||||||
* console.log(`Application name: ${application}`);
|
* console.log(`Application name: ${application}`);
|
||||||
|
|||||||
49
src/util/ApplicationFlags.js
Normal file
49
src/util/ApplicationFlags.js
Normal file
@@ -0,0 +1,49 @@
|
|||||||
|
'use strict';
|
||||||
|
|
||||||
|
const BitField = require('./BitField');
|
||||||
|
|
||||||
|
/**
|
||||||
|
* Data structure that makes it easy to interact with a {@link ClientApplication#flags} bitfield.
|
||||||
|
* @extends {BitField}
|
||||||
|
*/
|
||||||
|
class ApplicationFlags extends BitField {}
|
||||||
|
|
||||||
|
/**
|
||||||
|
* @name ApplicationFlags
|
||||||
|
* @kind constructor
|
||||||
|
* @memberof ApplicationFlags
|
||||||
|
* @param {BitFieldResolvable} [bits=0] Bit(s) to read from
|
||||||
|
*/
|
||||||
|
|
||||||
|
/**
|
||||||
|
* Bitfield of the packed bits
|
||||||
|
* @type {number}
|
||||||
|
* @name ApplicationFlags#bitfield
|
||||||
|
*/
|
||||||
|
|
||||||
|
/**
|
||||||
|
* Numeric application flags. All available properties:
|
||||||
|
* * `MANAGED_EMOJI`
|
||||||
|
* * `GROUP_DM_CREATE`
|
||||||
|
* * `RPC_HAS_CONNECTED`
|
||||||
|
* * `GATEWAY_PRESENCE`
|
||||||
|
* * `FATEWAY_PRESENCE_LIMITED`
|
||||||
|
* * `GATEWAY_GUILD_MEMBERS`
|
||||||
|
* * `GATEWAY_GUILD_MEMBERS_LIMITED`
|
||||||
|
* * `VERIFICATION_PENDING_GUILD_LIMIT`
|
||||||
|
* * `EMBEDDED`
|
||||||
|
* @type {Object}
|
||||||
|
*/
|
||||||
|
ApplicationFlags.FLAGS = {
|
||||||
|
MANAGED_EMOJI: 1 << 2,
|
||||||
|
GROUP_DM_CREATE: 1 << 4,
|
||||||
|
RPC_HAS_CONNECTED: 1 << 11,
|
||||||
|
GATEWAY_PRESENCE: 1 << 12,
|
||||||
|
GATEWAY_PRESENCE_LIMITED: 1 << 13,
|
||||||
|
GATEWAY_GUILD_MEMBERS: 1 << 14,
|
||||||
|
GATEWAY_GUILD_MEMBERS_LIMITED: 1 << 15,
|
||||||
|
VERIFICATION_PENDING_GUILD_LIMIT: 1 << 16,
|
||||||
|
EMBEDDED: 1 << 17,
|
||||||
|
};
|
||||||
|
|
||||||
|
module.exports = ApplicationFlags;
|
||||||
@@ -3,7 +3,7 @@
|
|||||||
const BitField = require('./BitField');
|
const BitField = require('./BitField');
|
||||||
|
|
||||||
/**
|
/**
|
||||||
* Data structure that makes it easy to interact with an {@link Message#flags} bitfield.
|
* Data structure that makes it easy to interact with a {@link Message#flags} bitfield.
|
||||||
* @extends {BitField}
|
* @extends {BitField}
|
||||||
*/
|
*/
|
||||||
class MessageFlags extends BitField {}
|
class MessageFlags extends BitField {}
|
||||||
|
|||||||
35
typings/index.d.ts
vendored
35
typings/index.d.ts
vendored
@@ -104,15 +104,20 @@ declare module 'discord.js' {
|
|||||||
constructor(client: Client, data: object);
|
constructor(client: Client, data: object);
|
||||||
public readonly createdAt: Date;
|
public readonly createdAt: Date;
|
||||||
public readonly createdTimestamp: number;
|
public readonly createdTimestamp: number;
|
||||||
public description: string;
|
public description: string | null;
|
||||||
public icon: string;
|
public icon: string | null;
|
||||||
public id: Snowflake;
|
public id: Snowflake;
|
||||||
public name: string;
|
public name: string | null;
|
||||||
public coverImage(options?: ImageURLOptions): string;
|
public coverImage(options?: ImageURLOptions): string | null;
|
||||||
public fetchAssets(): Promise<ApplicationAsset[]>;
|
public fetchAssets(): Promise<ApplicationAsset[]>;
|
||||||
public iconURL(options?: ImageURLOptions): string;
|
public iconURL(options?: ImageURLOptions): string | null;
|
||||||
public toJSON(): object;
|
public toJSON(): object;
|
||||||
public toString(): string;
|
public toString(): string | null;
|
||||||
|
}
|
||||||
|
|
||||||
|
export class ApplicationFlags extends BitField<ApplicationFlagsString> {
|
||||||
|
public static FLAGS: Record<ApplicationFlagsString, number>;
|
||||||
|
public static resolve(bit?: BitFieldResolvable<ApplicationFlagsString, number>): number;
|
||||||
}
|
}
|
||||||
|
|
||||||
export class Base {
|
export class Base {
|
||||||
@@ -203,6 +208,7 @@ declare module 'discord.js' {
|
|||||||
private _eval(script: string): any;
|
private _eval(script: string): any;
|
||||||
private _validateOptions(options: ClientOptions): void;
|
private _validateOptions(options: ClientOptions): void;
|
||||||
|
|
||||||
|
public application: ClientApplication | null;
|
||||||
public channels: ChannelManager;
|
public channels: ChannelManager;
|
||||||
public readonly emojis: BaseGuildEmojiManager;
|
public readonly emojis: BaseGuildEmojiManager;
|
||||||
public guilds: GuildManager;
|
public guilds: GuildManager;
|
||||||
@@ -217,13 +223,12 @@ declare module 'discord.js' {
|
|||||||
public voice: ClientVoiceManager;
|
public voice: ClientVoiceManager;
|
||||||
public ws: WebSocketManager;
|
public ws: WebSocketManager;
|
||||||
public destroy(): void;
|
public destroy(): void;
|
||||||
public fetchApplication(): Promise<ClientApplication>;
|
|
||||||
public fetchGuildPreview(guild: GuildResolvable): Promise<GuildPreview>;
|
public fetchGuildPreview(guild: GuildResolvable): Promise<GuildPreview>;
|
||||||
public fetchInvite(invite: InviteResolvable): Promise<Invite>;
|
public fetchInvite(invite: InviteResolvable): Promise<Invite>;
|
||||||
public fetchGuildTemplate(template: GuildTemplateResolvable): Promise<GuildTemplate>;
|
public fetchGuildTemplate(template: GuildTemplateResolvable): Promise<GuildTemplate>;
|
||||||
public fetchVoiceRegions(): Promise<Collection<string, VoiceRegion>>;
|
public fetchVoiceRegions(): Promise<Collection<string, VoiceRegion>>;
|
||||||
public fetchWebhook(id: Snowflake, token?: string): Promise<Webhook>;
|
public fetchWebhook(id: Snowflake, token?: string): Promise<Webhook>;
|
||||||
public generateInvite(options?: InviteGenerationOptions): Promise<string>;
|
public generateInvite(options?: InviteGenerationOptions): string;
|
||||||
public login(token?: string): Promise<string>;
|
public login(token?: string): Promise<string>;
|
||||||
public sweepMessages(lifetime?: number): number;
|
public sweepMessages(lifetime?: number): number;
|
||||||
public toJSON(): object;
|
public toJSON(): object;
|
||||||
@@ -257,8 +262,11 @@ declare module 'discord.js' {
|
|||||||
public botPublic: boolean | null;
|
public botPublic: boolean | null;
|
||||||
public botRequireCodeGrant: boolean | null;
|
public botRequireCodeGrant: boolean | null;
|
||||||
public cover: string | null;
|
public cover: string | null;
|
||||||
|
public flags: Readonly<ApplicationFlags>;
|
||||||
public owner: User | Team | null;
|
public owner: User | Team | null;
|
||||||
|
public readonly partial: boolean;
|
||||||
public rpcOrigins: string[];
|
public rpcOrigins: string[];
|
||||||
|
public fetch(): Promise<ClientApplication>;
|
||||||
}
|
}
|
||||||
|
|
||||||
export class ClientUser extends User {
|
export class ClientUser extends User {
|
||||||
@@ -2299,6 +2307,17 @@ declare module 'discord.js' {
|
|||||||
type: 'BIG' | 'SMALL';
|
type: 'BIG' | 'SMALL';
|
||||||
}
|
}
|
||||||
|
|
||||||
|
type ApplicationFlagsString =
|
||||||
|
| 'MANAGED_EMOJI'
|
||||||
|
| 'GROUP_DM_CREATE'
|
||||||
|
| 'RPC_HAS_CONNECTED'
|
||||||
|
| 'GATEWAY_PRESENCE'
|
||||||
|
| 'FATEWAY_PRESENCE_LIMITED'
|
||||||
|
| 'GATEWAY_GUILD_MEMBERS'
|
||||||
|
| 'GATEWAY_GUILD_MEMBERS_LIMITED'
|
||||||
|
| 'VERIFICATION_PENDING_GUILD_LIMIT'
|
||||||
|
| 'EMBEDDED';
|
||||||
|
|
||||||
interface AuditLogChange {
|
interface AuditLogChange {
|
||||||
key: string;
|
key: string;
|
||||||
old?: any;
|
old?: any;
|
||||||
|
|||||||
Reference in New Issue
Block a user