aboutsummaryrefslogtreecommitdiff
path: root/src/plugins/quickReply
diff options
context:
space:
mode:
authorV <vendicated@riseup.net>2023-09-24 16:02:18 +0200
committerV <vendicated@riseup.net>2023-09-24 16:02:18 +0200
commit30ac25607023752031aa98060cbf8a736109992d (patch)
tree79bb82b6634ef601db6c98e751275607ec54dbea /src/plugins/quickReply
parentd0e2a324717e600736a18b88fe89a21c640a406b (diff)
downloadVencord-30ac25607023752031aa98060cbf8a736109992d.tar.gz
Vencord-30ac25607023752031aa98060cbf8a736109992d.tar.bz2
Vencord-30ac25607023752031aa98060cbf8a736109992d.zip
migrate all plugins to folders
Diffstat (limited to 'src/plugins/quickReply')
-rw-r--r--src/plugins/quickReply/index.ts216
1 files changed, 216 insertions, 0 deletions
diff --git a/src/plugins/quickReply/index.ts b/src/plugins/quickReply/index.ts
new file mode 100644
index 0000000..06797bc
--- /dev/null
+++ b/src/plugins/quickReply/index.ts
@@ -0,0 +1,216 @@
+/*
+ * 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, Settings } from "@api/Settings";
+import { Devs } from "@utils/constants";
+import definePlugin, { OptionType } from "@utils/types";
+import { findByPropsLazy } from "@webpack";
+import { ChannelStore, FluxDispatcher as Dispatcher, MessageStore, SelectedChannelStore, UserStore } from "@webpack/common";
+import { Message } from "discord-types/general";
+
+const Kangaroo = findByPropsLazy("jumpToMessage");
+
+const isMac = navigator.platform.includes("Mac"); // bruh
+let replyIdx = -1;
+let editIdx = -1;
+
+
+const enum MentionOptions {
+ DISABLED,
+ ENABLED,
+ NO_REPLY_MENTION_PLUGIN
+}
+
+const settings = definePluginSettings({
+ shouldMention: {
+ type: OptionType.SELECT,
+ description: "Ping reply by default",
+ options: [
+ {
+ label: "Follow NoReplyMention",
+ value: MentionOptions.NO_REPLY_MENTION_PLUGIN,
+ default: true
+ },
+ { label: "Enabled", value: MentionOptions.ENABLED },
+ { label: "Disabled", value: MentionOptions.DISABLED },
+ ]
+ }
+});
+
+export default definePlugin({
+ name: "QuickReply",
+ authors: [Devs.obscurity, Devs.Ven, Devs.pylix],
+ description: "Reply to (ctrl + up/down) and edit (ctrl + shift + up/down) messages via keybinds",
+ settings,
+
+ start() {
+ Dispatcher.subscribe("DELETE_PENDING_REPLY", onDeletePendingReply);
+ Dispatcher.subscribe("MESSAGE_END_EDIT", onEndEdit);
+ Dispatcher.subscribe("MESSAGE_START_EDIT", onStartEdit);
+ Dispatcher.subscribe("CREATE_PENDING_REPLY", onCreatePendingReply);
+ document.addEventListener("keydown", onKeydown);
+ },
+
+ stop() {
+ Dispatcher.unsubscribe("DELETE_PENDING_REPLY", onDeletePendingReply);
+ Dispatcher.unsubscribe("MESSAGE_END_EDIT", onEndEdit);
+ Dispatcher.unsubscribe("MESSAGE_START_EDIT", onStartEdit);
+ Dispatcher.unsubscribe("CREATE_PENDING_REPLY", onCreatePendingReply);
+ document.removeEventListener("keydown", onKeydown);
+ },
+});
+
+const onDeletePendingReply = () => replyIdx = -1;
+const onEndEdit = () => editIdx = -1;
+
+function calculateIdx(messages: Message[], id: string) {
+ const idx = messages.findIndex(m => m.id === id);
+ return idx === -1
+ ? idx
+ : messages.length - idx - 1;
+}
+
+function onStartEdit({ channelId, messageId, _isQuickEdit }: any) {
+ if (_isQuickEdit) return;
+
+ const meId = UserStore.getCurrentUser().id;
+
+ const messages = MessageStore.getMessages(channelId)._array.filter(m => m.author.id === meId);
+ editIdx = calculateIdx(messages, messageId);
+}
+
+function onCreatePendingReply({ message, _isQuickReply }: { message: Message; _isQuickReply: boolean; }) {
+ if (_isQuickReply) return;
+
+ replyIdx = calculateIdx(MessageStore.getMessages(message.channel_id)._array, message.id);
+}
+
+const isCtrl = (e: KeyboardEvent) => isMac ? e.metaKey : e.ctrlKey;
+const isAltOrMeta = (e: KeyboardEvent) => e.altKey || (!isMac && e.metaKey);
+
+function onKeydown(e: KeyboardEvent) {
+ const isUp = e.key === "ArrowUp";
+ if (!isUp && e.key !== "ArrowDown") return;
+ if (!isCtrl(e) || isAltOrMeta(e)) return;
+
+ if (e.shiftKey)
+ nextEdit(isUp);
+ else
+ nextReply(isUp);
+}
+
+function jumpIfOffScreen(channelId: string, messageId: string) {
+ const element = document.getElementById("message-content-" + messageId);
+ if (!element) return;
+
+ const vh = Math.max(document.documentElement.clientHeight, window.innerHeight);
+ const rect = element.getBoundingClientRect();
+ const isOffscreen = rect.bottom < 200 || rect.top - vh >= -200;
+
+ if (isOffscreen) {
+ Kangaroo.jumpToMessage({
+ channelId,
+ messageId,
+ flash: false,
+ jumpType: "INSTANT"
+ });
+ }
+}
+
+function getNextMessage(isUp: boolean, isReply: boolean) {
+ let messages: Array<Message & { deleted?: boolean; }> = MessageStore.getMessages(SelectedChannelStore.getChannelId())._array;
+ if (!isReply) { // we are editing so only include own
+ const meId = UserStore.getCurrentUser().id;
+ messages = messages.filter(m => m.author.id === meId);
+ }
+
+ const mutate = (i: number) => isUp
+ ? Math.min(messages.length - 1, i + 1)
+ : Math.max(-1, i - 1);
+
+ const findNextNonDeleted = (i: number) => {
+ do {
+ i = mutate(i);
+ } while (i !== -1 && messages[messages.length - i - 1]?.deleted === true);
+ return i;
+ };
+
+ let i: number;
+ if (isReply)
+ replyIdx = i = findNextNonDeleted(replyIdx);
+ else
+ editIdx = i = findNextNonDeleted(editIdx);
+
+ return i === - 1 ? undefined : messages[messages.length - i - 1];
+}
+
+function shouldMention(message) {
+ const { enabled, userList, shouldPingListed } = Settings.plugins.NoReplyMention;
+ const shouldPing = !enabled || (shouldPingListed === userList.includes(message.author.id));
+
+ switch (settings.store.shouldMention) {
+ case MentionOptions.NO_REPLY_MENTION_PLUGIN: return shouldPing;
+ case MentionOptions.DISABLED: return false;
+ default: return true;
+ }
+}
+
+// handle next/prev reply
+function nextReply(isUp: boolean) {
+ const message = getNextMessage(isUp, true);
+
+ if (!message)
+ return void Dispatcher.dispatch({
+ type: "DELETE_PENDING_REPLY",
+ channelId: SelectedChannelStore.getChannelId(),
+ });
+
+ const channel = ChannelStore.getChannel(message.channel_id);
+ const meId = UserStore.getCurrentUser().id;
+
+ Dispatcher.dispatch({
+ type: "CREATE_PENDING_REPLY",
+ channel,
+ message,
+ shouldMention: shouldMention(message),
+ showMentionToggle: channel.guild_id !== null && message.author.id !== meId,
+ _isQuickReply: true
+ });
+ jumpIfOffScreen(channel.id, message.id);
+}
+
+// handle next/prev edit
+function nextEdit(isUp: boolean) {
+ const message = getNextMessage(isUp, false);
+
+ if (!message)
+ Dispatcher.dispatch({
+ type: "MESSAGE_END_EDIT",
+ channelId: SelectedChannelStore.getChannelId()
+ });
+ else {
+ Dispatcher.dispatch({
+ type: "MESSAGE_START_EDIT",
+ channelId: message.channel_id,
+ messageId: message.id,
+ content: message.content,
+ _isQuickEdit: true
+ });
+ jumpIfOffScreen(message.channel_id, message.id);
+ }
+}