mirror of
https://github.com/Vendicated/Vencord.git
synced 2025-01-12 18:46:22 +00:00
Merge branch 'dev' into plugin-text-emoji
Sync with dev.
This commit is contained in:
commit
1df841b406
71 changed files with 2853 additions and 759 deletions
2
.github/workflows/codeberg-mirror.yml
vendored
2
.github/workflows/codeberg-mirror.yml
vendored
|
@ -18,5 +18,5 @@ jobs:
|
||||||
fetch-depth: 0
|
fetch-depth: 0
|
||||||
- uses: pixta-dev/repository-mirroring-action@674e65a7d483ca28dafaacba0d07351bdcc8bd75 # v1.1.1
|
- uses: pixta-dev/repository-mirroring-action@674e65a7d483ca28dafaacba0d07351bdcc8bd75 # v1.1.1
|
||||||
with:
|
with:
|
||||||
target_repo_url: "git@codeberg.org:Ven/cord.git"
|
target_repo_url: "git@codeberg.org:Vee/cord.git"
|
||||||
ssh_private_key: ${{ secrets.CODEBERG_SSH_PRIVATE_KEY }}
|
ssh_private_key: ${{ secrets.CODEBERG_SSH_PRIVATE_KEY }}
|
||||||
|
|
|
@ -1,6 +1,6 @@
|
||||||
# Vencord
|
# Vencord
|
||||||
|
|
||||||
[![Codeberg Mirror](https://img.shields.io/static/v1?style=for-the-badge&label=Codeberg%20Mirror&message=codeberg.org/Ven/cord&color=2185D0&logo=)](https://codeberg.org/Ven/cord)
|
[![Codeberg Mirror](https://img.shields.io/static/v1?style=for-the-badge&label=Codeberg%20Mirror&message=codeberg.org/Vee/cord&color=2185D0&logo=)](https://codeberg.org/Vee/cord)
|
||||||
|
|
||||||
The cutest Discord client mod
|
The cutest Discord client mod
|
||||||
|
|
||||||
|
|
|
@ -1,7 +1,7 @@
|
||||||
{
|
{
|
||||||
"name": "vencord",
|
"name": "vencord",
|
||||||
"private": "true",
|
"private": "true",
|
||||||
"version": "1.6.5",
|
"version": "1.6.7",
|
||||||
"description": "The cutest Discord client mod",
|
"description": "The cutest Discord client mod",
|
||||||
"homepage": "https://github.com/Vendicated/Vencord#readme",
|
"homepage": "https://github.com/Vendicated/Vencord#readme",
|
||||||
"bugs": {
|
"bugs": {
|
||||||
|
|
|
@ -35,11 +35,11 @@ const ETAG_FILE = join(FILE_DIR, "etag.txt");
|
||||||
function getFilename() {
|
function getFilename() {
|
||||||
switch (process.platform) {
|
switch (process.platform) {
|
||||||
case "win32":
|
case "win32":
|
||||||
return "VencordInstaller.exe";
|
return "VencordInstallerCli.exe";
|
||||||
case "darwin":
|
case "darwin":
|
||||||
return "VencordInstaller.MacOS.zip";
|
return "VencordInstaller.MacOS.zip";
|
||||||
case "linux":
|
case "linux":
|
||||||
return "VencordInstaller-" + (process.env.WAYLAND_DISPLAY ? "wayland" : "x11");
|
return "VencordInstallerCli-linux";
|
||||||
default:
|
default:
|
||||||
throw new Error("Unsupported platform: " + process.platform);
|
throw new Error("Unsupported platform: " + process.platform);
|
||||||
}
|
}
|
||||||
|
@ -118,11 +118,15 @@ const installerBin = await ensureBinary();
|
||||||
|
|
||||||
console.log("Now running Installer...");
|
console.log("Now running Installer...");
|
||||||
|
|
||||||
execFileSync(installerBin, {
|
try {
|
||||||
stdio: "inherit",
|
execFileSync(installerBin, {
|
||||||
env: {
|
stdio: "inherit",
|
||||||
...process.env,
|
env: {
|
||||||
VENCORD_USER_DATA_DIR: BASE_DIR,
|
...process.env,
|
||||||
VENCORD_DEV_INSTALL: "1"
|
VENCORD_USER_DATA_DIR: BASE_DIR,
|
||||||
}
|
VENCORD_DEV_INSTALL: "1"
|
||||||
});
|
}
|
||||||
|
});
|
||||||
|
} catch {
|
||||||
|
console.error("Something went wrong. Please check the logs above.");
|
||||||
|
}
|
||||||
|
|
|
@ -44,7 +44,7 @@ async function syncSettings() {
|
||||||
// pre-check for local shared settings
|
// pre-check for local shared settings
|
||||||
if (
|
if (
|
||||||
Settings.cloud.authenticated &&
|
Settings.cloud.authenticated &&
|
||||||
await dsGet("Vencord_cloudSecret") === null // this has been enabled due to local settings share or some other bug
|
!await dsGet("Vencord_cloudSecret") // this has been enabled due to local settings share or some other bug
|
||||||
) {
|
) {
|
||||||
// show a notification letting them know and tell them how to fix it
|
// show a notification letting them know and tell them how to fix it
|
||||||
showNotification({
|
showNotification({
|
||||||
|
@ -145,4 +145,3 @@ document.addEventListener("DOMContentLoaded", () => {
|
||||||
}));
|
}));
|
||||||
}
|
}
|
||||||
}, { once: true });
|
}, { once: true });
|
||||||
|
|
||||||
|
|
4
src/api/ChatButton.css
Normal file
4
src/api/ChatButton.css
Normal file
|
@ -0,0 +1,4 @@
|
||||||
|
.vc-chatbar-button {
|
||||||
|
display: flex;
|
||||||
|
align-items: center;
|
||||||
|
}
|
128
src/api/ChatButtons.tsx
Normal file
128
src/api/ChatButtons.tsx
Normal file
|
@ -0,0 +1,128 @@
|
||||||
|
/*
|
||||||
|
* Vencord, a Discord client mod
|
||||||
|
* Copyright (c) 2024 Vendicated and contributors
|
||||||
|
* SPDX-License-Identifier: GPL-3.0-or-later
|
||||||
|
*/
|
||||||
|
|
||||||
|
import "./ChatButton.css";
|
||||||
|
|
||||||
|
import ErrorBoundary from "@components/ErrorBoundary";
|
||||||
|
import { Logger } from "@utils/Logger";
|
||||||
|
import { waitFor } from "@webpack";
|
||||||
|
import { Button, ButtonLooks, ButtonWrapperClasses, Tooltip } from "@webpack/common";
|
||||||
|
import { Channel } from "discord-types/general";
|
||||||
|
import { HTMLProps, MouseEventHandler, ReactNode } from "react";
|
||||||
|
|
||||||
|
let ChannelTextAreaClasses: Record<"button" | "buttonContainer", string>;
|
||||||
|
waitFor(["buttonContainer", "channelTextArea"], m => ChannelTextAreaClasses = m);
|
||||||
|
|
||||||
|
export interface ChatBarProps {
|
||||||
|
channel: Channel;
|
||||||
|
disabled: boolean;
|
||||||
|
isEmpty: boolean;
|
||||||
|
type: {
|
||||||
|
analyticsName: string;
|
||||||
|
attachments: boolean;
|
||||||
|
autocomplete: {
|
||||||
|
addReactionShortcut: boolean,
|
||||||
|
forceChatLayer: boolean,
|
||||||
|
reactions: boolean;
|
||||||
|
},
|
||||||
|
commands: {
|
||||||
|
enabled: boolean;
|
||||||
|
},
|
||||||
|
drafts: {
|
||||||
|
type: number,
|
||||||
|
commandType: number,
|
||||||
|
autoSave: boolean;
|
||||||
|
},
|
||||||
|
emojis: {
|
||||||
|
button: boolean;
|
||||||
|
},
|
||||||
|
gifs: {
|
||||||
|
button: boolean,
|
||||||
|
allowSending: boolean;
|
||||||
|
},
|
||||||
|
gifts: {
|
||||||
|
button: boolean;
|
||||||
|
},
|
||||||
|
permissions: {
|
||||||
|
requireSendMessages: boolean;
|
||||||
|
},
|
||||||
|
showThreadPromptOnReply: boolean,
|
||||||
|
stickers: {
|
||||||
|
button: boolean,
|
||||||
|
allowSending: boolean,
|
||||||
|
autoSuggest: boolean;
|
||||||
|
},
|
||||||
|
users: {
|
||||||
|
allowMentioning: boolean;
|
||||||
|
},
|
||||||
|
submit: {
|
||||||
|
button: boolean,
|
||||||
|
ignorePreference: boolean,
|
||||||
|
disableEnterToSubmit: boolean,
|
||||||
|
clearOnSubmit: boolean,
|
||||||
|
useDisabledStylesOnSubmit: boolean;
|
||||||
|
},
|
||||||
|
uploadLongMessages: boolean,
|
||||||
|
upsellLongMessages: {
|
||||||
|
iconOnly: boolean;
|
||||||
|
},
|
||||||
|
showCharacterCount: boolean,
|
||||||
|
sedReplace: boolean;
|
||||||
|
};
|
||||||
|
}
|
||||||
|
|
||||||
|
export type ChatBarButton = (props: ChatBarProps & { isMainChat: boolean; }) => JSX.Element | null;
|
||||||
|
|
||||||
|
const buttonFactories = new Map<string, ChatBarButton>();
|
||||||
|
const logger = new Logger("ChatButtons");
|
||||||
|
|
||||||
|
export function _injectButtons(buttons: ReactNode[], props: ChatBarProps) {
|
||||||
|
if (props.disabled) return;
|
||||||
|
|
||||||
|
for (const [key, Button] of buttonFactories) {
|
||||||
|
buttons.push(
|
||||||
|
<ErrorBoundary noop key={key} onError={e => logger.error(`Failed to render ${key}`, e.error)}>
|
||||||
|
<Button {...props} isMainChat={props.type.analyticsName === "normal"} />
|
||||||
|
</ErrorBoundary>
|
||||||
|
);
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
export const addChatBarButton = (id: string, button: ChatBarButton) => buttonFactories.set(id, button);
|
||||||
|
export const removeChatBarButton = (id: string) => buttonFactories.delete(id);
|
||||||
|
|
||||||
|
export interface ChatBarButtonProps {
|
||||||
|
children: ReactNode;
|
||||||
|
tooltip: string;
|
||||||
|
onClick: MouseEventHandler<HTMLButtonElement>;
|
||||||
|
onContextMenu?: MouseEventHandler<HTMLButtonElement>;
|
||||||
|
buttonProps?: Omit<HTMLProps<HTMLButtonElement>, "size" | "onClick" | "onContextMenu">;
|
||||||
|
}
|
||||||
|
export const ChatBarButton = ErrorBoundary.wrap((props: ChatBarButtonProps) => {
|
||||||
|
return (
|
||||||
|
<Tooltip text={props.tooltip}>
|
||||||
|
{({ onMouseEnter, onMouseLeave }) => (
|
||||||
|
<div className={`expression-picker-chat-input-button ${ChannelTextAreaClasses?.buttonContainer ?? ""} vc-chatbar-button`}>
|
||||||
|
<Button
|
||||||
|
aria-label={props.tooltip}
|
||||||
|
size=""
|
||||||
|
look={ButtonLooks.BLANK}
|
||||||
|
onMouseEnter={onMouseEnter}
|
||||||
|
onMouseLeave={onMouseLeave}
|
||||||
|
innerClassName={`${ButtonWrapperClasses.button} ${ChannelTextAreaClasses?.button}`}
|
||||||
|
onClick={props.onClick}
|
||||||
|
onContextMenu={props.onContextMenu}
|
||||||
|
{...props.buttonProps}
|
||||||
|
>
|
||||||
|
<div className={ButtonWrapperClasses.buttonWrapper}>
|
||||||
|
{props.children}
|
||||||
|
</div>
|
||||||
|
</Button>
|
||||||
|
</div>
|
||||||
|
)}
|
||||||
|
</Tooltip>
|
||||||
|
);
|
||||||
|
}, { noop: true });
|
|
@ -17,6 +17,7 @@
|
||||||
*/
|
*/
|
||||||
|
|
||||||
import * as $Badges from "./Badges";
|
import * as $Badges from "./Badges";
|
||||||
|
import * as $ChatButtons from "./ChatButtons";
|
||||||
import * as $Commands from "./Commands";
|
import * as $Commands from "./Commands";
|
||||||
import * as $ContextMenu from "./ContextMenu";
|
import * as $ContextMenu from "./ContextMenu";
|
||||||
import * as $DataStore from "./DataStore";
|
import * as $DataStore from "./DataStore";
|
||||||
|
@ -104,3 +105,8 @@ export const Notifications = $Notifications;
|
||||||
* An api allowing you to patch and add/remove items to/from context menus
|
* An api allowing you to patch and add/remove items to/from context menus
|
||||||
*/
|
*/
|
||||||
export const ContextMenu = $ContextMenu;
|
export const ContextMenu = $ContextMenu;
|
||||||
|
|
||||||
|
/**
|
||||||
|
* An API allowing you to add buttons to the chat input
|
||||||
|
*/
|
||||||
|
export const ChatButtons = $ChatButtons;
|
||||||
|
|
|
@ -21,9 +21,11 @@ import { classNameFactory } from "@api/Styles";
|
||||||
import { Flex } from "@components/Flex";
|
import { Flex } from "@components/Flex";
|
||||||
import { DeleteIcon } from "@components/Icons";
|
import { DeleteIcon } from "@components/Icons";
|
||||||
import { Link } from "@components/Link";
|
import { Link } from "@components/Link";
|
||||||
|
import PluginModal from "@components/PluginSettings/PluginModal";
|
||||||
import { openInviteModal } from "@utils/discord";
|
import { openInviteModal } from "@utils/discord";
|
||||||
import { Margins } from "@utils/margins";
|
import { Margins } from "@utils/margins";
|
||||||
import { classes } from "@utils/misc";
|
import { classes } from "@utils/misc";
|
||||||
|
import { openModal } from "@utils/modal";
|
||||||
import { showItemInFolder } from "@utils/native";
|
import { showItemInFolder } from "@utils/native";
|
||||||
import { useAwaiter } from "@utils/react";
|
import { useAwaiter } from "@utils/react";
|
||||||
import { findByPropsLazy, findLazy } from "@webpack";
|
import { findByPropsLazy, findLazy } from "@webpack";
|
||||||
|
@ -248,6 +250,21 @@ function ThemesTab() {
|
||||||
>
|
>
|
||||||
Edit QuickCSS
|
Edit QuickCSS
|
||||||
</Button>
|
</Button>
|
||||||
|
|
||||||
|
{Vencord.Settings.plugins.ClientTheme.enabled && (
|
||||||
|
<Button
|
||||||
|
onClick={() => openModal(modalProps => (
|
||||||
|
<PluginModal
|
||||||
|
{...modalProps}
|
||||||
|
plugin={Vencord.Plugins.plugins.ClientTheme}
|
||||||
|
onRestartNeeded={() => { }}
|
||||||
|
/>
|
||||||
|
))}
|
||||||
|
size={Button.Sizes.SMALL}
|
||||||
|
>
|
||||||
|
Edit ClientTheme
|
||||||
|
</Button>
|
||||||
|
)}
|
||||||
</>
|
</>
|
||||||
</Card>
|
</Card>
|
||||||
|
|
||||||
|
|
|
@ -81,9 +81,12 @@ function HashLink({ repo, hash, disabled = false }: { repo: string, hash: string
|
||||||
|
|
||||||
function Changes({ updates, repo, repoPending }: CommonProps & { updates: typeof changes; }) {
|
function Changes({ updates, repo, repoPending }: CommonProps & { updates: typeof changes; }) {
|
||||||
return (
|
return (
|
||||||
<Card style={{ padding: ".5em" }}>
|
<Card style={{ padding: "0 0.5em" }}>
|
||||||
{updates.map(({ hash, author, message }) => (
|
{updates.map(({ hash, author, message }) => (
|
||||||
<div>
|
<div style={{
|
||||||
|
marginTop: "0.5em",
|
||||||
|
marginBottom: "0.5em"
|
||||||
|
}}>
|
||||||
<code><HashLink {...{ repo, hash }} disabled={repoPending} /></code>
|
<code><HashLink {...{ repo, hash }} disabled={repoPending} /></code>
|
||||||
<span style={{
|
<span style={{
|
||||||
marginLeft: "0.5em",
|
marginLeft: "0.5em",
|
||||||
|
@ -113,7 +116,7 @@ function Updatable(props: CommonProps) {
|
||||||
</>
|
</>
|
||||||
) : (
|
) : (
|
||||||
<Forms.FormText className={Margins.bottom8}>
|
<Forms.FormText className={Margins.bottom8}>
|
||||||
{isOutdated ? `There are ${updates.length} Updates` : "Up to Date!"}
|
{isOutdated ? (updates.length === 1 ? "There is 1 Update" : `There are ${updates.length} Updates`) : "Up to Date!"}
|
||||||
</Forms.FormText>
|
</Forms.FormText>
|
||||||
)}
|
)}
|
||||||
|
|
||||||
|
|
|
@ -83,10 +83,10 @@ function VencordSettings() {
|
||||||
title: "Use Windows' native title bar instead of Discord's custom one",
|
title: "Use Windows' native title bar instead of Discord's custom one",
|
||||||
note: "Requires a full restart"
|
note: "Requires a full restart"
|
||||||
}),
|
}),
|
||||||
!IS_WEB && false /* This causes electron to freeze / white screen for some people */ && {
|
!IS_WEB && {
|
||||||
key: "transparent",
|
key: "transparent",
|
||||||
title: "Enable window transparency",
|
title: "Enable window transparency.",
|
||||||
note: "Requires a full restart"
|
note: "You need a theme that supports transparency or this will do nothing. Will stop the window from being resizable. Requires a full restart"
|
||||||
},
|
},
|
||||||
!IS_WEB && isWindows && {
|
!IS_WEB && isWindows && {
|
||||||
key: "winCtrlQ",
|
key: "winCtrlQ",
|
||||||
|
|
|
@ -139,8 +139,15 @@ export function initIpc(mainWindow: BrowserWindow) {
|
||||||
}
|
}
|
||||||
|
|
||||||
ipcMain.handle(IpcEvents.OPEN_MONACO_EDITOR, async () => {
|
ipcMain.handle(IpcEvents.OPEN_MONACO_EDITOR, async () => {
|
||||||
|
const title = "Vencord QuickCSS Editor";
|
||||||
|
const existingWindow = BrowserWindow.getAllWindows().find(w => w.title === title);
|
||||||
|
if (existingWindow && !existingWindow.isDestroyed()) {
|
||||||
|
existingWindow.focus();
|
||||||
|
return;
|
||||||
|
}
|
||||||
|
|
||||||
const win = new BrowserWindow({
|
const win = new BrowserWindow({
|
||||||
title: "Vencord QuickCSS Editor",
|
title,
|
||||||
autoHideMenuBar: true,
|
autoHideMenuBar: true,
|
||||||
darkTheme: true,
|
darkTheme: true,
|
||||||
webPreferences: {
|
webPreferences: {
|
||||||
|
|
|
@ -79,8 +79,7 @@ if (!IS_VANILLA) {
|
||||||
delete options.frame;
|
delete options.frame;
|
||||||
}
|
}
|
||||||
|
|
||||||
// This causes electron to freeze / white screen for some people
|
if (settings.transparent) {
|
||||||
if ((settings as any).transparentUNSAFE_USE_AT_OWN_RISK) {
|
|
||||||
options.transparent = true;
|
options.transparent = true;
|
||||||
options.backgroundColor = "#00000000";
|
options.backgroundColor = "#00000000";
|
||||||
}
|
}
|
||||||
|
@ -130,6 +129,15 @@ if (!IS_VANILLA) {
|
||||||
});
|
});
|
||||||
|
|
||||||
process.env.DATA_DIR = join(app.getPath("userData"), "..", "Vencord");
|
process.env.DATA_DIR = join(app.getPath("userData"), "..", "Vencord");
|
||||||
|
|
||||||
|
// Monkey patch commandLine to disable WidgetLayering: Fix DevTools context menus https://github.com/electron/electron/issues/38790
|
||||||
|
const originalAppend = app.commandLine.appendSwitch;
|
||||||
|
app.commandLine.appendSwitch = function (...args) {
|
||||||
|
if (args[0] === "disable-features" && !args[1]?.includes("WidgetLayering")) {
|
||||||
|
args[1] += ",WidgetLayering";
|
||||||
|
}
|
||||||
|
return originalAppend.apply(this, args);
|
||||||
|
};
|
||||||
} else {
|
} else {
|
||||||
console.log("[Vencord] Running in vanilla mode. Not loading Vencord");
|
console.log("[Vencord] Running in vanilla mode. Not loading Vencord");
|
||||||
}
|
}
|
||||||
|
|
|
@ -73,6 +73,8 @@ async function build() {
|
||||||
const command = isFlatpak ? "flatpak-spawn" : "node";
|
const command = isFlatpak ? "flatpak-spawn" : "node";
|
||||||
const args = isFlatpak ? ["--host", "node", "scripts/build/build.mjs"] : ["scripts/build/build.mjs"];
|
const args = isFlatpak ? ["--host", "node", "scripts/build/build.mjs"] : ["scripts/build/build.mjs"];
|
||||||
|
|
||||||
|
if (IS_DEV) args.push("--dev");
|
||||||
|
|
||||||
const res = await execFile(command, args, opts);
|
const res = await execFile(command, args, opts);
|
||||||
|
|
||||||
return !res.stderr.includes("Build failed");
|
return !res.stderr.includes("Build failed");
|
||||||
|
|
22
src/plugins/_api/chatButtons.ts
Normal file
22
src/plugins/_api/chatButtons.ts
Normal file
|
@ -0,0 +1,22 @@
|
||||||
|
/*
|
||||||
|
* Vencord, a Discord client mod
|
||||||
|
* Copyright (c) 2024 Vendicated and contributors
|
||||||
|
* SPDX-License-Identifier: GPL-3.0-or-later
|
||||||
|
*/
|
||||||
|
|
||||||
|
import { Devs } from "@utils/constants";
|
||||||
|
import definePlugin from "@utils/types";
|
||||||
|
|
||||||
|
export default definePlugin({
|
||||||
|
name: "ChatInputButtonAPI",
|
||||||
|
description: "API to add buttons to the chat input",
|
||||||
|
authors: [Devs.Ven],
|
||||||
|
|
||||||
|
patches: [{
|
||||||
|
find: 'location:"ChannelTextAreaButtons"',
|
||||||
|
replacement: {
|
||||||
|
match: /if\(!\i\.isMobile\)\{(?=.+?&&(\i)\.push\(.{0,50}"gift")/,
|
||||||
|
replace: "$&Vencord.Api.ChatButtons._injectButtons($1,arguments[0]);"
|
||||||
|
}
|
||||||
|
}]
|
||||||
|
});
|
|
@ -1,94 +0,0 @@
|
||||||
/*
|
|
||||||
* 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 { Settings } from "@api/Settings";
|
|
||||||
import { Devs } from "@utils/constants";
|
|
||||||
import definePlugin, { OptionType } from "@utils/types";
|
|
||||||
|
|
||||||
const enum Methods {
|
|
||||||
Random,
|
|
||||||
Consistent,
|
|
||||||
Timestamp,
|
|
||||||
}
|
|
||||||
|
|
||||||
const tarExtMatcher = /\.tar\.\w+$/;
|
|
||||||
|
|
||||||
export default definePlugin({
|
|
||||||
name: "AnonymiseFileNames",
|
|
||||||
authors: [Devs.obscurity],
|
|
||||||
description: "Anonymise uploaded file names",
|
|
||||||
patches: [
|
|
||||||
{
|
|
||||||
find: "instantBatchUpload:function",
|
|
||||||
replacement: {
|
|
||||||
match: /uploadFiles:(.{1,2}),/,
|
|
||||||
replace:
|
|
||||||
"uploadFiles:(...args)=>(args[0].uploads.forEach(f=>f.filename=$self.anonymise(f.filename)),$1(...args)),",
|
|
||||||
},
|
|
||||||
},
|
|
||||||
],
|
|
||||||
|
|
||||||
options: {
|
|
||||||
method: {
|
|
||||||
description: "Anonymising method",
|
|
||||||
type: OptionType.SELECT,
|
|
||||||
options: [
|
|
||||||
{ label: "Random Characters", value: Methods.Random, default: true },
|
|
||||||
{ label: "Consistent", value: Methods.Consistent },
|
|
||||||
{ label: "Timestamp (4chan-like)", value: Methods.Timestamp },
|
|
||||||
],
|
|
||||||
},
|
|
||||||
randomisedLength: {
|
|
||||||
description: "Random characters length",
|
|
||||||
type: OptionType.NUMBER,
|
|
||||||
default: 7,
|
|
||||||
disabled: () => Settings.plugins.AnonymiseFileNames.method !== Methods.Random,
|
|
||||||
},
|
|
||||||
consistent: {
|
|
||||||
description: "Consistent filename",
|
|
||||||
type: OptionType.STRING,
|
|
||||||
default: "image",
|
|
||||||
disabled: () => Settings.plugins.AnonymiseFileNames.method !== Methods.Consistent,
|
|
||||||
},
|
|
||||||
},
|
|
||||||
|
|
||||||
anonymise(file: string) {
|
|
||||||
let name = "image";
|
|
||||||
const tarMatch = tarExtMatcher.exec(file);
|
|
||||||
const extIdx = tarMatch?.index ?? file.lastIndexOf(".");
|
|
||||||
const ext = extIdx !== -1 ? file.slice(extIdx) : "";
|
|
||||||
|
|
||||||
switch (Settings.plugins.AnonymiseFileNames.method) {
|
|
||||||
case Methods.Random:
|
|
||||||
const chars = "ABCDEFGHIJKLMNOPQRSTUVWXYZabcdefghijklmnopqrstuvwxyz0123456789";
|
|
||||||
name = Array.from(
|
|
||||||
{ length: Settings.plugins.AnonymiseFileNames.randomisedLength },
|
|
||||||
() => chars[Math.floor(Math.random() * chars.length)]
|
|
||||||
).join("");
|
|
||||||
break;
|
|
||||||
case Methods.Consistent:
|
|
||||||
name = Settings.plugins.AnonymiseFileNames.consistent;
|
|
||||||
break;
|
|
||||||
case Methods.Timestamp:
|
|
||||||
// UNIX timestamp in nanos, i could not find a better dependency-less way
|
|
||||||
name = `${Math.floor(Date.now() / 1000)}${Math.floor(window.performance.now())}`;
|
|
||||||
break;
|
|
||||||
}
|
|
||||||
return name + ext;
|
|
||||||
},
|
|
||||||
});
|
|
130
src/plugins/anonymiseFileNames/index.tsx
Normal file
130
src/plugins/anonymiseFileNames/index.tsx
Normal file
|
@ -0,0 +1,130 @@
|
||||||
|
/*
|
||||||
|
* 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 { Upload } from "@api/MessageEvents";
|
||||||
|
import { definePluginSettings } from "@api/Settings";
|
||||||
|
import ErrorBoundary from "@components/ErrorBoundary";
|
||||||
|
import { Devs } from "@utils/constants";
|
||||||
|
import definePlugin, { OptionType } from "@utils/types";
|
||||||
|
import { findByCodeLazy, findByPropsLazy } from "@webpack";
|
||||||
|
|
||||||
|
type AnonUpload = Upload & { anonymise?: boolean; };
|
||||||
|
|
||||||
|
const ActionBarIcon = findByCodeLazy(".actionBarIcon)");
|
||||||
|
const UploadDraft = findByPropsLazy("popFirstFile", "update");
|
||||||
|
|
||||||
|
const enum Methods {
|
||||||
|
Random,
|
||||||
|
Consistent,
|
||||||
|
Timestamp,
|
||||||
|
}
|
||||||
|
|
||||||
|
const tarExtMatcher = /\.tar\.\w+$/;
|
||||||
|
|
||||||
|
const settings = definePluginSettings({
|
||||||
|
anonymiseByDefault: {
|
||||||
|
description: "Whether to anonymise file names by default",
|
||||||
|
type: OptionType.BOOLEAN,
|
||||||
|
default: true,
|
||||||
|
},
|
||||||
|
method: {
|
||||||
|
description: "Anonymising method",
|
||||||
|
type: OptionType.SELECT,
|
||||||
|
options: [
|
||||||
|
{ label: "Random Characters", value: Methods.Random, default: true },
|
||||||
|
{ label: "Consistent", value: Methods.Consistent },
|
||||||
|
{ label: "Timestamp", value: Methods.Timestamp },
|
||||||
|
],
|
||||||
|
},
|
||||||
|
randomisedLength: {
|
||||||
|
description: "Random characters length",
|
||||||
|
type: OptionType.NUMBER,
|
||||||
|
default: 7,
|
||||||
|
disabled: () => settings.store.method !== Methods.Random,
|
||||||
|
},
|
||||||
|
consistent: {
|
||||||
|
description: "Consistent filename",
|
||||||
|
type: OptionType.STRING,
|
||||||
|
default: "image",
|
||||||
|
disabled: () => settings.store.method !== Methods.Consistent,
|
||||||
|
},
|
||||||
|
});
|
||||||
|
|
||||||
|
export default definePlugin({
|
||||||
|
name: "AnonymiseFileNames",
|
||||||
|
authors: [Devs.obscurity],
|
||||||
|
description: "Anonymise uploaded file names",
|
||||||
|
patches: [
|
||||||
|
{
|
||||||
|
find: "instantBatchUpload:function",
|
||||||
|
replacement: {
|
||||||
|
match: /uploadFiles:(.{1,2}),/,
|
||||||
|
replace:
|
||||||
|
"uploadFiles:(...args)=>(args[0].uploads.forEach(f=>f.filename=$self.anonymise(f)),$1(...args)),",
|
||||||
|
},
|
||||||
|
},
|
||||||
|
{
|
||||||
|
find: ".Messages.ATTACHMENT_UTILITIES_SPOILER",
|
||||||
|
replacement: {
|
||||||
|
match: /(?<=children:\[)(?=.{10,80}tooltip:.{0,100}\i\.\i\.Messages\.ATTACHMENT_UTILITIES_SPOILER)/,
|
||||||
|
replace: "arguments[0].canEdit!==false?$self.renderIcon(arguments[0]):null,"
|
||||||
|
},
|
||||||
|
},
|
||||||
|
],
|
||||||
|
settings,
|
||||||
|
|
||||||
|
renderIcon: ErrorBoundary.wrap(({ upload, channelId, draftType }: { upload: AnonUpload; draftType: unknown; channelId: string; }) => {
|
||||||
|
const anonymise = upload.anonymise ?? settings.store.anonymiseByDefault;
|
||||||
|
return (
|
||||||
|
<ActionBarIcon
|
||||||
|
tooltip={anonymise ? "Using anonymous file name" : "Using normal file name"}
|
||||||
|
onClick={() => {
|
||||||
|
upload.anonymise = !anonymise;
|
||||||
|
UploadDraft.update(channelId, upload.id, draftType, {}); // dummy update so component rerenders
|
||||||
|
}}
|
||||||
|
>
|
||||||
|
{anonymise
|
||||||
|
? <svg xmlns="http://www.w3.org/2000/svg" viewBox="0 0 24 24"><path fill="currentColor" d="M17.06 13C15.2 13 13.64 14.33 13.24 16.1C12.29 15.69 11.42 15.8 10.76 16.09C10.35 14.31 8.79 13 6.94 13C4.77 13 3 14.79 3 17C3 19.21 4.77 21 6.94 21C9 21 10.68 19.38 10.84 17.32C11.18 17.08 12.07 16.63 13.16 17.34C13.34 19.39 15 21 17.06 21C19.23 21 21 19.21 21 17C21 14.79 19.23 13 17.06 13M6.94 19.86C5.38 19.86 4.13 18.58 4.13 17S5.39 14.14 6.94 14.14C8.5 14.14 9.75 15.42 9.75 17S8.5 19.86 6.94 19.86M17.06 19.86C15.5 19.86 14.25 18.58 14.25 17S15.5 14.14 17.06 14.14C18.62 14.14 19.88 15.42 19.88 17S18.61 19.86 17.06 19.86M22 10.5H2V12H22V10.5M15.53 2.63C15.31 2.14 14.75 1.88 14.22 2.05L12 2.79L9.77 2.05L9.72 2.04C9.19 1.89 8.63 2.17 8.43 2.68L6 9H18L15.56 2.68L15.53 2.63Z" /></svg>
|
||||||
|
: <svg xmlns="http://www.w3.org/2000/svg" viewBox="0 0 24 24" style={{ transform: "scale(-1,1)" }}><path fill="currentColor" d="M22.11 21.46L2.39 1.73L1.11 3L6.31 8.2L6 9H7.11L8.61 10.5H2V12H10.11L13.5 15.37C13.38 15.61 13.3 15.85 13.24 16.1C12.29 15.69 11.41 15.8 10.76 16.09C10.35 14.31 8.79 13 6.94 13C4.77 13 3 14.79 3 17C3 19.21 4.77 21 6.94 21C9 21 10.68 19.38 10.84 17.32C11.18 17.08 12.07 16.63 13.16 17.34C13.34 19.39 15 21 17.06 21C17.66 21 18.22 20.86 18.72 20.61L20.84 22.73L22.11 21.46M6.94 19.86C5.38 19.86 4.13 18.58 4.13 17C4.13 15.42 5.39 14.14 6.94 14.14C8.5 14.14 9.75 15.42 9.75 17C9.75 18.58 8.5 19.86 6.94 19.86M17.06 19.86C15.5 19.86 14.25 18.58 14.25 17C14.25 16.74 14.29 16.5 14.36 16.25L17.84 19.73C17.59 19.81 17.34 19.86 17.06 19.86M22 12H15.2L13.7 10.5H22V12M17.06 13C19.23 13 21 14.79 21 17C21 17.25 20.97 17.5 20.93 17.73L19.84 16.64C19.68 15.34 18.66 14.32 17.38 14.17L16.29 13.09C16.54 13.03 16.8 13 17.06 13M12.2 9L7.72 4.5L8.43 2.68C8.63 2.17 9.19 1.89 9.72 2.04L9.77 2.05L12 2.79L14.22 2.05C14.75 1.88 15.32 2.14 15.54 2.63L15.56 2.68L18 9H12.2Z" /></svg>
|
||||||
|
}
|
||||||
|
</ActionBarIcon>
|
||||||
|
);
|
||||||
|
}, { noop: true }),
|
||||||
|
|
||||||
|
anonymise(upload: AnonUpload) {
|
||||||
|
if ((upload.anonymise ?? settings.store.anonymiseByDefault) === false) return upload.filename;
|
||||||
|
|
||||||
|
const file = upload.filename;
|
||||||
|
const tarMatch = tarExtMatcher.exec(file);
|
||||||
|
const extIdx = tarMatch?.index ?? file.lastIndexOf(".");
|
||||||
|
const ext = extIdx !== -1 ? file.slice(extIdx) : "";
|
||||||
|
|
||||||
|
switch (settings.store.method) {
|
||||||
|
case Methods.Random:
|
||||||
|
const chars = "ABCDEFGHIJKLMNOPQRSTUVWXYZabcdefghijklmnopqrstuvwxyz0123456789";
|
||||||
|
return Array.from(
|
||||||
|
{ length: settings.store.randomisedLength },
|
||||||
|
() => chars[Math.floor(Math.random() * chars.length)]
|
||||||
|
).join("") + ext;
|
||||||
|
case Methods.Consistent:
|
||||||
|
return settings.store.consistent + ext;
|
||||||
|
case Methods.Timestamp:
|
||||||
|
return Date.now() + ext;
|
||||||
|
}
|
||||||
|
},
|
||||||
|
});
|
23
src/plugins/betterGifPicker/index.ts
Normal file
23
src/plugins/betterGifPicker/index.ts
Normal file
|
@ -0,0 +1,23 @@
|
||||||
|
/*
|
||||||
|
* Vencord, a Discord client mod
|
||||||
|
* Copyright (c) 2024 Vendicated and contributors
|
||||||
|
* SPDX-License-Identifier: GPL-3.0-or-later
|
||||||
|
*/
|
||||||
|
|
||||||
|
import { Devs } from "@utils/constants";
|
||||||
|
import definePlugin from "@utils/types";
|
||||||
|
|
||||||
|
export default definePlugin({
|
||||||
|
name: "BetterGifPicker",
|
||||||
|
description: "Makes the gif picker open the favourite category by default",
|
||||||
|
authors: [Devs.Samwich],
|
||||||
|
patches: [
|
||||||
|
{
|
||||||
|
find: ".GIFPickerResultTypes.SEARCH",
|
||||||
|
replacement: [{
|
||||||
|
match: "this.state={resultType:null}",
|
||||||
|
replace: 'this.state={resultType:"Favorites"}'
|
||||||
|
}]
|
||||||
|
}
|
||||||
|
]
|
||||||
|
});
|
|
@ -82,7 +82,7 @@ export const streamContextPatch: NavContextMenuPatchCallback = (children, { stre
|
||||||
};
|
};
|
||||||
|
|
||||||
export const userContextPatch: NavContextMenuPatchCallback = (children, { user }: UserContextProps) => {
|
export const userContextPatch: NavContextMenuPatchCallback = (children, { user }: UserContextProps) => {
|
||||||
return addViewStreamContext(children, { userId: user.id });
|
if (user) return addViewStreamContext(children, { userId: user.id });
|
||||||
};
|
};
|
||||||
|
|
||||||
export default definePlugin({
|
export default definePlugin({
|
||||||
|
|
|
@ -140,11 +140,11 @@ export const defaultRules = [
|
||||||
"tt_content",
|
"tt_content",
|
||||||
"lr@yandex.*",
|
"lr@yandex.*",
|
||||||
"redircnt@yandex.*",
|
"redircnt@yandex.*",
|
||||||
"feature@youtube.com",
|
"feature@*.youtube.com",
|
||||||
"kw@youtube.com",
|
"kw@*.youtube.com",
|
||||||
"si@youtube.com",
|
"si@*.youtube.com",
|
||||||
"pp@youtube.com",
|
"pp@*.youtube.com",
|
||||||
"si@youtu.be",
|
"si@*.youtu.be",
|
||||||
"wt_zmc",
|
"wt_zmc",
|
||||||
"utm_source",
|
"utm_source",
|
||||||
"utm_content",
|
"utm_content",
|
||||||
|
@ -153,5 +153,6 @@ export const defaultRules = [
|
||||||
"utm_term",
|
"utm_term",
|
||||||
"si@open.spotify.com",
|
"si@open.spotify.com",
|
||||||
"igshid",
|
"igshid",
|
||||||
|
"igsh",
|
||||||
"share_id@reddit.com",
|
"share_id@reddit.com",
|
||||||
];
|
];
|
||||||
|
|
|
@ -19,6 +19,16 @@
|
||||||
border: thin solid var(--background-modifier-accent) !important;
|
border: thin solid var(--background-modifier-accent) !important;
|
||||||
}
|
}
|
||||||
|
|
||||||
.client-theme-warning {
|
.client-theme-warning * {
|
||||||
color: var(--text-danger);
|
color: var(--text-danger);
|
||||||
}
|
}
|
||||||
|
|
||||||
|
.client-theme-contrast-warning {
|
||||||
|
background-color: var(--background-primary);
|
||||||
|
padding: 0.5rem;
|
||||||
|
border-radius: .5rem;
|
||||||
|
display: flex;
|
||||||
|
flex-direction: row;
|
||||||
|
justify-content: space-between;
|
||||||
|
align-items: center;
|
||||||
|
}
|
||||||
|
|
|
@ -8,19 +8,19 @@ import "./clientTheme.css";
|
||||||
|
|
||||||
import { definePluginSettings } from "@api/Settings";
|
import { definePluginSettings } from "@api/Settings";
|
||||||
import { Devs } from "@utils/constants";
|
import { Devs } from "@utils/constants";
|
||||||
import { getTheme, Theme } from "@utils/discord";
|
|
||||||
import { Margins } from "@utils/margins";
|
import { Margins } from "@utils/margins";
|
||||||
import { classes } from "@utils/misc";
|
import { classes } from "@utils/misc";
|
||||||
import definePlugin, { OptionType, StartAt } from "@utils/types";
|
import definePlugin, { OptionType, StartAt } from "@utils/types";
|
||||||
import { findComponentByCodeLazy } from "@webpack";
|
import { findByPropsLazy, findComponentByCodeLazy, findStoreLazy } from "@webpack";
|
||||||
import { Button, Forms } from "@webpack/common";
|
import { Button, Forms, lodash as _, useStateFromStores } from "@webpack/common";
|
||||||
|
|
||||||
const ColorPicker = findComponentByCodeLazy(".Messages.USER_SETTINGS_PROFILE_COLOR_SELECT_COLOR", ".BACKGROUND_PRIMARY)");
|
const ColorPicker = findComponentByCodeLazy(".Messages.USER_SETTINGS_PROFILE_COLOR_SELECT_COLOR", ".BACKGROUND_PRIMARY)");
|
||||||
|
|
||||||
const colorPresets = [
|
const colorPresets = [
|
||||||
"#1E1514", "#172019", "#13171B", "#1C1C28", "#402D2D",
|
"#1E1514", "#172019", "#13171B", "#1C1C28", "#402D2D",
|
||||||
"#3A483D", "#344242", "#313D4B", "#2D2F47", "#322B42",
|
"#3A483D", "#344242", "#313D4B", "#2D2F47", "#322B42",
|
||||||
"#3C2E42", "#422938"
|
"#3C2E42", "#422938", "#b6908f", "#bfa088", "#d3c77d",
|
||||||
|
"#86ac86", "#88aab3", "#8693b5", "#8a89ba", "#ad94bb",
|
||||||
];
|
];
|
||||||
|
|
||||||
function onPickColor(color: number) {
|
function onPickColor(color: number) {
|
||||||
|
@ -30,9 +30,35 @@ function onPickColor(color: number) {
|
||||||
updateColorVars(hexColor);
|
updateColorVars(hexColor);
|
||||||
}
|
}
|
||||||
|
|
||||||
|
const { saveClientTheme } = findByPropsLazy("saveClientTheme");
|
||||||
|
|
||||||
|
function setTheme(theme: string) {
|
||||||
|
saveClientTheme({ theme });
|
||||||
|
}
|
||||||
|
|
||||||
|
const ThemeStore = findStoreLazy("ThemeStore");
|
||||||
|
const NitroThemeStore = findStoreLazy("ClientThemesBackgroundStore");
|
||||||
|
|
||||||
function ThemeSettings() {
|
function ThemeSettings() {
|
||||||
const lightnessWarning = hexToLightness(settings.store.color) > 45;
|
const theme = useStateFromStores([ThemeStore], () => ThemeStore.theme);
|
||||||
const lightModeWarning = getTheme() === Theme.Light;
|
const isLightTheme = theme === "light";
|
||||||
|
const oppositeTheme = isLightTheme ? "dark" : "light";
|
||||||
|
|
||||||
|
const nitroTheme = useStateFromStores([NitroThemeStore], () => NitroThemeStore.gradientPreset);
|
||||||
|
const nitroThemeEnabled = nitroTheme !== undefined;
|
||||||
|
|
||||||
|
const selectedLuminance = relativeLuminance(settings.store.color);
|
||||||
|
|
||||||
|
let contrastWarning = false, fixableContrast = true;
|
||||||
|
if ((isLightTheme && selectedLuminance < 0.26) || !isLightTheme && selectedLuminance > 0.12)
|
||||||
|
contrastWarning = true;
|
||||||
|
if (selectedLuminance < 0.26 && selectedLuminance > 0.12)
|
||||||
|
fixableContrast = false;
|
||||||
|
// light mode with values greater than 65 leads to background colors getting crushed together and poor text contrast for muted channels
|
||||||
|
if (isLightTheme && selectedLuminance > 0.65) {
|
||||||
|
contrastWarning = true;
|
||||||
|
fixableContrast = false;
|
||||||
|
}
|
||||||
|
|
||||||
return (
|
return (
|
||||||
<div className="client-theme-settings">
|
<div className="client-theme-settings">
|
||||||
|
@ -48,15 +74,18 @@ function ThemeSettings() {
|
||||||
suggestedColors={colorPresets}
|
suggestedColors={colorPresets}
|
||||||
/>
|
/>
|
||||||
</div>
|
</div>
|
||||||
{lightnessWarning || lightModeWarning
|
{(contrastWarning || nitroThemeEnabled) && (<>
|
||||||
? <div>
|
<Forms.FormDivider className={classes(Margins.top8, Margins.bottom8)} />
|
||||||
<Forms.FormDivider className={classes(Margins.top8, Margins.bottom8)} />
|
<div className={`client-theme-contrast-warning ${contrastWarning ? (isLightTheme ? "theme-dark" : "theme-light") : ""}`}>
|
||||||
<Forms.FormText className="client-theme-warning">Your theme won't look good:</Forms.FormText>
|
<div className="client-theme-warning">
|
||||||
{lightnessWarning && <Forms.FormText className="client-theme-warning">Selected color is very light</Forms.FormText>}
|
<Forms.FormText>Warning, your theme won't look good:</Forms.FormText>
|
||||||
{lightModeWarning && <Forms.FormText className="client-theme-warning">Light mode isn't supported</Forms.FormText>}
|
{contrastWarning && <Forms.FormText>Selected color won't contrast well with text</Forms.FormText>}
|
||||||
|
{nitroThemeEnabled && <Forms.FormText>Nitro themes aren't supported</Forms.FormText>}
|
||||||
|
</div>
|
||||||
|
{(contrastWarning && fixableContrast) && <Button onClick={() => setTheme(oppositeTheme)} color={Button.Colors.RED}>Switch to {oppositeTheme} mode</Button>}
|
||||||
|
{(nitroThemeEnabled) && <Button onClick={() => setTheme(theme)} color={Button.Colors.RED}>Disable Nitro Theme</Button>}
|
||||||
</div>
|
</div>
|
||||||
: null
|
</>)}
|
||||||
}
|
|
||||||
</div>
|
</div>
|
||||||
);
|
);
|
||||||
}
|
}
|
||||||
|
@ -87,9 +116,12 @@ export default definePlugin({
|
||||||
settings,
|
settings,
|
||||||
|
|
||||||
startAt: StartAt.DOMContentLoaded,
|
startAt: StartAt.DOMContentLoaded,
|
||||||
start() {
|
async start() {
|
||||||
updateColorVars(settings.store.color);
|
updateColorVars(settings.store.color);
|
||||||
generateColorOffsets();
|
|
||||||
|
const styles = await getStyles();
|
||||||
|
generateColorOffsets(styles);
|
||||||
|
generateLightModeFixes(styles);
|
||||||
},
|
},
|
||||||
|
|
||||||
stop() {
|
stop() {
|
||||||
|
@ -98,56 +130,86 @@ export default definePlugin({
|
||||||
}
|
}
|
||||||
});
|
});
|
||||||
|
|
||||||
const variableRegex = /(--primary-[5-9]\d{2}-hsl):.*?(\S*)%;/g;
|
const variableRegex = /(--primary-\d{3}-hsl):.*?(\S*)%;/g;
|
||||||
|
const lightVariableRegex = /^--primary-[1-5]\d{2}-hsl/g;
|
||||||
|
const darkVariableRegex = /^--primary-[5-9]\d{2}-hsl/g;
|
||||||
|
|
||||||
async function generateColorOffsets() {
|
// generates variables per theme by:
|
||||||
|
// - matching regex (so we can limit what variables are included in light/dark theme, otherwise text becomes unreadable)
|
||||||
const styleLinkNodes = document.querySelectorAll('link[rel="stylesheet"]');
|
// - offset from specified center (light/dark theme get different offsets because light uses 100 for background-primary, while dark uses 600)
|
||||||
const variableLightness = {} as Record<string, number>;
|
function genThemeSpecificOffsets(variableLightness: Record<string, number>, regex: RegExp, centerVariable: string): string {
|
||||||
|
return Object.entries(variableLightness).filter(([key]) => key.search(regex) > -1)
|
||||||
// Search all stylesheets for color variables
|
|
||||||
for (const styleLinkNode of styleLinkNodes) {
|
|
||||||
const cssLink = styleLinkNode.getAttribute("href");
|
|
||||||
if (!cssLink) continue;
|
|
||||||
|
|
||||||
const res = await fetch(cssLink);
|
|
||||||
const cssString = await res.text();
|
|
||||||
|
|
||||||
// Get lightness values of --primary variables >=500
|
|
||||||
let variableMatch = variableRegex.exec(cssString);
|
|
||||||
while (variableMatch !== null) {
|
|
||||||
const [, variable, lightness] = variableMatch;
|
|
||||||
variableLightness[variable] = parseFloat(lightness);
|
|
||||||
variableMatch = variableRegex.exec(cssString);
|
|
||||||
}
|
|
||||||
}
|
|
||||||
|
|
||||||
// Generate offsets
|
|
||||||
const lightnessOffsets = Object.entries(variableLightness)
|
|
||||||
.map(([key, lightness]) => {
|
.map(([key, lightness]) => {
|
||||||
const lightnessOffset = lightness - variableLightness["--primary-600-hsl"];
|
const lightnessOffset = lightness - variableLightness[centerVariable];
|
||||||
const plusOrMinus = lightnessOffset >= 0 ? "+" : "-";
|
const plusOrMinus = lightnessOffset >= 0 ? "+" : "-";
|
||||||
return `${key}: var(--theme-h) var(--theme-s) calc(var(--theme-l) ${plusOrMinus} ${Math.abs(lightnessOffset).toFixed(2)}%);`;
|
return `${key}: var(--theme-h) var(--theme-s) calc(var(--theme-l) ${plusOrMinus} ${Math.abs(lightnessOffset).toFixed(2)}%);`;
|
||||||
})
|
})
|
||||||
.join("\n");
|
.join("\n");
|
||||||
|
}
|
||||||
|
|
||||||
const style = document.createElement("style");
|
|
||||||
style.setAttribute("id", "clientThemeOffsets");
|
function generateColorOffsets(styles) {
|
||||||
style.textContent = `:root:root {
|
const variableLightness = {} as Record<string, number>;
|
||||||
${lightnessOffsets}
|
|
||||||
}`;
|
// Get lightness values of --primary variables
|
||||||
document.head.appendChild(style);
|
let variableMatch = variableRegex.exec(styles);
|
||||||
|
while (variableMatch !== null) {
|
||||||
|
const [, variable, lightness] = variableMatch;
|
||||||
|
variableLightness[variable] = parseFloat(lightness);
|
||||||
|
variableMatch = variableRegex.exec(styles);
|
||||||
|
}
|
||||||
|
|
||||||
|
createStyleSheet("clientThemeOffsets", [
|
||||||
|
`.theme-light {\n ${genThemeSpecificOffsets(variableLightness, lightVariableRegex, "--primary-345-hsl")} \n}`,
|
||||||
|
`.theme-dark {\n ${genThemeSpecificOffsets(variableLightness, darkVariableRegex, "--primary-600-hsl")} \n}`,
|
||||||
|
].join("\n\n"));
|
||||||
|
}
|
||||||
|
|
||||||
|
function generateLightModeFixes(styles) {
|
||||||
|
const groupLightUsesW500Regex = /\.theme-light[^{]*\{[^}]*var\(--white-500\)[^}]*}/gm;
|
||||||
|
// get light capturing groups that mention --white-500
|
||||||
|
const relevantStyles = [...styles.matchAll(groupLightUsesW500Regex)].flat();
|
||||||
|
|
||||||
|
const groupBackgroundRegex = /^([^{]*)\{background:var\(--white-500\)/m;
|
||||||
|
const groupBackgroundColorRegex = /^([^{]*)\{background-color:var\(--white-500\)/m;
|
||||||
|
// find all capturing groups that assign background or background-color directly to w500
|
||||||
|
const backgroundGroups = mapReject(relevantStyles, entry => captureOne(entry, groupBackgroundRegex)).join(",\n");
|
||||||
|
const backgroundColorGroups = mapReject(relevantStyles, entry => captureOne(entry, groupBackgroundColorRegex)).join(",\n");
|
||||||
|
// create css to reassign them to --primary-100
|
||||||
|
const reassignBackgrounds = `${backgroundGroups} {\n background: var(--primary-100) \n}`;
|
||||||
|
const reassignBackgroundColors = `${backgroundColorGroups} {\n background-color: var(--primary-100) \n}`;
|
||||||
|
|
||||||
|
const groupBgVarRegex = /\.theme-light\{([^}]*--[^:}]*(?:background|bg)[^:}]*:var\(--white-500\)[^}]*)\}/m;
|
||||||
|
const bgVarRegex = /^(--[^:]*(?:background|bg)[^:]*):var\(--white-500\)/m;
|
||||||
|
// get all global variables used for backgrounds
|
||||||
|
const lightVars = mapReject(relevantStyles, style => captureOne(style, groupBgVarRegex)) // get the insides of capture groups that have at least one background var with w500
|
||||||
|
.map(str => str.split(";")).flat(); // captureGroupInsides[] -> cssRule[]
|
||||||
|
const lightBgVars = mapReject(lightVars, variable => captureOne(variable, bgVarRegex)); // remove vars that aren't for backgrounds or w500
|
||||||
|
// create css to reassign every var
|
||||||
|
const reassignVariables = `.theme-light {\n ${lightBgVars.map(variable => `${variable}: var(--primary-100);`).join("\n")} \n}`;
|
||||||
|
|
||||||
|
createStyleSheet("clientThemeLightModeFixes", [
|
||||||
|
reassignBackgrounds,
|
||||||
|
reassignBackgroundColors,
|
||||||
|
reassignVariables,
|
||||||
|
].join("\n\n"));
|
||||||
|
}
|
||||||
|
|
||||||
|
function captureOne(str, regex) {
|
||||||
|
const result = str.match(regex);
|
||||||
|
return (result === null) ? null : result[1];
|
||||||
|
}
|
||||||
|
|
||||||
|
function mapReject(arr, mapFunc, rejectFunc = _.isNull) {
|
||||||
|
return _.reject(arr.map(mapFunc), rejectFunc);
|
||||||
}
|
}
|
||||||
|
|
||||||
function updateColorVars(color: string) {
|
function updateColorVars(color: string) {
|
||||||
const { hue, saturation, lightness } = hexToHSL(color);
|
const { hue, saturation, lightness } = hexToHSL(color);
|
||||||
|
|
||||||
let style = document.getElementById("clientThemeVars");
|
let style = document.getElementById("clientThemeVars");
|
||||||
if (!style) {
|
if (!style)
|
||||||
style = document.createElement("style");
|
style = createStyleSheet("clientThemeVars");
|
||||||
style.setAttribute("id", "clientThemeVars");
|
|
||||||
document.head.appendChild(style);
|
|
||||||
}
|
|
||||||
|
|
||||||
style.textContent = `:root {
|
style.textContent = `:root {
|
||||||
--theme-h: ${hue};
|
--theme-h: ${hue};
|
||||||
|
@ -156,6 +218,28 @@ function updateColorVars(color: string) {
|
||||||
}`;
|
}`;
|
||||||
}
|
}
|
||||||
|
|
||||||
|
function createStyleSheet(id, content = "") {
|
||||||
|
const style = document.createElement("style");
|
||||||
|
style.setAttribute("id", id);
|
||||||
|
style.textContent = content.split("\n").map(line => line.trim()).join("\n");
|
||||||
|
document.body.appendChild(style);
|
||||||
|
return style;
|
||||||
|
}
|
||||||
|
|
||||||
|
// returns all of discord's native styles in a single string
|
||||||
|
async function getStyles(): Promise<string> {
|
||||||
|
let out = "";
|
||||||
|
const styleLinkNodes = document.querySelectorAll('link[rel="stylesheet"]');
|
||||||
|
for (const styleLinkNode of styleLinkNodes) {
|
||||||
|
const cssLink = styleLinkNode.getAttribute("href");
|
||||||
|
if (!cssLink) continue;
|
||||||
|
|
||||||
|
const res = await fetch(cssLink);
|
||||||
|
out += await res.text();
|
||||||
|
}
|
||||||
|
return out;
|
||||||
|
}
|
||||||
|
|
||||||
// https://css-tricks.com/converting-color-spaces-in-javascript/
|
// https://css-tricks.com/converting-color-spaces-in-javascript/
|
||||||
function hexToHSL(hexCode: string) {
|
function hexToHSL(hexCode: string) {
|
||||||
// Hex => RGB normalized to 0-1
|
// Hex => RGB normalized to 0-1
|
||||||
|
@ -198,17 +282,14 @@ function hexToHSL(hexCode: string) {
|
||||||
return { hue, saturation, lightness };
|
return { hue, saturation, lightness };
|
||||||
}
|
}
|
||||||
|
|
||||||
// Minimized math just for lightness, lowers lag when changing colors
|
// https://www.w3.org/TR/WCAG21/#dfn-relative-luminance
|
||||||
function hexToLightness(hexCode: string) {
|
function relativeLuminance(hexCode: string) {
|
||||||
// Hex => RGB normalized to 0-1
|
const normalize = (x: number) =>
|
||||||
const r = parseInt(hexCode.substring(0, 2), 16) / 255;
|
x <= 0.03928 ? x / 12.92 : ((x + 0.055) / 1.055) ** 2.4;
|
||||||
const g = parseInt(hexCode.substring(2, 4), 16) / 255;
|
|
||||||
const b = parseInt(hexCode.substring(4, 6), 16) / 255;
|
|
||||||
|
|
||||||
const cMax = Math.max(r, g, b);
|
const r = normalize(parseInt(hexCode.substring(0, 2), 16) / 255);
|
||||||
const cMin = Math.min(r, g, b);
|
const g = normalize(parseInt(hexCode.substring(2, 4), 16) / 255);
|
||||||
|
const b = normalize(parseInt(hexCode.substring(4, 6), 16) / 255);
|
||||||
|
|
||||||
const lightness = 100 * ((cMax + cMin) / 2);
|
return r * 0.2126 + g * 0.7152 + b * 0.0722;
|
||||||
|
|
||||||
return lightness;
|
|
||||||
}
|
}
|
||||||
|
|
|
@ -30,6 +30,8 @@ interface UserContextProps {
|
||||||
}
|
}
|
||||||
|
|
||||||
const UserContextMenuPatch: NavContextMenuPatchCallback = (children, { user }: UserContextProps) => () => {
|
const UserContextMenuPatch: NavContextMenuPatchCallback = (children, { user }: UserContextProps) => () => {
|
||||||
|
if (!user) return;
|
||||||
|
|
||||||
children.push(
|
children.push(
|
||||||
<Menu.MenuItem
|
<Menu.MenuItem
|
||||||
id="vc-copy-user-url"
|
id="vc-copy-user-url"
|
||||||
|
|
|
@ -6,21 +6,17 @@
|
||||||
|
|
||||||
import "./ui/styles.css";
|
import "./ui/styles.css";
|
||||||
|
|
||||||
import { definePluginSettings } from "@api/Settings";
|
|
||||||
import ErrorBoundary from "@components/ErrorBoundary";
|
import ErrorBoundary from "@components/ErrorBoundary";
|
||||||
import { Link } from "@components/Link";
|
|
||||||
import { Devs } from "@utils/constants";
|
import { Devs } from "@utils/constants";
|
||||||
import { Margins } from "@utils/margins";
|
import definePlugin from "@utils/types";
|
||||||
import { classes } from "@utils/misc";
|
|
||||||
import { closeAllModals } from "@utils/modal";
|
|
||||||
import definePlugin, { OptionType } from "@utils/types";
|
|
||||||
import { findByPropsLazy } from "@webpack";
|
import { findByPropsLazy } from "@webpack";
|
||||||
import { FluxDispatcher, Forms, UserStore } from "@webpack/common";
|
import { UserStore } from "@webpack/common";
|
||||||
|
|
||||||
import { CDN_URL, RAW_SKU_ID, SKU_ID } from "./lib/constants";
|
import { CDN_URL, RAW_SKU_ID, SKU_ID } from "./lib/constants";
|
||||||
import { useAuthorizationStore } from "./lib/stores/AuthorizationStore";
|
import { useAuthorizationStore } from "./lib/stores/AuthorizationStore";
|
||||||
import { useCurrentUserDecorationsStore } from "./lib/stores/CurrentUserDecorationsStore";
|
import { useCurrentUserDecorationsStore } from "./lib/stores/CurrentUserDecorationsStore";
|
||||||
import { useUserDecorAvatarDecoration, useUsersDecorationsStore } from "./lib/stores/UsersDecorationsStore";
|
import { useUserDecorAvatarDecoration, useUsersDecorationsStore } from "./lib/stores/UsersDecorationsStore";
|
||||||
|
import { settings } from "./settings";
|
||||||
import { setDecorationGridDecoration, setDecorationGridItem } from "./ui/components";
|
import { setDecorationGridDecoration, setDecorationGridItem } from "./ui/components";
|
||||||
import DecorSection from "./ui/components/DecorSection";
|
import DecorSection from "./ui/components/DecorSection";
|
||||||
|
|
||||||
|
@ -30,27 +26,6 @@ export interface AvatarDecoration {
|
||||||
skuId: string;
|
skuId: string;
|
||||||
}
|
}
|
||||||
|
|
||||||
const settings = definePluginSettings({
|
|
||||||
changeDecoration: {
|
|
||||||
type: OptionType.COMPONENT,
|
|
||||||
description: "Change your avatar decoration",
|
|
||||||
component() {
|
|
||||||
return <div>
|
|
||||||
<DecorSection hideTitle hideDivider noMargin />
|
|
||||||
<Forms.FormText type="description" className={classes(Margins.top8, Margins.bottom8)}>
|
|
||||||
You can also access Decor decorations from the <Link
|
|
||||||
href="/settings/profile-customization"
|
|
||||||
onClick={e => {
|
|
||||||
e.preventDefault();
|
|
||||||
closeAllModals();
|
|
||||||
FluxDispatcher.dispatch({ type: "USER_SETTINGS_MODAL_SET_SECTION", section: "Profile Customization" });
|
|
||||||
}}
|
|
||||||
>Profiles</Link> page.
|
|
||||||
</Forms.FormText>
|
|
||||||
</div>;
|
|
||||||
}
|
|
||||||
}
|
|
||||||
});
|
|
||||||
export default definePlugin({
|
export default definePlugin({
|
||||||
name: "Decor",
|
name: "Decor",
|
||||||
description: "Create and use your own custom avatar decorations, or pick your favorite from the presets.",
|
description: "Create and use your own custom avatar decorations, or pick your favorite from the presets.",
|
||||||
|
|
47
src/plugins/decor/settings.tsx
Normal file
47
src/plugins/decor/settings.tsx
Normal file
|
@ -0,0 +1,47 @@
|
||||||
|
/*
|
||||||
|
* Vencord, a Discord client mod
|
||||||
|
* Copyright (c) 2023 Vendicated and contributors
|
||||||
|
* SPDX-License-Identifier: GPL-3.0-or-later
|
||||||
|
*/
|
||||||
|
|
||||||
|
import { definePluginSettings } from "@api/Settings";
|
||||||
|
import { Link } from "@components/Link";
|
||||||
|
import { Margins } from "@utils/margins";
|
||||||
|
import { classes } from "@utils/misc";
|
||||||
|
import { closeAllModals } from "@utils/modal";
|
||||||
|
import { OptionType } from "@utils/types";
|
||||||
|
import { FluxDispatcher, Forms } from "@webpack/common";
|
||||||
|
|
||||||
|
import DecorSection from "./ui/components/DecorSection";
|
||||||
|
|
||||||
|
export const settings = definePluginSettings({
|
||||||
|
changeDecoration: {
|
||||||
|
type: OptionType.COMPONENT,
|
||||||
|
description: "Change your avatar decoration",
|
||||||
|
component() {
|
||||||
|
if (!Vencord.Plugins.plugins.Decor.started) return <Forms.FormText>
|
||||||
|
Enable Decor and restart your client to change your avatar decoration.
|
||||||
|
</Forms.FormText>;
|
||||||
|
|
||||||
|
return <div>
|
||||||
|
<DecorSection hideTitle hideDivider noMargin />
|
||||||
|
<Forms.FormText type="description" className={classes(Margins.top8, Margins.bottom8)}>
|
||||||
|
You can also access Decor decorations from the <Link
|
||||||
|
href="/settings/profile-customization"
|
||||||
|
onClick={e => {
|
||||||
|
e.preventDefault();
|
||||||
|
closeAllModals();
|
||||||
|
FluxDispatcher.dispatch({ type: "USER_SETTINGS_MODAL_SET_SECTION", section: "Profile Customization" });
|
||||||
|
}}
|
||||||
|
>Profiles</Link> page.
|
||||||
|
</Forms.FormText>
|
||||||
|
</div>;
|
||||||
|
}
|
||||||
|
},
|
||||||
|
agreedToGuidelines: {
|
||||||
|
type: OptionType.BOOLEAN,
|
||||||
|
description: "Agreed to guidelines",
|
||||||
|
hidden: true,
|
||||||
|
default: false
|
||||||
|
}
|
||||||
|
});
|
|
@ -19,7 +19,7 @@ export let DecorationGridItem: DecorationGridItemComponent;
|
||||||
export const setDecorationGridItem = v => DecorationGridItem = v;
|
export const setDecorationGridItem = v => DecorationGridItem = v;
|
||||||
|
|
||||||
export const AvatarDecorationModalPreview = LazyComponentWebpack(() => {
|
export const AvatarDecorationModalPreview = LazyComponentWebpack(() => {
|
||||||
const component = findComponentByCode("AvatarDecorationModalPreview");
|
const component = findComponentByCode(".shopPreviewBanner");
|
||||||
return React.memo(component);
|
return React.memo(component);
|
||||||
});
|
});
|
||||||
|
|
||||||
|
|
|
@ -5,9 +5,10 @@
|
||||||
*/
|
*/
|
||||||
|
|
||||||
import { classNameFactory } from "@api/Styles";
|
import { classNameFactory } from "@api/Styles";
|
||||||
import { extractAndLoadChunksLazy } from "@webpack";
|
import { extractAndLoadChunksLazy, findByPropsLazy } from "@webpack";
|
||||||
|
|
||||||
export const cl = classNameFactory("vc-decor-");
|
export const cl = classNameFactory("vc-decor-");
|
||||||
|
export const DecorationModalStyles = findByPropsLazy("modalFooterShopButton");
|
||||||
|
|
||||||
export const requireAvatarDecorationModal = extractAndLoadChunksLazy(["openAvatarDecorationModal:"]);
|
export const requireAvatarDecorationModal = extractAndLoadChunksLazy(["openAvatarDecorationModal:"]);
|
||||||
export const requireCreateStickerModal = extractAndLoadChunksLazy(["stickerInspected]:"]);
|
export const requireCreateStickerModal = extractAndLoadChunksLazy(["stickerInspected]:"]);
|
||||||
|
|
|
@ -4,12 +4,13 @@
|
||||||
* SPDX-License-Identifier: GPL-3.0-or-later
|
* SPDX-License-Identifier: GPL-3.0-or-later
|
||||||
*/
|
*/
|
||||||
|
|
||||||
|
import ErrorBoundary from "@components/ErrorBoundary";
|
||||||
import { Flex } from "@components/Flex";
|
import { Flex } from "@components/Flex";
|
||||||
import { openInviteModal } from "@utils/discord";
|
import { openInviteModal } from "@utils/discord";
|
||||||
import { Margins } from "@utils/margins";
|
import { Margins } from "@utils/margins";
|
||||||
import { classes } from "@utils/misc";
|
import { classes } from "@utils/misc";
|
||||||
import { closeAllModals, ModalCloseButton, ModalContent, ModalFooter, ModalHeader, ModalRoot, ModalSize, openModal } from "@utils/modal";
|
import { closeAllModals, ModalCloseButton, ModalContent, ModalFooter, ModalHeader, ModalProps, ModalRoot, ModalSize, openModal } from "@utils/modal";
|
||||||
import { findByPropsLazy, findComponentByCodeLazy } from "@webpack";
|
import { findComponentByCodeLazy } from "@webpack";
|
||||||
import { Alerts, Button, FluxDispatcher, Forms, GuildStore, NavigationRouter, Parser, Text, Tooltip, useEffect, UserStore, UserUtils, useState } from "@webpack/common";
|
import { Alerts, Button, FluxDispatcher, Forms, GuildStore, NavigationRouter, Parser, Text, Tooltip, useEffect, UserStore, UserUtils, useState } from "@webpack/common";
|
||||||
import { User } from "discord-types/general";
|
import { User } from "discord-types/general";
|
||||||
|
|
||||||
|
@ -18,16 +19,17 @@ import { GUILD_ID, INVITE_KEY } from "../../lib/constants";
|
||||||
import { useAuthorizationStore } from "../../lib/stores/AuthorizationStore";
|
import { useAuthorizationStore } from "../../lib/stores/AuthorizationStore";
|
||||||
import { useCurrentUserDecorationsStore } from "../../lib/stores/CurrentUserDecorationsStore";
|
import { useCurrentUserDecorationsStore } from "../../lib/stores/CurrentUserDecorationsStore";
|
||||||
import { decorationToAvatarDecoration } from "../../lib/utils/decoration";
|
import { decorationToAvatarDecoration } from "../../lib/utils/decoration";
|
||||||
import { cl, requireAvatarDecorationModal } from "../";
|
import { settings } from "../../settings";
|
||||||
|
import { cl, DecorationModalStyles, requireAvatarDecorationModal } from "../";
|
||||||
import { AvatarDecorationModalPreview } from "../components";
|
import { AvatarDecorationModalPreview } from "../components";
|
||||||
import DecorationGridCreate from "../components/DecorationGridCreate";
|
import DecorationGridCreate from "../components/DecorationGridCreate";
|
||||||
import DecorationGridNone from "../components/DecorationGridNone";
|
import DecorationGridNone from "../components/DecorationGridNone";
|
||||||
import DecorDecorationGridDecoration from "../components/DecorDecorationGridDecoration";
|
import DecorDecorationGridDecoration from "../components/DecorDecorationGridDecoration";
|
||||||
import SectionedGridList from "../components/SectionedGridList";
|
import SectionedGridList from "../components/SectionedGridList";
|
||||||
import { openCreateDecorationModal } from "./CreateDecorationModal";
|
import { openCreateDecorationModal } from "./CreateDecorationModal";
|
||||||
|
import { openGuidelinesModal } from "./GuidelinesModal";
|
||||||
|
|
||||||
const UserSummaryItem = findComponentByCodeLazy("defaultRenderUser", "showDefaultAvatarsForNullUsers");
|
const UserSummaryItem = findComponentByCodeLazy("defaultRenderUser", "showDefaultAvatarsForNullUsers");
|
||||||
const DecorationModalStyles = findByPropsLazy("modalFooterShopButton");
|
|
||||||
|
|
||||||
function usePresets() {
|
function usePresets() {
|
||||||
const [presets, setPresets] = useState<Preset[]>([]);
|
const [presets, setPresets] = useState<Preset[]>([]);
|
||||||
|
@ -83,7 +85,7 @@ function SectionHeader({ section }: { section: Section; }) {
|
||||||
</div>;
|
</div>;
|
||||||
}
|
}
|
||||||
|
|
||||||
export default function ChangeDecorationModal(props: any) {
|
function ChangeDecorationModal(props: ModalProps) {
|
||||||
// undefined = not trying, null = none, Decoration = selected
|
// undefined = not trying, null = none, Decoration = selected
|
||||||
const [tryingDecoration, setTryingDecoration] = useState<Decoration | null | undefined>(undefined);
|
const [tryingDecoration, setTryingDecoration] = useState<Decoration | null | undefined>(undefined);
|
||||||
const isTryingDecoration = typeof tryingDecoration !== "undefined";
|
const isTryingDecoration = typeof tryingDecoration !== "undefined";
|
||||||
|
@ -116,6 +118,7 @@ export default function ChangeDecorationModal(props: any) {
|
||||||
const data = [
|
const data = [
|
||||||
{
|
{
|
||||||
title: "Your Decorations",
|
title: "Your Decorations",
|
||||||
|
subtitle: "You can delete your own decorations by right clicking on them.",
|
||||||
sectionKey: "ownDecorations",
|
sectionKey: "ownDecorations",
|
||||||
items: ["none", ...ownDecorations, "create"]
|
items: ["none", ...ownDecorations, "create"]
|
||||||
},
|
},
|
||||||
|
@ -148,60 +151,62 @@ export default function ChangeDecorationModal(props: any) {
|
||||||
className={cl("change-decoration-modal-content")}
|
className={cl("change-decoration-modal-content")}
|
||||||
scrollbarType="none"
|
scrollbarType="none"
|
||||||
>
|
>
|
||||||
<SectionedGridList
|
<ErrorBoundary>
|
||||||
renderItem={item => {
|
<SectionedGridList
|
||||||
if (typeof item === "string") {
|
renderItem={item => {
|
||||||
switch (item) {
|
if (typeof item === "string") {
|
||||||
case "none":
|
switch (item) {
|
||||||
return <DecorationGridNone
|
case "none":
|
||||||
className={cl("change-decoration-modal-decoration")}
|
return <DecorationGridNone
|
||||||
isSelected={activeSelectedDecoration === null}
|
|
||||||
onSelect={() => setTryingDecoration(null)}
|
|
||||||
/>;
|
|
||||||
case "create":
|
|
||||||
return <Tooltip text="You already have a decoration pending review" shouldShow={hasDecorationPendingReview}>
|
|
||||||
{tooltipProps => <DecorationGridCreate
|
|
||||||
className={cl("change-decoration-modal-decoration")}
|
className={cl("change-decoration-modal-decoration")}
|
||||||
|
isSelected={activeSelectedDecoration === null}
|
||||||
|
onSelect={() => setTryingDecoration(null)}
|
||||||
|
/>;
|
||||||
|
case "create":
|
||||||
|
return <Tooltip text="You already have a decoration pending review" shouldShow={hasDecorationPendingReview}>
|
||||||
|
{tooltipProps => <DecorationGridCreate
|
||||||
|
className={cl("change-decoration-modal-decoration")}
|
||||||
|
{...tooltipProps}
|
||||||
|
onSelect={!hasDecorationPendingReview ? (settings.store.agreedToGuidelines ? openCreateDecorationModal : openGuidelinesModal) : () => { }}
|
||||||
|
/>}
|
||||||
|
</Tooltip>;
|
||||||
|
}
|
||||||
|
} else {
|
||||||
|
return <Tooltip text={"Pending review"} shouldShow={item.reviewed === false}>
|
||||||
|
{tooltipProps => (
|
||||||
|
<DecorDecorationGridDecoration
|
||||||
{...tooltipProps}
|
{...tooltipProps}
|
||||||
onSelect={!hasDecorationPendingReview ? openCreateDecorationModal : () => { }}
|
className={cl("change-decoration-modal-decoration")}
|
||||||
/>}
|
onSelect={item.reviewed !== false ? () => setTryingDecoration(item) : () => { }}
|
||||||
</Tooltip>;
|
isSelected={activeSelectedDecoration?.hash === item.hash}
|
||||||
|
decoration={item}
|
||||||
|
/>
|
||||||
|
)}
|
||||||
|
</Tooltip>;
|
||||||
}
|
}
|
||||||
} else {
|
}}
|
||||||
return <Tooltip text={"Pending review"} shouldShow={item.reviewed === false}>
|
getItemKey={item => typeof item === "string" ? item : item.hash}
|
||||||
{tooltipProps => (
|
getSectionKey={section => section.sectionKey}
|
||||||
<DecorDecorationGridDecoration
|
renderSectionHeader={section => <SectionHeader section={section} />}
|
||||||
{...tooltipProps}
|
sections={data}
|
||||||
className={cl("change-decoration-modal-decoration")}
|
|
||||||
onSelect={item.reviewed !== false ? () => setTryingDecoration(item) : () => { }}
|
|
||||||
isSelected={activeSelectedDecoration?.hash === item.hash}
|
|
||||||
decoration={item}
|
|
||||||
/>
|
|
||||||
)}
|
|
||||||
</Tooltip>;
|
|
||||||
}
|
|
||||||
}}
|
|
||||||
getItemKey={item => typeof item === "string" ? item : item.hash}
|
|
||||||
getSectionKey={section => section.sectionKey}
|
|
||||||
renderSectionHeader={section => <SectionHeader section={section} />}
|
|
||||||
sections={data}
|
|
||||||
/>
|
|
||||||
<div className={cl("change-decoration-modal-preview")}>
|
|
||||||
<AvatarDecorationModalPreview
|
|
||||||
avatarDecorationOverride={avatarDecorationOverride}
|
|
||||||
user={UserStore.getCurrentUser()}
|
|
||||||
/>
|
/>
|
||||||
{isActiveDecorationPreset && <Forms.FormTitle className="">Part of the {activeDecorationPreset.name} Preset</Forms.FormTitle>}
|
<div className={cl("change-decoration-modal-preview")}>
|
||||||
{typeof activeSelectedDecoration === "object" &&
|
<AvatarDecorationModalPreview
|
||||||
<Text
|
avatarDecorationOverride={avatarDecorationOverride}
|
||||||
variant="text-sm/semibold"
|
user={UserStore.getCurrentUser()}
|
||||||
color="header-primary"
|
/>
|
||||||
>
|
{isActiveDecorationPreset && <Forms.FormTitle className="">Part of the {activeDecorationPreset.name} Preset</Forms.FormTitle>}
|
||||||
{activeSelectedDecoration?.alt}
|
{typeof activeSelectedDecoration === "object" &&
|
||||||
</Text>
|
<Text
|
||||||
}
|
variant="text-sm/semibold"
|
||||||
{activeDecorationHasAuthor && <Text key={`createdBy-${activeSelectedDecoration.authorId}`}>Created by {Parser.parse(`<@${activeSelectedDecoration.authorId}>`)}</Text>}
|
color="header-primary"
|
||||||
</div>
|
>
|
||||||
|
{activeSelectedDecoration?.alt}
|
||||||
|
</Text>
|
||||||
|
}
|
||||||
|
{activeDecorationHasAuthor && <Text key={`createdBy-${activeSelectedDecoration.authorId}`}>Created by {Parser.parse(`<@${activeSelectedDecoration.authorId}>`)}</Text>}
|
||||||
|
</div>
|
||||||
|
</ErrorBoundary>
|
||||||
</ModalContent>
|
</ModalContent>
|
||||||
<ModalFooter className={classes(cl("change-decoration-modal-footer", cl("modal-footer")))}>
|
<ModalFooter className={classes(cl("change-decoration-modal-footer", cl("modal-footer")))}>
|
||||||
<div className={cl("change-decoration-modal-footer-btn-container")}>
|
<div className={cl("change-decoration-modal-footer-btn-container")}>
|
||||||
|
|
|
@ -4,23 +4,23 @@
|
||||||
* SPDX-License-Identifier: GPL-3.0-or-later
|
* SPDX-License-Identifier: GPL-3.0-or-later
|
||||||
*/
|
*/
|
||||||
|
|
||||||
|
import ErrorBoundary from "@components/ErrorBoundary";
|
||||||
import { Link } from "@components/Link";
|
import { Link } from "@components/Link";
|
||||||
import { openInviteModal } from "@utils/discord";
|
import { openInviteModal } from "@utils/discord";
|
||||||
import { Margins } from "@utils/margins";
|
import { Margins } from "@utils/margins";
|
||||||
import { closeAllModals, ModalCloseButton, ModalContent, ModalFooter, ModalHeader, ModalRoot, ModalSize, openModal } from "@utils/modal";
|
import { closeAllModals, ModalCloseButton, ModalContent, ModalFooter, ModalHeader, ModalProps, ModalRoot, ModalSize, openModal } from "@utils/modal";
|
||||||
import { findByPropsLazy, findComponentByCodeLazy } from "@webpack";
|
import { findByPropsLazy, findComponentByCodeLazy } from "@webpack";
|
||||||
import { Button, FluxDispatcher, Forms, GuildStore, NavigationRouter, Text, TextInput, useEffect, useMemo, UserStore, useState } from "@webpack/common";
|
import { Button, FluxDispatcher, Forms, GuildStore, NavigationRouter, Text, TextInput, useEffect, useMemo, UserStore, useState } from "@webpack/common";
|
||||||
|
|
||||||
import { GUILD_ID, INVITE_KEY, RAW_SKU_ID } from "../../lib/constants";
|
import { GUILD_ID, INVITE_KEY, RAW_SKU_ID } from "../../lib/constants";
|
||||||
import { useCurrentUserDecorationsStore } from "../../lib/stores/CurrentUserDecorationsStore";
|
import { useCurrentUserDecorationsStore } from "../../lib/stores/CurrentUserDecorationsStore";
|
||||||
import { cl, requireAvatarDecorationModal, requireCreateStickerModal } from "../";
|
import { cl, DecorationModalStyles, requireAvatarDecorationModal, requireCreateStickerModal } from "../";
|
||||||
import { AvatarDecorationModalPreview } from "../components";
|
import { AvatarDecorationModalPreview } from "../components";
|
||||||
|
|
||||||
|
|
||||||
const DecorationModalStyles = findByPropsLazy("modalFooterShopButton");
|
|
||||||
|
|
||||||
const FileUpload = findComponentByCodeLazy("fileUploadInput,");
|
const FileUpload = findComponentByCodeLazy("fileUploadInput,");
|
||||||
|
|
||||||
|
const { default: HelpMessage, HelpMessageTypes } = findByPropsLazy("HelpMessageTypes");
|
||||||
|
|
||||||
function useObjectURL(object: Blob | MediaSource | null) {
|
function useObjectURL(object: Blob | MediaSource | null) {
|
||||||
const [url, setUrl] = useState<string | null>(null);
|
const [url, setUrl] = useState<string | null>(null);
|
||||||
|
|
||||||
|
@ -39,7 +39,7 @@ function useObjectURL(object: Blob | MediaSource | null) {
|
||||||
return url;
|
return url;
|
||||||
}
|
}
|
||||||
|
|
||||||
export default function CreateDecorationModal(props) {
|
function CreateDecorationModal(props: ModalProps) {
|
||||||
const [name, setName] = useState("");
|
const [name, setName] = useState("");
|
||||||
const [file, setFile] = useState<File | null>(null);
|
const [file, setFile] = useState<File | null>(null);
|
||||||
const [submitting, setSubmitting] = useState(false);
|
const [submitting, setSubmitting] = useState(false);
|
||||||
|
@ -75,65 +75,69 @@ export default function CreateDecorationModal(props) {
|
||||||
className={cl("create-decoration-modal-content")}
|
className={cl("create-decoration-modal-content")}
|
||||||
scrollbarType="none"
|
scrollbarType="none"
|
||||||
>
|
>
|
||||||
<div className={cl("create-decoration-modal-form-preview-container")}>
|
<ErrorBoundary>
|
||||||
<div className={cl("create-decoration-modal-form")}>
|
<HelpMessage messageType={HelpMessageTypes.WARNING}>
|
||||||
{error !== null && <Text color="text-danger" variant="text-xs/normal">{error.message}</Text>}
|
Make sure your decoration does not violate <Link
|
||||||
<Forms.FormSection title="File">
|
href="https://github.com/decor-discord/.github/blob/main/GUIDELINES.md"
|
||||||
<FileUpload
|
>
|
||||||
filename={file?.name}
|
the guidelines
|
||||||
placeholder="Choose a file"
|
</Link> before submitting it.
|
||||||
buttonText="Browse"
|
</HelpMessage>
|
||||||
filters={[{ name: "Decoration file", extensions: ["png", "apng"] }]}
|
<div className={cl("create-decoration-modal-form-preview-container")}>
|
||||||
onFileSelect={setFile}
|
<div className={cl("create-decoration-modal-form")}>
|
||||||
|
{error !== null && <Text color="text-danger" variant="text-xs/normal">{error.message}</Text>}
|
||||||
|
<Forms.FormSection title="File">
|
||||||
|
<FileUpload
|
||||||
|
filename={file?.name}
|
||||||
|
placeholder="Choose a file"
|
||||||
|
buttonText="Browse"
|
||||||
|
filters={[{ name: "Decoration file", extensions: ["png", "apng"] }]}
|
||||||
|
onFileSelect={setFile}
|
||||||
|
/>
|
||||||
|
<Forms.FormText type="description" className={Margins.top8}>
|
||||||
|
File should be APNG or PNG.
|
||||||
|
</Forms.FormText>
|
||||||
|
</Forms.FormSection>
|
||||||
|
<Forms.FormSection title="Name">
|
||||||
|
<TextInput
|
||||||
|
placeholder="Companion Cube"
|
||||||
|
value={name}
|
||||||
|
onChange={setName}
|
||||||
|
/>
|
||||||
|
<Forms.FormText type="description" className={Margins.top8}>
|
||||||
|
This name will be used when referring to this decoration.
|
||||||
|
</Forms.FormText>
|
||||||
|
</Forms.FormSection>
|
||||||
|
</div>
|
||||||
|
<div>
|
||||||
|
<AvatarDecorationModalPreview
|
||||||
|
avatarDecorationOverride={decoration}
|
||||||
|
user={UserStore.getCurrentUser()}
|
||||||
/>
|
/>
|
||||||
<Forms.FormText type="description" className={Margins.top8}>
|
</div>
|
||||||
File should be APNG or PNG.
|
|
||||||
</Forms.FormText>
|
|
||||||
</Forms.FormSection>
|
|
||||||
<Forms.FormSection title="Name">
|
|
||||||
<TextInput
|
|
||||||
placeholder="Companion Cube"
|
|
||||||
value={name}
|
|
||||||
onChange={setName}
|
|
||||||
/>
|
|
||||||
<Forms.FormText type="description" className={Margins.top8}>
|
|
||||||
This name will be used when referring to this decoration.
|
|
||||||
</Forms.FormText>
|
|
||||||
</Forms.FormSection>
|
|
||||||
</div>
|
</div>
|
||||||
<div>
|
<Forms.FormText type="description" className={Margins.bottom16}>
|
||||||
<AvatarDecorationModalPreview
|
<br />You can receive updates on your decoration's review by joining <Link
|
||||||
avatarDecorationOverride={decoration}
|
href={`https://discord.gg/${INVITE_KEY}`}
|
||||||
user={UserStore.getCurrentUser()}
|
onClick={async e => {
|
||||||
/>
|
e.preventDefault();
|
||||||
</div>
|
if (!GuildStore.getGuild(GUILD_ID)) {
|
||||||
</div>
|
const inviteAccepted = await openInviteModal(INVITE_KEY);
|
||||||
<Forms.FormText type="description" className={Margins.bottom16}>
|
if (inviteAccepted) {
|
||||||
Make sure your decoration does not violate <Link
|
closeAllModals();
|
||||||
href="https://github.com/decor-discord/.github/blob/main/GUIDELINES.md"
|
FluxDispatcher.dispatch({ type: "LAYER_POP_ALL" });
|
||||||
>
|
}
|
||||||
the guidelines
|
} else {
|
||||||
</Link> before creating your decoration.
|
|
||||||
<br />You can receive updates on your decoration's review by joining <Link
|
|
||||||
href={`https://discord.gg/${INVITE_KEY}`}
|
|
||||||
onClick={async e => {
|
|
||||||
e.preventDefault();
|
|
||||||
if (!GuildStore.getGuild(GUILD_ID)) {
|
|
||||||
const inviteAccepted = await openInviteModal(INVITE_KEY);
|
|
||||||
if (inviteAccepted) {
|
|
||||||
closeAllModals();
|
closeAllModals();
|
||||||
FluxDispatcher.dispatch({ type: "LAYER_POP_ALL" });
|
FluxDispatcher.dispatch({ type: "LAYER_POP_ALL" });
|
||||||
|
NavigationRouter.transitionToGuild(GUILD_ID);
|
||||||
}
|
}
|
||||||
} else {
|
}}
|
||||||
closeAllModals();
|
>
|
||||||
FluxDispatcher.dispatch({ type: "LAYER_POP_ALL" });
|
Decor's Discord server
|
||||||
NavigationRouter.transitionToGuild(GUILD_ID);
|
</Link>.
|
||||||
}
|
</Forms.FormText>
|
||||||
}}
|
</ErrorBoundary>
|
||||||
>
|
|
||||||
Decor's Discord server
|
|
||||||
</Link>.
|
|
||||||
</Forms.FormText>
|
|
||||||
</ModalContent>
|
</ModalContent>
|
||||||
<ModalFooter className={cl("modal-footer")}>
|
<ModalFooter className={cl("modal-footer")}>
|
||||||
<Button
|
<Button
|
||||||
|
@ -145,7 +149,7 @@ export default function CreateDecorationModal(props) {
|
||||||
disabled={!file || !name}
|
disabled={!file || !name}
|
||||||
submitting={submitting}
|
submitting={submitting}
|
||||||
>
|
>
|
||||||
Create
|
Submit for Review
|
||||||
</Button>
|
</Button>
|
||||||
<Button
|
<Button
|
||||||
onClick={props.onClose}
|
onClick={props.onClose}
|
||||||
|
|
65
src/plugins/decor/ui/modals/GuidelinesModal.tsx
Normal file
65
src/plugins/decor/ui/modals/GuidelinesModal.tsx
Normal file
|
@ -0,0 +1,65 @@
|
||||||
|
/*
|
||||||
|
* Vencord, a Discord client mod
|
||||||
|
* Copyright (c) 2023 Vendicated and contributors
|
||||||
|
* SPDX-License-Identifier: GPL-3.0-or-later
|
||||||
|
*/
|
||||||
|
|
||||||
|
import { Link } from "@components/Link";
|
||||||
|
import { ModalCloseButton, ModalContent, ModalFooter, ModalHeader, ModalProps, ModalRoot, ModalSize, openModal } from "@utils/modal";
|
||||||
|
import { Button, Forms, Text } from "@webpack/common";
|
||||||
|
|
||||||
|
import { settings } from "../../settings";
|
||||||
|
import { cl, DecorationModalStyles, requireAvatarDecorationModal } from "../";
|
||||||
|
import { openCreateDecorationModal } from "./CreateDecorationModal";
|
||||||
|
|
||||||
|
function GuidelinesModal(props: ModalProps) {
|
||||||
|
return <ModalRoot
|
||||||
|
{...props}
|
||||||
|
size={ModalSize.SMALL}
|
||||||
|
className={DecorationModalStyles.modal}
|
||||||
|
>
|
||||||
|
<ModalHeader separator={false} className={cl("modal-header")}>
|
||||||
|
<Text
|
||||||
|
color="header-primary"
|
||||||
|
variant="heading-lg/semibold"
|
||||||
|
tag="h1"
|
||||||
|
style={{ flexGrow: 1 }}
|
||||||
|
>
|
||||||
|
Hold on
|
||||||
|
</Text>
|
||||||
|
<ModalCloseButton onClick={props.onClose} />
|
||||||
|
</ModalHeader>
|
||||||
|
<ModalContent
|
||||||
|
scrollbarType="none"
|
||||||
|
>
|
||||||
|
<Forms.FormText>
|
||||||
|
By submitting a decoration, you agree to <Link
|
||||||
|
href="https://github.com/decor-discord/.github/blob/main/GUIDELINES.md"
|
||||||
|
>
|
||||||
|
the guidelines
|
||||||
|
</Link>. Not reading these guidelines may get your account suspended from creating more decorations in the future.
|
||||||
|
</Forms.FormText>
|
||||||
|
</ModalContent>
|
||||||
|
<ModalFooter className={cl("modal-footer")}>
|
||||||
|
<Button
|
||||||
|
onClick={() => {
|
||||||
|
settings.store.agreedToGuidelines = true;
|
||||||
|
props.onClose();
|
||||||
|
openCreateDecorationModal();
|
||||||
|
}}
|
||||||
|
>
|
||||||
|
Continue
|
||||||
|
</Button>
|
||||||
|
<Button
|
||||||
|
onClick={props.onClose}
|
||||||
|
color={Button.Colors.PRIMARY}
|
||||||
|
look={Button.Looks.LINK}
|
||||||
|
>
|
||||||
|
Go Back
|
||||||
|
</Button>
|
||||||
|
</ModalFooter>
|
||||||
|
</ModalRoot>;
|
||||||
|
}
|
||||||
|
|
||||||
|
export const openGuidelinesModal = () =>
|
||||||
|
requireAvatarDecorationModal().then(() => openModal(props => <GuidelinesModal {...props} />));
|
|
@ -8,7 +8,7 @@
|
||||||
display: flex;
|
display: flex;
|
||||||
border-radius: 5px 5px 0 0;
|
border-radius: 5px 5px 0 0;
|
||||||
padding: 0 16px;
|
padding: 0 16px;
|
||||||
gap: 4px
|
gap: 4px;
|
||||||
}
|
}
|
||||||
|
|
||||||
.vc-decor-change-decoration-modal-preview {
|
.vc-decor-change-decoration-modal-preview {
|
||||||
|
@ -72,7 +72,7 @@
|
||||||
.vc-decor-sectioned-grid-list-grid {
|
.vc-decor-sectioned-grid-list-grid {
|
||||||
display: flex;
|
display: flex;
|
||||||
flex-wrap: wrap;
|
flex-wrap: wrap;
|
||||||
gap: 8px
|
gap: 8px;
|
||||||
}
|
}
|
||||||
|
|
||||||
.vc-decor-section-remove-margin {
|
.vc-decor-section-remove-margin {
|
||||||
|
|
35
src/plugins/fixCodeblockGap/index.ts
Normal file
35
src/plugins/fixCodeblockGap/index.ts
Normal file
|
@ -0,0 +1,35 @@
|
||||||
|
/*
|
||||||
|
* 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 { Devs } from "@utils/constants";
|
||||||
|
import definePlugin from "@utils/types";
|
||||||
|
|
||||||
|
export default definePlugin({
|
||||||
|
name: "FixCodeblockGap",
|
||||||
|
description: "Removes the gap between codeblocks and text below it",
|
||||||
|
authors: [Devs.Grzesiek11],
|
||||||
|
patches: [
|
||||||
|
{
|
||||||
|
find: ".default.Messages.DELETED_ROLE_PLACEHOLDER",
|
||||||
|
replacement: {
|
||||||
|
match: String.raw`/^${"```"}(?:([a-z0-9_+\-.#]+?)\n)?\n*([^\n][^]*?)\n*${"```"}`,
|
||||||
|
replace: "$&\\n?",
|
||||||
|
},
|
||||||
|
},
|
||||||
|
],
|
||||||
|
});
|
5
src/plugins/fixYoutubeEmbeds.desktop/README.md
Normal file
5
src/plugins/fixYoutubeEmbeds.desktop/README.md
Normal file
|
@ -0,0 +1,5 @@
|
||||||
|
# FixYoutubeEmbeds
|
||||||
|
|
||||||
|
Bypasses youtube videos being blocked from display on Discord (for example by UMG)
|
||||||
|
|
||||||
|
![](https://github.com/Vendicated/Vencord/assets/45497981/7a5fdcaa-217c-4c63-acae-f0d6af2f79be)
|
14
src/plugins/fixYoutubeEmbeds.desktop/index.ts
Normal file
14
src/plugins/fixYoutubeEmbeds.desktop/index.ts
Normal file
|
@ -0,0 +1,14 @@
|
||||||
|
/*
|
||||||
|
* Vencord, a Discord client mod
|
||||||
|
* Copyright (c) 2023 Vendicated and contributors
|
||||||
|
* SPDX-License-Identifier: GPL-3.0-or-later
|
||||||
|
*/
|
||||||
|
|
||||||
|
import { Devs } from "@utils/constants";
|
||||||
|
import definePlugin from "@utils/types";
|
||||||
|
|
||||||
|
export default definePlugin({
|
||||||
|
name: "FixYoutubeEmbeds",
|
||||||
|
description: "Bypasses youtube videos being blocked from display on Discord (for example by UMG)",
|
||||||
|
authors: [Devs.coolelectronics]
|
||||||
|
});
|
27
src/plugins/fixYoutubeEmbeds.desktop/native.ts
Normal file
27
src/plugins/fixYoutubeEmbeds.desktop/native.ts
Normal file
|
@ -0,0 +1,27 @@
|
||||||
|
/*
|
||||||
|
* Vencord, a Discord client mod
|
||||||
|
* Copyright (c) 2023 Vendicated and contributors
|
||||||
|
* SPDX-License-Identifier: GPL-3.0-or-later
|
||||||
|
*/
|
||||||
|
|
||||||
|
import { app } from "electron";
|
||||||
|
import { getSettings } from "main/ipcMain";
|
||||||
|
|
||||||
|
app.on("browser-window-created", (_, win) => {
|
||||||
|
win.webContents.on("frame-created", (_, { frame }) => {
|
||||||
|
frame.once("dom-ready", () => {
|
||||||
|
if (frame.url.startsWith("https://www.youtube.com/")) {
|
||||||
|
const settings = getSettings().plugins?.FixYoutubeEmbeds;
|
||||||
|
if (!settings?.enabled) return;
|
||||||
|
|
||||||
|
frame.executeJavaScript(`
|
||||||
|
new MutationObserver(() => {
|
||||||
|
if(
|
||||||
|
document.querySelector('div.ytp-error-content-wrap-subreason a[href^="https://www.youtube.com/watch?v="]')
|
||||||
|
) location.reload()
|
||||||
|
}).observe(document.body, { childList: true, subtree:true });
|
||||||
|
`);
|
||||||
|
}
|
||||||
|
});
|
||||||
|
});
|
||||||
|
});
|
|
@ -16,13 +16,14 @@
|
||||||
* along with this program. If not, see <https://www.gnu.org/licenses/>.
|
* along with this program. If not, see <https://www.gnu.org/licenses/>.
|
||||||
*/
|
*/
|
||||||
|
|
||||||
|
import { addChatBarButton, ChatBarButton } from "@api/ChatButtons";
|
||||||
import { addButton, removeButton } from "@api/MessagePopover";
|
import { addButton, removeButton } from "@api/MessagePopover";
|
||||||
import { definePluginSettings } from "@api/Settings";
|
import { definePluginSettings } from "@api/Settings";
|
||||||
import ErrorBoundary from "@components/ErrorBoundary";
|
import ErrorBoundary from "@components/ErrorBoundary";
|
||||||
import { Devs } from "@utils/constants";
|
import { Devs } from "@utils/constants";
|
||||||
import { getStegCloak } from "@utils/dependencies";
|
import { getStegCloak } from "@utils/dependencies";
|
||||||
import definePlugin, { OptionType } from "@utils/types";
|
import definePlugin, { OptionType } from "@utils/types";
|
||||||
import { Button, ButtonLooks, ButtonWrapperClasses, ChannelStore, FluxDispatcher, RestAPI, Tooltip } from "@webpack/common";
|
import { ChannelStore, FluxDispatcher, RestAPI, Tooltip } from "@webpack/common";
|
||||||
import { Message } from "discord-types/general";
|
import { Message } from "discord-types/general";
|
||||||
|
|
||||||
import { buildDecModal } from "./components/DecryptionModal";
|
import { buildDecModal } from "./components/DecryptionModal";
|
||||||
|
@ -64,54 +65,31 @@ function Indicator() {
|
||||||
|
|
||||||
}
|
}
|
||||||
|
|
||||||
function ChatBarIcon(chatBoxProps: {
|
const ChatBarIcon: ChatBarButton = ({ isMainChat }) => {
|
||||||
type: {
|
if (!isMainChat) return null;
|
||||||
analyticsName: string;
|
|
||||||
};
|
|
||||||
}) {
|
|
||||||
if (chatBoxProps.type.analyticsName !== "normal") return null;
|
|
||||||
|
|
||||||
return (
|
return (
|
||||||
<Tooltip text="Encrypt Message">
|
<ChatBarButton
|
||||||
{({ onMouseEnter, onMouseLeave }) => (
|
tooltip="Encrypt Message"
|
||||||
// size="" = Button.Sizes.NONE
|
onClick={() => buildEncModal()}
|
||||||
/*
|
|
||||||
many themes set "> button" to display: none, as the gift button is
|
buttonProps={{
|
||||||
the only directly descending button (all the other elements are divs.)
|
"aria-haspopup": "dialog",
|
||||||
Thus, wrap in a div here to avoid getting hidden by that.
|
}}
|
||||||
flex is for some reason necessary as otherwise the button goes flying off
|
>
|
||||||
*/
|
<svg
|
||||||
<div style={{ display: "flex" }}>
|
aria-hidden
|
||||||
<Button
|
role="img"
|
||||||
aria-haspopup="dialog"
|
width="24"
|
||||||
aria-label="Encrypt Message"
|
height="24"
|
||||||
size=""
|
viewBox={"0 0 64 64"}
|
||||||
look={ButtonLooks.BLANK}
|
style={{ scale: "1.39", translate: "0 -1px" }}
|
||||||
onMouseEnter={onMouseEnter}
|
>
|
||||||
onMouseLeave={onMouseLeave}
|
<path fill="currentColor" d="M 32 9 C 24.832 9 19 14.832 19 22 L 19 27.347656 C 16.670659 28.171862 15 30.388126 15 33 L 15 49 C 15 52.314 17.686 55 21 55 L 43 55 C 46.314 55 49 52.314 49 49 L 49 33 C 49 30.388126 47.329341 28.171862 45 27.347656 L 45 22 C 45 14.832 39.168 9 32 9 z M 32 13 C 36.963 13 41 17.038 41 22 L 41 27 L 23 27 L 23 22 C 23 17.038 27.037 13 32 13 z" />
|
||||||
innerClassName={ButtonWrapperClasses.button}
|
</svg>
|
||||||
onClick={() => buildEncModal()}
|
</ChatBarButton>
|
||||||
style={{ padding: "0 2px", scale: "0.9" }}
|
|
||||||
>
|
|
||||||
<div className={ButtonWrapperClasses.buttonWrapper}>
|
|
||||||
<svg
|
|
||||||
aria-hidden
|
|
||||||
role="img"
|
|
||||||
width="32"
|
|
||||||
height="32"
|
|
||||||
viewBox={"0 0 64 64"}
|
|
||||||
style={{ scale: "1.1" }}
|
|
||||||
>
|
|
||||||
<path fill="currentColor" d="M 32 9 C 24.832 9 19 14.832 19 22 L 19 27.347656 C 16.670659 28.171862 15 30.388126 15 33 L 15 49 C 15 52.314 17.686 55 21 55 L 43 55 C 46.314 55 49 52.314 49 49 L 49 33 C 49 30.388126 47.329341 28.171862 45 27.347656 L 45 22 C 45 14.832 39.168 9 32 9 z M 32 13 C 36.963 13 41 17.038 41 22 L 41 27 L 23 27 L 23 22 C 23 17.038 27.037 13 32 13 z" />
|
|
||||||
</svg>
|
|
||||||
</div>
|
|
||||||
</Button>
|
|
||||||
</div>
|
|
||||||
)
|
|
||||||
}
|
|
||||||
</Tooltip >
|
|
||||||
);
|
);
|
||||||
}
|
};
|
||||||
|
|
||||||
const settings = definePluginSettings({
|
const settings = definePluginSettings({
|
||||||
savedPasswords: {
|
savedPasswords: {
|
||||||
|
@ -125,7 +103,7 @@ export default definePlugin({
|
||||||
name: "InvisibleChat",
|
name: "InvisibleChat",
|
||||||
description: "Encrypt your Messages in a non-suspicious way!",
|
description: "Encrypt your Messages in a non-suspicious way!",
|
||||||
authors: [Devs.SammCheese],
|
authors: [Devs.SammCheese],
|
||||||
dependencies: ["MessagePopoverAPI"],
|
dependencies: ["MessagePopoverAPI", "ChatInputButtonAPI"],
|
||||||
patches: [
|
patches: [
|
||||||
{
|
{
|
||||||
// Indicator
|
// Indicator
|
||||||
|
@ -135,13 +113,6 @@ export default definePlugin({
|
||||||
replace: "try {$1 && $self.INV_REGEX.test($1.message.content) ? $1.content.push($self.indicator()) : null } catch {};$&"
|
replace: "try {$1 && $self.INV_REGEX.test($1.message.content) ? $1.content.push($self.indicator()) : null } catch {};$&"
|
||||||
}
|
}
|
||||||
},
|
},
|
||||||
{
|
|
||||||
find: "ChannelTextAreaButtons",
|
|
||||||
replacement: {
|
|
||||||
match: /(\i)\.push.{1,30}disabled:(\i),.{1,20}\},"gift"\)\)/,
|
|
||||||
replace: "$&,(()=>{try{$2||$1.push($self.chatBarIcon(arguments[0]))}catch{}})()",
|
|
||||||
}
|
|
||||||
},
|
|
||||||
],
|
],
|
||||||
|
|
||||||
EMBED_API_URL: "https://embed.sammcheese.net",
|
EMBED_API_URL: "https://embed.sammcheese.net",
|
||||||
|
@ -151,10 +122,7 @@ export default definePlugin({
|
||||||
),
|
),
|
||||||
settings,
|
settings,
|
||||||
async start() {
|
async start() {
|
||||||
const { default: StegCloak } = await getStegCloak();
|
addButton("InvisibleChat", message => {
|
||||||
steggo = new StegCloak(true, false);
|
|
||||||
|
|
||||||
addButton("invDecrypt", message => {
|
|
||||||
return this.INV_REGEX.test(message?.content)
|
return this.INV_REGEX.test(message?.content)
|
||||||
? {
|
? {
|
||||||
label: "Decrypt Message",
|
label: "Decrypt Message",
|
||||||
|
@ -170,10 +138,16 @@ export default definePlugin({
|
||||||
}
|
}
|
||||||
: null;
|
: null;
|
||||||
});
|
});
|
||||||
|
|
||||||
|
addChatBarButton("InvisibleChat", ChatBarIcon);
|
||||||
|
|
||||||
|
const { default: StegCloak } = await getStegCloak();
|
||||||
|
steggo = new StegCloak(true, false);
|
||||||
},
|
},
|
||||||
|
|
||||||
stop() {
|
stop() {
|
||||||
removeButton("invDecrypt");
|
removeButton("InvisibleChat");
|
||||||
|
removeButton("InvisibleChat");
|
||||||
},
|
},
|
||||||
|
|
||||||
// Gets the Embed of a Link
|
// Gets the Embed of a Link
|
||||||
|
@ -216,7 +190,6 @@ export default definePlugin({
|
||||||
});
|
});
|
||||||
},
|
},
|
||||||
|
|
||||||
chatBarIcon: ErrorBoundary.wrap(ChatBarIcon, { noop: true }),
|
|
||||||
popOverIcon: () => <PopOverIcon />,
|
popOverIcon: () => <PopOverIcon />,
|
||||||
indicator: ErrorBoundary.wrap(Indicator, { noop: true })
|
indicator: ErrorBoundary.wrap(Indicator, { noop: true })
|
||||||
});
|
});
|
||||||
|
|
|
@ -198,7 +198,7 @@ export default definePlugin({
|
||||||
replacement: [
|
replacement: [
|
||||||
// make the tag show the right text
|
// make the tag show the right text
|
||||||
{
|
{
|
||||||
match: /(switch\((\i)\){.+?)case (\i(?:\.\i)?)\.BOT:default:(\i)=(\i\.\i\.Messages)\.BOT_TAG_BOT/,
|
match: /(switch\((\i)\){.+?)case (\i(?:\.\i)?)\.BOT:default:(\i)=.{0,40}(\i\.\i\.Messages)\.BOT_TAG_BOT/,
|
||||||
replace: (_, origSwitch, variant, tags, displayedText, strings) =>
|
replace: (_, origSwitch, variant, tags, displayedText, strings) =>
|
||||||
`${origSwitch}default:{${displayedText} = $self.getTagText(${tags}[${variant}], ${strings})}`
|
`${origSwitch}default:{${displayedText} = $self.getTagText(${tags}[${variant}], ${strings})}`
|
||||||
},
|
},
|
||||||
|
|
|
@ -126,7 +126,9 @@ function MenuItem(guildId: string, id?: string, type?: MenuItemParentType) {
|
||||||
|
|
||||||
function makeContextMenuPatch(childId: string | string[], type?: MenuItemParentType): NavContextMenuPatchCallback {
|
function makeContextMenuPatch(childId: string | string[], type?: MenuItemParentType): NavContextMenuPatchCallback {
|
||||||
return (children, props) => () => {
|
return (children, props) => () => {
|
||||||
if (!props || (type === MenuItemParentType.User && !props.user) || (type === MenuItemParentType.Guild && !props.guild)) return children;
|
if (!props) return;
|
||||||
|
if ((type === MenuItemParentType.User && !props.user) || (type === MenuItemParentType.Guild && !props.guild) || (type === MenuItemParentType.Channel && (!props.channel || !props.guild)))
|
||||||
|
return children;
|
||||||
|
|
||||||
const group = findGroupChildrenByChildId(childId, children);
|
const group = findGroupChildrenByChildId(childId, children);
|
||||||
|
|
||||||
|
|
|
@ -24,7 +24,7 @@ const settings = definePluginSettings({
|
||||||
export default definePlugin({
|
export default definePlugin({
|
||||||
name: "PictureInPicture",
|
name: "PictureInPicture",
|
||||||
description: "Adds picture in picture to videos (next to the Download button)",
|
description: "Adds picture in picture to videos (next to the Download button)",
|
||||||
authors: [Devs.Lumap],
|
authors: [Devs.Nobody],
|
||||||
settings,
|
settings,
|
||||||
patches: [
|
patches: [
|
||||||
{
|
{
|
||||||
|
|
|
@ -16,22 +16,14 @@
|
||||||
* along with this program. If not, see <https://www.gnu.org/licenses/>.
|
* along with this program. If not, see <https://www.gnu.org/licenses/>.
|
||||||
*/
|
*/
|
||||||
|
|
||||||
|
import { addChatBarButton, ChatBarButton, removeChatBarButton } from "@api/ChatButtons";
|
||||||
import { generateId, sendBotMessage } from "@api/Commands";
|
import { generateId, sendBotMessage } from "@api/Commands";
|
||||||
import ErrorBoundary from "@components/ErrorBoundary";
|
|
||||||
import { Devs } from "@utils/constants";
|
import { Devs } from "@utils/constants";
|
||||||
import definePlugin from "@utils/types";
|
import definePlugin, { StartAt } from "@utils/types";
|
||||||
import { findByPropsLazy } from "@webpack";
|
import { findByPropsLazy } from "@webpack";
|
||||||
import { Button, ButtonLooks, ButtonWrapperClasses, DraftStore, DraftType, SelectedChannelStore, Tooltip, UserStore, useStateFromStores } from "@webpack/common";
|
import { DraftStore, DraftType, SelectedChannelStore, UserStore, useStateFromStores } from "@webpack/common";
|
||||||
import { MessageAttachment } from "discord-types/general";
|
import { MessageAttachment } from "discord-types/general";
|
||||||
|
|
||||||
interface Props {
|
|
||||||
type: {
|
|
||||||
analyticsName: string;
|
|
||||||
isEmpty: boolean;
|
|
||||||
attachments: boolean;
|
|
||||||
};
|
|
||||||
}
|
|
||||||
|
|
||||||
const UploadStore = findByPropsLazy("getUploads");
|
const UploadStore = findByPropsLazy("getUploads");
|
||||||
|
|
||||||
const getDraft = (channelId: string) => DraftStore.getDraft(channelId, DraftType.ChannelMessage);
|
const getDraft = (channelId: string) => DraftStore.getDraft(channelId, DraftType.ChannelMessage);
|
||||||
|
@ -81,13 +73,11 @@ const getAttachments = async (channelId: string) =>
|
||||||
);
|
);
|
||||||
|
|
||||||
|
|
||||||
export function PreviewButton(chatBoxProps: Props) {
|
const PreviewButton: ChatBarButton = ({ isMainChat, isEmpty, type: { attachments } }) => {
|
||||||
const { isEmpty, attachments } = chatBoxProps.type;
|
|
||||||
|
|
||||||
const channelId = SelectedChannelStore.getChannelId();
|
const channelId = SelectedChannelStore.getChannelId();
|
||||||
const draft = useStateFromStores([DraftStore], () => getDraft(channelId));
|
const draft = useStateFromStores([DraftStore], () => getDraft(channelId));
|
||||||
|
|
||||||
if (chatBoxProps.type.analyticsName !== "normal") return null;
|
if (!isMainChat) return null;
|
||||||
|
|
||||||
const hasAttachments = attachments && UploadStore.getUploads(channelId, DraftType.ChannelMessage).length > 0;
|
const hasAttachments = attachments && UploadStore.getUploads(channelId, DraftType.ChannelMessage).length > 0;
|
||||||
const hasContent = !isEmpty && draft?.length > 0;
|
const hasContent = !isEmpty && draft?.length > 0;
|
||||||
|
@ -95,47 +85,47 @@ export function PreviewButton(chatBoxProps: Props) {
|
||||||
if (!hasContent && !hasAttachments) return null;
|
if (!hasContent && !hasAttachments) return null;
|
||||||
|
|
||||||
return (
|
return (
|
||||||
<Tooltip text="Preview Message">
|
<ChatBarButton
|
||||||
{tooltipProps => (
|
tooltip="Preview Message"
|
||||||
<Button
|
onClick={async () =>
|
||||||
{...tooltipProps}
|
sendBotMessage(
|
||||||
onClick={async () =>
|
channelId,
|
||||||
sendBotMessage(
|
{
|
||||||
channelId,
|
content: getDraft(channelId),
|
||||||
{
|
author: UserStore.getCurrentUser(),
|
||||||
content: getDraft(channelId),
|
attachments: hasAttachments ? await getAttachments(channelId) : undefined,
|
||||||
author: UserStore.getCurrentUser(),
|
}
|
||||||
attachments: hasAttachments ? await getAttachments(channelId) : undefined,
|
)}
|
||||||
}
|
buttonProps={{
|
||||||
)}
|
style: {
|
||||||
size=""
|
translate: "0 2px"
|
||||||
look={ButtonLooks.BLANK}
|
}
|
||||||
innerClassName={ButtonWrapperClasses.button}
|
}}
|
||||||
style={{ padding: "0 2px", height: "100%" }}
|
>
|
||||||
>
|
<svg
|
||||||
<div className={ButtonWrapperClasses.buttonWrapper}>
|
fill="currentColor"
|
||||||
<img width={24} height={24} src="https://discord.com/assets/4c5a77a89716352686f590a6f014770c.svg" />
|
fillRule="evenodd"
|
||||||
</div>
|
width="24"
|
||||||
</Button>
|
height="24"
|
||||||
)}
|
viewBox="0 0 24 24"
|
||||||
</Tooltip>
|
style={{ scale: "1.096", translate: "0 -1px" }}
|
||||||
|
>
|
||||||
|
<path d="M22.89 11.7c.07.2.07.4 0 .6C22.27 13.9 19.1 21 12 21c-7.11 0-10.27-7.11-10.89-8.7a.83.83 0 0 1 0-.6C1.73 10.1 4.9 3 12 3c7.11 0 10.27 7.11 10.89 8.7Zm-4.5-3.62A15.11 15.11 0 0 1 20.85 12c-.38.88-1.18 2.47-2.46 3.92C16.87 17.62 14.8 19 12 19c-2.8 0-4.87-1.38-6.39-3.08A15.11 15.11 0 0 1 3.15 12c.38-.88 1.18-2.47 2.46-3.92C7.13 6.38 9.2 5 12 5c2.8 0 4.87 1.38 6.39 3.08ZM15.56 11.77c.2-.1.44.02.44.23a4 4 0 1 1-4-4c.21 0 .33.25.23.44a2.5 2.5 0 0 0 3.32 3.32Z" />
|
||||||
|
</svg>
|
||||||
|
</ChatBarButton>
|
||||||
);
|
);
|
||||||
|
|
||||||
}
|
};
|
||||||
|
|
||||||
export default definePlugin({
|
export default definePlugin({
|
||||||
name: "PreviewMessage",
|
name: "PreviewMessage",
|
||||||
description: "Lets you preview your message before sending it.",
|
description: "Lets you preview your message before sending it.",
|
||||||
authors: [Devs.Aria],
|
authors: [Devs.Aria],
|
||||||
patches: [
|
dependencies: ["ChatInputButtonAPI"],
|
||||||
{
|
// start early to ensure we're the first plugin to add our button
|
||||||
find: "ChannelTextAreaButtons",
|
// This makes the popping in less awkward
|
||||||
replacement: {
|
startAt: StartAt.Init,
|
||||||
match: /(\i)\.push.{1,30}disabled:(\i),.{1,20}\},"gift"\)\)/,
|
|
||||||
replace: "$&,(()=>{try{$2||$1.push($self.chatBarIcon(arguments[0]))}catch{}})()",
|
|
||||||
}
|
|
||||||
},
|
|
||||||
],
|
|
||||||
|
|
||||||
chatBarIcon: ErrorBoundary.wrap(PreviewButton, { noop: true }),
|
start: () => addChatBarButton("previewMessage", PreviewButton),
|
||||||
|
stop: () => removeChatBarButton("previewMessage"),
|
||||||
});
|
});
|
||||||
|
|
|
@ -25,16 +25,17 @@ function onClick() {
|
||||||
const channels: Array<any> = [];
|
const channels: Array<any> = [];
|
||||||
|
|
||||||
Object.values(GuildStore.getGuilds()).forEach(guild => {
|
Object.values(GuildStore.getGuilds()).forEach(guild => {
|
||||||
GuildChannelStore.getChannels(guild.id).SELECTABLE.forEach((c: { channel: { id: string; }; }) => {
|
GuildChannelStore.getChannels(guild.id).SELECTABLE
|
||||||
if (!ReadStateStore.hasUnread(c.channel.id)) return;
|
.concat(GuildChannelStore.getChannels(guild.id).VOCAL)
|
||||||
|
.forEach((c: { channel: { id: string; }; }) => {
|
||||||
|
if (!ReadStateStore.hasUnread(c.channel.id)) return;
|
||||||
|
|
||||||
channels.push({
|
channels.push({
|
||||||
channelId: c.channel.id,
|
channelId: c.channel.id,
|
||||||
// messageId: c.channel?.lastMessageId,
|
messageId: ReadStateStore.lastMessageId(c.channel.id),
|
||||||
messageId: ReadStateStore.lastMessageId(c.channel.id),
|
readStateType: 0
|
||||||
readStateType: 0
|
});
|
||||||
});
|
});
|
||||||
});
|
|
||||||
});
|
});
|
||||||
|
|
||||||
FluxDispatcher.dispatch({
|
FluxDispatcher.dispatch({
|
||||||
|
|
|
@ -36,62 +36,68 @@ function search(src: string, engine: string) {
|
||||||
open(engine + encodeURIComponent(src), "_blank");
|
open(engine + encodeURIComponent(src), "_blank");
|
||||||
}
|
}
|
||||||
|
|
||||||
const imageContextMenuPatch: NavContextMenuPatchCallback = (children, props) => () => {
|
function makeSearchItem(src: string) {
|
||||||
if (!props) return;
|
return (
|
||||||
const { reverseImageSearchType, itemHref, itemSrc } = props;
|
<Menu.MenuItem
|
||||||
|
label="Search Image"
|
||||||
|
key="search-image"
|
||||||
|
id="search-image"
|
||||||
|
>
|
||||||
|
{Object.keys(Engines).map((engine, i) => {
|
||||||
|
const key = "search-image-" + engine;
|
||||||
|
return (
|
||||||
|
<Menu.MenuItem
|
||||||
|
key={key}
|
||||||
|
id={key}
|
||||||
|
label={
|
||||||
|
<Flex style={{ alignItems: "center", gap: "0.5em" }}>
|
||||||
|
<img
|
||||||
|
style={{
|
||||||
|
borderRadius: i >= 3 // Do not round Google, Yandex & SauceNAO
|
||||||
|
? "50%"
|
||||||
|
: void 0
|
||||||
|
}}
|
||||||
|
aria-hidden="true"
|
||||||
|
height={16}
|
||||||
|
width={16}
|
||||||
|
src={new URL("/favicon.ico", Engines[engine]).toString().replace("lens.", "")}
|
||||||
|
/>
|
||||||
|
{engine}
|
||||||
|
</Flex>
|
||||||
|
}
|
||||||
|
action={() => search(src, Engines[engine])}
|
||||||
|
/>
|
||||||
|
);
|
||||||
|
})}
|
||||||
|
<Menu.MenuItem
|
||||||
|
key="search-image-all"
|
||||||
|
id="search-image-all"
|
||||||
|
label={
|
||||||
|
<Flex style={{ alignItems: "center", gap: "0.5em" }}>
|
||||||
|
<OpenExternalIcon height={16} width={16} />
|
||||||
|
All
|
||||||
|
</Flex>
|
||||||
|
}
|
||||||
|
action={() => Object.values(Engines).forEach(e => search(src, e))}
|
||||||
|
/>
|
||||||
|
</Menu.MenuItem>
|
||||||
|
);
|
||||||
|
}
|
||||||
|
|
||||||
if (!reverseImageSearchType || reverseImageSearchType !== "img") return;
|
const messageContextMenuPatch: NavContextMenuPatchCallback = (children, props) => () => {
|
||||||
|
if (props?.reverseImageSearchType !== "img") return;
|
||||||
|
|
||||||
const src = itemHref ?? itemSrc;
|
const src = props.itemHref ?? props.itemSrc;
|
||||||
|
|
||||||
const group = findGroupChildrenByChildId("copy-link", children);
|
const group = findGroupChildrenByChildId("copy-link", children);
|
||||||
if (group) {
|
group?.push(makeSearchItem(src));
|
||||||
group.push((
|
};
|
||||||
<Menu.MenuItem
|
|
||||||
label="Search Image"
|
const imageContextMenuPatch: NavContextMenuPatchCallback = (children, props) => () => {
|
||||||
key="search-image"
|
if (!props?.src) return;
|
||||||
id="search-image"
|
|
||||||
>
|
const group = findGroupChildrenByChildId("copy-native-link", children) ?? children;
|
||||||
{Object.keys(Engines).map((engine, i) => {
|
group.push(makeSearchItem(props.src));
|
||||||
const key = "search-image-" + engine;
|
|
||||||
return (
|
|
||||||
<Menu.MenuItem
|
|
||||||
key={key}
|
|
||||||
id={key}
|
|
||||||
label={
|
|
||||||
<Flex style={{ alignItems: "center", gap: "0.5em" }}>
|
|
||||||
<img
|
|
||||||
style={{
|
|
||||||
borderRadius: i >= 3 // Do not round Google, Yandex & SauceNAO
|
|
||||||
? "50%"
|
|
||||||
: void 0
|
|
||||||
}}
|
|
||||||
aria-hidden="true"
|
|
||||||
height={16}
|
|
||||||
width={16}
|
|
||||||
src={new URL("/favicon.ico", Engines[engine]).toString().replace("lens.", "")}
|
|
||||||
/>
|
|
||||||
{engine}
|
|
||||||
</Flex>
|
|
||||||
}
|
|
||||||
action={() => search(src, Engines[engine])}
|
|
||||||
/>
|
|
||||||
);
|
|
||||||
})}
|
|
||||||
<Menu.MenuItem
|
|
||||||
key="search-image-all"
|
|
||||||
id="search-image-all"
|
|
||||||
label={
|
|
||||||
<Flex style={{ alignItems: "center", gap: "0.5em" }}>
|
|
||||||
<OpenExternalIcon height={16} width={16} />
|
|
||||||
All
|
|
||||||
</Flex>
|
|
||||||
}
|
|
||||||
action={() => Object.values(Engines).forEach(e => search(src, e))}
|
|
||||||
/>
|
|
||||||
</Menu.MenuItem>
|
|
||||||
));
|
|
||||||
}
|
|
||||||
};
|
};
|
||||||
|
|
||||||
export default definePlugin({
|
export default definePlugin({
|
||||||
|
@ -111,10 +117,12 @@ export default definePlugin({
|
||||||
],
|
],
|
||||||
|
|
||||||
start() {
|
start() {
|
||||||
addContextMenuPatch("message", imageContextMenuPatch);
|
addContextMenuPatch("message", messageContextMenuPatch);
|
||||||
|
addContextMenuPatch("image-context", imageContextMenuPatch);
|
||||||
},
|
},
|
||||||
|
|
||||||
stop() {
|
stop() {
|
||||||
removeContextMenuPatch("message", imageContextMenuPatch);
|
removeContextMenuPatch("message", messageContextMenuPatch);
|
||||||
|
removeContextMenuPatch("image-context", imageContextMenuPatch);
|
||||||
}
|
}
|
||||||
});
|
});
|
||||||
|
|
81
src/plugins/reviewDB/auth.tsx
Normal file
81
src/plugins/reviewDB/auth.tsx
Normal file
|
@ -0,0 +1,81 @@
|
||||||
|
/*
|
||||||
|
* Vencord, a Discord client mod
|
||||||
|
* Copyright (c) 2023 Vendicated and contributors
|
||||||
|
* SPDX-License-Identifier: GPL-3.0-or-later
|
||||||
|
*/
|
||||||
|
|
||||||
|
import { DataStore } from "@api/index";
|
||||||
|
import { Logger } from "@utils/Logger";
|
||||||
|
import { openModal } from "@utils/modal";
|
||||||
|
import { findByPropsLazy } from "@webpack";
|
||||||
|
import { showToast, Toasts, UserStore } from "@webpack/common";
|
||||||
|
|
||||||
|
import { ReviewDBAuth } from "./entities";
|
||||||
|
|
||||||
|
const DATA_STORE_KEY = "rdb-auth";
|
||||||
|
|
||||||
|
const { OAuth2AuthorizeModal } = findByPropsLazy("OAuth2AuthorizeModal");
|
||||||
|
|
||||||
|
export let Auth: ReviewDBAuth = {};
|
||||||
|
|
||||||
|
export async function initAuth() {
|
||||||
|
Auth = await getAuth() ?? {};
|
||||||
|
}
|
||||||
|
|
||||||
|
export async function getAuth(): Promise<ReviewDBAuth | undefined> {
|
||||||
|
const auth = await DataStore.get(DATA_STORE_KEY);
|
||||||
|
return auth?.[UserStore.getCurrentUser()?.id];
|
||||||
|
}
|
||||||
|
|
||||||
|
export async function getToken() {
|
||||||
|
const auth = await getAuth();
|
||||||
|
return auth?.token;
|
||||||
|
}
|
||||||
|
|
||||||
|
export async function updateAuth(newAuth: ReviewDBAuth) {
|
||||||
|
return DataStore.update(DATA_STORE_KEY, auth => {
|
||||||
|
auth ??= {};
|
||||||
|
Auth = auth[UserStore.getCurrentUser().id] ??= {};
|
||||||
|
|
||||||
|
if (newAuth.token) Auth.token = newAuth.token;
|
||||||
|
if (newAuth.user) Auth.user = newAuth.user;
|
||||||
|
|
||||||
|
return auth;
|
||||||
|
});
|
||||||
|
}
|
||||||
|
|
||||||
|
export function authorize(callback?: any) {
|
||||||
|
openModal(props =>
|
||||||
|
<OAuth2AuthorizeModal
|
||||||
|
{...props}
|
||||||
|
scopes={["identify"]}
|
||||||
|
responseType="code"
|
||||||
|
redirectUri="https://manti.vendicated.dev/api/reviewdb/auth"
|
||||||
|
permissions={0n}
|
||||||
|
clientId="915703782174752809"
|
||||||
|
cancelCompletesFlow={false}
|
||||||
|
callback={async (response: any) => {
|
||||||
|
try {
|
||||||
|
const url = new URL(response.location);
|
||||||
|
url.searchParams.append("clientMod", "vencord");
|
||||||
|
const res = await fetch(url, {
|
||||||
|
headers: new Headers({ Accept: "application/json" })
|
||||||
|
});
|
||||||
|
|
||||||
|
if (!res.ok) {
|
||||||
|
const { message } = await res.json();
|
||||||
|
showToast(message || "An error occured while authorizing", Toasts.Type.FAILURE);
|
||||||
|
return;
|
||||||
|
}
|
||||||
|
|
||||||
|
const { token } = await res.json();
|
||||||
|
updateAuth({ token });
|
||||||
|
showToast("Successfully logged in!", Toasts.Type.SUCCESS);
|
||||||
|
callback?.();
|
||||||
|
} catch (e) {
|
||||||
|
new Logger("ReviewDB").error("Failed to authorize", e);
|
||||||
|
}
|
||||||
|
}}
|
||||||
|
/>
|
||||||
|
);
|
||||||
|
}
|
99
src/plugins/reviewDB/components/BlockedUserModal.tsx
Normal file
99
src/plugins/reviewDB/components/BlockedUserModal.tsx
Normal file
|
@ -0,0 +1,99 @@
|
||||||
|
/*
|
||||||
|
* Vencord, a Discord client mod
|
||||||
|
* Copyright (c) 2024 Vendicated and contributors
|
||||||
|
* SPDX-License-Identifier: GPL-3.0-or-later
|
||||||
|
*/
|
||||||
|
|
||||||
|
import { Logger } from "@utils/Logger";
|
||||||
|
import { ModalCloseButton, ModalContent, ModalHeader, ModalRoot, openModal } from "@utils/modal";
|
||||||
|
import { useAwaiter } from "@utils/react";
|
||||||
|
import { Forms, Tooltip, useState } from "@webpack/common";
|
||||||
|
|
||||||
|
import { Auth } from "../auth";
|
||||||
|
import { ReviewDBUser } from "../entities";
|
||||||
|
import { fetchBlocks, unblockUser } from "../reviewDbApi";
|
||||||
|
import { cl } from "../utils";
|
||||||
|
|
||||||
|
function UnblockButton(props: { onClick?(): void; }) {
|
||||||
|
return (
|
||||||
|
<Tooltip text="Unblock user">
|
||||||
|
{tooltipProps => (
|
||||||
|
<div
|
||||||
|
{...tooltipProps}
|
||||||
|
role="button"
|
||||||
|
onClick={props.onClick}
|
||||||
|
className={cl("block-modal-unblock")}
|
||||||
|
>
|
||||||
|
<svg height="20" viewBox="0 -960 960 960" width="20" fill="var(--status-danger)">
|
||||||
|
<path d="M480-80q-83 0-156-31.5T197-197q-54-54-85.5-127T80-480q0-83 31.5-156T197-763q54-54 127-85.5T480-880q83 0 156 31.5T763-763q54 54 85.5 127T880-480q0 83-31.5 156T763-197q-54 54-127 85.5T480-80Zm0-80q54 0 104-17.5t92-50.5L228-676q-33 42-50.5 92T160-480q0 134 93 227t227 93Zm252-124q33-42 50.5-92T800-480q0-134-93-227t-227-93q-54 0-104 17.5T284-732l448 448Z" />
|
||||||
|
</svg>
|
||||||
|
</div>
|
||||||
|
)}
|
||||||
|
</Tooltip>
|
||||||
|
);
|
||||||
|
}
|
||||||
|
|
||||||
|
function BlockedUser({ user, isBusy, setIsBusy }: { user: ReviewDBUser; isBusy: boolean; setIsBusy(v: boolean): void; }) {
|
||||||
|
const [gone, setGone] = useState(false);
|
||||||
|
if (gone) return null;
|
||||||
|
|
||||||
|
return (
|
||||||
|
<div className={cl("block-modal-row")}>
|
||||||
|
<img src={user.profilePhoto} alt="" />
|
||||||
|
<Forms.FormText className={cl("block-modal-username")}>{user.username}</Forms.FormText>
|
||||||
|
<UnblockButton
|
||||||
|
onClick={isBusy ? undefined : async () => {
|
||||||
|
setIsBusy(true);
|
||||||
|
try {
|
||||||
|
await unblockUser(user.discordID);
|
||||||
|
setGone(true);
|
||||||
|
} finally {
|
||||||
|
setIsBusy(false);
|
||||||
|
}
|
||||||
|
}}
|
||||||
|
/>
|
||||||
|
</div>
|
||||||
|
);
|
||||||
|
}
|
||||||
|
|
||||||
|
function Modal() {
|
||||||
|
const [isBusy, setIsBusy] = useState(false);
|
||||||
|
const [blocks, error, pending] = useAwaiter(fetchBlocks, {
|
||||||
|
onError: e => new Logger("ReviewDB").error("Failed to fetch blocks", e),
|
||||||
|
fallbackValue: [],
|
||||||
|
});
|
||||||
|
|
||||||
|
if (pending)
|
||||||
|
return null;
|
||||||
|
if (error)
|
||||||
|
return <Forms.FormText>Failed to fetch blocks: ${String(error)}</Forms.FormText>;
|
||||||
|
if (!blocks.length)
|
||||||
|
return <Forms.FormText>No blocked users.</Forms.FormText>;
|
||||||
|
|
||||||
|
return (
|
||||||
|
<>
|
||||||
|
{blocks.map(b => (
|
||||||
|
<BlockedUser
|
||||||
|
key={b.discordID}
|
||||||
|
user={b}
|
||||||
|
isBusy={isBusy}
|
||||||
|
setIsBusy={setIsBusy}
|
||||||
|
/>
|
||||||
|
))}
|
||||||
|
</>
|
||||||
|
);
|
||||||
|
}
|
||||||
|
|
||||||
|
export function openBlockModal() {
|
||||||
|
openModal(modalProps => (
|
||||||
|
<ModalRoot {...modalProps}>
|
||||||
|
<ModalHeader className={cl("block-modal-header")}>
|
||||||
|
<Forms.FormTitle style={{ margin: 0 }}>Blocked Users</Forms.FormTitle>
|
||||||
|
<ModalCloseButton onClick={modalProps.onClose} />
|
||||||
|
</ModalHeader>
|
||||||
|
<ModalContent className={cl("block-modal")}>
|
||||||
|
{Auth.token ? <Modal /> : <Forms.FormText>You are not logged into ReviewDB!</Forms.FormText>}
|
||||||
|
</ModalContent>
|
||||||
|
</ModalRoot>
|
||||||
|
));
|
||||||
|
}
|
85
src/plugins/reviewDB/components/MessageButton.tsx
Normal file
85
src/plugins/reviewDB/components/MessageButton.tsx
Normal file
|
@ -0,0 +1,85 @@
|
||||||
|
/*
|
||||||
|
* 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 { DeleteIcon } from "@components/Icons";
|
||||||
|
import { classes } from "@utils/misc";
|
||||||
|
import { findByPropsLazy } from "@webpack";
|
||||||
|
import { Tooltip } from "@webpack/common";
|
||||||
|
|
||||||
|
const iconClasses = findByPropsLazy("button", "wrapper", "disabled", "separator");
|
||||||
|
|
||||||
|
export function DeleteButton({ onClick }: { onClick(): void; }) {
|
||||||
|
return (
|
||||||
|
<Tooltip text="Delete Review">
|
||||||
|
{props => (
|
||||||
|
<div
|
||||||
|
{...props}
|
||||||
|
className={classes(iconClasses.button, iconClasses.dangerous)}
|
||||||
|
onClick={onClick}
|
||||||
|
role="button"
|
||||||
|
>
|
||||||
|
<DeleteIcon width="20" height="20" />
|
||||||
|
</div>
|
||||||
|
)}
|
||||||
|
</Tooltip>
|
||||||
|
);
|
||||||
|
}
|
||||||
|
|
||||||
|
export function ReportButton({ onClick }: { onClick(): void; }) {
|
||||||
|
return (
|
||||||
|
<Tooltip text="Report Review">
|
||||||
|
{props => (
|
||||||
|
<div
|
||||||
|
{...props}
|
||||||
|
className={iconClasses.button}
|
||||||
|
onClick={onClick}
|
||||||
|
role="button"
|
||||||
|
>
|
||||||
|
<svg width="20" height="20" viewBox="0 0 24 24">
|
||||||
|
<path
|
||||||
|
fill="currentColor"
|
||||||
|
d="M20,6.002H14V3.002C14,2.45 13.553,2.002 13,2.002H4C3.447,2.002 3,2.45 3,3.002V22.002H5V14.002H10.586L8.293,16.295C8.007,16.581 7.922,17.011 8.076,17.385C8.23,17.759 8.596,18.002 9,18.002H20C20.553,18.002 21,17.554 21,17.002V7.002C21,6.45 20.553,6.002 20,6.002Z"
|
||||||
|
/>
|
||||||
|
</svg>
|
||||||
|
</div>
|
||||||
|
)}
|
||||||
|
</Tooltip>
|
||||||
|
);
|
||||||
|
}
|
||||||
|
|
||||||
|
export function BlockButton({ onClick, isBlocked }: { onClick(): void; isBlocked: boolean; }) {
|
||||||
|
return (
|
||||||
|
<Tooltip text={`${isBlocked ? "Unblock" : "Block"} user`}>
|
||||||
|
{props => (
|
||||||
|
<div
|
||||||
|
{...props}
|
||||||
|
className={iconClasses.button}
|
||||||
|
onClick={onClick}
|
||||||
|
role="button"
|
||||||
|
>
|
||||||
|
<svg height="20" viewBox="0 -960 960 960" width="20" fill="currentColor">
|
||||||
|
{isBlocked
|
||||||
|
? <path d="M480-80q-83 0-156-31.5T197-197q-54-54-85.5-127T80-480q0-83 31.5-156T197-763q54-54 127-85.5T480-880q83 0 156 31.5T763-763q54 54 85.5 127T880-480q0 83-31.5 156T763-197q-54 54-127 85.5T480-80Zm0-80q134 0 227-93t93-227q0-134-93-227t-227-93q-134 0-227 93t-93 227q0 134 93 227t227 93Zm0-320Z" />
|
||||||
|
: <path d="M480-80q-83 0-156-31.5T197-197q-54-54-85.5-127T80-480q0-83 31.5-156T197-763q54-54 127-85.5T480-880q83 0 156 31.5T763-763q54 54 85.5 127T880-480q0 83-31.5 156T763-197q-54 54-127 85.5T480-80Zm0-80q54 0 104-17.5t92-50.5L228-676q-33 42-50.5 92T160-480q0 134 93 227t227 93Zm252-124q33-42 50.5-92T800-480q0-134-93-227t-227-93q-54 0-104 17.5T284-732l448 448Z" />
|
||||||
|
}
|
||||||
|
</svg>
|
||||||
|
</div>
|
||||||
|
)}
|
||||||
|
</Tooltip>
|
||||||
|
);
|
||||||
|
}
|
50
src/plugins/reviewDB/components/ReviewBadge.tsx
Normal file
50
src/plugins/reviewDB/components/ReviewBadge.tsx
Normal file
|
@ -0,0 +1,50 @@
|
||||||
|
/*
|
||||||
|
* 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 { MaskedLink, React, Tooltip } from "@webpack/common";
|
||||||
|
import { HTMLAttributes } from "react";
|
||||||
|
|
||||||
|
import { Badge } from "../entities";
|
||||||
|
import { cl } from "../utils";
|
||||||
|
|
||||||
|
export default function ReviewBadge(badge: Badge & { onClick?(): void; }) {
|
||||||
|
const Wrapper = badge.redirectURL
|
||||||
|
? MaskedLink
|
||||||
|
: (props: HTMLAttributes<HTMLDivElement>) => (
|
||||||
|
<span {...props} role="button">{props.children}</span>
|
||||||
|
);
|
||||||
|
|
||||||
|
return (
|
||||||
|
<Tooltip
|
||||||
|
text={badge.name}>
|
||||||
|
{({ onMouseEnter, onMouseLeave }) => (
|
||||||
|
<Wrapper className={cl("blocked-badge")} href={badge.redirectURL!} onClick={badge.onClick}>
|
||||||
|
<img
|
||||||
|
className={cl("badge")}
|
||||||
|
width="22px"
|
||||||
|
height="22px"
|
||||||
|
onMouseEnter={onMouseEnter}
|
||||||
|
onMouseLeave={onMouseLeave}
|
||||||
|
src={badge.icon}
|
||||||
|
alt={badge.description}
|
||||||
|
/>
|
||||||
|
</Wrapper>
|
||||||
|
)}
|
||||||
|
</Tooltip>
|
||||||
|
);
|
||||||
|
}
|
191
src/plugins/reviewDB/components/ReviewComponent.tsx
Normal file
191
src/plugins/reviewDB/components/ReviewComponent.tsx
Normal file
|
@ -0,0 +1,191 @@
|
||||||
|
/*
|
||||||
|
* 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 { openUserProfile } from "@utils/discord";
|
||||||
|
import { classes } from "@utils/misc";
|
||||||
|
import { LazyComponent } from "@utils/react";
|
||||||
|
import { filters, findBulk } from "@webpack";
|
||||||
|
import { Alerts, moment, Parser, Timestamp, useState } from "@webpack/common";
|
||||||
|
|
||||||
|
import { Auth, getToken } from "../auth";
|
||||||
|
import { Review, ReviewType } from "../entities";
|
||||||
|
import { blockUser, deleteReview, reportReview, unblockUser } from "../reviewDbApi";
|
||||||
|
import { settings } from "../settings";
|
||||||
|
import { canBlockReviewAuthor, canDeleteReview, canReportReview, cl, showToast } from "../utils";
|
||||||
|
import { openBlockModal } from "./BlockedUserModal";
|
||||||
|
import { BlockButton, DeleteButton, ReportButton } from "./MessageButton";
|
||||||
|
import ReviewBadge from "./ReviewBadge";
|
||||||
|
|
||||||
|
export default LazyComponent(() => {
|
||||||
|
// this is terrible, blame mantika
|
||||||
|
const p = filters.byProps;
|
||||||
|
const [
|
||||||
|
{ cozyMessage, buttons, message, buttonsInner, groupStart },
|
||||||
|
{ container, isHeader },
|
||||||
|
{ avatar, clickable, username, wrapper, cozy },
|
||||||
|
buttonClasses,
|
||||||
|
botTag
|
||||||
|
] = findBulk(
|
||||||
|
p("cozyMessage"),
|
||||||
|
p("container", "isHeader"),
|
||||||
|
p("avatar", "zalgo"),
|
||||||
|
p("button", "wrapper", "selected"),
|
||||||
|
p("botTag", "botTagRegular")
|
||||||
|
);
|
||||||
|
|
||||||
|
const dateFormat = new Intl.DateTimeFormat();
|
||||||
|
|
||||||
|
return function ReviewComponent({ review, refetch, profileId }: { review: Review; refetch(): void; profileId: string; }) {
|
||||||
|
const [showAll, setShowAll] = useState(false);
|
||||||
|
|
||||||
|
function openModal() {
|
||||||
|
openUserProfile(review.sender.discordID);
|
||||||
|
}
|
||||||
|
|
||||||
|
function delReview() {
|
||||||
|
Alerts.show({
|
||||||
|
title: "Are you sure?",
|
||||||
|
body: "Do you really want to delete this review?",
|
||||||
|
confirmText: "Delete",
|
||||||
|
cancelText: "Nevermind",
|
||||||
|
onConfirm: async () => {
|
||||||
|
if (!(await getToken())) {
|
||||||
|
return showToast("You must be logged in to delete reviews.");
|
||||||
|
} else {
|
||||||
|
deleteReview(review.id).then(res => {
|
||||||
|
if (res) {
|
||||||
|
refetch();
|
||||||
|
}
|
||||||
|
});
|
||||||
|
}
|
||||||
|
}
|
||||||
|
});
|
||||||
|
}
|
||||||
|
|
||||||
|
function reportRev() {
|
||||||
|
Alerts.show({
|
||||||
|
title: "Are you sure?",
|
||||||
|
body: "Do you really you want to report this review?",
|
||||||
|
confirmText: "Report",
|
||||||
|
cancelText: "Nevermind",
|
||||||
|
// confirmColor: "red", this just adds a class name and breaks the submit button guh
|
||||||
|
onConfirm: async () => {
|
||||||
|
if (!(await getToken())) {
|
||||||
|
return showToast("You must be logged in to report reviews.");
|
||||||
|
} else {
|
||||||
|
reportReview(review.id);
|
||||||
|
}
|
||||||
|
}
|
||||||
|
});
|
||||||
|
}
|
||||||
|
|
||||||
|
const isAuthorBlocked = Auth?.user?.blockedUsers?.includes(review.sender.discordID) ?? false;
|
||||||
|
|
||||||
|
function blockReviewSender() {
|
||||||
|
if (isAuthorBlocked)
|
||||||
|
return unblockUser(review.sender.discordID);
|
||||||
|
|
||||||
|
Alerts.show({
|
||||||
|
title: "Are you sure?",
|
||||||
|
body: "Do you really you want to block this user? They will be unable to leave further reviews on your profile. You can unblock users in the plugin settings.",
|
||||||
|
confirmText: "Block",
|
||||||
|
cancelText: "Nevermind",
|
||||||
|
// confirmColor: "red", this just adds a class name and breaks the submit button guh
|
||||||
|
onConfirm: async () => {
|
||||||
|
if (!(await getToken())) {
|
||||||
|
return showToast("You must be logged in to block users.");
|
||||||
|
} else {
|
||||||
|
blockUser(review.sender.discordID);
|
||||||
|
}
|
||||||
|
}
|
||||||
|
});
|
||||||
|
}
|
||||||
|
|
||||||
|
return (
|
||||||
|
<div className={classes(cl("review"), cozyMessage, wrapper, message, groupStart, cozy)} style={
|
||||||
|
{
|
||||||
|
marginLeft: "0px",
|
||||||
|
paddingLeft: "52px", // wth is this
|
||||||
|
// nobody knows anymore
|
||||||
|
}
|
||||||
|
}>
|
||||||
|
|
||||||
|
<img
|
||||||
|
className={classes(avatar, clickable)}
|
||||||
|
onClick={openModal}
|
||||||
|
src={review.sender.profilePhoto || "/assets/1f0bfc0865d324c2587920a7d80c609b.png?size=128"}
|
||||||
|
style={{ left: "0px", zIndex: 0 }}
|
||||||
|
/>
|
||||||
|
<div style={{ display: "inline-flex", justifyContent: "center", alignItems: "center" }}>
|
||||||
|
<span
|
||||||
|
className={classes(clickable, username)}
|
||||||
|
style={{ color: "var(--channels-default)", fontSize: "14px" }}
|
||||||
|
onClick={() => openModal()}
|
||||||
|
>
|
||||||
|
{review.sender.username}
|
||||||
|
</span>
|
||||||
|
|
||||||
|
{review.type === ReviewType.System && (
|
||||||
|
<span
|
||||||
|
className={classes(botTag.botTagVerified, botTag.botTagRegular, botTag.botTag, botTag.px, botTag.rem)}
|
||||||
|
style={{ marginLeft: "4px" }}>
|
||||||
|
<span className={botTag.botText}>
|
||||||
|
System
|
||||||
|
</span>
|
||||||
|
</span>
|
||||||
|
)}
|
||||||
|
</div>
|
||||||
|
{isAuthorBlocked && (
|
||||||
|
<ReviewBadge
|
||||||
|
name="You have blocked this user"
|
||||||
|
description="You have blocked this user"
|
||||||
|
icon="/assets/aaee57e0090991557b66.svg"
|
||||||
|
type={0}
|
||||||
|
onClick={() => openBlockModal()}
|
||||||
|
/>
|
||||||
|
)}
|
||||||
|
{review.sender.badges.map(badge => <ReviewBadge {...badge} />)}
|
||||||
|
|
||||||
|
{
|
||||||
|
!settings.store.hideTimestamps && review.type !== ReviewType.System && (
|
||||||
|
<Timestamp timestamp={moment(review.timestamp * 1000)} >
|
||||||
|
{dateFormat.format(review.timestamp * 1000)}
|
||||||
|
</Timestamp>)
|
||||||
|
}
|
||||||
|
|
||||||
|
<div className={cl("review-comment")}>
|
||||||
|
{(review.comment.length > 200 && !showAll)
|
||||||
|
? [Parser.parseGuildEventDescription(review.comment.substring(0, 200)), "...", <br />, (<a onClick={() => setShowAll(true)}>Read more</a>)]
|
||||||
|
: Parser.parseGuildEventDescription(review.comment)}
|
||||||
|
</div>
|
||||||
|
|
||||||
|
{review.id !== 0 && (
|
||||||
|
<div className={classes(container, isHeader, buttons)} style={{
|
||||||
|
padding: "0px",
|
||||||
|
}}>
|
||||||
|
<div className={classes(buttonClasses.wrapper, buttonsInner)} >
|
||||||
|
{canReportReview(review) && <ReportButton onClick={reportRev} />}
|
||||||
|
{canBlockReviewAuthor(profileId, review) && <BlockButton isBlocked={isAuthorBlocked} onClick={blockReviewSender} />}
|
||||||
|
{canDeleteReview(profileId, review) && <DeleteButton onClick={delReview} />}
|
||||||
|
</div>
|
||||||
|
</div>
|
||||||
|
)}
|
||||||
|
</div>
|
||||||
|
);
|
||||||
|
};
|
||||||
|
});
|
105
src/plugins/reviewDB/components/ReviewModal.tsx
Normal file
105
src/plugins/reviewDB/components/ReviewModal.tsx
Normal file
|
@ -0,0 +1,105 @@
|
||||||
|
/*
|
||||||
|
* 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 ErrorBoundary from "@components/ErrorBoundary";
|
||||||
|
import { ModalCloseButton, ModalContent, ModalFooter, ModalHeader, ModalRoot, ModalSize, openModal } from "@utils/modal";
|
||||||
|
import { useForceUpdater } from "@utils/react";
|
||||||
|
import { Paginator, Text, useRef, useState } from "@webpack/common";
|
||||||
|
|
||||||
|
import { Auth } from "../auth";
|
||||||
|
import { Response, REVIEWS_PER_PAGE } from "../reviewDbApi";
|
||||||
|
import { cl } from "../utils";
|
||||||
|
import ReviewComponent from "./ReviewComponent";
|
||||||
|
import ReviewsView, { ReviewsInputComponent } from "./ReviewsView";
|
||||||
|
|
||||||
|
function Modal({ modalProps, discordId, name }: { modalProps: any; discordId: string; name: string; }) {
|
||||||
|
const [data, setData] = useState<Response>();
|
||||||
|
const [signal, refetch] = useForceUpdater(true);
|
||||||
|
const [page, setPage] = useState(1);
|
||||||
|
|
||||||
|
const ref = useRef<HTMLDivElement>(null);
|
||||||
|
|
||||||
|
const reviewCount = data?.reviewCount;
|
||||||
|
const ownReview = data?.reviews.find(r => r.sender.discordID === Auth.user?.discordID);
|
||||||
|
|
||||||
|
return (
|
||||||
|
<ErrorBoundary>
|
||||||
|
<ModalRoot {...modalProps} size={ModalSize.MEDIUM}>
|
||||||
|
<ModalHeader>
|
||||||
|
<Text variant="heading-lg/semibold" className={cl("modal-header")}>
|
||||||
|
{name}'s Reviews
|
||||||
|
{!!reviewCount && <span> ({reviewCount} Reviews)</span>}
|
||||||
|
</Text>
|
||||||
|
<ModalCloseButton onClick={modalProps.onClose} />
|
||||||
|
</ModalHeader>
|
||||||
|
|
||||||
|
<ModalContent scrollerRef={ref}>
|
||||||
|
<div className={cl("modal-reviews")}>
|
||||||
|
<ReviewsView
|
||||||
|
discordId={discordId}
|
||||||
|
name={name}
|
||||||
|
page={page}
|
||||||
|
refetchSignal={signal}
|
||||||
|
onFetchReviews={setData}
|
||||||
|
scrollToTop={() => ref.current?.scrollTo({ top: 0, behavior: "smooth" })}
|
||||||
|
hideOwnReview
|
||||||
|
/>
|
||||||
|
</div>
|
||||||
|
</ModalContent>
|
||||||
|
|
||||||
|
<ModalFooter className={cl("modal-footer")}>
|
||||||
|
<div>
|
||||||
|
{ownReview && (
|
||||||
|
<ReviewComponent
|
||||||
|
refetch={refetch}
|
||||||
|
review={ownReview}
|
||||||
|
profileId={discordId}
|
||||||
|
/>
|
||||||
|
)}
|
||||||
|
<ReviewsInputComponent
|
||||||
|
isAuthor={ownReview != null}
|
||||||
|
discordId={discordId}
|
||||||
|
name={name}
|
||||||
|
refetch={refetch}
|
||||||
|
/>
|
||||||
|
|
||||||
|
{!!reviewCount && (
|
||||||
|
<Paginator
|
||||||
|
currentPage={page}
|
||||||
|
maxVisiblePages={5}
|
||||||
|
pageSize={REVIEWS_PER_PAGE}
|
||||||
|
totalCount={reviewCount}
|
||||||
|
onPageChange={setPage}
|
||||||
|
/>
|
||||||
|
)}
|
||||||
|
</div>
|
||||||
|
</ModalFooter>
|
||||||
|
</ModalRoot>
|
||||||
|
</ErrorBoundary>
|
||||||
|
);
|
||||||
|
}
|
||||||
|
|
||||||
|
export function openReviewsModal(discordId: string, name: string) {
|
||||||
|
openModal(props => (
|
||||||
|
<Modal
|
||||||
|
modalProps={props}
|
||||||
|
discordId={discordId}
|
||||||
|
name={name}
|
||||||
|
/>
|
||||||
|
));
|
||||||
|
}
|
197
src/plugins/reviewDB/components/ReviewsView.tsx
Normal file
197
src/plugins/reviewDB/components/ReviewsView.tsx
Normal file
|
@ -0,0 +1,197 @@
|
||||||
|
/*
|
||||||
|
* 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 { LazyComponent, useAwaiter, useForceUpdater } from "@utils/react";
|
||||||
|
import { find, findByPropsLazy } from "@webpack";
|
||||||
|
import { Forms, React, RelationshipStore, useRef, UserStore } from "@webpack/common";
|
||||||
|
|
||||||
|
import { Auth, authorize } from "../auth";
|
||||||
|
import { Review } from "../entities";
|
||||||
|
import { addReview, getReviews, Response, REVIEWS_PER_PAGE } from "../reviewDbApi";
|
||||||
|
import { settings } from "../settings";
|
||||||
|
import { cl, showToast } from "../utils";
|
||||||
|
import ReviewComponent from "./ReviewComponent";
|
||||||
|
|
||||||
|
|
||||||
|
const { Editor, Transforms } = findByPropsLazy("Editor", "Transforms");
|
||||||
|
const { ChatInputTypes } = findByPropsLazy("ChatInputTypes");
|
||||||
|
|
||||||
|
const InputComponent = LazyComponent(() => find(m => m.default?.type?.render?.toString().includes("default.CHANNEL_TEXT_AREA")).default);
|
||||||
|
|
||||||
|
interface UserProps {
|
||||||
|
discordId: string;
|
||||||
|
name: string;
|
||||||
|
}
|
||||||
|
|
||||||
|
interface Props extends UserProps {
|
||||||
|
onFetchReviews(data: Response): void;
|
||||||
|
refetchSignal?: unknown;
|
||||||
|
showInput?: boolean;
|
||||||
|
page?: number;
|
||||||
|
scrollToTop?(): void;
|
||||||
|
hideOwnReview?: boolean;
|
||||||
|
}
|
||||||
|
|
||||||
|
export default function ReviewsView({
|
||||||
|
discordId,
|
||||||
|
name,
|
||||||
|
onFetchReviews,
|
||||||
|
refetchSignal,
|
||||||
|
scrollToTop,
|
||||||
|
page = 1,
|
||||||
|
showInput = false,
|
||||||
|
hideOwnReview = false,
|
||||||
|
}: Props) {
|
||||||
|
const [signal, refetch] = useForceUpdater(true);
|
||||||
|
|
||||||
|
const [reviewData] = useAwaiter(() => getReviews(discordId, (page - 1) * REVIEWS_PER_PAGE), {
|
||||||
|
fallbackValue: null,
|
||||||
|
deps: [refetchSignal, signal, page],
|
||||||
|
onSuccess: data => {
|
||||||
|
if (settings.store.hideBlockedUsers)
|
||||||
|
data!.reviews = data!.reviews?.filter(r => !RelationshipStore.isBlocked(r.sender.discordID));
|
||||||
|
|
||||||
|
scrollToTop?.();
|
||||||
|
onFetchReviews(data!);
|
||||||
|
}
|
||||||
|
});
|
||||||
|
|
||||||
|
if (!reviewData) return null;
|
||||||
|
|
||||||
|
return (
|
||||||
|
<>
|
||||||
|
<ReviewList
|
||||||
|
refetch={refetch}
|
||||||
|
reviews={reviewData!.reviews}
|
||||||
|
hideOwnReview={hideOwnReview}
|
||||||
|
profileId={discordId}
|
||||||
|
/>
|
||||||
|
|
||||||
|
{showInput && (
|
||||||
|
<ReviewsInputComponent
|
||||||
|
name={name}
|
||||||
|
discordId={discordId}
|
||||||
|
refetch={refetch}
|
||||||
|
isAuthor={reviewData!.reviews?.some(r => r.sender.discordID === UserStore.getCurrentUser().id)}
|
||||||
|
/>
|
||||||
|
)}
|
||||||
|
</>
|
||||||
|
);
|
||||||
|
}
|
||||||
|
|
||||||
|
function ReviewList({ refetch, reviews, hideOwnReview, profileId }: { refetch(): void; reviews: Review[]; hideOwnReview: boolean; profileId: string; }) {
|
||||||
|
const myId = UserStore.getCurrentUser().id;
|
||||||
|
|
||||||
|
return (
|
||||||
|
<div className={cl("view")}>
|
||||||
|
{reviews?.map(review =>
|
||||||
|
(review.sender.discordID !== myId || !hideOwnReview) &&
|
||||||
|
<ReviewComponent
|
||||||
|
key={review.id}
|
||||||
|
review={review}
|
||||||
|
refetch={refetch}
|
||||||
|
profileId={profileId}
|
||||||
|
/>
|
||||||
|
)}
|
||||||
|
|
||||||
|
{reviews?.length === 0 && (
|
||||||
|
<Forms.FormText className={cl("placeholder")}>
|
||||||
|
Looks like nobody reviewed this user yet. You could be the first!
|
||||||
|
</Forms.FormText>
|
||||||
|
)}
|
||||||
|
</div>
|
||||||
|
);
|
||||||
|
}
|
||||||
|
|
||||||
|
|
||||||
|
export function ReviewsInputComponent({ discordId, isAuthor, refetch, name }: { discordId: string, name: string; isAuthor: boolean; refetch(): void; }) {
|
||||||
|
const { token } = Auth;
|
||||||
|
const editorRef = useRef<any>(null);
|
||||||
|
const inputType = ChatInputTypes.FORM;
|
||||||
|
inputType.disableAutoFocus = true;
|
||||||
|
|
||||||
|
const channel = {
|
||||||
|
flags_: 256,
|
||||||
|
guild_id_: null,
|
||||||
|
id: "0",
|
||||||
|
getGuildId: () => null,
|
||||||
|
isPrivate: () => true,
|
||||||
|
isActiveThread: () => false,
|
||||||
|
isArchivedLockedThread: () => false,
|
||||||
|
isDM: () => true,
|
||||||
|
roles: { "0": { permissions: 0n } },
|
||||||
|
getRecipientId: () => "0",
|
||||||
|
hasFlag: () => false,
|
||||||
|
};
|
||||||
|
|
||||||
|
return (
|
||||||
|
<>
|
||||||
|
<div onClick={() => {
|
||||||
|
if (!token) {
|
||||||
|
showToast("Opening authorization window...");
|
||||||
|
authorize();
|
||||||
|
}
|
||||||
|
}}>
|
||||||
|
<InputComponent
|
||||||
|
className={cl("input")}
|
||||||
|
channel={channel}
|
||||||
|
placeholder={
|
||||||
|
!token
|
||||||
|
? "You need to authorize to review users!"
|
||||||
|
: isAuthor
|
||||||
|
? `Update review for @${name}`
|
||||||
|
: `Review @${name}`
|
||||||
|
}
|
||||||
|
type={inputType}
|
||||||
|
disableThemedBackground={true}
|
||||||
|
setEditorRef={ref => editorRef.current = ref}
|
||||||
|
textValue=""
|
||||||
|
onSubmit={
|
||||||
|
async res => {
|
||||||
|
const response = await addReview({
|
||||||
|
userid: discordId,
|
||||||
|
comment: res.value,
|
||||||
|
});
|
||||||
|
|
||||||
|
if (response) {
|
||||||
|
refetch();
|
||||||
|
|
||||||
|
const slateEditor = editorRef.current.ref.current.getSlateEditor();
|
||||||
|
|
||||||
|
// clear editor
|
||||||
|
Transforms.delete(slateEditor, {
|
||||||
|
at: {
|
||||||
|
anchor: Editor.start(slateEditor, []),
|
||||||
|
focus: Editor.end(slateEditor, []),
|
||||||
|
}
|
||||||
|
});
|
||||||
|
}
|
||||||
|
|
||||||
|
// even tho we need to return this, it doesnt do anything
|
||||||
|
return {
|
||||||
|
shouldClear: false,
|
||||||
|
shouldRefocus: true,
|
||||||
|
};
|
||||||
|
}
|
||||||
|
}
|
||||||
|
/>
|
||||||
|
</div>
|
||||||
|
|
||||||
|
</>
|
||||||
|
);
|
||||||
|
}
|
100
src/plugins/reviewDB/entities.ts
Normal file
100
src/plugins/reviewDB/entities.ts
Normal file
|
@ -0,0 +1,100 @@
|
||||||
|
/*
|
||||||
|
* 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/>.
|
||||||
|
*/
|
||||||
|
|
||||||
|
export const enum UserType {
|
||||||
|
Banned = -1,
|
||||||
|
Normal = 0,
|
||||||
|
Admin = 1
|
||||||
|
}
|
||||||
|
|
||||||
|
export const enum ReviewType {
|
||||||
|
User = 0,
|
||||||
|
Server = 1,
|
||||||
|
Support = 2,
|
||||||
|
System = 3
|
||||||
|
}
|
||||||
|
|
||||||
|
export const enum NotificationType {
|
||||||
|
Info = 0,
|
||||||
|
Ban = 1,
|
||||||
|
Unban = 2,
|
||||||
|
Warning = 3
|
||||||
|
}
|
||||||
|
|
||||||
|
export interface ReviewDBAuth {
|
||||||
|
token?: string;
|
||||||
|
user?: ReviewDBCurrentUser;
|
||||||
|
}
|
||||||
|
|
||||||
|
export interface Badge {
|
||||||
|
name: string;
|
||||||
|
description: string;
|
||||||
|
icon: string;
|
||||||
|
redirectURL?: string;
|
||||||
|
type: number;
|
||||||
|
}
|
||||||
|
|
||||||
|
export interface BanInfo {
|
||||||
|
id: string;
|
||||||
|
discordID: string;
|
||||||
|
reviewID: number;
|
||||||
|
reviewContent: string;
|
||||||
|
banEndDate: number;
|
||||||
|
}
|
||||||
|
|
||||||
|
export interface Notification {
|
||||||
|
id: number;
|
||||||
|
title: string;
|
||||||
|
content: string;
|
||||||
|
type: NotificationType;
|
||||||
|
}
|
||||||
|
|
||||||
|
export interface ReviewDBUser {
|
||||||
|
ID: number;
|
||||||
|
discordID: string;
|
||||||
|
username: string;
|
||||||
|
type: UserType;
|
||||||
|
profilePhoto: string;
|
||||||
|
badges: any[];
|
||||||
|
}
|
||||||
|
|
||||||
|
export interface ReviewDBCurrentUser extends ReviewDBUser {
|
||||||
|
warningCount: number;
|
||||||
|
clientMod: string;
|
||||||
|
banInfo: BanInfo | null;
|
||||||
|
notification: Notification | null;
|
||||||
|
lastReviewID: number;
|
||||||
|
blockedUsers?: string[];
|
||||||
|
}
|
||||||
|
|
||||||
|
export interface ReviewAuthor {
|
||||||
|
id: number,
|
||||||
|
discordID: string,
|
||||||
|
username: string,
|
||||||
|
profilePhoto: string,
|
||||||
|
badges: Badge[];
|
||||||
|
}
|
||||||
|
|
||||||
|
export interface Review {
|
||||||
|
comment: string,
|
||||||
|
id: number,
|
||||||
|
star: number,
|
||||||
|
sender: ReviewAuthor,
|
||||||
|
timestamp: number;
|
||||||
|
type?: ReviewType;
|
||||||
|
}
|
158
src/plugins/reviewDB/index.tsx
Normal file
158
src/plugins/reviewDB/index.tsx
Normal file
|
@ -0,0 +1,158 @@
|
||||||
|
/*
|
||||||
|
* 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 "./style.css";
|
||||||
|
|
||||||
|
import { addContextMenuPatch, NavContextMenuPatchCallback, removeContextMenuPatch } from "@api/ContextMenu";
|
||||||
|
import ErrorBoundary from "@components/ErrorBoundary";
|
||||||
|
import ExpandableHeader from "@components/ExpandableHeader";
|
||||||
|
import { OpenExternalIcon } from "@components/Icons";
|
||||||
|
import { Devs } from "@utils/constants";
|
||||||
|
import { Logger } from "@utils/Logger";
|
||||||
|
import definePlugin from "@utils/types";
|
||||||
|
import { Alerts, Menu, Parser, useState } from "@webpack/common";
|
||||||
|
import { Guild, User } from "discord-types/general";
|
||||||
|
|
||||||
|
import { Auth, initAuth, updateAuth } from "./auth";
|
||||||
|
import { openReviewsModal } from "./components/ReviewModal";
|
||||||
|
import ReviewsView from "./components/ReviewsView";
|
||||||
|
import { NotificationType } from "./entities";
|
||||||
|
import { getCurrentUserInfo, readNotification } from "./reviewDbApi";
|
||||||
|
import { settings } from "./settings";
|
||||||
|
import { showToast } from "./utils";
|
||||||
|
|
||||||
|
const guildPopoutPatch: NavContextMenuPatchCallback = (children, props: { guild: Guild, onClose(): void; }) => () => {
|
||||||
|
children.push(
|
||||||
|
<Menu.MenuItem
|
||||||
|
label="View Reviews"
|
||||||
|
id="vc-rdb-server-reviews"
|
||||||
|
icon={OpenExternalIcon}
|
||||||
|
action={() => openReviewsModal(props.guild.id, props.guild.name)}
|
||||||
|
/>
|
||||||
|
);
|
||||||
|
};
|
||||||
|
|
||||||
|
export default definePlugin({
|
||||||
|
name: "ReviewDB",
|
||||||
|
description: "Review other users (Adds a new settings to profiles)",
|
||||||
|
authors: [Devs.mantikafasi, Devs.Ven],
|
||||||
|
|
||||||
|
settings,
|
||||||
|
|
||||||
|
patches: [
|
||||||
|
{
|
||||||
|
find: "showBorder:null",
|
||||||
|
replacement: {
|
||||||
|
match: /user:(\i),setNote:\i,canDM.+?\}\)/,
|
||||||
|
replace: "$&,$self.getReviewsComponent($1)"
|
||||||
|
}
|
||||||
|
}
|
||||||
|
],
|
||||||
|
|
||||||
|
flux: {
|
||||||
|
CONNECTION_OPEN: initAuth,
|
||||||
|
},
|
||||||
|
|
||||||
|
async start() {
|
||||||
|
addContextMenuPatch("guild-header-popout", guildPopoutPatch);
|
||||||
|
|
||||||
|
const s = settings.store;
|
||||||
|
const { lastReviewId, notifyReviews } = s;
|
||||||
|
|
||||||
|
const legacy = s as any as { token?: string; };
|
||||||
|
if (legacy.token) {
|
||||||
|
await updateAuth({ token: legacy.token });
|
||||||
|
legacy.token = undefined;
|
||||||
|
new Logger("ReviewDB").info("Migrated legacy settings");
|
||||||
|
}
|
||||||
|
|
||||||
|
await initAuth();
|
||||||
|
|
||||||
|
setTimeout(async () => {
|
||||||
|
if (!Auth.token) return;
|
||||||
|
|
||||||
|
const user = await getCurrentUserInfo(Auth.token);
|
||||||
|
updateAuth({ user });
|
||||||
|
|
||||||
|
if (notifyReviews) {
|
||||||
|
if (lastReviewId && lastReviewId < user.lastReviewID) {
|
||||||
|
s.lastReviewId = user.lastReviewID;
|
||||||
|
if (user.lastReviewID !== 0)
|
||||||
|
showToast("You have new reviews on your profile!");
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
if (user.notification) {
|
||||||
|
const props = user.notification.type === NotificationType.Ban ? {
|
||||||
|
cancelText: "Appeal",
|
||||||
|
confirmText: "Ok",
|
||||||
|
onCancel: async () =>
|
||||||
|
VencordNative.native.openExternal(
|
||||||
|
"https://reviewdb.mantikafasi.dev/api/redirect?"
|
||||||
|
+ new URLSearchParams({
|
||||||
|
token: Auth.token!,
|
||||||
|
page: "dashboard/appeal"
|
||||||
|
})
|
||||||
|
)
|
||||||
|
} : {};
|
||||||
|
|
||||||
|
Alerts.show({
|
||||||
|
title: user.notification.title,
|
||||||
|
body: (
|
||||||
|
Parser.parse(
|
||||||
|
user.notification.content,
|
||||||
|
false
|
||||||
|
)
|
||||||
|
),
|
||||||
|
...props
|
||||||
|
});
|
||||||
|
|
||||||
|
readNotification(user.notification.id);
|
||||||
|
}
|
||||||
|
}, 4000);
|
||||||
|
},
|
||||||
|
|
||||||
|
stop() {
|
||||||
|
removeContextMenuPatch("guild-header-popout", guildPopoutPatch);
|
||||||
|
},
|
||||||
|
|
||||||
|
getReviewsComponent: ErrorBoundary.wrap((user: User) => {
|
||||||
|
const [reviewCount, setReviewCount] = useState<number>();
|
||||||
|
|
||||||
|
return (
|
||||||
|
<ExpandableHeader
|
||||||
|
headerText="User Reviews"
|
||||||
|
onMoreClick={() => openReviewsModal(user.id, user.username)}
|
||||||
|
moreTooltipText={
|
||||||
|
reviewCount && reviewCount > 50
|
||||||
|
? `View all ${reviewCount} reviews`
|
||||||
|
: "Open Review Modal"
|
||||||
|
}
|
||||||
|
onDropDownClick={state => settings.store.reviewsDropdownState = !state}
|
||||||
|
defaultState={settings.store.reviewsDropdownState}
|
||||||
|
>
|
||||||
|
<ReviewsView
|
||||||
|
discordId={user.id}
|
||||||
|
name={user.username}
|
||||||
|
onFetchReviews={r => setReviewCount(r.reviewCount)}
|
||||||
|
showInput
|
||||||
|
/>
|
||||||
|
</ExpandableHeader>
|
||||||
|
);
|
||||||
|
}, { message: "Failed to render Reviews" })
|
||||||
|
});
|
202
src/plugins/reviewDB/reviewDbApi.ts
Normal file
202
src/plugins/reviewDB/reviewDbApi.ts
Normal file
|
@ -0,0 +1,202 @@
|
||||||
|
/*
|
||||||
|
* 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 { Toasts } from "@webpack/common";
|
||||||
|
|
||||||
|
import { Auth, authorize, getToken, updateAuth } from "./auth";
|
||||||
|
import { Review, ReviewDBCurrentUser, ReviewDBUser, ReviewType } from "./entities";
|
||||||
|
import { settings } from "./settings";
|
||||||
|
import { showToast } from "./utils";
|
||||||
|
|
||||||
|
const API_URL = "https://manti.vendicated.dev/api/reviewdb";
|
||||||
|
|
||||||
|
export const REVIEWS_PER_PAGE = 50;
|
||||||
|
|
||||||
|
export interface Response {
|
||||||
|
message: string;
|
||||||
|
reviews: Review[];
|
||||||
|
updated: boolean;
|
||||||
|
hasNextPage: boolean;
|
||||||
|
reviewCount: number;
|
||||||
|
}
|
||||||
|
|
||||||
|
const WarningFlag = 0b00000010;
|
||||||
|
|
||||||
|
async function rdbRequest(path: string, options: RequestInit = {}) {
|
||||||
|
return fetch(API_URL + path, {
|
||||||
|
...options,
|
||||||
|
headers: {
|
||||||
|
...options.headers,
|
||||||
|
Authorization: await getToken() || "",
|
||||||
|
}
|
||||||
|
});
|
||||||
|
}
|
||||||
|
|
||||||
|
export async function getReviews(id: string, offset = 0): Promise<Response> {
|
||||||
|
let flags = 0;
|
||||||
|
if (!settings.store.showWarning) flags |= WarningFlag;
|
||||||
|
|
||||||
|
const params = new URLSearchParams({
|
||||||
|
flags: String(flags),
|
||||||
|
offset: String(offset)
|
||||||
|
});
|
||||||
|
const req = await fetch(`${API_URL}/users/${id}/reviews?${params}`);
|
||||||
|
|
||||||
|
const res = (req.ok)
|
||||||
|
? await req.json() as Response
|
||||||
|
: {
|
||||||
|
message: req.status === 429 ? "You are sending requests too fast. Wait a few seconds and try again." : "An Error occured while fetching reviews. Please try again later.",
|
||||||
|
reviews: [],
|
||||||
|
updated: false,
|
||||||
|
hasNextPage: false,
|
||||||
|
reviewCount: 0
|
||||||
|
};
|
||||||
|
|
||||||
|
if (!req.ok) {
|
||||||
|
showToast(res.message, Toasts.Type.FAILURE);
|
||||||
|
return {
|
||||||
|
...res,
|
||||||
|
reviews: [
|
||||||
|
{
|
||||||
|
id: 0,
|
||||||
|
comment: res.message,
|
||||||
|
star: 0,
|
||||||
|
timestamp: 0,
|
||||||
|
type: ReviewType.System,
|
||||||
|
sender: {
|
||||||
|
id: 0,
|
||||||
|
username: "ReviewDB",
|
||||||
|
profilePhoto: "https://cdn.discordapp.com/avatars/1134864775000629298/3f87ad315b32ee464d84f1270c8d1b37.png?size=256&format=webp&quality=lossless",
|
||||||
|
discordID: "1134864775000629298",
|
||||||
|
badges: []
|
||||||
|
}
|
||||||
|
}
|
||||||
|
]
|
||||||
|
};
|
||||||
|
}
|
||||||
|
|
||||||
|
return res;
|
||||||
|
}
|
||||||
|
|
||||||
|
export async function addReview(review: any): Promise<Response | null> {
|
||||||
|
|
||||||
|
const token = await getToken();
|
||||||
|
if (!token) {
|
||||||
|
showToast("Please authorize to add a review.");
|
||||||
|
authorize();
|
||||||
|
return null;
|
||||||
|
}
|
||||||
|
|
||||||
|
return await rdbRequest(`/users/${review.userid}/reviews`, {
|
||||||
|
method: "PUT",
|
||||||
|
body: JSON.stringify(review),
|
||||||
|
headers: {
|
||||||
|
"Content-Type": "application/json",
|
||||||
|
}
|
||||||
|
}).then(async r => {
|
||||||
|
const data = await r.json() as Response;
|
||||||
|
showToast(data.message);
|
||||||
|
return r.ok ? data : null;
|
||||||
|
});
|
||||||
|
}
|
||||||
|
|
||||||
|
export async function deleteReview(id: number): Promise<Response | null> {
|
||||||
|
return await rdbRequest(`/users/${id}/reviews`, {
|
||||||
|
method: "DELETE",
|
||||||
|
headers: new Headers({
|
||||||
|
"Content-Type": "application/json",
|
||||||
|
Accept: "application/json",
|
||||||
|
}),
|
||||||
|
body: JSON.stringify({
|
||||||
|
reviewid: id
|
||||||
|
})
|
||||||
|
}).then(async r => {
|
||||||
|
const data = await r.json() as Response;
|
||||||
|
showToast(data.message);
|
||||||
|
return r.ok ? data : null;
|
||||||
|
});
|
||||||
|
}
|
||||||
|
|
||||||
|
export async function reportReview(id: number) {
|
||||||
|
const res = await rdbRequest("/reports", {
|
||||||
|
method: "PUT",
|
||||||
|
headers: new Headers({
|
||||||
|
"Content-Type": "application/json",
|
||||||
|
Accept: "application/json",
|
||||||
|
}),
|
||||||
|
body: JSON.stringify({
|
||||||
|
reviewid: id,
|
||||||
|
})
|
||||||
|
}).then(r => r.json()) as Response;
|
||||||
|
|
||||||
|
showToast(res.message);
|
||||||
|
}
|
||||||
|
|
||||||
|
async function patchBlock(action: "block" | "unblock", userId: string) {
|
||||||
|
const res = await rdbRequest("/blocks", {
|
||||||
|
method: "PATCH",
|
||||||
|
headers: new Headers({
|
||||||
|
"Content-Type": "application/json",
|
||||||
|
Accept: "application/json",
|
||||||
|
}),
|
||||||
|
body: JSON.stringify({
|
||||||
|
action: action,
|
||||||
|
discordId: userId
|
||||||
|
})
|
||||||
|
});
|
||||||
|
|
||||||
|
if (!res.ok) {
|
||||||
|
showToast(`Failed to ${action} user`, Toasts.Type.FAILURE);
|
||||||
|
} else {
|
||||||
|
showToast(`Successfully ${action}ed user`, Toasts.Type.SUCCESS);
|
||||||
|
|
||||||
|
if (Auth?.user?.blockedUsers) {
|
||||||
|
const newBlockedUsers = action === "block"
|
||||||
|
? [...Auth.user.blockedUsers, userId]
|
||||||
|
: Auth.user.blockedUsers.filter(id => id !== userId);
|
||||||
|
updateAuth({ user: { ...Auth.user, blockedUsers: newBlockedUsers } });
|
||||||
|
}
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
export const blockUser = (userId: string) => patchBlock("block", userId);
|
||||||
|
export const unblockUser = (userId: string) => patchBlock("unblock", userId);
|
||||||
|
|
||||||
|
export async function fetchBlocks(): Promise<ReviewDBUser[]> {
|
||||||
|
const res = await rdbRequest("/blocks", {
|
||||||
|
method: "GET",
|
||||||
|
headers: new Headers({
|
||||||
|
Accept: "application/json",
|
||||||
|
})
|
||||||
|
});
|
||||||
|
|
||||||
|
if (!res.ok) throw new Error(`${res.status}: ${res.statusText}`);
|
||||||
|
return res.json();
|
||||||
|
}
|
||||||
|
|
||||||
|
export function getCurrentUserInfo(token: string): Promise<ReviewDBCurrentUser> {
|
||||||
|
return rdbRequest("/users", {
|
||||||
|
method: "POST",
|
||||||
|
}).then(r => r.json());
|
||||||
|
}
|
||||||
|
|
||||||
|
export async function readNotification(id: number) {
|
||||||
|
return rdbRequest(`/notifications?id=${id}`, {
|
||||||
|
method: "PATCH"
|
||||||
|
});
|
||||||
|
}
|
96
src/plugins/reviewDB/settings.tsx
Normal file
96
src/plugins/reviewDB/settings.tsx
Normal file
|
@ -0,0 +1,96 @@
|
||||||
|
/*
|
||||||
|
* 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 { definePluginSettings } from "@api/Settings";
|
||||||
|
import { OptionType } from "@utils/types";
|
||||||
|
import { Button } from "@webpack/common";
|
||||||
|
|
||||||
|
import { authorize, getToken } from "./auth";
|
||||||
|
import { openBlockModal } from "./components/BlockedUserModal";
|
||||||
|
import { cl } from "./utils";
|
||||||
|
|
||||||
|
export const settings = definePluginSettings({
|
||||||
|
authorize: {
|
||||||
|
type: OptionType.COMPONENT,
|
||||||
|
description: "Authorize with ReviewDB",
|
||||||
|
component: () => (
|
||||||
|
<Button onClick={() => authorize()}>
|
||||||
|
Authorize with ReviewDB
|
||||||
|
</Button>
|
||||||
|
)
|
||||||
|
},
|
||||||
|
notifyReviews: {
|
||||||
|
type: OptionType.BOOLEAN,
|
||||||
|
description: "Notify about new reviews on startup",
|
||||||
|
default: true,
|
||||||
|
},
|
||||||
|
showWarning: {
|
||||||
|
type: OptionType.BOOLEAN,
|
||||||
|
description: "Display warning to be respectful at the top of the reviews list",
|
||||||
|
default: true,
|
||||||
|
},
|
||||||
|
hideTimestamps: {
|
||||||
|
type: OptionType.BOOLEAN,
|
||||||
|
description: "Hide timestamps on reviews",
|
||||||
|
default: false,
|
||||||
|
},
|
||||||
|
hideBlockedUsers: {
|
||||||
|
type: OptionType.BOOLEAN,
|
||||||
|
description: "Hide reviews from blocked users",
|
||||||
|
default: true,
|
||||||
|
},
|
||||||
|
buttons: {
|
||||||
|
type: OptionType.COMPONENT,
|
||||||
|
description: "ReviewDB buttons",
|
||||||
|
component: () => (
|
||||||
|
<div className={cl("button-grid")} >
|
||||||
|
<Button onClick={openBlockModal}>Manage Blocked Users</Button>
|
||||||
|
|
||||||
|
<Button
|
||||||
|
color={Button.Colors.GREEN}
|
||||||
|
onClick={() => {
|
||||||
|
VencordNative.native.openExternal("https://github.com/sponsors/mantikafasi");
|
||||||
|
}}
|
||||||
|
>
|
||||||
|
Support ReviewDB development
|
||||||
|
</Button>
|
||||||
|
|
||||||
|
<Button onClick={async () => {
|
||||||
|
let url = "https://reviewdb.mantikafasi.dev/";
|
||||||
|
const token = await getToken();
|
||||||
|
if (token)
|
||||||
|
url += "/api/redirect?token=" + encodeURIComponent(token);
|
||||||
|
|
||||||
|
VencordNative.native.openExternal(url);
|
||||||
|
}}>
|
||||||
|
ReviewDB website
|
||||||
|
</Button>
|
||||||
|
|
||||||
|
|
||||||
|
<Button onClick={() => {
|
||||||
|
VencordNative.native.openExternal("https://discord.gg/eWPBSbvznt");
|
||||||
|
}}>
|
||||||
|
ReviewDB Support Server
|
||||||
|
</Button>
|
||||||
|
</div >
|
||||||
|
)
|
||||||
|
}
|
||||||
|
}).withPrivateSettings<{
|
||||||
|
lastReviewId?: number;
|
||||||
|
reviewsDropdownState?: boolean;
|
||||||
|
}>();
|
140
src/plugins/reviewDB/style.css
Normal file
140
src/plugins/reviewDB/style.css
Normal file
|
@ -0,0 +1,140 @@
|
||||||
|
[class|="section"]:not([class|="lastSection"]) + .vc-rdb-view {
|
||||||
|
margin-top: 12px;
|
||||||
|
}
|
||||||
|
|
||||||
|
.vc-rdb-badge {
|
||||||
|
vertical-align: middle;
|
||||||
|
margin-left: 4px;
|
||||||
|
}
|
||||||
|
|
||||||
|
.vc-rdb-input {
|
||||||
|
margin-top: 6px;
|
||||||
|
margin-bottom: 12px;
|
||||||
|
resize: none;
|
||||||
|
overflow: hidden;
|
||||||
|
background: transparent;
|
||||||
|
border: 1px solid var(--profile-message-input-border-color);
|
||||||
|
}
|
||||||
|
|
||||||
|
.vc-rdb-modal-footer > div {
|
||||||
|
width: 100%;
|
||||||
|
margin: 6px 16px;
|
||||||
|
}
|
||||||
|
|
||||||
|
/* When input becomes disabled(while sending review), input adds unneccesary padding to left, this prevents it */
|
||||||
|
.vc-rdb-input > div > div {
|
||||||
|
padding-left: 0 !important;
|
||||||
|
}
|
||||||
|
|
||||||
|
.vc-rdb-placeholder {
|
||||||
|
margin-bottom: 4px;
|
||||||
|
font-weight: bold;
|
||||||
|
font-style: italic;
|
||||||
|
color: var(--text-muted);
|
||||||
|
}
|
||||||
|
|
||||||
|
.vc-rdb-input * {
|
||||||
|
font-size: 14px;
|
||||||
|
}
|
||||||
|
|
||||||
|
.vc-rdb-modal-footer {
|
||||||
|
padding: 0;
|
||||||
|
}
|
||||||
|
|
||||||
|
.vc-rdb-modal-footer .vc-rdb-input {
|
||||||
|
margin-bottom: 0;
|
||||||
|
background: var(--input-background);
|
||||||
|
}
|
||||||
|
|
||||||
|
.vc-rdb-modal-footer [class|="pageControlContainer"] {
|
||||||
|
margin-top: 0;
|
||||||
|
}
|
||||||
|
|
||||||
|
.vc-rdb-modal-header {
|
||||||
|
flex-grow: 1;
|
||||||
|
}
|
||||||
|
|
||||||
|
.vc-rdb-modal-reviews {
|
||||||
|
margin-top: 16px;
|
||||||
|
}
|
||||||
|
|
||||||
|
.vc-rdb-review {
|
||||||
|
padding-top: 8px !important;
|
||||||
|
padding-bottom: 8px !important;
|
||||||
|
padding-right: 32px !important;
|
||||||
|
}
|
||||||
|
|
||||||
|
.vc-rdb-review:hover {
|
||||||
|
background: var(--background-message-hover) !important;
|
||||||
|
border-radius: 8px;
|
||||||
|
}
|
||||||
|
|
||||||
|
.vc-rdb-review-comment img {
|
||||||
|
vertical-align: text-top;
|
||||||
|
}
|
||||||
|
|
||||||
|
.vc-rdb-review-comment {
|
||||||
|
overflow-y: hidden;
|
||||||
|
margin-top: 1px;
|
||||||
|
margin-bottom: 8px;
|
||||||
|
color: var(--text-normal);
|
||||||
|
font-size: 15px;
|
||||||
|
}
|
||||||
|
|
||||||
|
.vc-rdb-blocked-badge {
|
||||||
|
cursor: pointer;
|
||||||
|
}
|
||||||
|
|
||||||
|
.vc-rdb-block-modal-header {
|
||||||
|
display: flex;
|
||||||
|
justify-content: space-between;
|
||||||
|
align-items: center;
|
||||||
|
}
|
||||||
|
|
||||||
|
.vc-rdb-block-modal {
|
||||||
|
padding: 1em;
|
||||||
|
display: grid;
|
||||||
|
gap: 0.75em;
|
||||||
|
}
|
||||||
|
|
||||||
|
.vc-rdb-button-grid {
|
||||||
|
display: grid;
|
||||||
|
grid-template-columns: 1fr 1fr;
|
||||||
|
gap: 10px;
|
||||||
|
}
|
||||||
|
|
||||||
|
/* stylelint-disable-next-line media-feature-range-notation */
|
||||||
|
@media (max-width: 600px) {
|
||||||
|
.vc-rdb-button-grid {
|
||||||
|
grid-template-columns: 1fr;
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
.vc-rdb-block-modal-row {
|
||||||
|
display: flex;
|
||||||
|
height: 2em;
|
||||||
|
gap: 0.5em;
|
||||||
|
align-items: center;
|
||||||
|
}
|
||||||
|
|
||||||
|
.vc-rdb-block-modal-row img {
|
||||||
|
border-radius: 50%;
|
||||||
|
height: 2em;
|
||||||
|
width: 2em;
|
||||||
|
}
|
||||||
|
|
||||||
|
.vc-rdb-block-modal img::before {
|
||||||
|
content: "";
|
||||||
|
display: block;
|
||||||
|
width: 100%;
|
||||||
|
height: 100%;
|
||||||
|
background-color: var(--background-modifier-accent);
|
||||||
|
}
|
||||||
|
|
||||||
|
.vc-rdb-block-modal-username {
|
||||||
|
flex-grow: 1;
|
||||||
|
}
|
||||||
|
|
||||||
|
.vc-rdb-block-modal-unblock {
|
||||||
|
cursor: pointer;
|
||||||
|
}
|
54
src/plugins/reviewDB/utils.tsx
Normal file
54
src/plugins/reviewDB/utils.tsx
Normal file
|
@ -0,0 +1,54 @@
|
||||||
|
/*
|
||||||
|
* 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 { classNameFactory } from "@api/Styles";
|
||||||
|
import { Toasts, UserStore } from "@webpack/common";
|
||||||
|
|
||||||
|
import { Auth } from "./auth";
|
||||||
|
import { Review, UserType } from "./entities";
|
||||||
|
|
||||||
|
export const cl = classNameFactory("vc-rdb-");
|
||||||
|
|
||||||
|
export function canDeleteReview(profileId: string, review: Review) {
|
||||||
|
const myId = UserStore.getCurrentUser().id;
|
||||||
|
return (
|
||||||
|
myId === profileId
|
||||||
|
|| review.sender.discordID === myId
|
||||||
|
|| Auth.user?.type === UserType.Admin
|
||||||
|
);
|
||||||
|
}
|
||||||
|
|
||||||
|
export function canBlockReviewAuthor(profileId: string, review: Review) {
|
||||||
|
const myId = UserStore.getCurrentUser().id;
|
||||||
|
return profileId === myId && review.sender.discordID !== myId;
|
||||||
|
}
|
||||||
|
|
||||||
|
export function canReportReview(review: Review) {
|
||||||
|
return review.sender.discordID !== UserStore.getCurrentUser().id;
|
||||||
|
}
|
||||||
|
|
||||||
|
export function showToast(message: string, type = Toasts.Type.MESSAGE) {
|
||||||
|
Toasts.show({
|
||||||
|
id: Toasts.genId(),
|
||||||
|
message,
|
||||||
|
type,
|
||||||
|
options: {
|
||||||
|
position: Toasts.Position.BOTTOM, // NOBODY LIKES TOASTS AT THE TOP
|
||||||
|
},
|
||||||
|
});
|
||||||
|
}
|
|
@ -72,10 +72,6 @@ export default definePlugin({
|
||||||
{
|
{
|
||||||
find: 'tutorialId:"whos-online',
|
find: 'tutorialId:"whos-online',
|
||||||
replacement: [
|
replacement: [
|
||||||
{
|
|
||||||
match: /\i.roleIcon,\.\.\.\i/,
|
|
||||||
replace: "$&,color:$self.roleGroupColor(arguments[0])"
|
|
||||||
},
|
|
||||||
{
|
{
|
||||||
match: /null,\i," — ",\i\]/,
|
match: /null,\i," — ",\i\]/,
|
||||||
replace: "null,$self.roleGroupColor(arguments[0])]"
|
replace: "null,$self.roleGroupColor(arguments[0])]"
|
||||||
|
@ -83,6 +79,16 @@ export default definePlugin({
|
||||||
],
|
],
|
||||||
predicate: () => settings.store.memberList,
|
predicate: () => settings.store.memberList,
|
||||||
},
|
},
|
||||||
|
{
|
||||||
|
find: ".Messages.THREAD_BROWSER_PRIVATE",
|
||||||
|
replacement: [
|
||||||
|
{
|
||||||
|
match: /children:\[\i," — ",\i\]/,
|
||||||
|
replace: "children:[$self.roleGroupColor(arguments[0])]"
|
||||||
|
},
|
||||||
|
],
|
||||||
|
predicate: () => settings.store.memberList,
|
||||||
|
},
|
||||||
{
|
{
|
||||||
find: "renderPrioritySpeaker",
|
find: "renderPrioritySpeaker",
|
||||||
replacement: [
|
replacement: [
|
||||||
|
|
|
@ -18,14 +18,24 @@
|
||||||
|
|
||||||
import "./styles.css";
|
import "./styles.css";
|
||||||
|
|
||||||
|
import { addChatBarButton, ChatBarButton, removeChatBarButton } from "@api/ChatButtons";
|
||||||
import { addPreSendListener, removePreSendListener } from "@api/MessageEvents";
|
import { addPreSendListener, removePreSendListener } from "@api/MessageEvents";
|
||||||
|
import { definePluginSettings } from "@api/Settings";
|
||||||
import { classNameFactory } from "@api/Styles";
|
import { classNameFactory } from "@api/Styles";
|
||||||
import { Devs } from "@utils/constants";
|
import { Devs } from "@utils/constants";
|
||||||
import { getTheme, insertTextIntoChatInputBox, Theme } from "@utils/discord";
|
import { getTheme, insertTextIntoChatInputBox, Theme } from "@utils/discord";
|
||||||
import { Margins } from "@utils/margins";
|
import { Margins } from "@utils/margins";
|
||||||
import { closeModal, ModalCloseButton, ModalContent, ModalFooter, ModalHeader, ModalProps, ModalRoot, openModal } from "@utils/modal";
|
import { closeModal, ModalCloseButton, ModalContent, ModalFooter, ModalHeader, ModalProps, ModalRoot, openModal } from "@utils/modal";
|
||||||
import definePlugin from "@utils/types";
|
import definePlugin, { OptionType } from "@utils/types";
|
||||||
import { Button, ButtonLooks, ButtonWrapperClasses, Forms, Parser, Select, Tooltip, useMemo, useState } from "@webpack/common";
|
import { Button, Forms, Parser, Select, useMemo, useState } from "@webpack/common";
|
||||||
|
|
||||||
|
const settings = definePluginSettings({
|
||||||
|
replaceMessageContents: {
|
||||||
|
description: "Replace timestamps in message contents",
|
||||||
|
type: OptionType.BOOLEAN,
|
||||||
|
default: true,
|
||||||
|
},
|
||||||
|
});
|
||||||
|
|
||||||
function parseTime(time: string) {
|
function parseTime(time: string) {
|
||||||
const cleanTime = time.slice(1, -1).replace(/(\d)(AM|PM)$/i, "$1 $2");
|
const cleanTime = time.slice(1, -1).replace(/(\d)(AM|PM)$/i, "$1 $2");
|
||||||
|
@ -113,79 +123,61 @@ function PickerModal({ rootProps, close }: { rootProps: ModalProps, close(): voi
|
||||||
);
|
);
|
||||||
}
|
}
|
||||||
|
|
||||||
|
const ChatBarIcon: ChatBarButton = ({ isMainChat }) => {
|
||||||
|
if (!isMainChat) return null;
|
||||||
|
|
||||||
|
return (
|
||||||
|
<ChatBarButton
|
||||||
|
tooltip="Insert Timestamp"
|
||||||
|
onClick={() => {
|
||||||
|
const key = openModal(props => (
|
||||||
|
<PickerModal
|
||||||
|
rootProps={props}
|
||||||
|
close={() => closeModal(key)}
|
||||||
|
/>
|
||||||
|
));
|
||||||
|
}}
|
||||||
|
buttonProps={{ "aria-haspopup": "dialog" }}
|
||||||
|
>
|
||||||
|
<svg
|
||||||
|
aria-hidden="true"
|
||||||
|
role="img"
|
||||||
|
width="24"
|
||||||
|
height="24"
|
||||||
|
viewBox="0 0 24 24"
|
||||||
|
style={{ scale: "1.2" }}
|
||||||
|
>
|
||||||
|
<g fill="none" fill-rule="evenodd">
|
||||||
|
<path fill="currentColor" d="M19 3h-1V1h-2v2H8V1H6v2H5c-1.11 0-1.99.9-1.99 2L3 19a2 2 0 0 0 2 2h14c1.1 0 2-.9 2-2V5c0-1.1-.9-2-2-2zm0 16H5V8h14v11zM7 10h5v5H7v-5z" />
|
||||||
|
<rect width="24" height="24" />
|
||||||
|
</g>
|
||||||
|
</svg>
|
||||||
|
</ChatBarButton>
|
||||||
|
);
|
||||||
|
};
|
||||||
|
|
||||||
export default definePlugin({
|
export default definePlugin({
|
||||||
name: "SendTimestamps",
|
name: "SendTimestamps",
|
||||||
description: "Send timestamps easily via chat box button & text shortcuts. Read the extended description!",
|
description: "Send timestamps easily via chat box button & text shortcuts. Read the extended description!",
|
||||||
authors: [Devs.Ven, Devs.Tyler],
|
authors: [Devs.Ven, Devs.Tyler, Devs.Grzesiek11],
|
||||||
dependencies: ["MessageEventsAPI"],
|
dependencies: ["MessageEventsAPI", "ChatInputButtonAPI"],
|
||||||
|
|
||||||
patches: [
|
settings,
|
||||||
{
|
|
||||||
find: "ChannelTextAreaButtons",
|
|
||||||
replacement: {
|
|
||||||
match: /(\i)\.push.{1,30}disabled:(\i),.{1,20}\},"gift"\)\)/,
|
|
||||||
replace: "$&,(()=>{try{$2||$1.push($self.chatBarIcon(arguments[0]))}catch{}})()",
|
|
||||||
}
|
|
||||||
},
|
|
||||||
],
|
|
||||||
|
|
||||||
start() {
|
start() {
|
||||||
|
addChatBarButton("SendTimestamps", ChatBarIcon);
|
||||||
this.listener = addPreSendListener((_, msg) => {
|
this.listener = addPreSendListener((_, msg) => {
|
||||||
msg.content = msg.content.replace(/`\d{1,2}:\d{2} ?(?:AM|PM)?`/gi, parseTime);
|
if (settings.store.replaceMessageContents) {
|
||||||
|
msg.content = msg.content.replace(/`\d{1,2}:\d{2} ?(?:AM|PM)?`/gi, parseTime);
|
||||||
|
}
|
||||||
});
|
});
|
||||||
},
|
},
|
||||||
|
|
||||||
stop() {
|
stop() {
|
||||||
|
removeChatBarButton("SendTimestamps");
|
||||||
removePreSendListener(this.listener);
|
removePreSendListener(this.listener);
|
||||||
},
|
},
|
||||||
|
|
||||||
chatBarIcon(chatBoxProps: { type: { analyticsName: string; }; }) {
|
|
||||||
if (chatBoxProps.type.analyticsName !== "normal") return null;
|
|
||||||
|
|
||||||
return (
|
|
||||||
<Tooltip text="Insert Timestamp">
|
|
||||||
{({ onMouseEnter, onMouseLeave }) => (
|
|
||||||
<div style={{ display: "flex" }}>
|
|
||||||
<Button
|
|
||||||
aria-haspopup="dialog"
|
|
||||||
aria-label="Insert Timestamp"
|
|
||||||
size=""
|
|
||||||
look={ButtonLooks.BLANK}
|
|
||||||
onMouseEnter={onMouseEnter}
|
|
||||||
onMouseLeave={onMouseLeave}
|
|
||||||
innerClassName={ButtonWrapperClasses.button}
|
|
||||||
onClick={() => {
|
|
||||||
const key = openModal(props => (
|
|
||||||
<PickerModal
|
|
||||||
rootProps={props}
|
|
||||||
close={() => closeModal(key)}
|
|
||||||
/>
|
|
||||||
));
|
|
||||||
}}
|
|
||||||
className={cl("button")}
|
|
||||||
>
|
|
||||||
<div className={ButtonWrapperClasses.buttonWrapper}>
|
|
||||||
<svg
|
|
||||||
aria-hidden="true"
|
|
||||||
role="img"
|
|
||||||
width="24"
|
|
||||||
height="24"
|
|
||||||
viewBox="0 0 24 24"
|
|
||||||
>
|
|
||||||
<g fill="none" fill-rule="evenodd">
|
|
||||||
<path fill="currentColor" d="M19 3h-1V1h-2v2H8V1H6v2H5c-1.11 0-1.99.9-1.99 2L3 19a2 2 0 0 0 2 2h14c1.1 0 2-.9 2-2V5c0-1.1-.9-2-2-2zm0 16H5V8h14v11zM7 10h5v5H7v-5z" />
|
|
||||||
<rect width="24" height="24" />
|
|
||||||
</g>
|
|
||||||
</svg>
|
|
||||||
</div>
|
|
||||||
</Button>
|
|
||||||
</div>
|
|
||||||
)
|
|
||||||
}
|
|
||||||
</Tooltip >
|
|
||||||
);
|
|
||||||
},
|
|
||||||
|
|
||||||
settingsAboutComponent() {
|
settingsAboutComponent() {
|
||||||
const samples = [
|
const samples = [
|
||||||
"12:00",
|
"12:00",
|
||||||
|
|
|
@ -42,10 +42,6 @@
|
||||||
margin-bottom: 1em;
|
margin-bottom: 1em;
|
||||||
}
|
}
|
||||||
|
|
||||||
.vc-st-button {
|
|
||||||
padding: 0 6px;
|
|
||||||
}
|
|
||||||
|
|
||||||
.vc-st-button svg {
|
.vc-st-button svg {
|
||||||
transform: scale(1.1) translateY(1px);
|
transform: scale(1.1) translateY(1px);
|
||||||
}
|
}
|
||||||
|
|
|
@ -16,12 +16,12 @@
|
||||||
* along with this program. If not, see <https://www.gnu.org/licenses/>.
|
* along with this program. If not, see <https://www.gnu.org/licenses/>.
|
||||||
*/
|
*/
|
||||||
|
|
||||||
|
import { addChatBarButton, ChatBarButton, removeChatBarButton } from "@api/ChatButtons";
|
||||||
import { addPreSendListener, removePreSendListener, SendListener } from "@api/MessageEvents";
|
import { addPreSendListener, removePreSendListener, SendListener } from "@api/MessageEvents";
|
||||||
import { definePluginSettings } from "@api/Settings";
|
import { definePluginSettings } from "@api/Settings";
|
||||||
import ErrorBoundary from "@components/ErrorBoundary";
|
|
||||||
import { Devs } from "@utils/constants";
|
import { Devs } from "@utils/constants";
|
||||||
import definePlugin, { OptionType } from "@utils/types";
|
import definePlugin, { OptionType } from "@utils/types";
|
||||||
import { Button, ButtonLooks, ButtonWrapperClasses, React, Tooltip } from "@webpack/common";
|
import { React, useEffect, useState } from "@webpack/common";
|
||||||
|
|
||||||
let lastState = false;
|
let lastState = false;
|
||||||
|
|
||||||
|
@ -41,19 +41,15 @@ const settings = definePluginSettings({
|
||||||
}
|
}
|
||||||
});
|
});
|
||||||
|
|
||||||
function SilentMessageToggle(chatBoxProps: {
|
const SilentMessageToggle: ChatBarButton = ({ isMainChat }) => {
|
||||||
type: {
|
const [enabled, setEnabled] = useState(lastState);
|
||||||
analyticsName: string;
|
|
||||||
};
|
|
||||||
}) {
|
|
||||||
const [enabled, setEnabled] = React.useState(lastState);
|
|
||||||
|
|
||||||
function setEnabledValue(value: boolean) {
|
function setEnabledValue(value: boolean) {
|
||||||
if (settings.store.persistState) lastState = value;
|
if (settings.store.persistState) lastState = value;
|
||||||
setEnabled(value);
|
setEnabled(value);
|
||||||
}
|
}
|
||||||
|
|
||||||
React.useEffect(() => {
|
useEffect(() => {
|
||||||
const listener: SendListener = (_, message) => {
|
const listener: SendListener = (_, message) => {
|
||||||
if (enabled) {
|
if (enabled) {
|
||||||
if (settings.store.autoDisable) setEnabledValue(false);
|
if (settings.store.autoDisable) setEnabledValue(false);
|
||||||
|
@ -65,55 +61,39 @@ function SilentMessageToggle(chatBoxProps: {
|
||||||
return () => void removePreSendListener(listener);
|
return () => void removePreSendListener(listener);
|
||||||
}, [enabled]);
|
}, [enabled]);
|
||||||
|
|
||||||
if (chatBoxProps.type.analyticsName !== "normal") return null;
|
if (!isMainChat) return null;
|
||||||
|
|
||||||
return (
|
return (
|
||||||
<Tooltip text={enabled ? "Disable Silent Message" : "Enable Silent Message"}>
|
<ChatBarButton
|
||||||
{tooltipProps => (
|
tooltip={enabled ? "Disable Silent Message" : "Enable Silent Message"}
|
||||||
<div style={{ display: "flex" }}>
|
onClick={() => setEnabledValue(!enabled)}
|
||||||
<Button
|
>
|
||||||
{...tooltipProps}
|
<svg
|
||||||
onClick={() => setEnabledValue(!enabled)}
|
width="24"
|
||||||
size=""
|
height="24"
|
||||||
look={ButtonLooks.BLANK}
|
viewBox="0 0 24 24"
|
||||||
innerClassName={ButtonWrapperClasses.button}
|
style={{ scale: "1.2" }}
|
||||||
style={{ padding: "0 6px" }}
|
>
|
||||||
>
|
<path fill="currentColor" mask="url(#_)" d="M18 10.7101C15.1085 9.84957 13 7.17102 13 4c0-.30736.0198-.6101.0582-.907C12.7147 3.03189 12.3611 3 12 3 8.686 3 6 5.686 6 9v5c0 1.657-1.344 3-3 3v1h18v-1c-1.656 0-3-1.343-3-3v-3.2899ZM8.55493 19c.693 1.19 1.96897 2 3.44497 2s2.752-.81 3.445-2H8.55493ZM18.2624 5.50209 21 2.5V1h-4.9651v1.49791h2.4411L16 5.61088V7h5V5.50209h-2.7376Z" />
|
||||||
<div className={ButtonWrapperClasses.buttonWrapper}>
|
{!enabled && <>
|
||||||
<svg width="24" height="24" viewBox="0 0 24 24">
|
<mask id="_">
|
||||||
<path fill="currentColor" mask="url(#_)" d="M18 10.7101C15.1085 9.84957 13 7.17102 13 4c0-.30736.0198-.6101.0582-.907C12.7147 3.03189 12.3611 3 12 3 8.686 3 6 5.686 6 9v5c0 1.657-1.344 3-3 3v1h18v-1c-1.656 0-3-1.343-3-3v-3.2899ZM8.55493 19c.693 1.19 1.96897 2 3.44497 2s2.752-.81 3.445-2H8.55493ZM18.2624 5.50209 21 2.5V1h-4.9651v1.49791h2.4411L16 5.61088V7h5V5.50209h-2.7376Z" />
|
<path fill="#fff" d="M0 0h24v24H0Z" />
|
||||||
{!enabled && <>
|
<path stroke="#000" stroke-width="5.99068" d="M0 24 24 0" />
|
||||||
<mask id="_">
|
</mask>
|
||||||
<path fill="#fff" d="M0 0h24v24H0Z" />
|
<path fill="var(--status-danger)" d="m21.178 1.70703 1.414 1.414L4.12103 21.593l-1.414-1.415L21.178 1.70703Z" />
|
||||||
<path stroke="#000" stroke-width="5.99068" d="M0 24 24 0" />
|
</>}
|
||||||
</mask>
|
</svg>
|
||||||
<path fill="var(--status-danger)" d="m21.178 1.70703 1.414 1.414L4.12103 21.593l-1.414-1.415L21.178 1.70703Z" />
|
</ChatBarButton>
|
||||||
</>}
|
|
||||||
</svg>
|
|
||||||
</div>
|
|
||||||
</Button>
|
|
||||||
</div>
|
|
||||||
)}
|
|
||||||
</Tooltip>
|
|
||||||
);
|
);
|
||||||
}
|
};
|
||||||
|
|
||||||
export default definePlugin({
|
export default definePlugin({
|
||||||
name: "SilentMessageToggle",
|
name: "SilentMessageToggle",
|
||||||
authors: [Devs.Nuckyz, Devs.CatNoir],
|
authors: [Devs.Nuckyz, Devs.CatNoir],
|
||||||
description: "Adds a button to the chat bar to toggle sending a silent message.",
|
description: "Adds a button to the chat bar to toggle sending a silent message.",
|
||||||
dependencies: ["MessageEventsAPI"],
|
dependencies: ["MessageEventsAPI", "ChatInputButtonAPI"],
|
||||||
|
|
||||||
settings,
|
settings,
|
||||||
patches: [
|
|
||||||
{
|
|
||||||
find: "ChannelTextAreaButtons",
|
|
||||||
replacement: {
|
|
||||||
match: /(\i)\.push.{1,30}disabled:(\i),.{1,20}\},"gift"\)\)/,
|
|
||||||
replace: "$&,(()=>{try{$2||$1.push($self.chatBarIcon(arguments[0]))}catch{}})()",
|
|
||||||
}
|
|
||||||
},
|
|
||||||
],
|
|
||||||
|
|
||||||
chatBarIcon: ErrorBoundary.wrap(SilentMessageToggle, { noop: true }),
|
start: () => addChatBarButton("SilentMessageToggle", SilentMessageToggle),
|
||||||
|
stop: () => removeChatBarButton("SilentMessageToggle")
|
||||||
});
|
});
|
||||||
|
|
|
@ -16,12 +16,12 @@
|
||||||
* along with this program. If not, see <https://www.gnu.org/licenses/>.
|
* along with this program. If not, see <https://www.gnu.org/licenses/>.
|
||||||
*/
|
*/
|
||||||
|
|
||||||
|
import { addChatBarButton, ChatBarButton, removeChatBarButton } from "@api/ChatButtons";
|
||||||
import { ApplicationCommandInputType, ApplicationCommandOptionType, findOption, sendBotMessage } from "@api/Commands";
|
import { ApplicationCommandInputType, ApplicationCommandOptionType, findOption, sendBotMessage } from "@api/Commands";
|
||||||
import { definePluginSettings } from "@api/Settings";
|
import { definePluginSettings } from "@api/Settings";
|
||||||
import ErrorBoundary from "@components/ErrorBoundary";
|
|
||||||
import { Devs } from "@utils/constants";
|
import { Devs } from "@utils/constants";
|
||||||
import definePlugin, { OptionType } from "@utils/types";
|
import definePlugin, { OptionType } from "@utils/types";
|
||||||
import { Button, ButtonLooks, ButtonWrapperClasses, FluxDispatcher, React, Tooltip } from "@webpack/common";
|
import { FluxDispatcher, React } from "@webpack/common";
|
||||||
|
|
||||||
const settings = definePluginSettings({
|
const settings = definePluginSettings({
|
||||||
showIcon: {
|
showIcon: {
|
||||||
|
@ -37,45 +37,32 @@ const settings = definePluginSettings({
|
||||||
}
|
}
|
||||||
});
|
});
|
||||||
|
|
||||||
function SilentTypingToggle(chatBoxProps: {
|
const SilentTypingToggle: ChatBarButton = ({ isMainChat }) => {
|
||||||
type: {
|
const { isEnabled, showIcon } = settings.use(["isEnabled", "showIcon"]);
|
||||||
analyticsName: string;
|
|
||||||
};
|
|
||||||
}) {
|
|
||||||
const { isEnabled } = settings.use(["isEnabled"]);
|
|
||||||
const toggle = () => settings.store.isEnabled = !settings.store.isEnabled;
|
const toggle = () => settings.store.isEnabled = !settings.store.isEnabled;
|
||||||
|
|
||||||
if (chatBoxProps.type.analyticsName !== "normal") return null;
|
if (!isMainChat || !showIcon) return null;
|
||||||
|
|
||||||
return (
|
return (
|
||||||
<Tooltip text={isEnabled ? "Disable Silent Typing" : "Enable Silent Typing"}>
|
<ChatBarButton
|
||||||
{(tooltipProps: any) => (
|
tooltip={isEnabled ? "Disable Silent Typing" : "Enable Silent Typing"}
|
||||||
<div style={{ display: "flex" }}>
|
onClick={toggle}
|
||||||
<Button
|
>
|
||||||
{...tooltipProps}
|
<svg width="24" height="24" xmlns="http://www.w3.org/2000/svg" viewBox="0 0 576 512">
|
||||||
onClick={toggle}
|
<path fill="currentColor" d="M528 448H48c-26.51 0-48-21.49-48-48V112c0-26.51 21.49-48 48-48h480c26.51 0 48 21.49 48 48v288c0 26.51-21.49 48-48 48zM128 180v-40c0-6.627-5.373-12-12-12H76c-6.627 0-12 5.373-12 12v40c0 6.627 5.373 12 12 12h40c6.627 0 12-5.373 12-12zm96 0v-40c0-6.627-5.373-12-12-12h-40c-6.627 0-12 5.373-12 12v40c0 6.627 5.373 12 12 12h40c6.627 0 12-5.373 12-12zm96 0v-40c0-6.627-5.373-12-12-12h-40c-6.627 0-12 5.373-12 12v40c0 6.627 5.373 12 12 12h40c6.627 0 12-5.373 12-12zm96 0v-40c0-6.627-5.373-12-12-12h-40c-6.627 0-12 5.373-12 12v40c0 6.627 5.373 12 12 12h40c6.627 0 12-5.373 12-12zm96 0v-40c0-6.627-5.373-12-12-12h-40c-6.627 0-12 5.373-12 12v40c0 6.627 5.373 12 12 12h40c6.627 0 12-5.373 12-12zm-336 96v-40c0-6.627-5.373-12-12-12h-40c-6.627 0-12 5.373-12 12v40c0 6.627 5.373 12 12 12h40c6.627 0 12-5.373 12-12zm96 0v-40c0-6.627-5.373-12-12-12h-40c-6.627 0-12 5.373-12 12v40c0 6.627 5.373 12 12 12h40c6.627 0 12-5.373 12-12zm96 0v-40c0-6.627-5.373-12-12-12h-40c-6.627 0-12 5.373-12 12v40c0 6.627 5.373 12 12 12h40c6.627 0 12-5.373 12-12zm96 0v-40c0-6.627-5.373-12-12-12h-40c-6.627 0-12 5.373-12 12v40c0 6.627 5.373 12 12 12h40c6.627 0 12-5.373 12-12zm-336 96v-40c0-6.627-5.373-12-12-12H76c-6.627 0-12 5.373-12 12v40c0 6.627 5.373 12 12 12h40c6.627 0 12-5.373 12-12zm288 0v-40c0-6.627-5.373-12-12-12H172c-6.627 0-12 5.373-12 12v40c0 6.627 5.373 12 12 12h232c6.627 0 12-5.373 12-12zm96 0v-40c0-6.627-5.373-12-12-12h-40c-6.627 0-12 5.373-12 12v40c0 6.627 5.373 12 12 12h40c6.627 0 12-5.373 12-12z" />
|
||||||
size=""
|
{isEnabled && <path d="M13 432L590 48" stroke="var(--red-500)" stroke-width="72" stroke-linecap="round" />}
|
||||||
look={ButtonLooks.BLANK}
|
</svg>
|
||||||
innerClassName={ButtonWrapperClasses.button}
|
</ChatBarButton>
|
||||||
style={{ padding: "0 6px" }}
|
|
||||||
>
|
|
||||||
<div className={ButtonWrapperClasses.buttonWrapper}>
|
|
||||||
<svg width="24" height="24" xmlns="http://www.w3.org/2000/svg" viewBox="0 0 576 512">
|
|
||||||
<path fill="currentColor" d="M528 448H48c-26.51 0-48-21.49-48-48V112c0-26.51 21.49-48 48-48h480c26.51 0 48 21.49 48 48v288c0 26.51-21.49 48-48 48zM128 180v-40c0-6.627-5.373-12-12-12H76c-6.627 0-12 5.373-12 12v40c0 6.627 5.373 12 12 12h40c6.627 0 12-5.373 12-12zm96 0v-40c0-6.627-5.373-12-12-12h-40c-6.627 0-12 5.373-12 12v40c0 6.627 5.373 12 12 12h40c6.627 0 12-5.373 12-12zm96 0v-40c0-6.627-5.373-12-12-12h-40c-6.627 0-12 5.373-12 12v40c0 6.627 5.373 12 12 12h40c6.627 0 12-5.373 12-12zm96 0v-40c0-6.627-5.373-12-12-12h-40c-6.627 0-12 5.373-12 12v40c0 6.627 5.373 12 12 12h40c6.627 0 12-5.373 12-12zm96 0v-40c0-6.627-5.373-12-12-12h-40c-6.627 0-12 5.373-12 12v40c0 6.627 5.373 12 12 12h40c6.627 0 12-5.373 12-12zm-336 96v-40c0-6.627-5.373-12-12-12h-40c-6.627 0-12 5.373-12 12v40c0 6.627 5.373 12 12 12h40c6.627 0 12-5.373 12-12zm96 0v-40c0-6.627-5.373-12-12-12h-40c-6.627 0-12 5.373-12 12v40c0 6.627 5.373 12 12 12h40c6.627 0 12-5.373 12-12zm96 0v-40c0-6.627-5.373-12-12-12h-40c-6.627 0-12 5.373-12 12v40c0 6.627 5.373 12 12 12h40c6.627 0 12-5.373 12-12zm96 0v-40c0-6.627-5.373-12-12-12h-40c-6.627 0-12 5.373-12 12v40c0 6.627 5.373 12 12 12h40c6.627 0 12-5.373 12-12zm-336 96v-40c0-6.627-5.373-12-12-12H76c-6.627 0-12 5.373-12 12v40c0 6.627 5.373 12 12 12h40c6.627 0 12-5.373 12-12zm288 0v-40c0-6.627-5.373-12-12-12H172c-6.627 0-12 5.373-12 12v40c0 6.627 5.373 12 12 12h232c6.627 0 12-5.373 12-12zm96 0v-40c0-6.627-5.373-12-12-12h-40c-6.627 0-12 5.373-12 12v40c0 6.627 5.373 12 12 12h40c6.627 0 12-5.373 12-12z" />
|
|
||||||
{isEnabled && <path d="M13 432L590 48" stroke="var(--red-500)" stroke-width="72" stroke-linecap="round" />}
|
|
||||||
</svg>
|
|
||||||
</div>
|
|
||||||
</Button>
|
|
||||||
</div>
|
|
||||||
)}
|
|
||||||
</Tooltip>
|
|
||||||
);
|
);
|
||||||
}
|
};
|
||||||
|
|
||||||
export default definePlugin({
|
export default definePlugin({
|
||||||
name: "SilentTyping",
|
name: "SilentTyping",
|
||||||
authors: [Devs.Ven, Devs.Rini],
|
authors: [Devs.Ven, Devs.Rini],
|
||||||
description: "Hide that you are typing",
|
description: "Hide that you are typing",
|
||||||
|
dependencies: ["CommandsAPI", "ChatInputButtonAPI"],
|
||||||
|
settings,
|
||||||
|
|
||||||
patches: [
|
patches: [
|
||||||
{
|
{
|
||||||
find: '.dispatch({type:"TYPING_START_LOCAL"',
|
find: '.dispatch({type:"TYPING_START_LOCAL"',
|
||||||
|
@ -84,17 +71,8 @@ export default definePlugin({
|
||||||
replace: "startTyping:$self.startTyping,stop"
|
replace: "startTyping:$self.startTyping,stop"
|
||||||
}
|
}
|
||||||
},
|
},
|
||||||
{
|
|
||||||
find: "ChannelTextAreaButtons",
|
|
||||||
predicate: () => settings.store.showIcon,
|
|
||||||
replacement: {
|
|
||||||
match: /(\i)\.push.{1,30}disabled:(\i),.{1,20}\},"gift"\)\)/,
|
|
||||||
replace: "$&,(()=>{try{$2||$1.push($self.chatBarIcon(arguments[0]))}catch{}})()",
|
|
||||||
}
|
|
||||||
},
|
|
||||||
],
|
],
|
||||||
dependencies: ["CommandsAPI"],
|
|
||||||
settings,
|
|
||||||
commands: [{
|
commands: [{
|
||||||
name: "silenttype",
|
name: "silenttype",
|
||||||
description: "Toggle whether you're hiding that you're typing or not.",
|
description: "Toggle whether you're hiding that you're typing or not.",
|
||||||
|
@ -120,5 +98,6 @@ export default definePlugin({
|
||||||
FluxDispatcher.dispatch({ type: "TYPING_START_LOCAL", channelId });
|
FluxDispatcher.dispatch({ type: "TYPING_START_LOCAL", channelId });
|
||||||
},
|
},
|
||||||
|
|
||||||
chatBarIcon: ErrorBoundary.wrap(SilentTypingToggle, { noop: true }),
|
start: () => addChatBarButton("SilentTyping", SilentTypingToggle),
|
||||||
|
stop: () => removeChatBarButton("SilentTyping"),
|
||||||
});
|
});
|
||||||
|
|
|
@ -213,7 +213,7 @@ function applyRules(content: string): string {
|
||||||
|
|
||||||
if (stringRules) {
|
if (stringRules) {
|
||||||
for (const rule of stringRules) {
|
for (const rule of stringRules) {
|
||||||
if (!rule.find || !rule.replace) continue;
|
if (!rule.find) continue;
|
||||||
if (rule.onlyIfIncludes && !content.includes(rule.onlyIfIncludes)) continue;
|
if (rule.onlyIfIncludes && !content.includes(rule.onlyIfIncludes)) continue;
|
||||||
|
|
||||||
content = ` ${content} `.replaceAll(rule.find, rule.replace.replaceAll("\\n", "\n")).replace(/^\s|\s$/g, "");
|
content = ` ${content} `.replaceAll(rule.find, rule.replace.replaceAll("\\n", "\n")).replace(/^\s|\s$/g, "");
|
||||||
|
@ -222,7 +222,7 @@ function applyRules(content: string): string {
|
||||||
|
|
||||||
if (regexRules) {
|
if (regexRules) {
|
||||||
for (const rule of regexRules) {
|
for (const rule of regexRules) {
|
||||||
if (!rule.find || !rule.replace) continue;
|
if (!rule.find) continue;
|
||||||
if (rule.onlyIfIncludes && !content.includes(rule.onlyIfIncludes)) continue;
|
if (rule.onlyIfIncludes && !content.includes(rule.onlyIfIncludes)) continue;
|
||||||
|
|
||||||
try {
|
try {
|
||||||
|
|
|
@ -16,9 +16,11 @@
|
||||||
* along with this program. If not, see <https://www.gnu.org/licenses/>.
|
* along with this program. If not, see <https://www.gnu.org/licenses/>.
|
||||||
*/
|
*/
|
||||||
|
|
||||||
|
import { ChatBarButton } from "@api/ChatButtons";
|
||||||
|
import { Margins } from "@utils/margins";
|
||||||
import { classes } from "@utils/misc";
|
import { classes } from "@utils/misc";
|
||||||
import { openModal } from "@utils/modal";
|
import { openModal } from "@utils/modal";
|
||||||
import { Button, ButtonLooks, ButtonWrapperClasses, Tooltip } from "@webpack/common";
|
import { Alerts, Forms } from "@webpack/common";
|
||||||
|
|
||||||
import { settings } from "./settings";
|
import { settings } from "./settings";
|
||||||
import { TranslateModal } from "./TranslateModal";
|
import { TranslateModal } from "./TranslateModal";
|
||||||
|
@ -37,42 +39,49 @@ export function TranslateIcon({ height = 24, width = 24, className }: { height?:
|
||||||
);
|
);
|
||||||
}
|
}
|
||||||
|
|
||||||
export function TranslateChatBarIcon({ slateProps }: { slateProps: { type: { analyticsName: string; }; }; }) {
|
export const TranslateChatBarIcon: ChatBarButton = ({ isMainChat }) => {
|
||||||
const { autoTranslate } = settings.use(["autoTranslate"]);
|
const { autoTranslate } = settings.use(["autoTranslate"]);
|
||||||
|
|
||||||
if (slateProps.type.analyticsName !== "normal")
|
if (!isMainChat) return null;
|
||||||
return null;
|
|
||||||
|
|
||||||
const toggle = () => settings.store.autoTranslate = !autoTranslate;
|
const toggle = () => {
|
||||||
|
const newState = !autoTranslate;
|
||||||
|
settings.store.autoTranslate = newState;
|
||||||
|
if (newState && settings.store.showAutoTranslateAlert !== false)
|
||||||
|
Alerts.show({
|
||||||
|
title: "Vencord Auto-Translate Enabled",
|
||||||
|
body: <>
|
||||||
|
<Forms.FormText>
|
||||||
|
You just enabled auto translate (by right clicking the Translate icon). Any message you send will automatically be translated before being sent.
|
||||||
|
</Forms.FormText>
|
||||||
|
<Forms.FormText className={Margins.top16}>
|
||||||
|
If this was an accident, disable it again, or it will change your message content before sending.
|
||||||
|
</Forms.FormText>
|
||||||
|
</>,
|
||||||
|
cancelText: "Disable Auto-Translate",
|
||||||
|
confirmText: "Got it",
|
||||||
|
secondaryConfirmText: "Don't show again",
|
||||||
|
onConfirmSecondary: () => settings.store.showAutoTranslateAlert = false,
|
||||||
|
onCancel: () => settings.store.autoTranslate = false
|
||||||
|
});
|
||||||
|
};
|
||||||
|
|
||||||
return (
|
return (
|
||||||
<Tooltip text="Open Translate Modal">
|
<ChatBarButton
|
||||||
{({ onMouseEnter, onMouseLeave }) => (
|
tooltip="Open Translate Modal"
|
||||||
<div style={{ display: "flex" }}>
|
onClick={e => {
|
||||||
<Button
|
if (e.shiftKey) return toggle();
|
||||||
aria-haspopup="dialog"
|
|
||||||
aria-label="Open Translate Modal"
|
|
||||||
size=""
|
|
||||||
look={ButtonLooks.BLANK}
|
|
||||||
onMouseEnter={onMouseEnter}
|
|
||||||
onMouseLeave={onMouseLeave}
|
|
||||||
innerClassName={ButtonWrapperClasses.button}
|
|
||||||
onClick={e => {
|
|
||||||
if (e.shiftKey) return toggle();
|
|
||||||
|
|
||||||
openModal(props => (
|
openModal(props => (
|
||||||
<TranslateModal rootProps={props} />
|
<TranslateModal rootProps={props} />
|
||||||
));
|
));
|
||||||
}}
|
}}
|
||||||
onContextMenu={() => toggle()}
|
onContextMenu={() => toggle()}
|
||||||
style={{ padding: "0 4px" }}
|
buttonProps={{
|
||||||
>
|
"aria-haspopup": "dialog"
|
||||||
<div className={ButtonWrapperClasses.buttonWrapper}>
|
}}
|
||||||
<TranslateIcon className={cl({ "auto-translate": autoTranslate })} />
|
>
|
||||||
</div>
|
<TranslateIcon className={cl({ "auto-translate": autoTranslate, "chat-button": true })} />
|
||||||
</Button>
|
</ChatBarButton>
|
||||||
</div>
|
|
||||||
)}
|
|
||||||
</Tooltip>
|
|
||||||
);
|
);
|
||||||
}
|
};
|
||||||
|
|
|
@ -18,11 +18,11 @@
|
||||||
|
|
||||||
import "./styles.css";
|
import "./styles.css";
|
||||||
|
|
||||||
|
import { addChatBarButton, removeChatBarButton } from "@api/ChatButtons";
|
||||||
import { addContextMenuPatch, findGroupChildrenByChildId, NavContextMenuPatchCallback, removeContextMenuPatch } from "@api/ContextMenu";
|
import { addContextMenuPatch, findGroupChildrenByChildId, NavContextMenuPatchCallback, removeContextMenuPatch } from "@api/ContextMenu";
|
||||||
import { addAccessory, removeAccessory } from "@api/MessageAccessories";
|
import { addAccessory, removeAccessory } from "@api/MessageAccessories";
|
||||||
import { addPreSendListener, removePreSendListener } from "@api/MessageEvents";
|
import { addPreSendListener, removePreSendListener } from "@api/MessageEvents";
|
||||||
import { addButton, removeButton } from "@api/MessagePopover";
|
import { addButton, removeButton } from "@api/MessagePopover";
|
||||||
import ErrorBoundary from "@components/ErrorBoundary";
|
|
||||||
import { Devs } from "@utils/constants";
|
import { Devs } from "@utils/constants";
|
||||||
import definePlugin from "@utils/types";
|
import definePlugin from "@utils/types";
|
||||||
import { ChannelStore, Menu } from "@webpack/common";
|
import { ChannelStore, Menu } from "@webpack/common";
|
||||||
|
@ -55,25 +55,16 @@ export default definePlugin({
|
||||||
name: "Translate",
|
name: "Translate",
|
||||||
description: "Translate messages with Google Translate",
|
description: "Translate messages with Google Translate",
|
||||||
authors: [Devs.Ven],
|
authors: [Devs.Ven],
|
||||||
dependencies: ["MessageAccessoriesAPI", "MessagePopoverAPI", "MessageEventsAPI"],
|
dependencies: ["MessageAccessoriesAPI", "MessagePopoverAPI", "MessageEventsAPI", "ChatInputButtonAPI"],
|
||||||
settings,
|
settings,
|
||||||
// not used, just here in case some other plugin wants it or w/e
|
// not used, just here in case some other plugin wants it or w/e
|
||||||
translate,
|
translate,
|
||||||
|
|
||||||
patches: [
|
|
||||||
{
|
|
||||||
find: "ChannelTextAreaButtons",
|
|
||||||
replacement: {
|
|
||||||
match: /(\i)\.push.{1,30}disabled:(\i),.{1,20}\},"gift"\)\)/,
|
|
||||||
replace: "$&,(()=>{try{$2||$1.push($self.chatBarIcon(arguments[0]))}catch{}})()",
|
|
||||||
}
|
|
||||||
},
|
|
||||||
],
|
|
||||||
|
|
||||||
start() {
|
start() {
|
||||||
addAccessory("vc-translation", props => <TranslationAccessory message={props.message} />);
|
addAccessory("vc-translation", props => <TranslationAccessory message={props.message} />);
|
||||||
|
|
||||||
addContextMenuPatch("message", messageCtxPatch);
|
addContextMenuPatch("message", messageCtxPatch);
|
||||||
|
addChatBarButton("vc-translate", TranslateChatBarIcon);
|
||||||
|
|
||||||
addButton("vc-translate", message => {
|
addButton("vc-translate", message => {
|
||||||
if (!message.content) return null;
|
if (!message.content) return null;
|
||||||
|
@ -101,13 +92,8 @@ export default definePlugin({
|
||||||
stop() {
|
stop() {
|
||||||
removePreSendListener(this.preSend);
|
removePreSendListener(this.preSend);
|
||||||
removeContextMenuPatch("message", messageCtxPatch);
|
removeContextMenuPatch("message", messageCtxPatch);
|
||||||
|
removeChatBarButton("vc-translate");
|
||||||
removeButton("vc-translate");
|
removeButton("vc-translate");
|
||||||
removeAccessory("vc-translation");
|
removeAccessory("vc-translation");
|
||||||
},
|
},
|
||||||
|
|
||||||
chatBarIcon: (slateProps: any) => (
|
|
||||||
<ErrorBoundary noop>
|
|
||||||
<TranslateChatBarIcon slateProps={slateProps} />
|
|
||||||
</ErrorBoundary>
|
|
||||||
)
|
|
||||||
});
|
});
|
||||||
|
|
|
@ -49,4 +49,6 @@ export const settings = definePluginSettings({
|
||||||
description: "Automatically translate your messages before sending. You can also shift/right click the translate button to toggle this",
|
description: "Automatically translate your messages before sending. You can also shift/right click the translate button to toggle this",
|
||||||
default: false
|
default: false
|
||||||
}
|
}
|
||||||
});
|
}).withPrivateSettings<{
|
||||||
|
showAutoTranslateAlert: boolean;
|
||||||
|
}>();
|
||||||
|
|
|
@ -35,3 +35,7 @@
|
||||||
.vc-trans-auto-translate {
|
.vc-trans-auto-translate {
|
||||||
color: var(--green-360);
|
color: var(--green-360);
|
||||||
}
|
}
|
||||||
|
|
||||||
|
.vc-trans-chat-button {
|
||||||
|
scale: 1.085;
|
||||||
|
}
|
||||||
|
|
13
src/plugins/urbanDictionary/README.md
Normal file
13
src/plugins/urbanDictionary/README.md
Normal file
|
@ -0,0 +1,13 @@
|
||||||
|
# Urban Dictionary
|
||||||
|
|
||||||
|
Use /urban slash command to search for a definition for a word on [Urban Dictionary](https://www.urbandictionary.com/).
|
||||||
|
|
||||||
|
## Preview
|
||||||
|
|
||||||
|
![preview](https://i.imgur.com/1zwzj38.png)
|
||||||
|
|
||||||
|
## Usage
|
||||||
|
|
||||||
|
- Enable this plugin
|
||||||
|
- Set plugin settings as desired
|
||||||
|
- Type /urban and start getting definitions right into your Discord client.
|
|
@ -18,14 +18,24 @@
|
||||||
|
|
||||||
import { ApplicationCommandOptionType, sendBotMessage } from "@api/Commands";
|
import { ApplicationCommandOptionType, sendBotMessage } from "@api/Commands";
|
||||||
import { ApplicationCommandInputType } from "@api/Commands/types";
|
import { ApplicationCommandInputType } from "@api/Commands/types";
|
||||||
|
import { definePluginSettings } from "@api/Settings";
|
||||||
import { Devs } from "@utils/constants";
|
import { Devs } from "@utils/constants";
|
||||||
import definePlugin from "@utils/types";
|
import definePlugin, { OptionType } from "@utils/types";
|
||||||
|
|
||||||
|
const settings = definePluginSettings({
|
||||||
|
resultsAmount: {
|
||||||
|
type: OptionType.NUMBER,
|
||||||
|
description: "The amount of results you want to get (more gives better results, but is slower)",
|
||||||
|
default: 10
|
||||||
|
}
|
||||||
|
});
|
||||||
|
|
||||||
export default definePlugin({
|
export default definePlugin({
|
||||||
name: "UrbanDictionary",
|
name: "UrbanDictionary",
|
||||||
description: "Search for a word on Urban Dictionary via /urban slash command",
|
description: "Search for a word on Urban Dictionary via /urban slash command",
|
||||||
authors: [Devs.jewdev],
|
authors: [Devs.jewdev],
|
||||||
dependencies: ["CommandsAPI"],
|
dependencies: ["CommandsAPI"],
|
||||||
|
settings,
|
||||||
commands: [
|
commands: [
|
||||||
{
|
{
|
||||||
name: "urban",
|
name: "urban",
|
||||||
|
@ -41,12 +51,16 @@ export default definePlugin({
|
||||||
],
|
],
|
||||||
execute: async (args, ctx) => {
|
execute: async (args, ctx) => {
|
||||||
try {
|
try {
|
||||||
const query = encodeURIComponent(args[0].value);
|
const query: string = encodeURIComponent(args[0].value);
|
||||||
const { list: [definition] } = await (await fetch(`https://api.urbandictionary.com/v0/define?term=${query}`)).json();
|
const { list } = await fetch(`https://api.urbandictionary.com/v0/define?term=${query}&per_page=${settings.store.resultsAmount}`).then(response => response.json());
|
||||||
|
|
||||||
if (!definition)
|
if (!list.length)
|
||||||
return void sendBotMessage(ctx.channel.id, { content: "No results found." });
|
return void sendBotMessage(ctx.channel.id, { content: "No results found." });
|
||||||
|
|
||||||
|
const definition = list.reduce((prev, curr) => {
|
||||||
|
return prev.thumbs_up > curr.thumbs_up ? prev : curr;
|
||||||
|
});
|
||||||
|
|
||||||
const linkify = (text: string) => text
|
const linkify = (text: string) => text
|
||||||
.replaceAll("\r\n", "\n")
|
.replaceAll("\r\n", "\n")
|
||||||
.replace(/([*>_`~\\])/gsi, "\\$1")
|
.replace(/([*>_`~\\])/gsi, "\\$1")
|
||||||
|
|
|
@ -33,7 +33,7 @@ function VencordPopout(onClose: () => void) {
|
||||||
const pluginEntries = [] as ReactNode[];
|
const pluginEntries = [] as ReactNode[];
|
||||||
|
|
||||||
for (const plugin of Object.values(Vencord.Plugins.plugins)) {
|
for (const plugin of Object.values(Vencord.Plugins.plugins)) {
|
||||||
if (plugin.toolboxActions) {
|
if (plugin.toolboxActions && Vencord.Plugins.isPluginEnabled(plugin.name)) {
|
||||||
pluginEntries.push(
|
pluginEntries.push(
|
||||||
<Menu.MenuGroup
|
<Menu.MenuGroup
|
||||||
label={plugin.name}
|
label={plugin.name}
|
||||||
|
|
|
@ -27,7 +27,7 @@ import { Margins } from "@utils/margins";
|
||||||
import { copyWithToast } from "@utils/misc";
|
import { copyWithToast } from "@utils/misc";
|
||||||
import { closeModal, ModalCloseButton, ModalContent, ModalFooter, ModalHeader, ModalRoot, ModalSize, openModal } from "@utils/modal";
|
import { closeModal, ModalCloseButton, ModalContent, ModalFooter, ModalHeader, ModalRoot, ModalSize, openModal } from "@utils/modal";
|
||||||
import definePlugin, { OptionType } from "@utils/types";
|
import definePlugin, { OptionType } from "@utils/types";
|
||||||
import { Button, ChannelStore, Forms, Menu, Text } from "@webpack/common";
|
import { Button, ChannelStore, Forms, i18n, Menu, Text } from "@webpack/common";
|
||||||
import { Message } from "discord-types/general";
|
import { Message } from "discord-types/general";
|
||||||
|
|
||||||
|
|
||||||
|
@ -117,22 +117,26 @@ const settings = definePluginSettings({
|
||||||
}
|
}
|
||||||
});
|
});
|
||||||
|
|
||||||
function MakeContextCallback(name: string) {
|
function MakeContextCallback(name: "Guild" | "User" | "Channel") {
|
||||||
const callback: NavContextMenuPatchCallback = (children, props) => () => {
|
const callback: NavContextMenuPatchCallback = (children, props) => () => {
|
||||||
if ((name === "Guild" && !props.guild) || (name === "User" && !props.user)) return;
|
const value = props[name.toLowerCase()];
|
||||||
|
if (!value) return;
|
||||||
|
if (props.label === i18n.Messages.CHANNEL_ACTIONS_MENU_LABEL) return; // random shit like notification settings
|
||||||
|
|
||||||
const lastChild = children.at(-1);
|
const lastChild = children.at(-1);
|
||||||
if (lastChild?.key === "developer-actions") {
|
if (lastChild?.key === "developer-actions") {
|
||||||
const p = lastChild.props;
|
const p = lastChild.props;
|
||||||
if (!Array.isArray(p.children))
|
if (!Array.isArray(p.children))
|
||||||
p.children = [p.children];
|
p.children = [p.children];
|
||||||
({ children } = p);
|
|
||||||
|
children = p.children;
|
||||||
}
|
}
|
||||||
|
|
||||||
children.splice(-1, 0,
|
children.splice(-1, 0,
|
||||||
<Menu.MenuItem
|
<Menu.MenuItem
|
||||||
id={`vc-view-${name.toLowerCase()}-raw`}
|
id={`vc-view-${name.toLowerCase()}-raw`}
|
||||||
label="View Raw"
|
label="View Raw"
|
||||||
action={() => openViewRawModal(JSON.stringify(props[name.toLowerCase()], null, 4), name)}
|
action={() => openViewRawModal(JSON.stringify(value, null, 4), name)}
|
||||||
icon={CopyIcon}
|
icon={CopyIcon}
|
||||||
/>
|
/>
|
||||||
);
|
);
|
||||||
|
|
|
@ -46,6 +46,23 @@ const settings = definePluginSettings({
|
||||||
}
|
}
|
||||||
});
|
});
|
||||||
|
|
||||||
|
const MEDIA_PROXY_URL = "https://media.discordapp.net";
|
||||||
|
const CDN_URL = "https://cdn.discordapp.com";
|
||||||
|
|
||||||
|
function fixImageUrl(urlString: string, explodeWebp: boolean) {
|
||||||
|
const url = new URL(urlString);
|
||||||
|
if (url.origin === CDN_URL) return urlString;
|
||||||
|
if (url.origin === MEDIA_PROXY_URL) return CDN_URL + url.pathname;
|
||||||
|
|
||||||
|
url.searchParams.delete("width");
|
||||||
|
url.searchParams.delete("height");
|
||||||
|
url.searchParams.set("quality", "lossless");
|
||||||
|
if (explodeWebp && url.searchParams.get("format") === "webp")
|
||||||
|
url.searchParams.set("format", "png");
|
||||||
|
|
||||||
|
return url.toString();
|
||||||
|
}
|
||||||
|
|
||||||
export default definePlugin({
|
export default definePlugin({
|
||||||
name: "WebContextMenus",
|
name: "WebContextMenus",
|
||||||
description: "Re-adds context menus missing in the web version of Discord: Links & Images (Copy/Open Link/Image), Text Area (Copy, Cut, Paste, SpellCheck)",
|
description: "Re-adds context menus missing in the web version of Discord: Links & Images (Copy/Open Link/Image), Text Area (Copy, Cut, Paste, SpellCheck)",
|
||||||
|
@ -169,32 +186,53 @@ export default definePlugin({
|
||||||
match: /let\{text:\i=""/,
|
match: /let\{text:\i=""/,
|
||||||
replace: "return [null,null];$&"
|
replace: "return [null,null];$&"
|
||||||
}
|
}
|
||||||
|
},
|
||||||
|
|
||||||
|
// Add back "Show My Camera" context menu
|
||||||
|
{
|
||||||
|
find: '.default("MediaEngineWebRTC");',
|
||||||
|
replacement: {
|
||||||
|
match: /supports\(\i\)\{switch\(\i\)\{case (\i).Features/,
|
||||||
|
replace: "$&.DISABLE_VIDEO:return true;case $1.Features"
|
||||||
|
}
|
||||||
}
|
}
|
||||||
],
|
],
|
||||||
|
|
||||||
async copyImage(url: string) {
|
async copyImage(url: string) {
|
||||||
// Clipboard only supports image/png, jpeg and similar won't work. Thus, we need to convert it to png
|
url = fixImageUrl(url, true);
|
||||||
// via canvas first
|
|
||||||
const img = new Image();
|
|
||||||
img.onload = () => {
|
|
||||||
const canvas = document.createElement("canvas");
|
|
||||||
canvas.width = img.naturalWidth;
|
|
||||||
canvas.height = img.naturalHeight;
|
|
||||||
canvas.getContext("2d")!.drawImage(img, 0, 0);
|
|
||||||
|
|
||||||
canvas.toBlob(data => {
|
let imageData = await fetch(url).then(r => r.blob());
|
||||||
navigator.clipboard.write([
|
if (imageData.type !== "image/png") {
|
||||||
new ClipboardItem({
|
const bitmap = await createImageBitmap(imageData);
|
||||||
"image/png": data!
|
|
||||||
})
|
const canvas = document.createElement("canvas");
|
||||||
]);
|
canvas.width = bitmap.width;
|
||||||
}, "image/png");
|
canvas.height = bitmap.height;
|
||||||
};
|
canvas.getContext("2d")!.drawImage(bitmap, 0, 0);
|
||||||
img.crossOrigin = "anonymous";
|
|
||||||
img.src = url;
|
await new Promise<void>(done => {
|
||||||
|
canvas.toBlob(data => {
|
||||||
|
imageData = data!;
|
||||||
|
done();
|
||||||
|
}, "image/png");
|
||||||
|
});
|
||||||
|
}
|
||||||
|
|
||||||
|
if (IS_VESKTOP && VesktopNative.clipboard) {
|
||||||
|
VesktopNative.clipboard.copyImage(await imageData.arrayBuffer(), url);
|
||||||
|
return;
|
||||||
|
} else {
|
||||||
|
navigator.clipboard.write([
|
||||||
|
new ClipboardItem({
|
||||||
|
"image/png": imageData
|
||||||
|
})
|
||||||
|
]);
|
||||||
|
}
|
||||||
},
|
},
|
||||||
|
|
||||||
async saveImage(url: string) {
|
async saveImage(url: string) {
|
||||||
|
url = fixImageUrl(url, false);
|
||||||
|
|
||||||
const data = await fetchImage(url);
|
const data = await fetchImage(url);
|
||||||
if (!data) return;
|
if (!data) return;
|
||||||
|
|
||||||
|
|
|
@ -42,6 +42,10 @@ export interface Dev {
|
||||||
* If you are fine with attribution but don't want the badge, add badge: false
|
* If you are fine with attribution but don't want the badge, add badge: false
|
||||||
*/
|
*/
|
||||||
export const Devs = /* #__PURE__*/ Object.freeze({
|
export const Devs = /* #__PURE__*/ Object.freeze({
|
||||||
|
Nobody: {
|
||||||
|
name: "Nobody",
|
||||||
|
id: 0n,
|
||||||
|
},
|
||||||
Ven: {
|
Ven: {
|
||||||
name: "Vendicated",
|
name: "Vendicated",
|
||||||
id: 343383572805058560n
|
id: 343383572805058560n
|
||||||
|
@ -359,10 +363,6 @@ export const Devs = /* #__PURE__*/ Object.freeze({
|
||||||
name: "bb010g",
|
name: "bb010g",
|
||||||
id: 72791153467990016n,
|
id: 72791153467990016n,
|
||||||
},
|
},
|
||||||
Lumap: {
|
|
||||||
name: "lumap",
|
|
||||||
id: 635383782576357407n
|
|
||||||
},
|
|
||||||
Dolfies: {
|
Dolfies: {
|
||||||
name: "Dolfies",
|
name: "Dolfies",
|
||||||
id: 852892297661906993n,
|
id: 852892297661906993n,
|
||||||
|
@ -403,6 +403,14 @@ export const Devs = /* #__PURE__*/ Object.freeze({
|
||||||
name: "Grzesiek11",
|
name: "Grzesiek11",
|
||||||
id: 368475654662127616n,
|
id: 368475654662127616n,
|
||||||
},
|
},
|
||||||
|
Samwich: {
|
||||||
|
name: "Samwich",
|
||||||
|
id: 976176454511509554n,
|
||||||
|
},
|
||||||
|
coolelectronics: {
|
||||||
|
name: "coolelectronics",
|
||||||
|
id: 696392247205298207n,
|
||||||
|
}
|
||||||
} satisfies Record<string, Dev>);
|
} satisfies Record<string, Dev>);
|
||||||
|
|
||||||
// iife so #__PURE__ works correctly
|
// iife so #__PURE__ works correctly
|
||||||
|
|
Loading…
Reference in a new issue