aboutsummaryrefslogtreecommitdiff
path: root/src/plugins/fakeProfileThemes
diff options
context:
space:
mode:
authorV <vendicated@riseup.net>2023-09-24 16:02:18 +0200
committerV <vendicated@riseup.net>2023-09-24 16:02:18 +0200
commit30ac25607023752031aa98060cbf8a736109992d (patch)
tree79bb82b6634ef601db6c98e751275607ec54dbea /src/plugins/fakeProfileThemes
parentd0e2a324717e600736a18b88fe89a21c640a406b (diff)
downloadVencord-30ac25607023752031aa98060cbf8a736109992d.tar.gz
Vencord-30ac25607023752031aa98060cbf8a736109992d.tar.bz2
Vencord-30ac25607023752031aa98060cbf8a736109992d.zip
migrate all plugins to folders
Diffstat (limited to 'src/plugins/fakeProfileThemes')
-rw-r--r--src/plugins/fakeProfileThemes/index.tsx145
1 files changed, 145 insertions, 0 deletions
diff --git a/src/plugins/fakeProfileThemes/index.tsx b/src/plugins/fakeProfileThemes/index.tsx
new file mode 100644
index 0000000..70003e5
--- /dev/null
+++ b/src/plugins/fakeProfileThemes/index.tsx
@@ -0,0 +1,145 @@
+/*
+ * 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/>.
+*/
+
+// This plugin is a port from Alyxia's Vendetta plugin
+import { definePluginSettings } from "@api/Settings";
+import ErrorBoundary from "@components/ErrorBoundary";
+import { Devs } from "@utils/constants";
+import { Margins } from "@utils/margins";
+import { copyWithToast } from "@utils/misc";
+import definePlugin, { OptionType } from "@utils/types";
+import { Button, Forms } from "@webpack/common";
+import { User } from "discord-types/general";
+import virtualMerge from "virtual-merge";
+
+interface UserProfile extends User {
+ themeColors?: Array<number>;
+}
+
+interface Colors {
+ primary: number;
+ accent: number;
+}
+
+function encode(primary: number, accent: number): string {
+ const message = `[#${primary.toString(16).padStart(6, "0")},#${accent.toString(16).padStart(6, "0")}]`;
+ const padding = "";
+ const encoded = Array.from(message)
+ .map(x => x.codePointAt(0))
+ .filter(x => x! >= 0x20 && x! <= 0x7f)
+ .map(x => String.fromCodePoint(x! + 0xe0000))
+ .join("");
+
+ return (padding || "") + " " + encoded;
+}
+
+// Courtesy of Cynthia.
+function decode(bio: string): Array<number> | null {
+ if (bio == null) return null;
+
+ const colorString = bio.match(
+ /\u{e005b}\u{e0023}([\u{e0061}-\u{e0066}\u{e0041}-\u{e0046}\u{e0030}-\u{e0039}]+?)\u{e002c}\u{e0023}([\u{e0061}-\u{e0066}\u{e0041}-\u{e0046}\u{e0030}-\u{e0039}]+?)\u{e005d}/u,
+ );
+ if (colorString != null) {
+ const parsed = [...colorString[0]]
+ .map(x => String.fromCodePoint(x.codePointAt(0)! - 0xe0000))
+ .join("");
+ const colors = parsed
+ .substring(1, parsed.length - 1)
+ .split(",")
+ .map(x => parseInt(x.replace("#", "0x"), 16));
+
+ return colors;
+ } else {
+ return null;
+ }
+}
+
+const settings = definePluginSettings({
+ nitroFirst: {
+ description: "Default color source if both are present",
+ type: OptionType.SELECT,
+ options: [
+ { label: "Nitro colors", value: true, default: true },
+ { label: "Fake colors", value: false },
+ ]
+ }
+});
+
+export default definePlugin({
+ name: "FakeProfileThemes",
+ description: "Allows profile theming by hiding the colors in your bio thanks to invisible 3y3 encoding",
+ authors: [Devs.Alyxia, Devs.Remty],
+ patches: [
+ {
+ find: "getUserProfile=",
+ replacement: {
+ match: /(?<=getUserProfile=function\(\i\){return )(\i\[\i\])/,
+ replace: "$self.colorDecodeHook($1)"
+ }
+ }, {
+ find: ".USER_SETTINGS_PROFILE_THEME_ACCENT",
+ replacement: {
+ match: /RESET_PROFILE_THEME}\)(?<=},color:(\i).+?},color:(\i).+?)/,
+ replace: "$&,$self.addCopy3y3Button({primary:$1,accent:$2})"
+ }
+ }
+ ],
+ settingsAboutComponent: () => (
+ <Forms.FormSection>
+ <Forms.FormTitle tag="h3">Usage</Forms.FormTitle>
+ <Forms.FormText>
+ After enabling this plugin, you will see custom colors in the profiles of other people using compatible plugins. <br />
+ To set your own colors:
+ <ul>
+ <li>• go to your profile settings</li>
+ <li>• choose your own colors in the Nitro preview</li>
+ <li>• click the "Copy 3y3" button</li>
+ <li>• paste the invisible text anywhere in your bio</li>
+ </ul><br />
+ <b>Please note:</b> if you are using a theme which hides nitro ads, you should disable it temporarily to set colors.
+ </Forms.FormText>
+ </Forms.FormSection>),
+ settings,
+ colorDecodeHook(user: UserProfile) {
+ if (user) {
+ // don't replace colors if already set with nitro
+ if (settings.store.nitroFirst && user.themeColors) return user;
+ const colors = decode(user.bio);
+ if (colors) {
+ return virtualMerge(user, {
+ premiumType: 2,
+ themeColors: colors
+ });
+ }
+ }
+ return user;
+ },
+ addCopy3y3Button: ErrorBoundary.wrap(function ({ primary, accent }: Colors) {
+ return <Button
+ onClick={() => {
+ const colorString = encode(primary, accent);
+ copyWithToast(colorString);
+ }}
+ color={Button.Colors.PRIMARY}
+ size={Button.Sizes.XLARGE}
+ className={Margins.left16}
+ >Copy 3y3
+ </Button >;
+ }, { noop: true }),
+});