refactor(RichPresence): for Selfbot
This commit is contained in:
parent
0bb0b0e59d
commit
fd17a6a6f0
@ -156,38 +156,44 @@ client.user.setActivity(custom);
|
|||||||
```
|
```
|
||||||
|
|
||||||
Rich Presence [Custom]
|
Rich Presence [Custom]
|
||||||
Usage: see this module https://www.npmjs.com/package/discordrpcgenerator
|
|
||||||
```js
|
```js
|
||||||
const RPC = require('discord-rpc-contructor');
|
const r = new Discord.RichPresence()
|
||||||
const r = new RPC.Rpc()
|
|
||||||
.setApplicationId('817229550684471297')
|
.setApplicationId('817229550684471297')
|
||||||
.setType(0)
|
.setType('STREAMING')
|
||||||
|
.setURL('https://youtube.com/watch?v=dQw4w9WgXcQ')
|
||||||
.setState('State')
|
.setState('State')
|
||||||
.setName('Name')
|
.setName('Name')
|
||||||
.setDetails('Details')
|
.setDetails('Details')
|
||||||
.setParty({
|
.setParty({
|
||||||
size: [1, 2],
|
max: 9,
|
||||||
id: RPC.uuid(),
|
current: 1,
|
||||||
|
id: Discord.getUUID(),
|
||||||
})
|
})
|
||||||
.setStartTimestamp(Date.now())
|
.setStartTimestamp(Date.now())
|
||||||
.setAssetsLargeImage('929325841350000660')
|
.setAssetsLargeImage('929325841350000660')
|
||||||
.setAssetsLargeText('Youtube')
|
.setAssetsLargeText('Youtube')
|
||||||
.setAssetsSmallImage('895316294222635008')
|
.setAssetsSmallImage('895316294222635008')
|
||||||
.setAssetsSmallText('Bot')
|
.setAssetsSmallText('Bot')
|
||||||
client.user.setActivity(r.toDiscord().game);
|
.addButton('name', 'https://link.com/')
|
||||||
// Button not working
|
client.user.setActivity(r.toJSON());
|
||||||
```
|
```
|
||||||
<img src='https://cdn.discordapp.com/attachments/820557032016969751/955767445220646922/unknown.png'>
|
<img src='https://cdn.discordapp.com/attachments/820557032016969751/994300662378676264/unknown.png'>
|
||||||
|
|
||||||
Rich Presence with Twitch / Spotify
|
Rich Presence with Spotify
|
||||||
```js
|
```js
|
||||||
Update soon ~
|
const r = new Discord.SpotifyRPC(client)
|
||||||
|
.setAssetsLargeImage("spotify:ab67616d00001e02768629f8bc5b39b68797d1bb") // Image ID
|
||||||
|
.setAssetsSmallImage("spotify:ab67616d00001e02768629f8bc5b39b68797d1bb")
|
||||||
|
.setState('state')
|
||||||
|
.setDetails('details')
|
||||||
|
.setSongId('667eE4CFfNtJloC6Lvmgrx'); // Song ID
|
||||||
|
client.user.setActivity(r.toJSON());
|
||||||
```
|
```
|
||||||
|
<img src='https://cdn.discordapp.com/attachments/820557032016969751/994308354535075980/unknown.png'>
|
||||||
|
|
||||||
|
|
||||||
<strong>New: You can now add custom images for RPC !</strong>
|
<strong>New: You can now add custom images for RPC !</strong>
|
||||||
> Tutorial:
|
> Tutorial:
|
||||||
|
|
||||||
<strong>Method 1</strong>
|
|
||||||
|
|
||||||
+ Step 1: Send photos by embed.thumbnail
|
+ Step 1: Send photos by embed.thumbnail
|
||||||
|
|
||||||
@ -203,67 +209,27 @@ const proxyURL = msg.embeds[0].thumbnail.proxyURL;
|
|||||||
+ Step 3: Put the URL in the constructor
|
+ Step 3: Put the URL in the constructor
|
||||||
|
|
||||||
```js
|
```js
|
||||||
const RPC = require('discord-rpc-contructor');
|
const r = new Discord.RichPresence()
|
||||||
const r = new RPC.Rpc()
|
|
||||||
.setApplicationId('817229550684471297')
|
.setApplicationId('817229550684471297')
|
||||||
.setType(0)
|
.setType('STREAMING')
|
||||||
|
.setURL('https://youtube.com/watch?v=dQw4w9WgXcQ')
|
||||||
.setState('State')
|
.setState('State')
|
||||||
.setName('Name')
|
.setName('Name')
|
||||||
.setDetails('Details')
|
.setDetails('Details')
|
||||||
.setAssetsLargeImage(proxyURL) // Custom image
|
.setParty({
|
||||||
|
max: 9,
|
||||||
|
current: 1,
|
||||||
|
id: Discord.getUUID(),
|
||||||
|
})
|
||||||
|
.setStartTimestamp(Date.now())
|
||||||
|
.setAssetsLargeImage(proxyURL)
|
||||||
.setAssetsLargeText('Youtube')
|
.setAssetsLargeText('Youtube')
|
||||||
.setAssetsSmallImage('895316294222635008')
|
.setAssetsSmallImage('895316294222635008')
|
||||||
.setAssetsSmallText('Bot')
|
.setAssetsSmallText('Bot')
|
||||||
|
.addButton('name', 'https://link.com/')
|
||||||
client.user.setActivity(r.toDiscord().game);
|
client.user.setActivity(r.toDiscord().game);
|
||||||
```
|
```
|
||||||
|
|
||||||
<strong>Method 2 - require ver 1.1.1 - latest</strong>
|
|
||||||
|
|
||||||
```
|
|
||||||
Note: If you use time-varying RPC, or if it takes too long to place images, use the 2nd method because it spams me too much .-.
|
|
||||||
```
|
|
||||||
|
|
||||||
![image](https://user-images.githubusercontent.com/71698422/164040950-9679cec3-b47e-4521-b5f6-ee83a79a1978.png)
|
|
||||||
|
|
||||||
|
|
||||||
```js
|
|
||||||
const RPC = require('discord-rpc-contructor');
|
|
||||||
const r = new RPC.Rpc()
|
|
||||||
.setApplicationId('817229550684471297')
|
|
||||||
.setType(0)
|
|
||||||
.setState('State')
|
|
||||||
.setName('Name')
|
|
||||||
.setDetails('Details')
|
|
||||||
.setAssetsLargeImage(await RPC.getImageCustom(imageURL1)) // Custom image
|
|
||||||
.setAssetsLargeText('Youtube')
|
|
||||||
.setAssetsSmallImage(await RPC.getImageCustom(imageURL2)) // Custom image
|
|
||||||
.setAssetsSmallText('Bot')
|
|
||||||
client.user.setActivity(r.toDiscord().game);
|
|
||||||
```
|
|
||||||
|
|
||||||
|
|
||||||
<strong>How to get AssetID ?</strong>
|
|
||||||
|
|
||||||
Code
|
|
||||||
|
|
||||||
```js
|
|
||||||
const RPC = require('discord-rpc-contructor');
|
|
||||||
// Bot ID
|
|
||||||
RPC.getRpcImages('817229550684471297').then(console.log);
|
|
||||||
```
|
|
||||||
Return
|
|
||||||
```js
|
|
||||||
// ID is AssetID
|
|
||||||
[
|
|
||||||
{ id: '838629816881381376', type: 1, name: 'honkai' },
|
|
||||||
{ id: '853533658250084352', type: 1, name: 'vscode' },
|
|
||||||
{ id: '895316294222635008', type: 1, name: 'botsagiri' },
|
|
||||||
{ id: '929324633063292929', type: 1, name: 'soundcloud' },
|
|
||||||
{ id: '929324634858479666', type: 1, name: 'spotify' },
|
|
||||||
{ id: '929325841350000660', type: 1, name: 'youtube' }
|
|
||||||
]
|
|
||||||
```
|
|
||||||
You can cache to use these files, do not run this function too much because it will be rate limit
|
|
||||||
And you can change the status 5 times every 20 seconds!
|
And you can change the status 5 times every 20 seconds!
|
||||||
</details>
|
</details>
|
||||||
|
|
||||||
|
File diff suppressed because one or more lines are too long
29
package-lock.json
generated
29
package-lock.json
generated
@ -1,12 +1,12 @@
|
|||||||
{
|
{
|
||||||
"name": "discord.js-selfbot-v13",
|
"name": "discord.js-selfbot-v13",
|
||||||
"version": "2.3.62",
|
"version": "2.3.63",
|
||||||
"lockfileVersion": 2,
|
"lockfileVersion": 2,
|
||||||
"requires": true,
|
"requires": true,
|
||||||
"packages": {
|
"packages": {
|
||||||
"": {
|
"": {
|
||||||
"name": "discord.js-selfbot-v13",
|
"name": "discord.js-selfbot-v13",
|
||||||
"version": "2.3.62",
|
"version": "2.3.63",
|
||||||
"license": "GNU General Public License v3.0",
|
"license": "GNU General Public License v3.0",
|
||||||
"dependencies": {
|
"dependencies": {
|
||||||
"@aikochan2k6/qrcode-terminal": "^0.12.0",
|
"@aikochan2k6/qrcode-terminal": "^0.12.0",
|
||||||
@ -21,7 +21,6 @@
|
|||||||
"chalk": "^4.1.2",
|
"chalk": "^4.1.2",
|
||||||
"discord-api-types": "^0.36.0",
|
"discord-api-types": "^0.36.0",
|
||||||
"discord-bettermarkdown": "^1.2.0",
|
"discord-bettermarkdown": "^1.2.0",
|
||||||
"discord-rpc-contructor": "^1.1.5",
|
|
||||||
"form-data": "^4.0.0",
|
"form-data": "^4.0.0",
|
||||||
"json-bigint": "^1.0.0",
|
"json-bigint": "^1.0.0",
|
||||||
"node-fetch": "^2.6.1",
|
"node-fetch": "^2.6.1",
|
||||||
@ -4442,15 +4441,6 @@
|
|||||||
"discord.js": "^13.6.0"
|
"discord.js": "^13.6.0"
|
||||||
}
|
}
|
||||||
},
|
},
|
||||||
"node_modules/discord-rpc-contructor": {
|
|
||||||
"version": "1.1.5",
|
|
||||||
"resolved": "https://registry.npmjs.org/discord-rpc-contructor/-/discord-rpc-contructor-1.1.5.tgz",
|
|
||||||
"integrity": "sha512-qMwlQXPRT5DoXXRXn3RqHfH28HV7bUOKhPEBS/BelUj5fUFKG9ZotwJ4NeAxk46SRtjz39kIDybomXfLV8vi2A==",
|
|
||||||
"dependencies": {
|
|
||||||
"lodash": "^4.17.21",
|
|
||||||
"node-fetch": "^2.6.7"
|
|
||||||
}
|
|
||||||
},
|
|
||||||
"node_modules/discord.js": {
|
"node_modules/discord.js": {
|
||||||
"version": "13.8.0",
|
"version": "13.8.0",
|
||||||
"resolved": "https://registry.npmjs.org/discord.js/-/discord.js-13.8.0.tgz",
|
"resolved": "https://registry.npmjs.org/discord.js/-/discord.js-13.8.0.tgz",
|
||||||
@ -8542,7 +8532,8 @@
|
|||||||
"node_modules/lodash": {
|
"node_modules/lodash": {
|
||||||
"version": "4.17.21",
|
"version": "4.17.21",
|
||||||
"resolved": "https://registry.npmjs.org/lodash/-/lodash-4.17.21.tgz",
|
"resolved": "https://registry.npmjs.org/lodash/-/lodash-4.17.21.tgz",
|
||||||
"integrity": "sha512-v2kDEe57lecTulaDIuNTPy3Ry4gLGJ6Z1O3vE1krgXZNrsQ+LFTGHVxVjcXPs17LhbZVGedAJv8XZ1tvj5FvSg=="
|
"integrity": "sha512-v2kDEe57lecTulaDIuNTPy3Ry4gLGJ6Z1O3vE1krgXZNrsQ+LFTGHVxVjcXPs17LhbZVGedAJv8XZ1tvj5FvSg==",
|
||||||
|
"dev": true
|
||||||
},
|
},
|
||||||
"node_modules/lodash.camelcase": {
|
"node_modules/lodash.camelcase": {
|
||||||
"version": "4.3.0",
|
"version": "4.3.0",
|
||||||
@ -15954,15 +15945,6 @@
|
|||||||
"discord.js": "^13.6.0"
|
"discord.js": "^13.6.0"
|
||||||
}
|
}
|
||||||
},
|
},
|
||||||
"discord-rpc-contructor": {
|
|
||||||
"version": "1.1.5",
|
|
||||||
"resolved": "https://registry.npmjs.org/discord-rpc-contructor/-/discord-rpc-contructor-1.1.5.tgz",
|
|
||||||
"integrity": "sha512-qMwlQXPRT5DoXXRXn3RqHfH28HV7bUOKhPEBS/BelUj5fUFKG9ZotwJ4NeAxk46SRtjz39kIDybomXfLV8vi2A==",
|
|
||||||
"requires": {
|
|
||||||
"lodash": "^4.17.21",
|
|
||||||
"node-fetch": "^2.6.7"
|
|
||||||
}
|
|
||||||
},
|
|
||||||
"discord.js": {
|
"discord.js": {
|
||||||
"version": "13.8.0",
|
"version": "13.8.0",
|
||||||
"resolved": "https://registry.npmjs.org/discord.js/-/discord.js-13.8.0.tgz",
|
"resolved": "https://registry.npmjs.org/discord.js/-/discord.js-13.8.0.tgz",
|
||||||
@ -19095,7 +19077,8 @@
|
|||||||
"lodash": {
|
"lodash": {
|
||||||
"version": "4.17.21",
|
"version": "4.17.21",
|
||||||
"resolved": "https://registry.npmjs.org/lodash/-/lodash-4.17.21.tgz",
|
"resolved": "https://registry.npmjs.org/lodash/-/lodash-4.17.21.tgz",
|
||||||
"integrity": "sha512-v2kDEe57lecTulaDIuNTPy3Ry4gLGJ6Z1O3vE1krgXZNrsQ+LFTGHVxVjcXPs17LhbZVGedAJv8XZ1tvj5FvSg=="
|
"integrity": "sha512-v2kDEe57lecTulaDIuNTPy3Ry4gLGJ6Z1O3vE1krgXZNrsQ+LFTGHVxVjcXPs17LhbZVGedAJv8XZ1tvj5FvSg==",
|
||||||
|
"dev": true
|
||||||
},
|
},
|
||||||
"lodash.camelcase": {
|
"lodash.camelcase": {
|
||||||
"version": "4.3.0",
|
"version": "4.3.0",
|
||||||
|
@ -1,6 +1,6 @@
|
|||||||
{
|
{
|
||||||
"name": "discord.js-selfbot-v13",
|
"name": "discord.js-selfbot-v13",
|
||||||
"version": "2.3.62",
|
"version": "2.3.63",
|
||||||
"description": "A unofficial discord.js fork for creating selfbots [Based on discord.js v13]",
|
"description": "A unofficial discord.js fork for creating selfbots [Based on discord.js v13]",
|
||||||
"main": "./src/index.js",
|
"main": "./src/index.js",
|
||||||
"types": "./typings/index.d.ts",
|
"types": "./typings/index.d.ts",
|
||||||
@ -63,7 +63,6 @@
|
|||||||
"chalk": "^4.1.2",
|
"chalk": "^4.1.2",
|
||||||
"discord-api-types": "^0.36.0",
|
"discord-api-types": "^0.36.0",
|
||||||
"discord-bettermarkdown": "^1.2.0",
|
"discord-bettermarkdown": "^1.2.0",
|
||||||
"discord-rpc-contructor": "^1.1.5",
|
|
||||||
"form-data": "^4.0.0",
|
"form-data": "^4.0.0",
|
||||||
"json-bigint": "^1.0.0",
|
"json-bigint": "^1.0.0",
|
||||||
"node-fetch": "^2.6.1",
|
"node-fetch": "^2.6.1",
|
||||||
|
@ -135,6 +135,12 @@ exports.ReactionCollector = require('./structures/ReactionCollector');
|
|||||||
exports.ReactionEmoji = require('./structures/ReactionEmoji');
|
exports.ReactionEmoji = require('./structures/ReactionEmoji');
|
||||||
exports.RichPresenceAssets = require('./structures/Presence').RichPresenceAssets;
|
exports.RichPresenceAssets = require('./structures/Presence').RichPresenceAssets;
|
||||||
exports.Role = require('./structures/Role').Role;
|
exports.Role = require('./structures/Role').Role;
|
||||||
|
// RPC
|
||||||
|
exports.getUUID = require('./structures/RichPresence').getUUID;
|
||||||
|
exports.CustomStatus = require('./structures/RichPresence').CustomStatus;
|
||||||
|
exports.RichPresence = require('./structures/RichPresence').RichPresence;
|
||||||
|
exports.SpotifyRPC = require('./structures/RichPresence').SpotifyRPC;
|
||||||
|
//
|
||||||
exports.SelectMenuInteraction = require('./structures/SelectMenuInteraction');
|
exports.SelectMenuInteraction = require('./structures/SelectMenuInteraction');
|
||||||
exports.StageChannel = require('./structures/StageChannel');
|
exports.StageChannel = require('./structures/StageChannel');
|
||||||
exports.StageInstance = require('./structures/StageInstance').StageInstance;
|
exports.StageInstance = require('./structures/StageInstance').StageInstance;
|
||||||
|
512
src/structures/RichPresence.js
Normal file
512
src/structures/RichPresence.js
Normal file
@ -0,0 +1,512 @@
|
|||||||
|
'use strict';
|
||||||
|
const crypto = require('crypto');
|
||||||
|
const { ActivityTypes } = require('../util/Constants');
|
||||||
|
const { resolvePartialEmoji } = require('../util/Util');
|
||||||
|
|
||||||
|
// eslint-disable-next-line
|
||||||
|
const getUUID = () =>
|
||||||
|
([1e7] + -1e3 + -4e3 + -8e3 + -1e11).replace(/[018]/g, a => (a ^ ((Math.random() * 16) >> (a / 4))).toString(16));
|
||||||
|
// Function check url valid (ok copilot)
|
||||||
|
// eslint-disable-next-line
|
||||||
|
const checkUrl = url => /^(?:http(s)?:\/\/)?[\w.-]+(?:\.[\w\.-]+)+[\w\-\._~:/?#[\]@!\$&'\(\)\*\+,;=.]+$/.test(url);
|
||||||
|
|
||||||
|
class CustomStatus {
|
||||||
|
/**
|
||||||
|
* @typedef {Object} CustomStatusOptions
|
||||||
|
* @property {string} [state] The state to be displayed
|
||||||
|
* @property {EmojiIdentifierResolvable} [emoji] The emoji to be displayed
|
||||||
|
*/
|
||||||
|
|
||||||
|
/**
|
||||||
|
* @param {CustomStatus|CustomStatusOptions} [data={}] CustomStatus to clone or raw data
|
||||||
|
*/
|
||||||
|
constructor(data = {}) {
|
||||||
|
this.name = 'Custom Status';
|
||||||
|
/**
|
||||||
|
* The emoji to be displayed
|
||||||
|
* @type {?EmojiIdentifierResolvable}
|
||||||
|
*/
|
||||||
|
this.emoji = null;
|
||||||
|
this.type = ActivityTypes.CUSTOM;
|
||||||
|
/**
|
||||||
|
* The state to be displayed
|
||||||
|
* @type {?string}
|
||||||
|
*/
|
||||||
|
this.state = null;
|
||||||
|
this.setup(data);
|
||||||
|
}
|
||||||
|
/**
|
||||||
|
* Sets the status from a JSON object
|
||||||
|
* @param {CustomStatus|CustomStatusOptions} data CustomStatus to clone or raw data
|
||||||
|
* @private
|
||||||
|
*/
|
||||||
|
setup(data) {
|
||||||
|
this.emoji = data.emoji ? resolvePartialEmoji(data.emoji) : null;
|
||||||
|
this.state = data.state;
|
||||||
|
}
|
||||||
|
/**
|
||||||
|
* Set the emoji of this activity
|
||||||
|
* @param {EmojiIdentifierResolvable} emoji The emoji to be displayed
|
||||||
|
* @returns {CustomStatus}
|
||||||
|
*/
|
||||||
|
setEmoji(emoji) {
|
||||||
|
this.emoji = resolvePartialEmoji(emoji);
|
||||||
|
return this;
|
||||||
|
}
|
||||||
|
/**
|
||||||
|
* Set state of this activity
|
||||||
|
* @param {string | null} state The state to be displayed
|
||||||
|
* @returns {CustomStatus}
|
||||||
|
*/
|
||||||
|
setState(state) {
|
||||||
|
this.state = state;
|
||||||
|
return this;
|
||||||
|
}
|
||||||
|
|
||||||
|
/**
|
||||||
|
* Returns an object that can be used to set the status
|
||||||
|
* @returns {CustomStatus}
|
||||||
|
* @example
|
||||||
|
*/
|
||||||
|
toJSON() {
|
||||||
|
if (!this.emoji & !this.state) throw new Error('CustomStatus must have at least one of emoji or state');
|
||||||
|
return {
|
||||||
|
name: this.name,
|
||||||
|
emoji: this.emoji,
|
||||||
|
type: this.type,
|
||||||
|
state: this.state,
|
||||||
|
};
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
class RichPresence {
|
||||||
|
/**
|
||||||
|
* @param {Client} client Discord client
|
||||||
|
* @param {RichPresence} [data={}] RichPresence to clone or raw data
|
||||||
|
*/
|
||||||
|
constructor(client, data = {}) {
|
||||||
|
Object.defineProperty(this, 'client', { value: client });
|
||||||
|
/**
|
||||||
|
* The activity's name
|
||||||
|
* @type {string}
|
||||||
|
*/
|
||||||
|
this.name = null;
|
||||||
|
/**
|
||||||
|
* The activity status's type
|
||||||
|
* @type {ActivityType}
|
||||||
|
*/
|
||||||
|
this.type = ActivityTypes.PLAYING;
|
||||||
|
/**
|
||||||
|
* If the activity is being streamed, a link to the stream
|
||||||
|
* @type {?string}
|
||||||
|
*/
|
||||||
|
this.url = null;
|
||||||
|
/**
|
||||||
|
* The id of the application associated with this activity
|
||||||
|
* @type {?Snowflake}
|
||||||
|
*/
|
||||||
|
this.application_id = null;
|
||||||
|
/**
|
||||||
|
* State of the activity
|
||||||
|
* @type {?string}
|
||||||
|
*/
|
||||||
|
this.state = null;
|
||||||
|
/**
|
||||||
|
* Details about the activity
|
||||||
|
* @type {?string}
|
||||||
|
*/
|
||||||
|
this.details = null;
|
||||||
|
/**
|
||||||
|
* Party of the activity
|
||||||
|
* @type {?ActivityParty}
|
||||||
|
*/
|
||||||
|
this.party = null;
|
||||||
|
/**
|
||||||
|
* Timestamps for the activity
|
||||||
|
* @type {?ActivityTimestamps}
|
||||||
|
*/
|
||||||
|
this.timestamps = null;
|
||||||
|
/**
|
||||||
|
* Assets for rich presence
|
||||||
|
* @type {?RichPresenceAssets}
|
||||||
|
*/
|
||||||
|
this.assets = null;
|
||||||
|
/**
|
||||||
|
* The labels of the buttons of this rich presence
|
||||||
|
* @type {string[]}
|
||||||
|
*/
|
||||||
|
this.buttons = null;
|
||||||
|
this.setup(data);
|
||||||
|
}
|
||||||
|
/**
|
||||||
|
* Sets the status from a JSON object
|
||||||
|
* @param {RichPresence} data data
|
||||||
|
* @private
|
||||||
|
*/
|
||||||
|
setup(data) {
|
||||||
|
this.name = data.name;
|
||||||
|
this.type = typeof data.type != 'number' ? ActivityTypes[data.type?.toUpperCase()] : data.type;
|
||||||
|
this.application_id = data.application_id;
|
||||||
|
this.url = data.url;
|
||||||
|
this.state = data.state;
|
||||||
|
this.details = data.details;
|
||||||
|
this.party = data.party;
|
||||||
|
this.timestamps = data.timestamps;
|
||||||
|
this.created_at = data.created_at;
|
||||||
|
this.secrets = data.secrets;
|
||||||
|
this.assets = data.assets;
|
||||||
|
this.buttons = data.buttons;
|
||||||
|
this.metadata = data.metadata;
|
||||||
|
}
|
||||||
|
/**
|
||||||
|
* Set the large image of this activity
|
||||||
|
* @param {?Snowflake} image The large image asset's id
|
||||||
|
* @returns {RichPresence}
|
||||||
|
*/
|
||||||
|
setAssetsLargeImage(image) {
|
||||||
|
// Gif support
|
||||||
|
if (
|
||||||
|
typeof image == 'string' &&
|
||||||
|
(image.startsWith('https://') || image.startsWith('http://')) &&
|
||||||
|
image.includes('external') &&
|
||||||
|
image.includes('discord')
|
||||||
|
) {
|
||||||
|
image = `mp:external${image.split('external')[1]}`;
|
||||||
|
}
|
||||||
|
if (!(this.assets instanceof Object)) this.assets = {};
|
||||||
|
this.assets.large_image = image;
|
||||||
|
return this;
|
||||||
|
}
|
||||||
|
/**
|
||||||
|
* Set the small image of this activity
|
||||||
|
* @param {?Snowflake} image The small image asset's id
|
||||||
|
* @returns {RichPresence}
|
||||||
|
*/
|
||||||
|
setAssetsSmallImage(image) {
|
||||||
|
// Gif support
|
||||||
|
if (
|
||||||
|
typeof image == 'string' &&
|
||||||
|
(image.startsWith('https://') || image.startsWith('http://')) &&
|
||||||
|
image.includes('external') &&
|
||||||
|
image.includes('discord')
|
||||||
|
) {
|
||||||
|
image = `mp:external${image.split('external')[1]}`;
|
||||||
|
}
|
||||||
|
if (!(this.assets instanceof Object)) this.assets = {};
|
||||||
|
this.assets.small_image = image;
|
||||||
|
return this;
|
||||||
|
}
|
||||||
|
/**
|
||||||
|
* Hover text for the large image
|
||||||
|
* @param {string} text Assets text
|
||||||
|
* @returns {RichPresence}
|
||||||
|
*/
|
||||||
|
setAssetsLargeText(text) {
|
||||||
|
if (typeof this.assets !== 'object') this.assets = {};
|
||||||
|
this.assets.large_text = text;
|
||||||
|
return this;
|
||||||
|
}
|
||||||
|
/**
|
||||||
|
* Hover text for the small image
|
||||||
|
* @param {string} text Assets text
|
||||||
|
* @returns {RichPresence}
|
||||||
|
*/
|
||||||
|
setAssetsSmallText(text) {
|
||||||
|
if (typeof this.assets !== 'object') this.assets = {};
|
||||||
|
this.assets.small_text = text;
|
||||||
|
return this;
|
||||||
|
}
|
||||||
|
/**
|
||||||
|
* Set the name of the activity
|
||||||
|
* @param {?string} name The activity's name
|
||||||
|
* @returns {RichPresence}
|
||||||
|
*/
|
||||||
|
setName(name) {
|
||||||
|
this.name = name;
|
||||||
|
return this;
|
||||||
|
}
|
||||||
|
/**
|
||||||
|
* If the activity is being streamed, a link to the stream
|
||||||
|
* @param {?string} url URL of the stream
|
||||||
|
* @returns {RichPresence}
|
||||||
|
*/
|
||||||
|
setURL(url) {
|
||||||
|
if (typeof url == 'string' && !checkUrl(url)) throw new Error('URL must be a valid URL');
|
||||||
|
if (typeof url != 'string') url = null;
|
||||||
|
this.url = url;
|
||||||
|
return this;
|
||||||
|
}
|
||||||
|
/**
|
||||||
|
* The activity status's type
|
||||||
|
* @param {?ActivityTypes} type The type of activity
|
||||||
|
* @returns {RichPresence}
|
||||||
|
*/
|
||||||
|
setType(type) {
|
||||||
|
this.type = ActivityTypes[type?.toUpperCase()];
|
||||||
|
if (typeof this.type == 'string') this.type = ActivityTypes[this.type];
|
||||||
|
if (typeof this.type != 'number') throw new Error('Type must be a valid ActivityType');
|
||||||
|
return this;
|
||||||
|
}
|
||||||
|
/**
|
||||||
|
* Set the application id of this activity
|
||||||
|
* @param {?Snowflake} id Bot's id
|
||||||
|
* @returns {RichPresence}
|
||||||
|
*/
|
||||||
|
setApplicationId(id) {
|
||||||
|
this.application_id = id;
|
||||||
|
return this;
|
||||||
|
}
|
||||||
|
/**
|
||||||
|
* Set the state of the activity
|
||||||
|
* @param {?string} state The state of the activity
|
||||||
|
* @returns {RichPresence}
|
||||||
|
*/
|
||||||
|
setState(state) {
|
||||||
|
this.state = state;
|
||||||
|
return this;
|
||||||
|
}
|
||||||
|
/**
|
||||||
|
* Set the details of the activity
|
||||||
|
* @param {?string} details The details of the activity
|
||||||
|
* @returns {RichPresence}
|
||||||
|
*/
|
||||||
|
setDetails(details) {
|
||||||
|
this.details = details;
|
||||||
|
return this;
|
||||||
|
}
|
||||||
|
/**
|
||||||
|
* @typedef {Object} RichParty
|
||||||
|
* @property {string} id The id of the party
|
||||||
|
* @property {number} max The maximum number of members in the party
|
||||||
|
* @property {number} current The current number of members in the party
|
||||||
|
*/
|
||||||
|
/**
|
||||||
|
* Set the party of this activity
|
||||||
|
* @param {?RichParty} party The party to be displayed
|
||||||
|
* @returns {RichPresence}
|
||||||
|
*/
|
||||||
|
setParty(party) {
|
||||||
|
if (typeof party == 'object') {
|
||||||
|
if (!party.max || typeof party.max != 'number') throw new Error('Party must have max number');
|
||||||
|
if (!party.current || typeof party.current != 'number') throw new Error('Party must have current');
|
||||||
|
if (party.current > party.max) throw new Error('Party current must be less than max number');
|
||||||
|
if (!party.id || typeof party.id != 'string') party.id = getUUID();
|
||||||
|
this.party = {
|
||||||
|
size: [party.current, party.max],
|
||||||
|
id: party.id,
|
||||||
|
};
|
||||||
|
} else {
|
||||||
|
this.party = null;
|
||||||
|
}
|
||||||
|
return this;
|
||||||
|
}
|
||||||
|
/**
|
||||||
|
* Sets the start timestamp of the activity
|
||||||
|
* @param {?number} timestamp The timestamp of the start of the activity
|
||||||
|
* @returns {RichPresence}
|
||||||
|
*/
|
||||||
|
setStartTimestamp(timestamp) {
|
||||||
|
if (!this.timestamps) this.timestamps = {};
|
||||||
|
this.timestamps.start = timestamp;
|
||||||
|
return this;
|
||||||
|
}
|
||||||
|
/**
|
||||||
|
* Sets the end timestamp of the activity
|
||||||
|
* @param {?number} timestamp The timestamp of the end of the activity
|
||||||
|
* @returns {RichPresence}
|
||||||
|
*/
|
||||||
|
setEndTimestamp(timestamp) {
|
||||||
|
if (!this.timestamps) this.timestamps = {};
|
||||||
|
this.timestamps.end = timestamp;
|
||||||
|
return this;
|
||||||
|
}
|
||||||
|
/**
|
||||||
|
* @typedef {object} RichButton
|
||||||
|
* @property {string} name The name of the button
|
||||||
|
* @property {string} url The url of the button
|
||||||
|
*/
|
||||||
|
/**
|
||||||
|
* Set the buttons of the rich presence
|
||||||
|
* @param {...?RichButton} button A list of buttons to set
|
||||||
|
* @returns {RichPresence}
|
||||||
|
*/
|
||||||
|
setButtons(...button) {
|
||||||
|
if (button.length == 0) {
|
||||||
|
this.buttons = null;
|
||||||
|
delete this.metadata;
|
||||||
|
return this;
|
||||||
|
} else if (button.length > 2) {
|
||||||
|
throw new Error('RichPresence can only have up to 2 buttons');
|
||||||
|
}
|
||||||
|
this.buttons = [];
|
||||||
|
this.metadata = {
|
||||||
|
button_urls: [],
|
||||||
|
};
|
||||||
|
button.flat(2).forEach(b => {
|
||||||
|
if (b.name && b.url) {
|
||||||
|
this.buttons.push(b.name);
|
||||||
|
if (!checkUrl(b.url)) throw new Error('Button url must be a valid url');
|
||||||
|
this.metadata.button_urls.push(b.url);
|
||||||
|
} else {
|
||||||
|
throw new Error('Button must have name and url');
|
||||||
|
}
|
||||||
|
});
|
||||||
|
return this;
|
||||||
|
}
|
||||||
|
/**
|
||||||
|
* Add a button to the rich presence
|
||||||
|
* @param {string} name The name of the button
|
||||||
|
* @param {string} url The url of the button
|
||||||
|
* @returns {RichPresence}
|
||||||
|
*/
|
||||||
|
addButton(name, url) {
|
||||||
|
if (!name || !url) {
|
||||||
|
throw new Error('Button must have name and url');
|
||||||
|
}
|
||||||
|
if (typeof name !== 'string') throw new Error('Button name must be a string');
|
||||||
|
if (!checkUrl(url)) throw new Error('Button url must be a valid url');
|
||||||
|
if (!this.buttons) {
|
||||||
|
this.buttons = [];
|
||||||
|
this.metadata = {
|
||||||
|
button_urls: [],
|
||||||
|
};
|
||||||
|
}
|
||||||
|
this.buttons.push(name);
|
||||||
|
this.metadata.button_urls.push(url);
|
||||||
|
return this;
|
||||||
|
}
|
||||||
|
/**
|
||||||
|
* Convert the rich presence to a JSON object
|
||||||
|
* @returns {RichPresence}
|
||||||
|
*/
|
||||||
|
toJSON() {
|
||||||
|
return {
|
||||||
|
name: this.name,
|
||||||
|
type: this.type,
|
||||||
|
application_id: this.application_id,
|
||||||
|
url: this.url,
|
||||||
|
state: this.state,
|
||||||
|
details: this.details,
|
||||||
|
party: this.party,
|
||||||
|
timestamps: this.timestamps,
|
||||||
|
secrets: this.secrets,
|
||||||
|
assets: this.assets,
|
||||||
|
buttons: this.buttons,
|
||||||
|
metadata: this.metadata,
|
||||||
|
};
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
class SpotifyRPC extends RichPresence {
|
||||||
|
/**
|
||||||
|
* Create a new RichPresence (Spotify style)
|
||||||
|
* @param {Client} client Discord Client
|
||||||
|
* @param {SpotifyRPC} options Options for the Spotify RPC
|
||||||
|
*/
|
||||||
|
constructor(client, options = {}) {
|
||||||
|
if (!client) throw new Error('Client must be set');
|
||||||
|
super(client, options);
|
||||||
|
this.setup(options);
|
||||||
|
}
|
||||||
|
/**
|
||||||
|
* Sets the status from a JSON object
|
||||||
|
* @param {SpotifyRPC} options data
|
||||||
|
* @private
|
||||||
|
*/
|
||||||
|
setup(options) {
|
||||||
|
this.party = {
|
||||||
|
id: `spotify:${this.client.user.id}`,
|
||||||
|
};
|
||||||
|
/**
|
||||||
|
* The Spotify song's id
|
||||||
|
* @type {?string}
|
||||||
|
*/
|
||||||
|
this.sync_id = options.sync_id;
|
||||||
|
/**
|
||||||
|
* The activity's id
|
||||||
|
* @type {string}
|
||||||
|
*/
|
||||||
|
this.id = 'spotify:1';
|
||||||
|
/**
|
||||||
|
* Creation date of the activity
|
||||||
|
* @type {number}
|
||||||
|
*/
|
||||||
|
this.created_at = Date.now();
|
||||||
|
/**
|
||||||
|
* Flags that describe the activity
|
||||||
|
* @type {Readonly<ActivityFlags>}
|
||||||
|
*/
|
||||||
|
this.flags = 48; // Sync + Play (ActivityFlags)
|
||||||
|
/**
|
||||||
|
* The game's or Spotify session's id
|
||||||
|
* @type {?string}
|
||||||
|
*/
|
||||||
|
this.session_id = this.client.ws.shards.first().sessionId;
|
||||||
|
|
||||||
|
this.secrets = {
|
||||||
|
join: crypto.randomBytes(20).toString('hex'), // SHA1 / SHA128
|
||||||
|
spectate: crypto.randomBytes(20).toString('hex'),
|
||||||
|
match: crypto.randomBytes(20).toString('hex'),
|
||||||
|
};
|
||||||
|
}
|
||||||
|
setAssetsLargeImage(image) {
|
||||||
|
if (image.startsWith('spotify:')) image = image.replace('spotify:', '');
|
||||||
|
if (!(this.assets instanceof Object)) {
|
||||||
|
this.assets = {
|
||||||
|
large_image: `spotify:${image}`,
|
||||||
|
};
|
||||||
|
} else {
|
||||||
|
this.assets.large_image = `spotify:${image}`;
|
||||||
|
}
|
||||||
|
return this;
|
||||||
|
}
|
||||||
|
setAssetsSmallImage(image) {
|
||||||
|
if (image.startsWith('spotify:')) image = image.replace('spotify:', '');
|
||||||
|
if (!(this.assets instanceof Object)) {
|
||||||
|
this.assets = {
|
||||||
|
small_image: `spotify:${image}`,
|
||||||
|
};
|
||||||
|
} else {
|
||||||
|
this.assets.small_image = `spotify:${image}`;
|
||||||
|
}
|
||||||
|
return this;
|
||||||
|
}
|
||||||
|
/**
|
||||||
|
* Set Spotify song id to sync with
|
||||||
|
* @param {string} id Song id
|
||||||
|
* @returns {SpotifyRPC}
|
||||||
|
*/
|
||||||
|
setSongId(id) {
|
||||||
|
this.sync_id = id;
|
||||||
|
return this;
|
||||||
|
}
|
||||||
|
|
||||||
|
toJSON() {
|
||||||
|
if (!this.sync_id) throw new Error('Song id is required');
|
||||||
|
return {
|
||||||
|
name: 'Spotify',
|
||||||
|
type: ActivityTypes.LISTENING,
|
||||||
|
application_id: this.application_id,
|
||||||
|
url: this.url,
|
||||||
|
state: this.state,
|
||||||
|
details: this.details,
|
||||||
|
party: this.party,
|
||||||
|
timestamps: this.timestamps,
|
||||||
|
secrets: this.secrets,
|
||||||
|
assets: this.assets,
|
||||||
|
session_id: this.session_id,
|
||||||
|
sync_id: this.sync_id,
|
||||||
|
flags: this.flags,
|
||||||
|
id: this.id,
|
||||||
|
created_at: this.created_at,
|
||||||
|
metadata: this.metadata,
|
||||||
|
};
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
module.exports = {
|
||||||
|
CustomStatus,
|
||||||
|
RichPresence,
|
||||||
|
SpotifyRPC,
|
||||||
|
getUUID,
|
||||||
|
};
|
85
typings/index.d.ts
vendored
85
typings/index.d.ts
vendored
@ -165,6 +165,91 @@ import DiscordAuthWebsocket from '../src/util/RemoteAuth.js';
|
|||||||
|
|
||||||
//#region Classes
|
//#region Classes
|
||||||
|
|
||||||
|
// RPC by aiko-chan-ai
|
||||||
|
export interface RichButton {
|
||||||
|
name: string;
|
||||||
|
url: string;
|
||||||
|
}
|
||||||
|
|
||||||
|
export abstract class RichPresence {
|
||||||
|
public constructor(client?: Client, data?: object);
|
||||||
|
public application_id: Snowflake | null;
|
||||||
|
public assets: RichPresenceAssets | null;
|
||||||
|
public buttons: string[];
|
||||||
|
public details: string | null;
|
||||||
|
public name: string;
|
||||||
|
public party: {
|
||||||
|
id: string | null;
|
||||||
|
size: [number, number];
|
||||||
|
} | null;
|
||||||
|
public state: string | null;
|
||||||
|
public timestamps: {
|
||||||
|
start: Date | null;
|
||||||
|
end: Date | null;
|
||||||
|
} | null;
|
||||||
|
public type: ActivityType;
|
||||||
|
public url: string | null;
|
||||||
|
public setAssetsLargeImage(image?: Snowflake): this;
|
||||||
|
public setAssetsLargeText(text?: string): this;
|
||||||
|
public setAssetsSmallImage(image?: Snowflake): this;
|
||||||
|
public setAssetsSmallText(text?: string): this;
|
||||||
|
public setName(name?: string): this;
|
||||||
|
public setURL(url?: string): this;
|
||||||
|
public setType(type?: ActivityType): this;
|
||||||
|
public setApplicationId(id?: Snowflake): this;
|
||||||
|
public setDetails(details?: string): this;
|
||||||
|
public setState(state?: string): this;
|
||||||
|
public setParty(party?: { max: number; current: number; id?: string }): this;
|
||||||
|
public setStartTimestamp(timestamp?: Date): this;
|
||||||
|
public setEndTimestamp(timestamp?: Date): this;
|
||||||
|
public setButtons(...button: RichButton[]): this;
|
||||||
|
public addButton(name: string, url: string): this;
|
||||||
|
public toJSON(): object;
|
||||||
|
}
|
||||||
|
|
||||||
|
export abstract class SpotifyRPC extends RichPresence {
|
||||||
|
public constructor(client: Client, data?: object);
|
||||||
|
public application_id: Snowflake | null;
|
||||||
|
public client: Client;
|
||||||
|
public assets: RichPresenceAssets | null;
|
||||||
|
public buttons: string[];
|
||||||
|
public details: string | null;
|
||||||
|
public name: string;
|
||||||
|
public sync_id: string;
|
||||||
|
public id: string;
|
||||||
|
public created_at: Date;
|
||||||
|
public flags: number;
|
||||||
|
public secrets: {
|
||||||
|
join: string;
|
||||||
|
spectate: string;
|
||||||
|
match: string;
|
||||||
|
};
|
||||||
|
public session_id: string;
|
||||||
|
public party: {
|
||||||
|
id: string | null;
|
||||||
|
size: [number, number];
|
||||||
|
} | null;
|
||||||
|
public state: string | null;
|
||||||
|
public timestamps: {
|
||||||
|
start: Date | null;
|
||||||
|
end: Date | null;
|
||||||
|
} | null;
|
||||||
|
public type: ActivityType;
|
||||||
|
public url: string | null;
|
||||||
|
public setAssetsLargeImage(image?: string): this;
|
||||||
|
public setAssetsSmallImage(image?: string): this;
|
||||||
|
public setSongId(id: string): this;
|
||||||
|
}
|
||||||
|
|
||||||
|
export abstract class CustomStatus {
|
||||||
|
public constructor(data?: object);
|
||||||
|
public emoji: EmojiIdentifierResolvable;
|
||||||
|
public state: string;
|
||||||
|
public setEmoji(emoji?: EmojiIdentifierResolvable): this;
|
||||||
|
public setState(state: string): this;
|
||||||
|
public toJSON(): object;
|
||||||
|
}
|
||||||
|
|
||||||
export class Activity {
|
export class Activity {
|
||||||
private constructor(presence: Presence, data?: RawActivityData);
|
private constructor(presence: Presence, data?: RawActivityData);
|
||||||
public applicationId: Snowflake | null;
|
public applicationId: Snowflake | null;
|
||||||
|
Loading…
Reference in New Issue
Block a user