2022-11-11 23:50:09 +01:00
|
|
|
/*
|
|
|
|
* 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/>.
|
|
|
|
*/
|
|
|
|
|
2023-04-05 22:45:14 +02:00
|
|
|
import { BadgePosition, BadgeUserArgs, ProfileBadge } from "@api/Badges";
|
2022-11-28 13:37:55 +01:00
|
|
|
import DonateButton from "@components/DonateButton";
|
|
|
|
import ErrorBoundary from "@components/ErrorBoundary";
|
|
|
|
import { Flex } from "@components/Flex";
|
|
|
|
import { Heart } from "@components/Heart";
|
|
|
|
import { Devs } from "@utils/constants";
|
2023-03-01 21:35:08 +01:00
|
|
|
import { Margins } from "@utils/margins";
|
2023-05-23 01:55:39 +02:00
|
|
|
import { isPluginDev } from "@utils/misc";
|
2022-11-28 13:37:55 +01:00
|
|
|
import { closeModal, Modals, openModal } from "@utils/modal";
|
|
|
|
import definePlugin from "@utils/types";
|
2023-05-02 02:55:38 +02:00
|
|
|
import { Forms, Toasts } from "@webpack/common";
|
2022-11-11 23:50:09 +01:00
|
|
|
|
2023-11-30 17:28:53 +01:00
|
|
|
const CONTRIBUTOR_BADGE = "https://vencord.dev/assets/favicon.png";
|
2022-11-11 23:50:09 +01:00
|
|
|
|
|
|
|
const ContributorBadge: ProfileBadge = {
|
2023-04-13 21:04:19 +02:00
|
|
|
description: "Vencord Contributor",
|
2022-11-11 23:50:09 +01:00
|
|
|
image: CONTRIBUTOR_BADGE,
|
|
|
|
position: BadgePosition.START,
|
2023-05-23 01:55:39 +02:00
|
|
|
shouldShow: ({ user }) => isPluginDev(user.id),
|
2024-05-02 14:58:31 +02:00
|
|
|
onClick(_, { user }) {
|
|
|
|
// circular import shenanigans
|
|
|
|
const { openContributorModal } = require("@components/PluginSettings/ContributorModal") as typeof import("@components/PluginSettings/ContributorModal");
|
|
|
|
// setImmediate is needed to run on later tick to workaround limitation in proxyLazy
|
|
|
|
setImmediate(() => openContributorModal(user));
|
|
|
|
}
|
2022-11-11 23:50:09 +01:00
|
|
|
};
|
|
|
|
|
2023-11-30 17:28:53 +01:00
|
|
|
let DonorBadges = {} as Record<string, Array<Record<"tooltip" | "badge", string>>>;
|
2022-11-17 00:21:13 +01:00
|
|
|
|
2023-05-02 02:55:38 +02:00
|
|
|
async function loadBadges(noCache = false) {
|
2023-05-02 03:11:40 +02:00
|
|
|
DonorBadges = {};
|
|
|
|
|
2023-05-02 02:55:38 +02:00
|
|
|
const init = {} as RequestInit;
|
|
|
|
if (noCache)
|
|
|
|
init.cache = "no-cache";
|
|
|
|
|
2023-11-30 17:28:53 +01:00
|
|
|
DonorBadges = await fetch("https://badges.vencord.dev/badges.json", init)
|
|
|
|
.then(r => r.json());
|
2023-05-02 02:55:38 +02:00
|
|
|
}
|
|
|
|
|
2022-11-11 23:50:09 +01:00
|
|
|
export default definePlugin({
|
|
|
|
name: "BadgeAPI",
|
|
|
|
description: "API to add badges to users.",
|
2023-04-05 22:45:14 +02:00
|
|
|
authors: [Devs.Megu, Devs.Ven, Devs.TheSun],
|
2022-11-11 23:50:09 +01:00
|
|
|
required: true,
|
|
|
|
patches: [
|
|
|
|
/* Patch the badge list component on user profiles */
|
|
|
|
{
|
2024-04-24 05:27:14 +02:00
|
|
|
find: 'id:"premium",',
|
2022-12-21 20:16:32 +01:00
|
|
|
replacement: [
|
2023-04-18 02:47:48 +02:00
|
|
|
{
|
2023-07-02 01:17:02 +02:00
|
|
|
match: /&&(\i)\.push\(\{id:"premium".+?\}\);/,
|
|
|
|
replace: "$&$1.unshift(...Vencord.Api.Badges._getBadges(arguments[0]));",
|
2023-04-18 02:47:48 +02:00
|
|
|
},
|
2022-12-21 20:16:32 +01:00
|
|
|
{
|
2023-04-13 21:04:19 +02:00
|
|
|
// alt: "", aria-hidden: false, src: originalSrc
|
2023-10-25 15:24:06 +02:00
|
|
|
match: /alt:" ","aria-hidden":!0,src:(?=(\i)\.src)/,
|
2023-04-13 21:04:19 +02:00
|
|
|
// ...badge.props, ..., src: badge.image ?? ...
|
|
|
|
replace: "...$1.props,$& $1.image??"
|
2022-12-21 20:16:32 +01:00
|
|
|
},
|
2023-10-25 15:24:06 +02:00
|
|
|
// replace their component with ours if applicable
|
2022-12-21 20:16:32 +01:00
|
|
|
{
|
2024-05-03 22:18:31 +02:00
|
|
|
match: /(?<=text:(\i)\.description,spacing:12,.{0,50})children:/,
|
2023-10-25 15:24:06 +02:00
|
|
|
replace: "children:$1.component ? () => $self.renderBadgeComponent($1) :"
|
2023-04-13 21:04:19 +02:00
|
|
|
},
|
2023-10-25 15:24:06 +02:00
|
|
|
// conditionally override their onClick with badge.onClick if it exists
|
2023-04-13 21:04:19 +02:00
|
|
|
{
|
2023-10-25 15:24:06 +02:00
|
|
|
match: /href:(\i)\.link/,
|
2024-05-02 14:58:31 +02:00
|
|
|
replace: "...($1.onClick && { onClick: vcE => $1.onClick(vcE, arguments[0]) }),$&"
|
2022-12-21 20:16:32 +01:00
|
|
|
}
|
|
|
|
]
|
2022-11-11 23:50:09 +01:00
|
|
|
}
|
|
|
|
],
|
2022-11-17 00:21:13 +01:00
|
|
|
|
2023-05-02 02:55:38 +02:00
|
|
|
toolboxActions: {
|
|
|
|
async "Refetch Badges"() {
|
|
|
|
await loadBadges(true);
|
|
|
|
Toasts.show({
|
|
|
|
id: Toasts.genId(),
|
|
|
|
message: "Successfully refetched badges!",
|
|
|
|
type: Toasts.Type.SUCCESS
|
|
|
|
});
|
|
|
|
}
|
|
|
|
},
|
|
|
|
|
|
|
|
async start() {
|
|
|
|
Vencord.Api.Badges.addBadge(ContributorBadge);
|
|
|
|
await loadBadges();
|
|
|
|
},
|
|
|
|
|
2023-04-05 22:45:14 +02:00
|
|
|
renderBadgeComponent: ErrorBoundary.wrap((badge: ProfileBadge & BadgeUserArgs) => {
|
|
|
|
const Component = badge.component!;
|
|
|
|
return <Component {...badge} />;
|
|
|
|
}, { noop: true }),
|
|
|
|
|
2022-11-17 00:21:13 +01:00
|
|
|
|
2023-04-29 02:06:52 +02:00
|
|
|
getDonorBadges(userId: string) {
|
|
|
|
return DonorBadges[userId]?.map(badge => ({
|
2023-11-30 17:28:53 +01:00
|
|
|
image: badge.badge,
|
|
|
|
description: badge.tooltip,
|
2023-04-29 02:06:52 +02:00
|
|
|
position: BadgePosition.START,
|
|
|
|
props: {
|
|
|
|
style: {
|
|
|
|
borderRadius: "50%",
|
|
|
|
transform: "scale(0.9)" // The image is a bit too big compared to default badges
|
|
|
|
}
|
|
|
|
},
|
|
|
|
onClick() {
|
|
|
|
const modalKey = openModal(props => (
|
|
|
|
<ErrorBoundary noop onError={() => {
|
|
|
|
closeModal(modalKey);
|
2023-05-02 02:50:51 +02:00
|
|
|
VencordNative.native.openExternal("https://github.com/sponsors/Vendicated");
|
2023-04-29 02:06:52 +02:00
|
|
|
}}>
|
|
|
|
<Modals.ModalRoot {...props}>
|
|
|
|
<Modals.ModalHeader>
|
|
|
|
<Flex style={{ width: "100%", justifyContent: "center" }}>
|
|
|
|
<Forms.FormTitle
|
|
|
|
tag="h2"
|
|
|
|
style={{
|
|
|
|
width: "100%",
|
|
|
|
textAlign: "center",
|
|
|
|
margin: 0
|
|
|
|
}}
|
|
|
|
>
|
|
|
|
<Heart />
|
|
|
|
Vencord Donor
|
|
|
|
</Forms.FormTitle>
|
|
|
|
</Flex>
|
|
|
|
</Modals.ModalHeader>
|
|
|
|
<Modals.ModalContent>
|
|
|
|
<Flex>
|
|
|
|
<img
|
|
|
|
role="presentation"
|
|
|
|
src="https://cdn.discordapp.com/emojis/1026533070955872337.png"
|
|
|
|
alt=""
|
|
|
|
style={{ margin: "auto" }}
|
|
|
|
/>
|
|
|
|
<img
|
|
|
|
role="presentation"
|
|
|
|
src="https://cdn.discordapp.com/emojis/1026533090627174460.png"
|
|
|
|
alt=""
|
|
|
|
style={{ margin: "auto" }}
|
|
|
|
/>
|
|
|
|
</Flex>
|
|
|
|
<div style={{ padding: "1em" }}>
|
|
|
|
<Forms.FormText>
|
|
|
|
This Badge is a special perk for Vencord Donors
|
|
|
|
</Forms.FormText>
|
|
|
|
<Forms.FormText className={Margins.top20}>
|
|
|
|
Please consider supporting the development of Vencord by becoming a donor. It would mean a lot!!
|
|
|
|
</Forms.FormText>
|
|
|
|
</div>
|
|
|
|
</Modals.ModalContent>
|
|
|
|
<Modals.ModalFooter>
|
|
|
|
<Flex style={{ width: "100%", justifyContent: "center" }}>
|
|
|
|
<DonateButton />
|
|
|
|
</Flex>
|
|
|
|
</Modals.ModalFooter>
|
|
|
|
</Modals.ModalRoot>
|
|
|
|
</ErrorBoundary>
|
|
|
|
));
|
|
|
|
},
|
|
|
|
}));
|
2022-11-17 00:21:13 +01:00
|
|
|
}
|
2022-11-11 23:50:09 +01:00
|
|
|
});
|