mirror of
https://github.com/Vendicated/Vencord.git
synced 2025-01-25 08:46:25 +00:00
c836270320
- FriendsSince: Don't show for friend requests - FakeNitro: Fix attempting to bypass unicode emojis #2503 - MessageLatency: ignore bots #2504 - CtrlEnterSend: use cmd+enter on macOS #2502 - ReplaceGoogleSearch: trim LF character #2488 Co-authored-by: AutumnVN <autumnvnchino@gmail.com> Co-authored-by: rushiiMachine <33725716+rushiiMachine@users.noreply.github.com> Co-authored-by: Lumap <lumap@duck.com> Co-authored-by: Mylloon <kennel.anri@tutanota.com>
207 lines
7.5 KiB
TypeScript
207 lines
7.5 KiB
TypeScript
/*
|
|
* Vencord, a Discord client mod
|
|
* Copyright (c) 2024 Vendicated and contributors
|
|
* SPDX-License-Identifier: GPL-3.0-or-later
|
|
*/
|
|
|
|
import { definePluginSettings } from "@api/Settings";
|
|
import ErrorBoundary from "@components/ErrorBoundary";
|
|
import { Devs } from "@utils/constants";
|
|
import { isNonNullish } from "@utils/guards";
|
|
import definePlugin, { OptionType } from "@utils/types";
|
|
import { findExportedComponentLazy } from "@webpack";
|
|
import { SnowflakeUtils, Tooltip } from "@webpack/common";
|
|
import { Message } from "discord-types/general";
|
|
|
|
type FillValue = ("status-danger" | "status-warning" | "status-positive" | "text-muted");
|
|
type Fill = [FillValue, FillValue, FillValue];
|
|
type DiffKey = keyof Diff;
|
|
|
|
interface Diff {
|
|
days: number,
|
|
hours: number,
|
|
minutes: number,
|
|
seconds: number;
|
|
milliseconds: number;
|
|
}
|
|
|
|
const DISCORD_KT_DELAY = 1471228928;
|
|
const HiddenVisually = findExportedComponentLazy("HiddenVisually");
|
|
|
|
export default definePlugin({
|
|
name: "MessageLatency",
|
|
description: "Displays an indicator for messages that took ≥n seconds to send",
|
|
authors: [Devs.arHSM],
|
|
|
|
settings: definePluginSettings({
|
|
latency: {
|
|
type: OptionType.NUMBER,
|
|
description: "Threshold in seconds for latency indicator",
|
|
default: 2
|
|
},
|
|
detectDiscordKotlin: {
|
|
type: OptionType.BOOLEAN,
|
|
description: "Detect old Discord Android clients",
|
|
default: true
|
|
},
|
|
showMillis: {
|
|
type: OptionType.BOOLEAN,
|
|
description: "Show milliseconds",
|
|
default: false
|
|
}
|
|
}),
|
|
|
|
patches: [
|
|
{
|
|
find: "showCommunicationDisabledStyles",
|
|
replacement: {
|
|
match: /(message:(\i),avatar:\i,username:\(0,\i.jsxs\)\(\i.Fragment,\{children:\[)(\i&&)/,
|
|
replace: "$1$self.Tooltip()({ message: $2 }),$3"
|
|
}
|
|
}
|
|
],
|
|
|
|
stringDelta(delta: number, showMillis: boolean) {
|
|
const diff: Diff = {
|
|
days: Math.round(delta / (60 * 60 * 24 * 1000)),
|
|
hours: Math.round((delta / (60 * 60 * 1000)) % 24),
|
|
minutes: Math.round((delta / (60 * 1000)) % 60),
|
|
seconds: Math.round(delta / 1000 % 60),
|
|
milliseconds: Math.round(delta % 1000)
|
|
};
|
|
|
|
const str = (k: DiffKey) => diff[k] > 0 ? `${diff[k]} ${diff[k] > 1 ? k : k.substring(0, k.length - 1)}` : null;
|
|
const keys = Object.keys(diff) as DiffKey[];
|
|
|
|
const ts = keys.reduce((prev, k) => {
|
|
const s = str(k);
|
|
|
|
return prev + (
|
|
isNonNullish(s)
|
|
? (prev !== ""
|
|
? (showMillis ? k === "milliseconds" : k === "seconds")
|
|
? " and "
|
|
: " "
|
|
: "") + s
|
|
: ""
|
|
);
|
|
}, "");
|
|
|
|
return ts || "0 seconds";
|
|
},
|
|
|
|
latencyTooltipData(message: Message) {
|
|
const { latency, detectDiscordKotlin, showMillis } = this.settings.store;
|
|
const { id, nonce } = message;
|
|
|
|
// Message wasn't received through gateway
|
|
if (!isNonNullish(nonce)) return null;
|
|
|
|
// Bots basically never send a nonce, and if someone does do it then it's usually not a snowflake
|
|
if (message.bot) return null;
|
|
|
|
let isDiscordKotlin = false;
|
|
let delta = SnowflakeUtils.extractTimestamp(id) - SnowflakeUtils.extractTimestamp(nonce); // milliseconds
|
|
if (!showMillis) {
|
|
delta = Math.round(delta / 1000) * 1000;
|
|
}
|
|
|
|
// Old Discord Android clients have a delay of around 17 days
|
|
// This is a workaround for that
|
|
if (-delta >= DISCORD_KT_DELAY - 86400000) { // One day of padding for good measure
|
|
isDiscordKotlin = detectDiscordKotlin;
|
|
delta += DISCORD_KT_DELAY;
|
|
}
|
|
|
|
// Thanks dziurwa (I hate you)
|
|
// This is when the user's clock is ahead
|
|
// Can't do anything if the clock is behind
|
|
const abs = Math.abs(delta);
|
|
const ahead = abs !== delta;
|
|
const latencyMillis = latency * 1000;
|
|
|
|
const stringDelta = abs >= latencyMillis ? this.stringDelta(abs, showMillis) : null;
|
|
|
|
// Also thanks dziurwa
|
|
// 2 minutes
|
|
const TROLL_LIMIT = 2 * 60 * 1000;
|
|
|
|
const fill: Fill = isDiscordKotlin
|
|
? ["status-positive", "status-positive", "text-muted"]
|
|
: delta >= TROLL_LIMIT || ahead
|
|
? ["text-muted", "text-muted", "text-muted"]
|
|
: delta >= (latencyMillis * 2)
|
|
? ["status-danger", "text-muted", "text-muted"]
|
|
: ["status-warning", "status-warning", "text-muted"];
|
|
|
|
return (abs >= latencyMillis || isDiscordKotlin) ? { delta: stringDelta, ahead, fill, isDiscordKotlin } : null;
|
|
},
|
|
|
|
Tooltip() {
|
|
return ErrorBoundary.wrap(({ message }: { message: Message; }) => {
|
|
const d = this.latencyTooltipData(message);
|
|
|
|
if (!isNonNullish(d)) return null;
|
|
|
|
let text: string;
|
|
if (!d.delta) {
|
|
text = "User is suspected to be on an old Discord Android client";
|
|
} else {
|
|
text = (d.ahead ? `This user's clock is ${d.delta} ahead.` : `This message was sent with a delay of ${d.delta}.`) + (d.isDiscordKotlin ? " User is suspected to be on an old Discord Android client." : "");
|
|
}
|
|
|
|
return <Tooltip
|
|
text={text}
|
|
position="top"
|
|
>
|
|
{
|
|
props => <>
|
|
{<this.Icon delta={d.delta} fill={d.fill} props={props} />}
|
|
{/* Time Out indicator uses this, I think this is for a11y */}
|
|
<HiddenVisually>Delayed Message</HiddenVisually>
|
|
</>
|
|
}
|
|
</Tooltip>;
|
|
});
|
|
},
|
|
|
|
Icon({ delta, fill, props }: {
|
|
delta: string | null;
|
|
fill: Fill,
|
|
props: {
|
|
onClick(): void;
|
|
onMouseEnter(): void;
|
|
onMouseLeave(): void;
|
|
onContextMenu(): void;
|
|
onFocus(): void;
|
|
onBlur(): void;
|
|
"aria-label"?: string;
|
|
};
|
|
}) {
|
|
return <svg
|
|
xmlns="http://www.w3.org/2000/svg"
|
|
viewBox="0 0 16 16"
|
|
width="12"
|
|
height="12"
|
|
role="img"
|
|
fill="none"
|
|
style={{ marginRight: "8px", verticalAlign: -1 }}
|
|
aria-label={delta ?? "Old Discord Android client"}
|
|
aria-hidden="false"
|
|
{...props}
|
|
>
|
|
<path
|
|
fill={`var(--${fill[0]})`}
|
|
d="M4.8001 12C4.8001 11.5576 4.51344 11.2 4.16023 11.2H2.23997C1.88676 11.2 1.6001 11.5576 1.6001 12V13.6C1.6001 14.0424 1.88676 14.4 2.23997 14.4H4.15959C4.5128 14.4 4.79946 14.0424 4.79946 13.6L4.8001 12Z"
|
|
/>
|
|
<path
|
|
fill={`var(--${fill[1]})`}
|
|
d="M9.6001 7.12724C9.6001 6.72504 9.31337 6.39998 8.9601 6.39998H7.0401C6.68684 6.39998 6.40011 6.72504 6.40011 7.12724V13.6727C6.40011 14.0749 6.68684 14.4 7.0401 14.4H8.9601C9.31337 14.4 9.6001 14.0749 9.6001 13.6727V7.12724Z"
|
|
/>
|
|
<path
|
|
fill={`var(--${fill[2]})`}
|
|
d="M14.4001 2.31109C14.4001 1.91784 14.1134 1.59998 13.7601 1.59998H11.8401C11.4868 1.59998 11.2001 1.91784 11.2001 2.31109V13.6888C11.2001 14.0821 11.4868 14.4 11.8401 14.4H13.7601C14.1134 14.4 14.4001 14.0821 14.4001 13.6888V2.31109Z"
|
|
/>
|
|
</svg>;
|
|
}
|
|
});
|