/* * 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 . */ import { generateId } from "@api/Commands"; import { useSettings } from "@api/Settings"; import { disableStyle, enableStyle } from "@api/Styles"; import ErrorBoundary from "@components/ErrorBoundary"; import { Flex } from "@components/Flex"; import { proxyLazy } from "@utils/lazy"; import { Margins } from "@utils/margins"; import { classes, isObjectEmpty } from "@utils/misc"; import { ModalCloseButton, ModalContent, ModalFooter, ModalHeader, ModalProps, ModalRoot, ModalSize } from "@utils/modal"; import { LazyComponent } from "@utils/react"; import { OptionType, Plugin } from "@utils/types"; import { findByCode, findByPropsLazy } from "@webpack"; import { Button, FluxDispatcher, Forms, React, Text, Tooltip, UserStore, UserUtils } from "@webpack/common"; import { User } from "discord-types/general"; import { Constructor } from "type-fest"; import { ISettingElementProps, SettingBooleanComponent, SettingCustomComponent, SettingNumericComponent, SettingSelectComponent, SettingSliderComponent, SettingTextComponent } from "./components"; import hideBotTagStyle from "./userPopoutHideBotTag.css?managed"; const UserSummaryItem = LazyComponent(() => findByCode("defaultRenderUser", "showDefaultAvatarsForNullUsers")); const AvatarStyles = findByPropsLazy("moreUsers", "emptyUser", "avatarContainer", "clickableAvatar"); const UserRecord: Constructor> = proxyLazy(() => UserStore.getCurrentUser().constructor) as any; interface PluginModalProps extends ModalProps { plugin: Plugin; onRestartNeeded(): void; } function makeDummyUser(user: { username: string; id?: string; avatar?: string; }) { const newUser = new UserRecord({ username: user.username, id: user.id ?? generateId(), avatar: user.avatar, /** To stop discord making unwanted requests... */ bot: true, }); FluxDispatcher.dispatch({ type: "USER_UPDATE", user: newUser, }); return newUser; } const Components: Record>> = { [OptionType.STRING]: SettingTextComponent, [OptionType.NUMBER]: SettingNumericComponent, [OptionType.BIGINT]: SettingNumericComponent, [OptionType.BOOLEAN]: SettingBooleanComponent, [OptionType.SELECT]: SettingSelectComponent, [OptionType.SLIDER]: SettingSliderComponent, [OptionType.COMPONENT]: SettingCustomComponent }; export default function PluginModal({ plugin, onRestartNeeded, onClose, transitionState }: PluginModalProps) { const [authors, setAuthors] = React.useState[]>([]); const pluginSettings = useSettings().plugins[plugin.name]; const [tempSettings, setTempSettings] = React.useState>({}); const [errors, setErrors] = React.useState>({}); const [saveError, setSaveError] = React.useState(null); const canSubmit = () => Object.values(errors).every(e => !e); const hasSettings = Boolean(pluginSettings && plugin.options && !isObjectEmpty(plugin.options)); React.useEffect(() => { enableStyle(hideBotTagStyle); let originalUser: User; (async () => { for (const user of plugin.authors.slice(0, 6)) { const author = user.id ? await UserUtils.fetchUser(`${user.id}`) // only show name & pfp and no actions so users cannot harass plugin devs for support (send dms, add as friend, etc) .then(u => (originalUser = u, makeDummyUser(u))) .catch(() => makeDummyUser({ username: user.name })) : makeDummyUser({ username: user.name }); setAuthors(a => [...a, author]); } })(); return () => { disableStyle(hideBotTagStyle); if (originalUser) FluxDispatcher.dispatch({ type: "USER_UPDATE", user: originalUser }); }; }, []); async function saveAndClose() { if (!plugin.options) { onClose(); return; } if (plugin.beforeSave) { const result = await Promise.resolve(plugin.beforeSave(tempSettings)); if (result !== true) { setSaveError(result); return; } } let restartNeeded = false; for (const [key, value] of Object.entries(tempSettings)) { const option = plugin.options[key]; pluginSettings[key] = value; option?.onChange?.(value); if (option?.restartNeeded) restartNeeded = true; } if (restartNeeded) onRestartNeeded(); onClose(); } function renderSettings() { if (!hasSettings || !plugin.options) { return There are no settings for this plugin.; } else { const options = Object.entries(plugin.options).map(([key, setting]) => { if (setting.hidden) return null; function onChange(newValue: any) { setTempSettings(s => ({ ...s, [key]: newValue })); } function onError(hasError: boolean) { setErrors(e => ({ ...e, [key]: hasError })); } const Component = Components[setting.type]; return ( ); }); return {options}; } } function renderMoreUsers(_label: string, count: number) { const sliceCount = plugin.authors.length - count; const sliceStart = plugin.authors.length - sliceCount; const sliceEnd = sliceStart + plugin.authors.length - count; return ( u.name).join(", ")}> {({ onMouseEnter, onMouseLeave }) => (
+{sliceCount}
)}
); } return ( {plugin.name} About {plugin.name} {plugin.description} Authors
{!!plugin.settingsAboutComponent && (
)} Settings {renderSettings()}
{hasSettings && {({ onMouseEnter, onMouseLeave }) => ( )} {saveError && Error while saving: {saveError}} }
); }