aboutsummaryrefslogtreecommitdiff
path: root/src/commands/utilities/activity.ts
blob: 3f17d0aa27951414bdf245d47019597ea4d31749 (plain)
1
2
3
4
5
6
7
8
9
10
11
12
13
14
15
16
17
18
19
20
21
22
23
24
25
26
27
28
29
30
31
32
33
34
35
36
37
38
39
40
41
42
43
44
45
46
47
48
49
50
51
52
53
54
55
56
57
58
59
60
61
62
63
64
65
66
67
68
69
70
71
72
73
74
75
76
77
78
79
80
81
82
83
84
85
86
87
88
89
90
91
92
93
94
95
96
97
98
99
100
101
102
103
104
105
106
107
108
109
110
111
112
113
114
115
116
117
118
119
120
121
122
123
124
125
126
127
128
129
130
131
132
133
134
135
136
137
138
139
140
141
142
143
144
145
146
147
148
149
150
151
152
153
154
155
156
157
158
159
160
161
162
163
164
165
166
167
168
169
170
171
172
173
174
175
176
177
178
179
180
181
182
183
184
185
186
187
188
189
190
191
192
193
194
195
import {
	BotCommand,
	emojis,
	regex,
	type ArgType,
	type BotArgumentTypeCaster,
	type CommandMessage,
	type SlashMessage
} from '#lib';
import { type ArgumentGeneratorReturn, type ArgumentTypeCaster } from '@notenoughupdates/discord-akairo';
import { ApplicationCommandOptionType, ChannelType, type DiscordAPIError, type Snowflake } from 'discord.js';

const activityMap = {
	'Poker Night': {
		id: '755827207812677713',
		aliases: ['poker']
	},
	'Betrayal.io': {
		id: '773336526917861400',
		aliases: ['betrayal']
	},
	'Fishington.io': {
		id: '814288819477020702',
		aliases: ['fish', 'fishing', 'fishington']
	},
	'YouTube Together': {
		id: '755600276941176913',
		aliases: ['youtube-together']
	},
	'Chess In The Park': {
		id: '832012774040141894',
		aliases: ['chess']
	},
	'Watch Together': {
		id: '880218394199220334',
		aliases: ['watch-together', 'yt', 'youtube']
	},
	'Doodle Crew': {
		id: '878067389634314250',
		aliases: ['doodle-crew', 'doodle']
	},
	'Wood Snacks': {
		id: '879863976006127627',
		aliases: ['wood-snacks', 'wood']
	},
	'Letter Tile': {
		id: '879863686565621790',
		aliases: ['letter-tile', 'letter']
	},
	'Spell Cast': {
		id: '852509694341283871',
		aliases: ['spell-cast', 'spell', 'cast']
	},
	'Checkers In The Park': {
		id: '832013003968348200',
		aliases: ['checkers']
	}
};

interface Activity {
	id: Snowflake;
	aliases: string[];
}

function map(phase: string): Activity | null {
	if (regex.snowflake.test(phase)) return { id: phase, aliases: [] };
	else if (phase in activityMap) return activityMap[phase as keyof typeof activityMap];

	for (const activity in activityMap) {
		if (activityMap[activity as keyof typeof activityMap].aliases.includes(phase.toLowerCase()))
			return activityMap[activity as keyof typeof activityMap];
	}

	return null;
}

const activityTypeCaster: BotArgumentTypeCaster<Snowflake | null> = (message: CommandMessage, phrase: string) => {
	const parsedPhrase = phrase ?? message.util.parsed?.alias !== 'activity' ? message.util.parsed?.alias : undefined;
	if (!parsedPhrase) return null;
	const mappedPhrase = map(parsedPhrase)?.id;
	return mappedPhrase ?? null;
};

export default class ActivityCommand extends BotCommand {
	public constructor() {
		super('activity', {
			aliases: ['activity', ...Object.values(activityMap).flatMap((a) => a.aliases)],
			category: 'utilities',
			description: 'Allows you to play discord activities in voice channels.',
			usage: [
				`activity <channel> <${Object.values(activityMap)
					.flatMap((a) => a.aliases)
					.map((a) => `'${a}'`)
					.join('|')}>`,
				'yt <channel>' // you do not need to specify the activity if you use its alias.
			],
			examples: ['yt 785281831788216364', 'activity 785281831788216364 yt'],
			args: [
				{
					id: 'channel',
					description: 'The channel to create the activity in.',
					type: 'voiceChannel',
					prompt: 'What channel would you like to use?',
					slashType: ApplicationCommandOptionType.Channel,
					channelTypes: [ChannelType.GuildVoice],
					only: 'slash'
				},
				{
					id: 'activity',
					description: 'The activity to create an invite for.',
					match: 'rest',
					type: activityTypeCaster,
					prompt: 'What activity would you like to play?',
					retry: `{error} You must choose one of the following options: ${Object.values(activityMap)
						.flatMap((a) => a.aliases)
						.map((a) => `\`${a}\``)
						.join(', ')}.`,
					slashType: ApplicationCommandOptionType.String,
					choices: Object.keys(activityMap).map((key) => ({
						name: key,
						value: activityMap[key as keyof typeof activityMap].id
					}))
				}
			],
			slash: true,
			clientPermissions: [],
			userPermissions: []
		});
	}

	public override *args(message: CommandMessage): ArgumentGeneratorReturn {
		const channel: ArgType<'voiceChannel'> = yield {
			id: 'channel',
			description: 'The channel to create the activity in.',
			type: 'voiceChannel',
			prompt: {
				start: 'What channel would you like to use?',
				retry: '{error} Choose a valid voice channel'
			}
		};

		const activity: string = yield {
			id: 'activity',
			description: 'The activity to create an invite for.',
			match: 'rest',
			type: <ArgumentTypeCaster>activityTypeCaster,
			prompt: {
				start: 'What activity would you like to play?',
				retry: `{error} You must choose one of the following options: ${Object.values(activityMap)
					.flatMap((a) => a.aliases)
					.map((a) => `\`${a}\``)
					.join(', ')}.`,
				optional: !!(message.util.parsed && message.util.parsed?.alias !== 'activity')
			},
			default: message.util.parsed?.alias !== 'activity' ? message.util.parsed?.alias : undefined
		};

		return { channel, activity };
	}

	public override async exec(
		message: CommandMessage | SlashMessage,
		args: { channel: ArgType<'voiceChannel'>; activity: string }
	) {
		const channel = typeof args.channel === 'string' ? message.guild?.channels.cache.get(args.channel) : args.channel;
		if (channel?.type !== ChannelType.GuildVoice) return await message.util.reply(`${emojis.error} Choose a valid voice channel`);

		const target_application_id = message.util.isSlashMessage(message)
			? args.activity
			: activityTypeCaster(message, args.activity);

		let response: string;
		const invite: any = await this.client.rest
			.post(`/channels/${channel.id}/invites`, {
				body: {
					validate: null,
					max_age: 604800,
					max_uses: 0,
					target_type: 2,
					target_application_id,
					temporary: false
				}
			})

			.catch((e: Error | DiscordAPIError) => {
				if ((e as DiscordAPIError)?.code === 50013) {
					response = `${emojis.error} I am missing permissions to make an invite in that channel.`;
					return;
				} else response = `${emojis.error} An error occurred while generating your invite: ${e?.message ?? e}`;
			});
		if (response! || !invite || !invite.code)
			return await message.util.reply(response! ?? `${emojis.error} An unknown error occurred while generating your invite.`);
		else return await message.util.send(`https://discord.gg/${invite.code}`);
	}
}