1
0
Fork 1
mirror of https://github.com/Vendicated/Vencord.git synced 2025-01-10 09:56:24 +00:00
This commit is contained in:
camila314 2024-06-17 23:42:50 -05:00
parent b9b037bdb9
commit 91f6541d8a
3 changed files with 288 additions and 126 deletions

View file

@ -1,3 +1,3 @@
# KeywordNotify # KeywordNotify
Sends a notification when a message matches any number of regex cases. Allows for custom regex-defined keywords to notify the user exactly how a ping would. Adds a custom inbox for viewing keywords next to the mentions.

View file

@ -4,40 +4,41 @@
* SPDX-License-Identifier: GPL-3.0-or-later * SPDX-License-Identifier: GPL-3.0-or-later
*/ */
import definePlugin, { OptionType } from "@utils/types";
import { DataStore } from "@api/index";
import { definePluginSettings } from "@api/Settings";
import { DeleteIcon } from "@components/Icons";
import { Devs } from "@utils/constants";
import { Flex } from "@components/Flex";
import { TextInput, useState, Forms, Button, UserStore, UserUtils, TabBar, ChannelStore, SelectedChannelStore } from "@webpack/common";
import { useForceUpdater } from "@utils/react";
import { findByCodeLazy, findByPropsLazy, mapMangledModuleLazy, filters } from "@webpack";
import "./style.css"; import "./style.css";
let regexes = []; import { DataStore } from "@api/index";
import { definePluginSettings } from "@api/Settings";
import { Flex } from "@components/Flex";
import { DeleteIcon } from "@components/Icons";
import { Devs } from "@utils/constants";
import { Margins } from "@utils/margins";
import { useForceUpdater } from "@utils/react";
import definePlugin, { OptionType } from "@utils/types";
import { findByCodeLazy, findByPropsLazy } from "@webpack";
import { Button, ChannelStore, Forms, SearchableSelect,SelectedChannelStore, TabBar, TextInput, UserStore, UserUtils, useState } from "@webpack/common";
import { Message, User } from "discord-types/general/index.js";
let keywordEntries: Array<{ regex: string, listIds: Array<string>, listType: ListType }> = [];
let currentUser: User;
let keywordLog: Array<any> = [];
const MenuHeader = findByCodeLazy(".useInDesktopNotificationCenterExperiment)()?"); const MenuHeader = findByCodeLazy(".useInDesktopNotificationCenterExperiment)()?");
const Popout = findByPropsLazy("ItemsPopout"); const Popout = findByCodeLazy("let{analyticsName:");
const recentMentionsPopoutClass = findByPropsLazy("recentMentionsPopout"); const recentMentionsPopoutClass = findByPropsLazy("recentMentionsPopout");
const KEYWORD_ENTRIES_KEY = "KeywordNotify_keywordEntries";
const KEYWORD_LOG_KEY = "KeywordNotify_log";
const { createMessageRecord } = findByPropsLazy("createMessageRecord", "updateMessageRecord"); const { createMessageRecord } = findByPropsLazy("createMessageRecord", "updateMessageRecord");
async function addKeywordEntry(updater: () => void) {
async function setRegexes(idx: number, reg: string) { keywordEntries.push({ regex: "", listIds: [], listType: ListType.BlackList });
regexes[idx] = reg; await DataStore.set(KEYWORD_ENTRIES_KEY, keywordEntries);
await DataStore.set("KeywordNotify_rules", regexes);
}
async function removeRegex(idx: number, updater: () => void) {
regexes.splice(idx, 1);
await DataStore.set("KeywordNotify_rules", regexes);
updater(); updater();
} }
async function addRegex(updater: () => void) { async function removeKeywordEntry(idx: number, updater: () => void) {
regexes.push(""); keywordEntries.splice(idx, 1);
await DataStore.set("KeywordNotify_rules", regexes); await DataStore.set(KEYWORD_ENTRIES_KEY, keywordEntries);
updater(); updater();
} }
@ -49,86 +50,201 @@ function safeMatchesRegex(s: string, r: string) {
} }
} }
enum ListType {
BlackList = "BlackList",
Whitelist = "Whitelist"
}
function highlightKeywords(s: string, r: Array<string>) { function highlightKeywords(s: string, r: Array<string>) {
let reg; let regex: RegExp;
try { try {
reg = new RegExp(r.join("|"), "g"); regex = new RegExp(r.join("|"), "g");
} catch { } catch {
return [s]; return [s];
} }
let matches = s.match(reg); const matches = s.match(regex);
if (!matches) if (!matches)
return [s]; return [s];
let parts = [...matches.map((e) => { const parts = [...matches.map(e => {
let idx = s.indexOf(e); const idx = s.indexOf(e);
let before = s.substring(0, idx); const before = s.substring(0, idx);
s = s.substring(idx + e.length); s = s.substring(idx + e.length);
return before; return before;
}, s), s]; }, s), s];
return parts.map(e => [ return parts.map(e => [
(<span>{e}</span>), (<span>{e}</span>),
matches.length ? (<span class="highlight">{matches.splice(0, 1)[0]}</span>) : [] matches!.length ? (<span className="highlight">{matches!.splice(0, 1)[0]}</span>) : []
]); ]);
} }
function Collapsible({ title, children }) {
const [isOpen, setIsOpen] = useState(false);
return (
<div>
<Button
onClick={() => setIsOpen(!isOpen)}
look={Button.Looks.BLANK}
size={Button.Sizes.ICON}
className="keywordnotify-collapsible">
<div style={{ display: "flex", alignItems: "center" }}>
<div style={{ marginLeft: "auto", color: "var(--text-muted)", paddingRight: "5px" }}>{isOpen ? "▼" : "▶"}</div>
<Forms.FormTitle tag="h4">{title}</Forms.FormTitle>
</div>
</Button>
{isOpen && children}
</div>
);
}
function ListedIds({ listIds, setListIds }) {
const update = useForceUpdater();
const [values] = useState(listIds);
async function onChange(e: string, index: number) {
values[index] = e;
setListIds(values);
update();
}
const elements = values.map((currentValue: string, index: number) => {
return (
<Flex flexDirection="row" style={{ marginBottom: "5px" }}>
<div style={{ flexGrow: 1 }}>
<TextInput
placeholder="ID"
spellCheck={false}
value={currentValue}
onChange={e => onChange(e, index)}
/>
</div>
<Button
onClick={() => {
values.splice(index, 1);
setListIds(values);
update();
}}
look={Button.Looks.BLANK}
size={Button.Sizes.ICON}
className="keywordnotify-delete">
<DeleteIcon/>
</Button>
</Flex>
);
});
return (
<>
{elements}
</>
);
}
function ListTypeSelector({ listType, setListType }) {
return (
<SearchableSelect
options={[
{ label: "Whitelist", value: ListType.Whitelist },
{ label: "Blacklist", value: ListType.BlackList }
]}
placeholder={"Select a list type"}
maxVisibleItems={2}
closeOnSelect={true}
value={listType}
onChange={setListType}
/>
);
}
function KeywordEntries() {
const update = useForceUpdater();
const [values] = useState(keywordEntries);
async function setRegex(index: number, value: string) {
keywordEntries[index].regex = value;
await DataStore.set(KEYWORD_ENTRIES_KEY, keywordEntries);
update();
}
async function setListType(index: number, value: ListType) {
keywordEntries[index].listType = value;
await DataStore.set(KEYWORD_ENTRIES_KEY, keywordEntries);
update();
}
async function setListIds(index: number, value: Array<string>) {
keywordEntries[index].listIds = value ?? [];
await DataStore.set(KEYWORD_ENTRIES_KEY, keywordEntries);
update();
}
const elements = keywordEntries.map((entry, i) => {
return (
<>
<Collapsible title={`Keyword Entry ${i + 1}`}>
<Flex flexDirection="row">
<div style={{ flexGrow: 1 }}>
<TextInput
placeholder="example|regex"
spellCheck={false}
value={values[i].regex}
onChange={e => setRegex(i, e)}
/>
</div>
<Button
onClick={() => removeKeywordEntry(i, update)}
look={Button.Looks.BLANK}
size={Button.Sizes.ICON}
className="keywordnotify-delete">
<DeleteIcon/>
</Button>
</Flex>
<Forms.FormDivider className={Margins.top8 + " " + Margins.bottom8}/>
<Forms.FormTitle tag="h5">Whitelist/Blacklist</Forms.FormTitle>
<Flex flexDirection="row">
<div style={{ flexGrow: 1 }}>
<ListedIds listIds={values[i].listIds} setListIds={e => setListIds(i, e)}/>
</div>
</Flex>
<div className={Margins.top8 + " " + Margins.bottom8}/>
<Flex flexDirection="row">
<Button onClick={() => {
values[i].listIds.push("");
update();
}}>Add ID</Button>
<div style={{ flexGrow: 1 }}>
<ListTypeSelector listType={values[i].listType} setListType={e => setListType(i, e)}/>
</div>
</Flex>
</Collapsible>
</>
);
});
return (
<>
{elements}
<div><Button onClick={() => addKeywordEntry(update)}>Add Keyword Entry</Button></div>
</>
);
}
const settings = definePluginSettings({ const settings = definePluginSettings({
ignoreBots: { ignoreBots: {
type: OptionType.BOOLEAN, type: OptionType.BOOLEAN,
description: "Ignore messages from bots", description: "Ignore messages from bots",
default: true default: true
}, },
keywords: { keywords: {
type: OptionType.COMPONENT, type: OptionType.COMPONENT,
description: "", description: "",
component: () => { component: () => <KeywordEntries/>
const update = useForceUpdater(); }
const [values, setValues] = useState(regexes);
const elements = regexes.map((a, i) => {
const setValue = (v: string) => {
let valuesCopy = [...values];
valuesCopy[i] = v;
setValues(valuesCopy);
}
return (
<>
<Forms.FormTitle tag="h4">Keyword Regex {i + 1}</Forms.FormTitle>
<Flex flexDirection="row">
<div style={{flexGrow: 1}}>
<TextInput
placeholder="example|regex"
spellCheck={false}
value={values[i]}
onChange={setValue}
onBlur={() => setRegexes(i, values[i])}
/>
</div>
<Button
onClick={() => removeRegex(i, update)}
look={Button.Looks.BLANK}
size={Button.Sizes.ICON}
className="keywordnotify-delete">
<DeleteIcon />
</Button>
</Flex>
</>
)
});
return (
<>
{elements}
<div><Button onClick={() => addRegex(update)}>Add Regex</Button></div>
</>
);
}
},
}); });
export default definePlugin({ export default definePlugin({
@ -138,12 +254,12 @@ export default definePlugin({
settings, settings,
patches: [ patches: [
{ {
find: "}_dispatch(", find: "}_dispatch(",
replacement: { replacement: {
match: /}_dispatch\((\i),\i\){/, match: /}_dispatch\((\i),\i\){/,
replace: "$&$1=$self.modify($1);" replace: "$&$1=$self.modify($1);"
} }
}, },
{ {
find: "Messages.UNREADS_TAB_LABEL}", find: "Messages.UNREADS_TAB_LABEL}",
replacement: { replacement: {
@ -168,49 +284,81 @@ export default definePlugin({
], ],
async start() { async start() {
regexes = await DataStore.get("KeywordNotify_rules") ?? []; keywordEntries = await DataStore.get(KEYWORD_ENTRIES_KEY) ?? [];
this.me = await UserUtils.getUser(UserStore.getCurrentUser().id); currentUser = await UserUtils.getUser(UserStore.getCurrentUser().id);
this.onUpdate = ()=>null; this.onUpdate = () => null;
this.keywordLog = [];
(await DataStore.get("KeywordNotify_log") ?? []).map((e) => JSON.parse(e)).forEach((e) => { (await DataStore.get(KEYWORD_LOG_KEY) ?? []).map(e => JSON.parse(e)).forEach(e => {
this.addToLog(e); this.addToLog(e);
}); });
}, },
applyRegexes(m) { applyKeywordEntries(m: Message) {
if (settings.store.ignoreBots && m.author.bot)
return;
let matches = false; let matches = false;
if (regexes.some(r => r != "" && (safeMatchesRegex(m.content, r)))) { keywordEntries.forEach(entry => {
matches = true; if (entry.regex === "") {
} return;
for (let embed of m.embeds) { }
if (regexes.some(r => r != "" && (safeMatchesRegex(embed.description, r) || safeMatchesRegex(embed.title, r)))) {
let listed = entry.listIds.some(id => id === m.channel_id || id === m.author.id);
if (!listed) {
const channel = ChannelStore.getChannel(m.channel_id);
if (channel != null) {
listed = entry.listIds.some(id => id === channel.guild_id);
}
}
const whitelistMode = entry.listType === ListType.Whitelist;
if (!whitelistMode && listed) {
return;
}
if (whitelistMode && !listed) {
return;
}
if (settings.store.ignoreBots && m.author.bot) {
if (!whitelistMode || !entry.listIds.includes(m.author.id)) {
return;
}
}
if (safeMatchesRegex(m.content, entry.regex)) {
matches = true; matches = true;
} }
}
for (const embed of m.embeds as any) {
if (safeMatchesRegex(embed.description, entry.regex) || safeMatchesRegex(embed.title, entry.regex)) {
matches = true;
} else if (embed.fields != null) {
for (const field of embed.fields as Array<{ name: string, value: string }>) {
if (safeMatchesRegex(field.value, entry.regex) || safeMatchesRegex(field.name, entry.regex)) {
matches = true;
}
}
}
}
});
if (matches) { if (matches) {
m.mentions.push(this.me); // @ts-ignore
m.mentions.push(currentUser);
if (m.author.id != this.me.id) if (m.author.id !== currentUser.id)
this.addToLog(m); this.addToLog(m);
} }
}, },
addToLog(m) { addToLog(m: Message) {
if (m == null || this.keywordLog.some((e) => e.id == m.id)) if (m == null || keywordLog.some(e => e.id === m.id))
return; return;
let thing = createMessageRecord(m); const thing = createMessageRecord(m);
this.keywordLog.push(thing); keywordLog.push(thing);
this.keywordLog.sort((a, b) => b.timestamp - a.timestamp); keywordLog.sort((a, b) => b.timestamp - a.timestamp);
if (this.keywordLog.length > 50) if (keywordLog.length > 50)
this.keywordLog.pop(); keywordLog.pop();
this.onUpdate(); this.onUpdate();
}, },
@ -225,27 +373,28 @@ export default definePlugin({
}, },
tryKeywordMenu(setTab, onJump, closePopout) { tryKeywordMenu(setTab, onJump, closePopout) {
let header = ( const header = (
<MenuHeader tab={5} setTab={setTab} closePopout={closePopout} badgeState={{badgeForYou: false}}/> <MenuHeader tab={5} setTab={setTab} closePopout={closePopout} badgeState={{ badgeForYou: false }}/>
); );
let channel = ChannelStore.getChannel(SelectedChannelStore.getChannelId()); const channel = ChannelStore.getChannel(SelectedChannelStore.getChannelId());
let [keywordLog, setKeywordLog] = useState(this.keywordLog); const [tempLogs, setKeywordLog] = useState(keywordLog);
this.onUpdate = () => { this.onUpdate = () => {
let newLog = [...this.keywordLog]; const newLog = [...keywordLog];
setKeywordLog(newLog); setKeywordLog(newLog);
DataStore.set("KeywordNotify_log", newLog.map((e) => JSON.stringify(e))); DataStore.set(KEYWORD_LOG_KEY, newLog.map(e => JSON.stringify(e)));
}; };
let onDelete = (m) => { const onDelete = m => {
this.keywordLog = this.keywordLog.filter((e) => e.id != m.id); keywordLog = keywordLog.filter(e => e.id !== m.id);
this.onUpdate(); this.onUpdate();
}; };
let messageRender = (e, t) => { const messageRender = (e, t) => {
let msg = this.renderMsg({ console.log(this);
const msg = this.renderMsg({
message: e, message: e,
gotoMessage: t, gotoMessage: t,
dismissible: true dismissible: true
@ -256,36 +405,42 @@ export default definePlugin({
msg.props.children[0].props.children.props.onClick = () => onDelete(e); msg.props.children[0].props.children.props.onClick = () => onDelete(e);
msg.props.children[1].props.children[1].props.message.customRenderedContent = { msg.props.children[1].props.children[1].props.message.customRenderedContent = {
content: highlightKeywords(e.content, regexes) content: highlightKeywords(e.content, keywordEntries.map(e => e.regex))
}; };
return [msg]; return [msg];
}; };
/*return (
<>
<p>hi uwu</p>
</>
);*/
return ( return (
<> <>
<Popout.default <Popout
className={recentMentionsPopoutClass.recentMentionsPopout} className={recentMentionsPopoutClass.recentMentionsPopout}
renderHeader={() => header} renderHeader={() => header}
renderMessage={messageRender} renderMessage={messageRender}
channel={channel} channel={channel}
onJump={onJump} onJump={onJump}
onFetch={()=>null} onFetch={() => null}
onCloseMessage={onDelete} onCloseMessage={onDelete}
loadMore={()=>null} loadMore={() => null}
messages={keywordLog} messages={tempLogs}
renderEmptyState={()=>null} renderEmptyState={() => null}
/> />
</> </>
); );
}, },
modify(e) { modify(e) {
if (e.type == "MESSAGE_CREATE") { if (e.type === "MESSAGE_CREATE") {
this.applyRegexes(e.message); this.applyKeywordEntries(e.message);
} else if (e.type == "LOAD_MESSAGES_SUCCESS") { } else if (e.type === "LOAD_MESSAGES_SUCCESS") {
for (let msg = 0; msg < e.messages.length; ++msg) { for (let msg = 0; msg < e.messages.length; ++msg) {
this.applyRegexes(e.messages[msg]); this.applyKeywordEntries(e.messages[msg]);
} }
} }
return e; return e;

View file

@ -3,7 +3,14 @@
} }
.keywordnotify-delete { .keywordnotify-delete {
padding: 0px, padding: 0;
color: var(--primary-400); color: var(--primary-400);
transition: color 0.2s ease-in-out; transition: color 0.2s ease-in-out;
} }
.keywordnotify-collapsible {
display: flex;
align-items: center;
padding: 8px;
cursor: pointer;
}