/* * 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 . */ 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 = 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() { switch (settings.store.shouldMention) { case MentionOptions.NO_REPLY_MENTION_PLUGIN: return !Settings.plugins.NoReplyMention.enabled; 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(), 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); } }