2024-09-27 12:23:12 -04:00
|
|
|
/*
|
|
|
|
* Vencord, a Discord client mod
|
|
|
|
* Copyright (c) 2023 Vendicated and contributors
|
|
|
|
* SPDX-License-Identifier: GPL-3.0-or-later
|
|
|
|
*/
|
|
|
|
|
|
|
|
import * as DataStore from "@api/DataStore";
|
|
|
|
import { definePluginSettings } from "@api/Settings";
|
|
|
|
import { disableStyle, enableStyle } from "@api/Styles";
|
|
|
|
import { Devs } from "@utils/constants";
|
|
|
|
import definePlugin, { OptionType, Plugin } from "@utils/types";
|
|
|
|
import { findStoreLazy } from "@webpack";
|
|
|
|
import { User } from "discord-types/general";
|
|
|
|
|
|
|
|
import style from "./index.css?managed";
|
|
|
|
|
|
|
|
interface iUSRBG extends Plugin {
|
2024-09-27 13:36:22 -04:00
|
|
|
userHasBackground(userId: string);
|
|
|
|
getImageUrl(userId: string): string | null;
|
2024-09-27 12:23:12 -04:00
|
|
|
}
|
|
|
|
|
|
|
|
const settings = definePluginSettings({
|
|
|
|
animate: {
|
2024-10-12 07:59:33 -04:00
|
|
|
description: "Animate banners",
|
2024-09-27 12:23:12 -04:00
|
|
|
type: OptionType.BOOLEAN,
|
|
|
|
default: false
|
|
|
|
},
|
|
|
|
});
|
|
|
|
|
|
|
|
const DATASTORE_KEY = "bannersEverywhere";
|
|
|
|
|
|
|
|
const UserProfileStore = findStoreLazy("UserProfileStore");
|
|
|
|
|
|
|
|
|
|
|
|
export default definePlugin({
|
|
|
|
name: "BannersEverywhere",
|
|
|
|
description: "Displays banners in the member list ",
|
|
|
|
authors: [Devs.ImLvna, Devs.AutumnVN],
|
|
|
|
settings,
|
|
|
|
patches: [
|
|
|
|
{
|
2024-11-03 21:25:19 -05:00
|
|
|
find: "#{intl::GUILD_OWNER}),",
|
2024-09-27 12:23:12 -04:00
|
|
|
replacement:
|
|
|
|
{
|
|
|
|
// We add the banner as a property while we can still access the user id
|
|
|
|
match: /verified:(\i).isVerifiedBot.*?name:null.*?(?=avatar:)/,
|
|
|
|
replace: "$&banner:$self.memberListBannerHook($1),",
|
|
|
|
},
|
|
|
|
},
|
|
|
|
{
|
|
|
|
find: "role:\"listitem\",innerRef",
|
|
|
|
replacement:
|
|
|
|
{
|
|
|
|
// We cant access the user id here, so we take the banner property we set earlier
|
|
|
|
match: /let{avatar:\i.*?focusProps:\i.*?=(\i).*?children:\[/,
|
|
|
|
replace: "$&$1.banner,"
|
|
|
|
}
|
|
|
|
}
|
|
|
|
],
|
|
|
|
|
|
|
|
data: {},
|
|
|
|
|
|
|
|
async start() {
|
|
|
|
enableStyle(style);
|
|
|
|
this.data = await DataStore.get(DATASTORE_KEY) || {};
|
|
|
|
},
|
|
|
|
|
|
|
|
stop() {
|
|
|
|
disableStyle(style);
|
|
|
|
DataStore.set(DATASTORE_KEY, this.data);
|
|
|
|
},
|
|
|
|
|
|
|
|
memberListBannerHook(user: User) {
|
|
|
|
let url = this.getBanner(user.id);
|
|
|
|
if (!url) return;
|
2024-10-12 07:59:33 -04:00
|
|
|
if (!settings.store.animate) {
|
|
|
|
// Discord Banners
|
|
|
|
url = url.replace(".gif", ".png");
|
|
|
|
// Usrbg Banners
|
|
|
|
this.gifToPng(url)
|
|
|
|
.then(pngUrl => {
|
|
|
|
const imgElement = document.getElementById(`vc-banners-everywhere-${user.id}`) as HTMLImageElement;
|
|
|
|
if (imgElement) {
|
|
|
|
imgElement.src = pngUrl;
|
|
|
|
}
|
|
|
|
})
|
|
|
|
.catch();
|
|
|
|
}
|
2024-09-27 12:23:12 -04:00
|
|
|
|
|
|
|
return (
|
2024-10-12 07:59:33 -04:00
|
|
|
<img id={`vc-banners-everywhere-${user.id}`} src={url} className="vc-banners-everywhere-memberlist"></img>
|
2024-09-27 12:23:12 -04:00
|
|
|
);
|
|
|
|
},
|
|
|
|
|
2024-10-12 07:59:33 -04:00
|
|
|
async checkImageExists(url: string): Promise<boolean> {
|
|
|
|
return new Promise(resolve => {
|
|
|
|
const img = new Image();
|
|
|
|
img.onload = () => resolve(true);
|
|
|
|
img.onerror = () => resolve(false);
|
|
|
|
img.src = url;
|
|
|
|
});
|
|
|
|
},
|
|
|
|
|
|
|
|
async gifToPng(url: string): Promise<string> {
|
|
|
|
const exists = await this.checkImageExists(url);
|
|
|
|
if (!exists) return "";
|
|
|
|
|
|
|
|
return new Promise((resolve, reject) => {
|
|
|
|
const img = new Image();
|
|
|
|
img.crossOrigin = "anonymous";
|
|
|
|
img.onload = () => {
|
|
|
|
const canvas = document.createElement("canvas");
|
|
|
|
canvas.width = img.width;
|
|
|
|
canvas.height = img.height;
|
|
|
|
const ctx = canvas.getContext("2d");
|
|
|
|
if (ctx) {
|
|
|
|
ctx.drawImage(img, 0, 0);
|
|
|
|
const pngDataUrl = canvas.toDataURL("image/png");
|
|
|
|
resolve(pngDataUrl);
|
|
|
|
} else {
|
|
|
|
reject(new Error("Failed to get canvas context."));
|
|
|
|
}
|
|
|
|
};
|
|
|
|
img.onerror = err => reject(err);
|
|
|
|
img.src = url;
|
|
|
|
});
|
|
|
|
},
|
|
|
|
|
2024-09-27 12:23:12 -04:00
|
|
|
getBanner(userId: string): string | undefined {
|
2024-09-27 13:36:22 -04:00
|
|
|
if (Vencord.Plugins.isPluginEnabled("USRBG") && (Vencord.Plugins.plugins.USRBG as iUSRBG).userHasBackground(userId)) {
|
|
|
|
let banner = (Vencord.Plugins.plugins.USRBG as iUSRBG).getImageUrl(userId);
|
|
|
|
if (banner === null) banner = "";
|
|
|
|
return banner;
|
2024-09-27 12:23:12 -04:00
|
|
|
}
|
|
|
|
const userProfile = UserProfileStore.getUserProfile(userId);
|
|
|
|
if (userProfile?.banner) {
|
|
|
|
this.data[userId] = `https://cdn.discordapp.com/banners/${userId}/${userProfile.banner}.${userProfile.banner.startsWith("a_") ? "gif" : "png"}`;
|
|
|
|
DataStore.set(DATASTORE_KEY, this.data);
|
|
|
|
}
|
|
|
|
return this.data[userId];
|
|
|
|
},
|
|
|
|
});
|