Vencord/src/plugins/moyai.ts

167 lines
5.2 KiB
TypeScript
Raw Normal View History

2022-10-21 23:17:06 +00:00
/*
* Vencord, a modification for Discord's desktop app
* Copyright (c) 2022 Vendicated and contributors
*
* This program is free software: you can redistribute it and/or modify
* it under the terms of the GNU General Public License as published by
* the Free Software Foundation, either version 3 of the License, or
* (at your option) any later version.
*
* This program is distributed in the hope that it will be useful,
* but WITHOUT ANY WARRANTY; without even the implied warranty of
* MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the
* GNU General Public License for more details.
*
* You should have received a copy of the GNU General Public License
* along with this program. If not, see <https://www.gnu.org/licenses/>.
*/
import { Settings } from "@api/settings";
import { makeRange } from "@components/PluginSettings/components/SettingSliderComponent";
import { Devs } from "@utils/constants";
import { sleep } from "@utils/misc";
import definePlugin, { OptionType } from "@utils/types";
import { FluxDispatcher, SelectedChannelStore, UserStore } from "@webpack/common";
2022-10-09 17:48:22 +00:00
import { Message, ReactionEmoji } from "discord-types/general";
2022-10-09 16:18:18 +00:00
interface IMessageCreate {
type: "MESSAGE_CREATE";
optimistic: boolean;
2022-10-09 17:48:22 +00:00
isPushNotification: boolean;
channelId: string;
2022-10-09 16:18:18 +00:00
message: Message;
}
2022-10-09 17:48:22 +00:00
interface IReactionAdd {
type: "MESSAGE_REACTION_ADD";
optimistic: boolean;
channelId: string;
messageId: string;
userId: "195136840355807232";
emoji: ReactionEmoji;
}
interface IVoiceChannelEffectSendEvent {
type: string;
emoji?: ReactionEmoji; // Just in case...
channelId: string;
userId: string;
animationType: number;
animationId: number;
}
2022-10-09 17:48:22 +00:00
const MOYAI = "🗿";
2022-10-09 16:18:18 +00:00
const MOYAI_URL =
"https://raw.githubusercontent.com/MeguminSama/VencordPlugins/main/plugins/moyai/moyai.mp3";
2022-10-09 16:18:18 +00:00
export default definePlugin({
name: "Moyai",
authors: [Devs.Megu, Devs.Nuckyz],
2022-10-09 16:18:18 +00:00
description: "🗿🗿🗿🗿🗿🗿🗿🗿",
2022-10-09 17:48:22 +00:00
async onMessage(e: IMessageCreate) {
if (e.optimistic || e.type !== "MESSAGE_CREATE") return;
if (e.message.state === "SENDING") return;
if (Settings.plugins.Moyai.ignoreBots && e.message.author?.bot) return;
2022-10-09 17:48:22 +00:00
if (!e.message.content) return;
if (e.channelId !== SelectedChannelStore.getChannelId()) return;
2022-10-09 16:18:18 +00:00
2022-10-09 17:48:22 +00:00
const moyaiCount = getMoyaiCount(e.message.content);
2022-10-09 16:18:18 +00:00
for (let i = 0; i < moyaiCount; i++) {
2022-10-09 17:48:22 +00:00
boom();
await sleep(300);
2022-10-09 16:18:18 +00:00
}
},
2022-10-09 17:48:22 +00:00
onReaction(e: IReactionAdd) {
if (e.optimistic || e.type !== "MESSAGE_REACTION_ADD") return;
if (Settings.plugins.Moyai.ignoreBots && UserStore.getUser(e.userId)?.bot) return;
2022-10-09 17:48:22 +00:00
if (e.channelId !== SelectedChannelStore.getChannelId()) return;
const name = e.emoji.name.toLowerCase();
if (name !== MOYAI && !name.includes("moyai") && !name.includes("moai")) return;
boom();
},
onVoiceChannelEffect(e: IVoiceChannelEffectSendEvent) {
if (!e.emoji?.name) return;
const name = e.emoji.name.toLowerCase();
if (name !== MOYAI && !name.includes("moyai") && !name.includes("moai")) return;
boom();
},
2022-10-09 16:27:50 +00:00
start() {
2022-10-09 17:48:22 +00:00
FluxDispatcher.subscribe("MESSAGE_CREATE", this.onMessage);
FluxDispatcher.subscribe("MESSAGE_REACTION_ADD", this.onReaction);
FluxDispatcher.subscribe("VOICE_CHANNEL_EFFECT_SEND", this.onVoiceChannelEffect);
2022-10-09 16:27:50 +00:00
},
2022-10-09 17:48:22 +00:00
2022-10-09 16:27:50 +00:00
stop() {
2022-10-09 17:48:22 +00:00
FluxDispatcher.unsubscribe("MESSAGE_CREATE", this.onMessage);
FluxDispatcher.unsubscribe("MESSAGE_REACTION_ADD", this.onReaction);
FluxDispatcher.unsubscribe("VOICE_CHANNEL_EFFECT_SEND", this.onVoiceChannelEffect);
2022-10-09 17:48:22 +00:00
},
options: {
volume: {
description: "Volume of the 🗿🗿🗿",
type: OptionType.SLIDER,
markers: makeRange(0, 1, 0.1),
default: 0.5,
stickToMarkers: false,
},
triggerWhenUnfocused: {
description: "Trigger the 🗿 even when the window is unfocused",
type: OptionType.BOOLEAN,
default: true,
restartNeeded: false,
},
ignoreBots: {
description: "Ignore bots",
type: OptionType.BOOLEAN,
default: true,
restartNeeded: false,
}
}
2022-10-09 16:18:18 +00:00
});
2022-10-09 17:48:22 +00:00
function countOccurrences(sourceString: string, subString: string) {
let i = 0;
let lastIdx = 0;
while ((lastIdx = sourceString.indexOf(subString, lastIdx) + 1) !== 0)
i++;
return i;
}
2022-10-09 16:18:18 +00:00
2022-10-09 17:48:22 +00:00
function countMatches(sourceString: string, pattern: RegExp) {
if (!pattern.global)
throw new Error("pattern must be global");
2022-10-09 16:18:18 +00:00
2022-10-09 17:48:22 +00:00
let i = 0;
while (pattern.test(sourceString))
i++;
2022-10-09 16:18:18 +00:00
2022-10-09 17:48:22 +00:00
return i;
}
2022-10-09 16:18:18 +00:00
2022-10-09 17:48:22 +00:00
const customMoyaiRe = /<a?:\w*moy?ai\w*:\d{17,20}>/gi;
function getMoyaiCount(message: string) {
2022-10-20 09:58:20 +00:00
const count = countOccurrences(message, MOYAI)
2022-10-09 17:48:22 +00:00
+ countMatches(message, customMoyaiRe);
return Math.min(count, 10);
}
2022-10-09 16:18:18 +00:00
2022-10-09 17:48:22 +00:00
function boom() {
if (!Settings.plugins.Moyai.triggerWhenUnfocused && !document.hasFocus()) return;
2022-10-09 17:48:22 +00:00
const audioElement = document.createElement("audio");
audioElement.src = MOYAI_URL;
audioElement.volume = Settings.plugins.Moyai.volume;
2022-10-09 17:48:22 +00:00
audioElement.play();
2022-10-09 16:18:18 +00:00
}