mirror of
https://github.com/Equicord/Equicord.git
synced 2025-06-07 13:43:03 -04:00
Merge branch 'dev' into main
This commit is contained in:
commit
1c1654738f
11 changed files with 186 additions and 730 deletions
|
@ -106,6 +106,7 @@ You can join our [discord server](https://discord.gg/5Xh2W87egW) for commits, ch
|
|||
- Meow by Samwich
|
||||
- MessageBurst by port
|
||||
- MessageColors by Hen
|
||||
- MessageFetchTimer by GroupXyz
|
||||
- MessageLinkTooltip by Kyuuhachi
|
||||
- MessageLoggerEnhanced by Aria
|
||||
- MessageTranslate by Samwich
|
||||
|
|
|
@ -236,7 +236,7 @@ export function migratePluginSettings(name: string, ...oldNames: string[]) {
|
|||
}
|
||||
}
|
||||
|
||||
export function migratePluginSetting(pluginName: string, oldSetting: string, newSetting: string) {
|
||||
export function migratePluginSetting(pluginName: string, newSetting: string, oldSetting: string) {
|
||||
const settings = SettingsStore.plain.plugins[pluginName];
|
||||
if (!settings) return;
|
||||
|
||||
|
|
|
@ -30,7 +30,7 @@ export const settings = definePluginSettings({
|
|||
},
|
||||
showInMiniProfile: {
|
||||
type: OptionType.BOOLEAN,
|
||||
description: "Only show a button in the mini profile",
|
||||
description: "Show full ui in the mini profile instead of just a button",
|
||||
default: true
|
||||
},
|
||||
});
|
||||
|
|
170
src/equicordplugins/messageFetchTimer/index.tsx
Normal file
170
src/equicordplugins/messageFetchTimer/index.tsx
Normal file
|
@ -0,0 +1,170 @@
|
|||
/*
|
||||
* Vencord, a Discord client mod
|
||||
* Copyright (c) 2025 Vendicated and contributors
|
||||
* SPDX-License-Identifier: GPL-3.0-or-later
|
||||
*/
|
||||
|
||||
import { ChatBarButton, ChatBarButtonFactory } from "@api/ChatButtons";
|
||||
import { definePluginSettings } from "@api/Settings";
|
||||
import { EquicordDevs } from "@utils/constants";
|
||||
import { getCurrentChannel } from "@utils/discord";
|
||||
import definePlugin, { OptionType } from "@utils/types";
|
||||
import { FluxDispatcher, React } from "@webpack/common";
|
||||
|
||||
interface FetchTiming {
|
||||
channelId: string;
|
||||
startTime: number;
|
||||
endTime?: number;
|
||||
duration?: number;
|
||||
timestamp?: Date;
|
||||
}
|
||||
|
||||
let currentFetch: FetchTiming | null = null;
|
||||
let currentChannelId: string | null = null;
|
||||
const channelTimings: Map<string, { time: number; timestamp: Date; }> = new Map();
|
||||
|
||||
const settings = definePluginSettings({
|
||||
showIcon: {
|
||||
type: OptionType.BOOLEAN,
|
||||
description: "Show fetch time icon in message bar",
|
||||
default: true,
|
||||
},
|
||||
showMs: {
|
||||
type: OptionType.BOOLEAN,
|
||||
description: "Show milliseconds in timing",
|
||||
default: true,
|
||||
},
|
||||
iconColor: {
|
||||
type: OptionType.STRING,
|
||||
description: "Icon color (CSS color value)",
|
||||
default: "#00d166",
|
||||
}
|
||||
});
|
||||
|
||||
const FetchTimeButton: ChatBarButtonFactory = ({ isMainChat }) => {
|
||||
const { showMs, iconColor } = settings.use(["showMs", "iconColor"]);
|
||||
|
||||
if (!isMainChat || !settings.store.showIcon || !currentChannelId) {
|
||||
return null;
|
||||
}
|
||||
|
||||
const channelData = channelTimings.get(currentChannelId);
|
||||
if (!channelData) {
|
||||
return null;
|
||||
}
|
||||
|
||||
const { time, timestamp } = channelData;
|
||||
const displayTime = showMs ? `${Math.round(time)}ms` : `${Math.round(time / 1000)}s`;
|
||||
|
||||
if (!showMs && Math.round(time / 1000) === 0) {
|
||||
return null;
|
||||
}
|
||||
|
||||
const timeAgo = formatTimeAgo(timestamp);
|
||||
|
||||
return (
|
||||
<ChatBarButton
|
||||
tooltip={`Messages loaded in ${Math.round(time)}ms (${timeAgo})`}
|
||||
onClick={() => { }}
|
||||
>
|
||||
<div style={{
|
||||
display: "flex",
|
||||
alignItems: "center",
|
||||
gap: "4px"
|
||||
}}>
|
||||
<svg
|
||||
width="16"
|
||||
height="16"
|
||||
viewBox="0 0 24 24"
|
||||
fill="currentColor"
|
||||
>
|
||||
<path d="M12,2A10,10 0 0,0 2,12A10,10 0 0,0 12,22A10,10 0 0,0 22,12A10,10 0 0,0 12,2M12,4A8,8 0 0,1 20,12A8,8 0 0,1 12,20A8,8 0 0,1 4,12A8,8 0 0,1 12,4M12.5,7V12.25L17,14.92L16.25,16.15L11,13V7H12.5Z" />
|
||||
</svg>
|
||||
<span style={{
|
||||
fontSize: "12px",
|
||||
color: iconColor,
|
||||
fontWeight: "500"
|
||||
}}>
|
||||
{displayTime}
|
||||
</span>
|
||||
</div>
|
||||
</ChatBarButton>
|
||||
);
|
||||
};
|
||||
|
||||
function formatTimeAgo(timestamp: Date): string {
|
||||
const now = new Date();
|
||||
const diff = now.getTime() - timestamp.getTime();
|
||||
const seconds = Math.floor(diff / 1000);
|
||||
const minutes = Math.floor(seconds / 60);
|
||||
const hours = Math.floor(minutes / 60);
|
||||
const days = Math.floor(hours / 24);
|
||||
|
||||
if (days > 0) {
|
||||
return `${days} day${days > 1 ? "s" : ""} ago`;
|
||||
} else if (hours > 0) {
|
||||
return `${hours} hour${hours > 1 ? "s" : ""} ago`;
|
||||
} else if (minutes > 0) {
|
||||
return `${minutes} minute${minutes > 1 ? "s" : ""} ago`;
|
||||
} else {
|
||||
return "just now";
|
||||
}
|
||||
}
|
||||
|
||||
function handleChannelSelect(data: any) {
|
||||
if (data.channelId && data.channelId !== currentChannelId) {
|
||||
currentChannelId = data.channelId;
|
||||
currentFetch = {
|
||||
channelId: data.channelId,
|
||||
startTime: performance.now()
|
||||
};
|
||||
}
|
||||
}
|
||||
|
||||
function handleMessageLoad(data: any) {
|
||||
if (!currentFetch || data.channelId !== currentFetch.channelId) return;
|
||||
|
||||
const existing = channelTimings.get(currentFetch.channelId);
|
||||
if (existing) return;
|
||||
|
||||
const endTime = performance.now();
|
||||
const duration = endTime - currentFetch.startTime;
|
||||
|
||||
channelTimings.set(currentFetch.channelId, {
|
||||
time: duration,
|
||||
timestamp: new Date()
|
||||
});
|
||||
|
||||
currentFetch = null;
|
||||
}
|
||||
|
||||
|
||||
export default definePlugin({
|
||||
name: "MessageFetchTimer",
|
||||
description: "Shows how long it took to fetch messages for the current channel",
|
||||
authors: [EquicordDevs.GroupXyz],
|
||||
settings,
|
||||
|
||||
start() {
|
||||
FluxDispatcher.subscribe("CHANNEL_SELECT", handleChannelSelect);
|
||||
FluxDispatcher.subscribe("LOAD_MESSAGES_SUCCESS", handleMessageLoad);
|
||||
FluxDispatcher.subscribe("MESSAGE_CREATE", handleMessageLoad);
|
||||
|
||||
const currentChannel = getCurrentChannel();
|
||||
if (currentChannel) {
|
||||
currentChannelId = currentChannel.id;
|
||||
}
|
||||
},
|
||||
|
||||
stop() {
|
||||
FluxDispatcher.unsubscribe("CHANNEL_SELECT", handleChannelSelect);
|
||||
FluxDispatcher.unsubscribe("LOAD_MESSAGES_SUCCESS", handleMessageLoad);
|
||||
FluxDispatcher.unsubscribe("MESSAGE_CREATE", handleMessageLoad);
|
||||
|
||||
currentFetch = null;
|
||||
channelTimings.clear();
|
||||
currentChannelId = null;
|
||||
},
|
||||
|
||||
renderChatBarButton: FetchTimeButton,
|
||||
});
|
|
@ -1,332 +0,0 @@
|
|||
/*
|
||||
* Vencord, a modification for Discord's desktop app
|
||||
* Copyright (c) 2022 Vendicated, Samu 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 { ApplicationCommandInputType, ApplicationCommandOptionType, findOption, OptionalMessageOption, RequiredMessageOption, sendBotMessage } from "@api/Commands";
|
||||
import { Devs } from "@utils/constants";
|
||||
import definePlugin from "@utils/types";
|
||||
|
||||
|
||||
function mock(input: string): string {
|
||||
let output = "";
|
||||
for (let i = 0; i < input.length; i++) {
|
||||
output += i % 2 ? input[i].toUpperCase() : input[i].toLowerCase();
|
||||
}
|
||||
return output;
|
||||
}
|
||||
|
||||
export default definePlugin({
|
||||
name: "MoreCommands",
|
||||
description: "Echo, Lenny, Mock, and More",
|
||||
authors: [Devs.Arjix, Devs.echo, Devs.Samu],
|
||||
commands: [
|
||||
{
|
||||
name: "echo",
|
||||
description: "Sends a message as Clyde (locally)",
|
||||
options: [OptionalMessageOption],
|
||||
inputType: ApplicationCommandInputType.BOT,
|
||||
execute: (opts, ctx) => {
|
||||
const content = findOption(opts, "message", "");
|
||||
|
||||
sendBotMessage(ctx.channel.id, { content });
|
||||
},
|
||||
},
|
||||
{
|
||||
name: "lenny",
|
||||
description: "Sends a lenny face",
|
||||
options: [OptionalMessageOption],
|
||||
execute: opts => ({
|
||||
content: findOption(opts, "message", "") + " ( ͡° ͜ʖ ͡°)"
|
||||
}),
|
||||
},
|
||||
{
|
||||
name: "mock",
|
||||
description: "mOcK PeOpLe",
|
||||
options: [RequiredMessageOption],
|
||||
execute: opts => ({
|
||||
content: mock(findOption(opts, "message", ""))
|
||||
}),
|
||||
},
|
||||
{
|
||||
name: "reverse",
|
||||
description: "Reverses the input message",
|
||||
options: [RequiredMessageOption],
|
||||
execute: opts => ({
|
||||
content: findOption(opts, "message", "").split("").reverse().join("")
|
||||
}),
|
||||
},
|
||||
{
|
||||
name: "uppercase",
|
||||
description: "Converts the message to uppercase",
|
||||
options: [RequiredMessageOption],
|
||||
execute: opts => ({
|
||||
content: findOption(opts, "message", "").toUpperCase()
|
||||
}),
|
||||
},
|
||||
{
|
||||
name: "lowercase",
|
||||
description: "Converts the message to lowercase",
|
||||
options: [RequiredMessageOption],
|
||||
execute: opts => ({
|
||||
content: findOption(opts, "message", "").toLowerCase()
|
||||
}),
|
||||
},
|
||||
{
|
||||
name: "wordcount",
|
||||
description: "Counts the number of words in a message",
|
||||
options: [RequiredMessageOption],
|
||||
inputType: ApplicationCommandInputType.BOT,
|
||||
execute: (opts, ctx) => {
|
||||
const message = findOption(opts, "message", "");
|
||||
const wordCount = message.trim().split(/\s+/).length;
|
||||
sendBotMessage(ctx.channel.id, {
|
||||
content: `The message contains ${wordCount} words.`
|
||||
});
|
||||
},
|
||||
},
|
||||
{
|
||||
name: "ping",
|
||||
description: "Pings the bot to check if it's responding",
|
||||
options: [],
|
||||
inputType: ApplicationCommandInputType.BOT,
|
||||
execute: (opts, ctx) => {
|
||||
sendBotMessage(ctx.channel.id, {
|
||||
content: "Pong!"
|
||||
});
|
||||
},
|
||||
},
|
||||
{
|
||||
name: "rolldice",
|
||||
description: "Roll a die with the specified number of sides",
|
||||
options: [RequiredMessageOption],
|
||||
execute: opts => {
|
||||
const sides = parseInt(findOption(opts, "message", "6"));
|
||||
const roll = Math.floor(Math.random() * sides) + 1;
|
||||
return {
|
||||
content: `You rolled a ${roll}!`
|
||||
};
|
||||
},
|
||||
},
|
||||
{
|
||||
name: "flipcoin",
|
||||
description: "Flips a coin and returns heads or tails",
|
||||
options: [],
|
||||
execute: (opts, ctx) => {
|
||||
const flip = Math.random() < 0.5 ? "Heads" : "Tails";
|
||||
return {
|
||||
content: `The coin landed on: ${flip}`
|
||||
};
|
||||
},
|
||||
},
|
||||
{
|
||||
name: "ask",
|
||||
description: "Ask a yes/no question and get an answer",
|
||||
options: [RequiredMessageOption],
|
||||
execute: opts => {
|
||||
const question = findOption(opts, "message", "");
|
||||
const responses = [
|
||||
"Yes", "No", "Maybe", "Ask again later", "Definitely not", "It is certain"
|
||||
];
|
||||
const response = responses[Math.floor(Math.random() * responses.length)];
|
||||
return {
|
||||
content: `${question} - ${response}`
|
||||
};
|
||||
},
|
||||
},
|
||||
{
|
||||
name: "randomcat",
|
||||
description: "Get a random cat picture",
|
||||
options: [],
|
||||
execute: (opts, ctx) => {
|
||||
return (async () => {
|
||||
try {
|
||||
const response = await fetch("https://api.thecatapi.com/v1/images/search");
|
||||
if (!response.ok) throw new Error("Failed to fetch cat image");
|
||||
const data = await response.json();
|
||||
return {
|
||||
content: data[0].url
|
||||
};
|
||||
} catch (err) {
|
||||
sendBotMessage(ctx.channel.id, {
|
||||
content: "Sorry, couldn't fetch a cat picture right now 😿"
|
||||
});
|
||||
}
|
||||
})();
|
||||
},
|
||||
},
|
||||
{
|
||||
name: "randomdog",
|
||||
description: "Get a random ddog picture",
|
||||
options: [],
|
||||
execute: (opts, ctx) => {
|
||||
return (async () => {
|
||||
try {
|
||||
const response = await fetch("https://api.thedogapi.com/v1/images/search");
|
||||
if (!response.ok) throw new Error("Failed to fetch dog image");
|
||||
const data = await response.json();
|
||||
return {
|
||||
content: data[0].url
|
||||
};
|
||||
} catch (err) {
|
||||
sendBotMessage(ctx.channel.id, {
|
||||
content: "Sorry, couldn't fetch a cat picture right now 🐶"
|
||||
});
|
||||
}
|
||||
})();
|
||||
},
|
||||
},
|
||||
{
|
||||
name: "randomnumber",
|
||||
description: "Generates a random number between two values",
|
||||
options: [
|
||||
{
|
||||
name: "min",
|
||||
description: "Minimum value",
|
||||
type: ApplicationCommandOptionType.INTEGER,
|
||||
required: true
|
||||
},
|
||||
{
|
||||
name: "max",
|
||||
description: "Maximum value",
|
||||
type: ApplicationCommandOptionType.INTEGER,
|
||||
required: true
|
||||
}
|
||||
],
|
||||
execute: opts => {
|
||||
const min = parseInt(findOption(opts, "min", "0"));
|
||||
const max = parseInt(findOption(opts, "max", "100"));
|
||||
const number = Math.floor(Math.random() * (max - min + 1)) + min;
|
||||
return {
|
||||
content: `Random number between ${min} and ${max}: ${number}`
|
||||
};
|
||||
}
|
||||
},
|
||||
{
|
||||
name: "countdown",
|
||||
description: "Starts a countdown from a specified number",
|
||||
options: [
|
||||
{
|
||||
name: "number",
|
||||
description: "Number to countdown from (max 10)",
|
||||
type: ApplicationCommandOptionType.INTEGER,
|
||||
required: true
|
||||
}
|
||||
],
|
||||
inputType: ApplicationCommandInputType.BOT,
|
||||
execute: async (opts, ctx) => {
|
||||
const number = Math.min(parseInt(findOption(opts, "number", "5")), 10);
|
||||
if (isNaN(number) || number < 1) {
|
||||
sendBotMessage(ctx.channel.id, {
|
||||
content: "Please provide a valid number between 1 and 10!"
|
||||
});
|
||||
return;
|
||||
}
|
||||
sendBotMessage(ctx.channel.id, {
|
||||
content: `Starting countdown from ${number}...`
|
||||
});
|
||||
for (let i = number; i >= 0; i--) {
|
||||
await new Promise(resolve => setTimeout(resolve, 1000));
|
||||
sendBotMessage(ctx.channel.id, {
|
||||
content: i === 0 ? "🎉 Go! 🎉" : `${i}...`
|
||||
});
|
||||
}
|
||||
},
|
||||
},
|
||||
{
|
||||
name: "choose",
|
||||
description: "Randomly chooses from provided options",
|
||||
options: [
|
||||
{
|
||||
name: "choices",
|
||||
description: "Comma-separated list of choices",
|
||||
type: ApplicationCommandOptionType.STRING,
|
||||
required: true
|
||||
}
|
||||
],
|
||||
execute: opts => {
|
||||
const choices = findOption(opts, "choices", "").split(",").map(c => c.trim());
|
||||
const choice = choices[Math.floor(Math.random() * choices.length)];
|
||||
return {
|
||||
content: `I choose: ${choice}`
|
||||
};
|
||||
}
|
||||
},
|
||||
{
|
||||
name: "systeminfo",
|
||||
description: "Shows system information",
|
||||
options: [],
|
||||
execute: async (opts, ctx) => {
|
||||
try {
|
||||
const { userAgent, hardwareConcurrency, onLine, languages } = navigator;
|
||||
const { width, height, colorDepth } = window.screen;
|
||||
const { deviceMemory, connection }: { deviceMemory: any, connection: any; } = navigator as any;
|
||||
const platform = userAgent.includes("Windows") ? "Windows" :
|
||||
userAgent.includes("Mac") ? "MacOS" :
|
||||
userAgent.includes("Linux") ? "Linux" : "Unknown";
|
||||
const isMobile = /Mobile|Android|iPhone/i.test(userAgent);
|
||||
const deviceType = isMobile ? "Mobile" : "Desktop";
|
||||
const browserInfo = userAgent.match(/(?:chrome|firefox|safari|edge|opr)\/?\s*(\d+)/i)?.[0] || "Unknown";
|
||||
const networkInfo = connection ? `${connection.effectiveType || "Unknown"}` : "Unknown";
|
||||
const info = [
|
||||
`> **Platform**: ${platform}`,
|
||||
`> **Device Type**: ${deviceType}`,
|
||||
`> **Browser**: ${browserInfo}`,
|
||||
`> **CPU Cores**: ${hardwareConcurrency || "N/A"}`,
|
||||
`> **Memory**: ${deviceMemory ? `${deviceMemory}GB` : "N/A"}`,
|
||||
`> **Screen**: ${width}x${height} (${colorDepth}bit)`,
|
||||
`> **Languages**: ${languages?.join(", ")}`,
|
||||
`> **Network**: ${networkInfo} (${onLine ? "Online" : "Offline"})`
|
||||
].join("\n");
|
||||
return { content: info };
|
||||
} catch (err) {
|
||||
sendBotMessage(ctx.channel.id, { content: "Failed to fetch system information" });
|
||||
}
|
||||
},
|
||||
},
|
||||
{
|
||||
name: "getUptime",
|
||||
description: "Returns the system uptime",
|
||||
execute: async (opts, ctx) => {
|
||||
const uptime = performance.now() / 1000;
|
||||
const uptimeInfo = `> **System Uptime**: ${Math.floor(uptime / 60)} minutes`;
|
||||
return { content: uptimeInfo };
|
||||
},
|
||||
},
|
||||
{
|
||||
name: "getTime",
|
||||
description: "Returns the current server time",
|
||||
execute: async (opts, ctx) => {
|
||||
const currentTime = new Date().toLocaleString();
|
||||
return { content: `> **Current Time**: ${currentTime}` };
|
||||
},
|
||||
},
|
||||
{
|
||||
name: "getLocation",
|
||||
description: "Returns the user's approximate location based on IP",
|
||||
execute: async (opts, ctx) => {
|
||||
try {
|
||||
const response = await fetch("https://ipapi.co/json/");
|
||||
const data = await response.json();
|
||||
const locationInfo = `> **Country**: ${data.country_name}\n> **Region**: ${data.region}\n> **City**: ${data.city}`;
|
||||
return { content: locationInfo };
|
||||
} catch (err) {
|
||||
sendBotMessage(ctx.channel.id, { content: "Failed to fetch location information" });
|
||||
}
|
||||
},
|
||||
}
|
||||
]
|
||||
});
|
|
@ -1,216 +0,0 @@
|
|||
/*
|
||||
* 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 { findOption, OptionalMessageOption } from "@api/Commands";
|
||||
import { Devs, EquicordDevs } from "@utils/constants";
|
||||
import definePlugin from "@utils/types";
|
||||
|
||||
export default definePlugin({
|
||||
name: "MoreKaomoji",
|
||||
description: "Adds more Kaomoji to discord. ヽ(´▽`)/",
|
||||
authors: [Devs.JacobTm, EquicordDevs.voidbbg],
|
||||
commands: [
|
||||
{
|
||||
name: "dissatisfaction",
|
||||
description: " >﹏<",
|
||||
options: [OptionalMessageOption],
|
||||
execute: opts => ({
|
||||
content: findOption(opts, "message", "") + " " + " >﹏<",
|
||||
}),
|
||||
},
|
||||
{
|
||||
name: "smug",
|
||||
description: "ಠ_ಠ",
|
||||
options: [OptionalMessageOption],
|
||||
execute: opts => ({
|
||||
content: findOption(opts, "message", "") + " " + "ಠ_ಠ",
|
||||
}),
|
||||
},
|
||||
{
|
||||
name: "happy",
|
||||
description: "ヽ(´▽`)/",
|
||||
options: [OptionalMessageOption],
|
||||
execute: opts => ({
|
||||
content: findOption(opts, "message", "") + " " + "ヽ(´▽`)/",
|
||||
}),
|
||||
},
|
||||
{
|
||||
name: "crying",
|
||||
description: "ಥ_ಥ",
|
||||
options: [OptionalMessageOption],
|
||||
execute: opts => ({
|
||||
content: findOption(opts, "message", "") + " " + "ಥ_ಥ",
|
||||
}),
|
||||
},
|
||||
{
|
||||
name: "angry",
|
||||
description: "ヽ(`Д´)ノ",
|
||||
options: [OptionalMessageOption],
|
||||
execute: opts => ({
|
||||
content: findOption(opts, "message", "") + " " + "ヽ(`Д´)ノ",
|
||||
}),
|
||||
},
|
||||
{
|
||||
name: "anger",
|
||||
description: "ヽ(o`皿′o)ノ",
|
||||
options: [OptionalMessageOption],
|
||||
execute: opts => ({
|
||||
content: findOption(opts, "message", "") + " " + "ヽ(o`皿′o)ノ",
|
||||
}),
|
||||
},
|
||||
{
|
||||
name: "joy",
|
||||
description: "<( ̄︶ ̄)>",
|
||||
options: [OptionalMessageOption],
|
||||
execute: opts => ({
|
||||
content: findOption(opts, "message", "") + " " + "<( ̄︶ ̄)>",
|
||||
}),
|
||||
},
|
||||
{
|
||||
name: "blush",
|
||||
description: "૮ ˶ᵔ ᵕ ᵔ˶ ა",
|
||||
options: [OptionalMessageOption],
|
||||
execute: opts => ({
|
||||
content: findOption(opts, "message", "") + " " + "૮ ˶ᵔ ᵕ ᵔ˶ ა",
|
||||
}),
|
||||
},
|
||||
{
|
||||
name: "confused",
|
||||
description: "(•ิ_•ิ)?",
|
||||
options: [OptionalMessageOption],
|
||||
execute: opts => ({
|
||||
content: findOption(opts, "message", "") + " " + "(•ิ_•ิ)?",
|
||||
}),
|
||||
},
|
||||
{
|
||||
name: "sleeping",
|
||||
description: "(ᴗ_ᴗ)",
|
||||
options: [OptionalMessageOption],
|
||||
execute: opts => ({
|
||||
content: findOption(opts, "message", "") + " " + "(ᴗ_ᴗ)",
|
||||
}),
|
||||
},
|
||||
{
|
||||
name: "laughing",
|
||||
description: "o(≧▽≦)o",
|
||||
options: [OptionalMessageOption],
|
||||
execute: opts => ({
|
||||
content: findOption(opts, "message", "") + " " + "o(≧▽≦)o",
|
||||
}),
|
||||
},
|
||||
/*
|
||||
even more kaomoji
|
||||
*/
|
||||
{
|
||||
name: "giving",
|
||||
description: "(ノ◕ヮ◕)ノ*:・゚✧",
|
||||
options: [OptionalMessageOption],
|
||||
execute: opts => ({
|
||||
content: findOption(opts, "message", "") + " " + "(ノ◕ヮ◕)ノ*:・゚✧",
|
||||
}),
|
||||
},
|
||||
{
|
||||
name: "peace",
|
||||
description: "✌(◕‿-)✌",
|
||||
options: [OptionalMessageOption],
|
||||
execute: opts => ({
|
||||
content: findOption(opts, "message", "") + " " + "✌(◕‿-)✌",
|
||||
}),
|
||||
},
|
||||
{
|
||||
name: "ending1",
|
||||
description: "Ꮺ ָ࣪ ۰ ͙⊹",
|
||||
options: [OptionalMessageOption],
|
||||
execute: opts => ({
|
||||
content: findOption(opts, "message", "") + " " + "Ꮺ ָ࣪ ۰ ͙⊹",
|
||||
}),
|
||||
},
|
||||
{
|
||||
name: "uwu",
|
||||
description: "(>⩊<)",
|
||||
options: [OptionalMessageOption],
|
||||
execute: opts => ({
|
||||
content: findOption(opts, "message", "") + " " + "(>⩊<)",
|
||||
}),
|
||||
},
|
||||
{
|
||||
name: "comfy",
|
||||
description: "(─‿‿─)♡",
|
||||
options: [OptionalMessageOption],
|
||||
execute: opts => ({
|
||||
content: findOption(opts, "message", "") + " " + "(─‿‿─)♡",
|
||||
}),
|
||||
},
|
||||
{
|
||||
name: "lovehappy",
|
||||
description: "(*≧ω≦*)",
|
||||
options: [OptionalMessageOption],
|
||||
execute: opts => ({
|
||||
content: findOption(opts, "message", "") + " " + "(*≧ω≦*)",
|
||||
}),
|
||||
},
|
||||
{
|
||||
name: "loveee",
|
||||
description: "(⁄ ⁄>⁄ ▽ ⁄<⁄ ⁄)",
|
||||
options: [OptionalMessageOption],
|
||||
execute: opts => ({
|
||||
content: findOption(opts, "message", "") + " " + "(⁄ ⁄>⁄ ▽ ⁄<⁄ ⁄)",
|
||||
}),
|
||||
},
|
||||
{
|
||||
name: "give",
|
||||
description: "(ノ= ⩊ = )ノ",
|
||||
options: [OptionalMessageOption],
|
||||
execute: opts => ({
|
||||
content: findOption(opts, "message", "") + " " + "(ノ= ⩊ = )ノ",
|
||||
}),
|
||||
},
|
||||
{
|
||||
name: "lovegive",
|
||||
description: "ღゝ◡╹)ノ♡",
|
||||
options: [OptionalMessageOption],
|
||||
execute: opts => ({
|
||||
content: findOption(opts, "message", "") + " " + "ღゝ◡╹)ノ♡",
|
||||
}),
|
||||
},
|
||||
{
|
||||
name: "music",
|
||||
description: "( ̄▽ ̄)/♫•¨•.¸¸♪",
|
||||
options: [OptionalMessageOption],
|
||||
execute: opts => ({
|
||||
content: findOption(opts, "message", "") + " " + "( ̄▽ ̄)/♫•¨•.¸¸♪",
|
||||
}),
|
||||
},
|
||||
{
|
||||
name: "stars",
|
||||
description: ".𖥔 ݁ ˖๋ ࣭ ⭑",
|
||||
options: [OptionalMessageOption],
|
||||
execute: opts => ({
|
||||
content: findOption(opts, "message", "") + " " + ".𖥔 ݁ ˖๋ ࣭ ⭑",
|
||||
}),
|
||||
},
|
||||
{
|
||||
name: "lovegiving",
|
||||
description: "⸜(。˃ ᵕ ˂ )⸝♡",
|
||||
options: [OptionalMessageOption],
|
||||
execute: opts => ({
|
||||
content: findOption(opts, "message", "") + " " + "⸜(。˃ ᵕ ˂ )⸝♡",
|
||||
}),
|
||||
}
|
||||
]
|
||||
});
|
|
@ -1,177 +0,0 @@
|
|||
/*
|
||||
* 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 { definePluginSettings } 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 { RelationshipStore, SelectedChannelStore, UserStore } from "@webpack/common";
|
||||
import { Message, ReactionEmoji } from "discord-types/general";
|
||||
|
||||
interface IMessageCreate {
|
||||
type: "MESSAGE_CREATE";
|
||||
optimistic: boolean;
|
||||
isPushNotification: boolean;
|
||||
channelId: string;
|
||||
message: Message;
|
||||
}
|
||||
|
||||
interface IReactionAdd {
|
||||
type: "MESSAGE_REACTION_ADD";
|
||||
optimistic: boolean;
|
||||
channelId: string;
|
||||
messageId: string;
|
||||
messageAuthorId: string;
|
||||
userId: "195136840355807232";
|
||||
emoji: ReactionEmoji;
|
||||
}
|
||||
|
||||
interface IVoiceChannelEffectSendEvent {
|
||||
type: string;
|
||||
emoji?: ReactionEmoji; // Just in case...
|
||||
channelId: string;
|
||||
userId: string;
|
||||
animationType: number;
|
||||
animationId: number;
|
||||
}
|
||||
|
||||
const MOYAI = "🗿";
|
||||
const MOYAI_URL =
|
||||
"https://github.com/Equicord/Equibored/raw/main/sounds/moyai/moyai.mp3";
|
||||
const MOYAI_URL_HD =
|
||||
"https://github.com/Equicord/Equibored/raw/main/sounds/moyai/moyai.wav";
|
||||
|
||||
const settings = definePluginSettings({
|
||||
volume: {
|
||||
description: "Volume of the 🗿🗿🗿",
|
||||
type: OptionType.SLIDER,
|
||||
markers: makeRange(0, 1, 0.1),
|
||||
default: 0.5,
|
||||
stickToMarkers: false
|
||||
},
|
||||
quality: {
|
||||
description: "Quality of the 🗿🗿🗿",
|
||||
type: OptionType.SELECT,
|
||||
options: [
|
||||
{ label: "Normal", value: "Normal", default: true },
|
||||
{ label: "HD", value: "HD" }
|
||||
],
|
||||
},
|
||||
triggerWhenUnfocused: {
|
||||
description: "Trigger the 🗿 even when the window is unfocused",
|
||||
type: OptionType.BOOLEAN,
|
||||
default: true
|
||||
},
|
||||
ignoreBots: {
|
||||
description: "Ignore bots",
|
||||
type: OptionType.BOOLEAN,
|
||||
default: true
|
||||
},
|
||||
ignoreBlocked: {
|
||||
description: "Ignore blocked users",
|
||||
type: OptionType.BOOLEAN,
|
||||
default: true
|
||||
}
|
||||
});
|
||||
|
||||
export default definePlugin({
|
||||
name: "Moyai",
|
||||
authors: [Devs.Megu, Devs.Nuckyz],
|
||||
description: "🗿🗿🗿🗿🗿🗿🗿🗿",
|
||||
settings,
|
||||
|
||||
flux: {
|
||||
async MESSAGE_CREATE({ optimistic, type, message, channelId }: IMessageCreate) {
|
||||
if (optimistic || type !== "MESSAGE_CREATE") return;
|
||||
if (message.state === "SENDING") return;
|
||||
if (settings.store.ignoreBots && message.author?.bot) return;
|
||||
if (settings.store.ignoreBlocked && RelationshipStore.isBlocked(message.author?.id)) return;
|
||||
if (!message.content) return;
|
||||
if (channelId !== SelectedChannelStore.getChannelId()) return;
|
||||
|
||||
const moyaiCount = getMoyaiCount(message.content);
|
||||
|
||||
for (let i = 0; i < moyaiCount; i++) {
|
||||
boom();
|
||||
await sleep(300);
|
||||
}
|
||||
},
|
||||
|
||||
MESSAGE_REACTION_ADD({ optimistic, type, channelId, userId, messageAuthorId, emoji }: IReactionAdd) {
|
||||
if (optimistic || type !== "MESSAGE_REACTION_ADD") return;
|
||||
if (settings.store.ignoreBots && UserStore.getUser(userId)?.bot) return;
|
||||
if (settings.store.ignoreBlocked && RelationshipStore.isBlocked(messageAuthorId)) return;
|
||||
if (channelId !== SelectedChannelStore.getChannelId()) return;
|
||||
|
||||
const name = emoji.name.toLowerCase();
|
||||
if (name !== MOYAI && !name.includes("moyai") && !name.includes("moai")) return;
|
||||
|
||||
boom();
|
||||
},
|
||||
|
||||
VOICE_CHANNEL_EFFECT_SEND({ emoji }: IVoiceChannelEffectSendEvent) {
|
||||
if (!emoji?.name) return;
|
||||
const name = emoji.name.toLowerCase();
|
||||
if (name !== MOYAI && !name.includes("moyai") && !name.includes("moai")) return;
|
||||
|
||||
boom();
|
||||
}
|
||||
}
|
||||
});
|
||||
|
||||
function countOccurrences(sourceString: string, subString: string) {
|
||||
let i = 0;
|
||||
let lastIdx = 0;
|
||||
while ((lastIdx = sourceString.indexOf(subString, lastIdx) + 1) !== 0)
|
||||
i++;
|
||||
|
||||
return i;
|
||||
}
|
||||
|
||||
function countMatches(sourceString: string, pattern: RegExp) {
|
||||
if (!pattern.global)
|
||||
throw new Error("pattern must be global");
|
||||
|
||||
let i = 0;
|
||||
while (pattern.test(sourceString))
|
||||
i++;
|
||||
|
||||
return i;
|
||||
}
|
||||
|
||||
const customMoyaiRe = /<a?:\w*moy?ai\w*:\d{17,20}>/gi;
|
||||
|
||||
function getMoyaiCount(message: string) {
|
||||
const count = countOccurrences(message, MOYAI)
|
||||
+ countMatches(message, customMoyaiRe);
|
||||
|
||||
return Math.min(count, 10);
|
||||
}
|
||||
|
||||
function boom() {
|
||||
if (!settings.store.triggerWhenUnfocused && !document.hasFocus()) return;
|
||||
const audioElement = document.createElement("audio");
|
||||
|
||||
audioElement.src = settings.store.quality === "HD"
|
||||
? MOYAI_URL_HD
|
||||
: MOYAI_URL;
|
||||
|
||||
audioElement.volume = settings.store.volume;
|
||||
audioElement.play();
|
||||
}
|
|
@ -33,7 +33,7 @@ interface MessageDeleteProps {
|
|||
}
|
||||
|
||||
// Remove this migration once enough time has passed
|
||||
migratePluginSetting("NoBlockedMessages", "ignoreBlockedMessages", "ignoreMessages");
|
||||
migratePluginSetting("NoBlockedMessages", "ignoreMessages", "ignoreBlockedMessages");
|
||||
const settings = definePluginSettings({
|
||||
ignoreMessages: {
|
||||
description: "Completely ignores incoming messages from blocked and ignored (if enabled) users",
|
||||
|
|
|
@ -201,7 +201,7 @@ function toggleMessageDecorators(enabled: boolean) {
|
|||
}
|
||||
}
|
||||
|
||||
migratePluginSetting("PlatformIndicators", "badges", "profiles");
|
||||
migratePluginSetting("PlatformIndicators", "profiles", "badges");
|
||||
const settings = definePluginSettings({
|
||||
list: {
|
||||
type: OptionType.BOOLEAN,
|
||||
|
|
|
@ -151,7 +151,13 @@ function CompactConnectionComponent({ connection, theme }: { connection: Connect
|
|||
: <button
|
||||
{...tooltipProps}
|
||||
className="vc-user-connection"
|
||||
onClick={() => copyWithToast(connection.name)}
|
||||
onClick={() => {
|
||||
if (connection.type === "xbox") {
|
||||
VencordNative.native.openExternal(`https://www.xbox.com/en-US/play/user/${encodeURIComponent(connection.name)}`);
|
||||
} else {
|
||||
copyWithToast(connection.name);
|
||||
}
|
||||
}}
|
||||
>
|
||||
{img}
|
||||
</button>
|
||||
|
|
|
@ -1082,6 +1082,10 @@ export const EquicordDevs = Object.freeze({
|
|||
name: "davidkra230",
|
||||
id: 652699312631054356n,
|
||||
},
|
||||
GroupXyz: {
|
||||
name: "GroupXyz",
|
||||
id: 950033410229944331n
|
||||
},
|
||||
} satisfies Record<string, Dev>);
|
||||
|
||||
// iife so #__PURE__ works correctly
|
||||
|
|
Loading…
Add table
Add a link
Reference in a new issue