aboutsummaryrefslogtreecommitdiff
path: root/src/plugins/textReplace
diff options
context:
space:
mode:
Diffstat (limited to 'src/plugins/textReplace')
-rw-r--r--src/plugins/textReplace/index.tsx267
1 files changed, 267 insertions, 0 deletions
diff --git a/src/plugins/textReplace/index.tsx b/src/plugins/textReplace/index.tsx
new file mode 100644
index 0000000..45fb6f9
--- /dev/null
+++ b/src/plugins/textReplace/index.tsx
@@ -0,0 +1,267 @@
+/*
+ * Vencord, a modification for Discord's desktop app
+ * Copyright (c) 2023 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 { DataStore } from "@api/index";
+import { addPreSendListener, removePreSendListener } from "@api/MessageEvents";
+import { definePluginSettings } from "@api/Settings";
+import { Flex } from "@components/Flex";
+import { Devs } from "@utils/constants";
+import { Logger } from "@utils/Logger";
+import { useForceUpdater } from "@utils/react";
+import definePlugin, { OptionType } from "@utils/types";
+import { Button, Forms, React, TextInput, useState } from "@webpack/common";
+
+const STRING_RULES_KEY = "TextReplace_rulesString";
+const REGEX_RULES_KEY = "TextReplace_rulesRegex";
+
+type Rule = Record<"find" | "replace" | "onlyIfIncludes", string>;
+
+interface TextReplaceProps {
+ title: string;
+ rulesArray: Rule[];
+ rulesKey: string;
+ update: () => void;
+}
+
+const makeEmptyRule: () => Rule = () => ({
+ find: "",
+ replace: "",
+ onlyIfIncludes: ""
+});
+const makeEmptyRuleArray = () => [makeEmptyRule()];
+
+let stringRules = makeEmptyRuleArray();
+let regexRules = makeEmptyRuleArray();
+
+const settings = definePluginSettings({
+ replace: {
+ type: OptionType.COMPONENT,
+ description: "",
+ component: () => {
+ const update = useForceUpdater();
+ return (
+ <>
+ <TextReplace
+ title="Using String"
+ rulesArray={stringRules}
+ rulesKey={STRING_RULES_KEY}
+ update={update}
+ />
+ <TextReplace
+ title="Using Regex"
+ rulesArray={regexRules}
+ rulesKey={REGEX_RULES_KEY}
+ update={update}
+ />
+ <TextReplaceTesting />
+ </>
+ );
+ }
+ },
+});
+
+function stringToRegex(str: string) {
+ const match = str.match(/^(\/)?(.+?)(?:\/([gimsuy]*))?$/); // Regex to match regex
+ return match
+ ? new RegExp(
+ match[2], // Pattern
+ match[3]
+ ?.split("") // Remove duplicate flags
+ .filter((char, pos, flagArr) => flagArr.indexOf(char) === pos)
+ .join("")
+ ?? "g"
+ )
+ : new RegExp(str); // Not a regex, return string
+}
+
+function renderFindError(find: string) {
+ try {
+ stringToRegex(find);
+ return null;
+ } catch (e) {
+ return (
+ <span style={{ color: "var(--text-danger)" }}>
+ {String(e)}
+ </span>
+ );
+ }
+}
+
+function Input({ initialValue, onChange, placeholder }: {
+ placeholder: string;
+ initialValue: string;
+ onChange(value: string): void;
+}) {
+ const [value, setValue] = useState(initialValue);
+ return (
+ <TextInput
+ placeholder={placeholder}
+ value={value}
+ onChange={setValue}
+ spellCheck={false}
+ onBlur={() => value !== initialValue && onChange(value)}
+ />
+ );
+}
+
+function TextReplace({ title, rulesArray, rulesKey, update }: TextReplaceProps) {
+ const isRegexRules = title === "Using Regex";
+
+ async function onClickRemove(index: number) {
+ if (index === rulesArray.length - 1) return;
+ rulesArray.splice(index, 1);
+
+ await DataStore.set(rulesKey, rulesArray);
+ update();
+ }
+
+ async function onChange(e: string, index: number, key: string) {
+ if (index === rulesArray.length - 1)
+ rulesArray.push(makeEmptyRule());
+
+ rulesArray[index][key] = e;
+
+ if (rulesArray[index].find === "" && rulesArray[index].replace === "" && rulesArray[index].onlyIfIncludes === "" && index !== rulesArray.length - 1)
+ rulesArray.splice(index, 1);
+
+ await DataStore.set(rulesKey, rulesArray);
+ update();
+ }
+
+ return (
+ <>
+ <Forms.FormTitle tag="h4">{title}</Forms.FormTitle>
+ <Flex flexDirection="column" style={{ gap: "0.5em" }}>
+ {
+ rulesArray.map((rule, index) =>
+ <React.Fragment key={`${rule.find}-${index}`}>
+ <Flex flexDirection="row" style={{ gap: 0 }}>
+ <Flex flexDirection="row" style={{ flexGrow: 1, gap: "0.5em" }}>
+ <Input
+ placeholder="Find"
+ initialValue={rule.find}
+ onChange={e => onChange(e, index, "find")}
+ />
+ <Input
+ placeholder="Replace"
+ initialValue={rule.replace}
+ onChange={e => onChange(e, index, "replace")}
+ />
+ <Input
+ placeholder="Only if includes"
+ initialValue={rule.onlyIfIncludes}
+ onChange={e => onChange(e, index, "onlyIfIncludes")}
+ />
+ </Flex>
+ <Button
+ size={Button.Sizes.MIN}
+ onClick={() => onClickRemove(index)}
+ style={{
+ background: "none",
+ ...(index === rulesArray.length - 1
+ ? {
+ visibility: "hidden",
+ pointerEvents: "none"
+ }
+ : {}
+ )
+ }}
+ >
+ <svg width="24" height="24" viewBox="0 0 24 24">
+ <title>Delete Rule</title>
+ <path fill="var(--status-danger)" d="M15 3.999V2H9V3.999H3V5.999H21V3.999H15Z" />
+ <path fill="var(--status-danger)" d="M5 6.99902V18.999C5 20.101 5.897 20.999 7 20.999H17C18.103 20.999 19 20.101 19 18.999V6.99902H5ZM11 17H9V11H11V17ZM15 17H13V11H15V17Z" />
+ </svg>
+ </Button>
+ </Flex>
+ {isRegexRules && renderFindError(rule.find)}
+ </React.Fragment>
+ )
+ }
+ </Flex>
+ </>
+ );
+}
+
+function TextReplaceTesting() {
+ const [value, setValue] = useState("");
+ return (
+ <>
+ <Forms.FormTitle tag="h4">Test Rules</Forms.FormTitle>
+ <TextInput placeholder="Type a message" onChange={setValue} />
+ <TextInput placeholder="Message with rules applied" editable={false} value={applyRules(value)} />
+ </>
+ );
+}
+
+function applyRules(content: string): string {
+ if (content.length === 0)
+ return content;
+
+ if (stringRules) {
+ for (const rule of stringRules) {
+ if (!rule.find || !rule.replace) continue;
+ if (rule.onlyIfIncludes && !content.includes(rule.onlyIfIncludes)) continue;
+
+ content = ` ${content} `.replaceAll(rule.find, rule.replace.replaceAll("\\n", "\n")).replace(/^\s|\s$/g, "");
+ }
+ }
+
+ if (regexRules) {
+ for (const rule of regexRules) {
+ if (!rule.find || !rule.replace) continue;
+ if (rule.onlyIfIncludes && !content.includes(rule.onlyIfIncludes)) continue;
+
+ try {
+ const regex = stringToRegex(rule.find);
+ content = content.replace(regex, rule.replace.replaceAll("\\n", "\n"));
+ } catch (e) {
+ new Logger("TextReplace").error(`Invalid regex: ${rule.find}`);
+ }
+ }
+ }
+
+ content = content.trim();
+ return content;
+}
+
+const TEXT_REPLACE_RULES_CHANNEL_ID = "1102784112584040479";
+
+export default definePlugin({
+ name: "TextReplace",
+ description: "Replace text in your messages. You can find pre-made rules in the #textreplace-rules channel in Vencord's Server",
+ authors: [Devs.AutumnVN, Devs.TheKodeToad],
+ dependencies: ["MessageEventsAPI"],
+
+ settings,
+
+ async start() {
+ stringRules = await DataStore.get(STRING_RULES_KEY) ?? makeEmptyRuleArray();
+ regexRules = await DataStore.get(REGEX_RULES_KEY) ?? makeEmptyRuleArray();
+
+ this.preSend = addPreSendListener((channelId, msg) => {
+ // Channel used for sharing rules, applying rules here would be messy
+ if (channelId === TEXT_REPLACE_RULES_CHANNEL_ID) return;
+ msg.content = applyRules(msg.content);
+ });
+ },
+
+ stop() {
+ removePreSendListener(this.preSend);
+ }
+});