2023-05-02 00:55:38 +00:00
|
|
|
/*
|
|
|
|
* 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/>.
|
|
|
|
*/
|
|
|
|
|
2023-05-05 16:41:01 +00:00
|
|
|
import "./index.css";
|
|
|
|
|
2023-05-02 00:55:38 +00:00
|
|
|
import { openNotificationLogModal } from "@api/Notifications/notificationLog";
|
2023-08-12 00:14:07 +00:00
|
|
|
import { Settings } from "@api/Settings";
|
2023-05-02 00:55:38 +00:00
|
|
|
import ErrorBoundary from "@components/ErrorBoundary";
|
|
|
|
import { Devs } from "@utils/constants";
|
|
|
|
import definePlugin from "@utils/types";
|
2023-11-25 00:32:21 +00:00
|
|
|
import { filters, find, LazyComponentWebpack } from "@webpack";
|
2023-05-02 00:55:38 +00:00
|
|
|
import { Menu, Popout, useState } from "@webpack/common";
|
|
|
|
import type { ReactNode } from "react";
|
|
|
|
|
2023-11-25 00:32:21 +00:00
|
|
|
const HeaderBarIcon = LazyComponentWebpack(() => {
|
2023-10-25 21:00:04 +00:00
|
|
|
const filter = filters.byCode(".HEADER_BAR_BADGE");
|
2023-11-25 00:32:21 +00:00
|
|
|
return find(m => m.Icon && filter(m.Icon))?.Icon;
|
2023-10-25 21:00:04 +00:00
|
|
|
});
|
2023-05-02 00:55:38 +00:00
|
|
|
|
|
|
|
function VencordPopout(onClose: () => void) {
|
|
|
|
const pluginEntries = [] as ReactNode[];
|
|
|
|
|
|
|
|
for (const plugin of Object.values(Vencord.Plugins.plugins)) {
|
|
|
|
if (plugin.toolboxActions) {
|
|
|
|
pluginEntries.push(
|
|
|
|
<Menu.MenuGroup
|
|
|
|
label={plugin.name}
|
|
|
|
key={`vc-toolbox-${plugin.name}`}
|
|
|
|
>
|
|
|
|
{Object.entries(plugin.toolboxActions).map(([text, action]) => {
|
|
|
|
const key = `vc-toolbox-${plugin.name}-${text}`;
|
|
|
|
|
|
|
|
return (
|
|
|
|
<Menu.MenuItem
|
|
|
|
id={key}
|
|
|
|
key={key}
|
|
|
|
label={text}
|
|
|
|
action={action}
|
|
|
|
/>
|
|
|
|
);
|
|
|
|
})}
|
|
|
|
</Menu.MenuGroup>
|
|
|
|
);
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
return (
|
|
|
|
<Menu.Menu
|
|
|
|
navId="vc-toolbox"
|
|
|
|
onClose={onClose}
|
|
|
|
>
|
|
|
|
<Menu.MenuItem
|
|
|
|
id="vc-toolbox-notifications"
|
|
|
|
label="Open Notification Log"
|
|
|
|
action={openNotificationLogModal}
|
|
|
|
/>
|
2023-08-12 00:14:07 +00:00
|
|
|
<Menu.MenuCheckboxItem
|
|
|
|
id="vc-toolbox-quickcss-toggle"
|
|
|
|
checked={Settings.useQuickCss}
|
|
|
|
label={"Enable QuickCSS"}
|
|
|
|
action={() => {
|
|
|
|
Settings.useQuickCss = !Settings.useQuickCss;
|
|
|
|
onClose();
|
|
|
|
}}
|
|
|
|
/>
|
2023-05-02 00:55:38 +00:00
|
|
|
<Menu.MenuItem
|
|
|
|
id="vc-toolbox-quickcss"
|
|
|
|
label="Open QuickCSS"
|
2023-05-02 00:57:25 +00:00
|
|
|
action={() => VencordNative.quickCss.openEditor()}
|
2023-05-02 00:55:38 +00:00
|
|
|
/>
|
|
|
|
{...pluginEntries}
|
|
|
|
</Menu.Menu>
|
|
|
|
);
|
|
|
|
}
|
|
|
|
|
2023-10-28 00:21:35 +00:00
|
|
|
function VencordPopoutIcon(isShown: boolean) {
|
2023-05-02 00:55:38 +00:00
|
|
|
return (
|
2023-10-28 00:21:35 +00:00
|
|
|
<svg xmlns="http://www.w3.org/2000/svg" viewBox="0 0 27 27" width={24} height={24}>
|
|
|
|
<path fill="currentColor" d={isShown ? "M9 0h1v1h1v2h1v2h3V3h1V1h1V0h1v2h1v2h1v7h-1v-1h-3V9h1V6h-1v4h-3v1h1v-1h2v1h3v1h-1v1h-3v2h1v1h1v1h1v3h-1v4h-2v-1h-1v-4h-1v4h-1v1h-2v-4H9v-3h1v-1h1v-1h1v-2H9v-1H8v-1h3V6h-1v3h1v1H8v1H7V4h1V2h1M5 19h2v1h1v1h1v3H4v-1h2v-1H4v-2h1m15-1h2v1h1v2h-2v1h2v1h-5v-3h1v-1h1m4 3h4v1h-4" : "M0 0h7v1H6v1H5v1H4v1H3v1H2v1h5v1H0V6h1V5h1V4h1V3h1V2h1V1H0m13 2h5v1h-1v1h-1v1h-1v1h3v1h-5V7h1V6h1V5h1V4h-3m8 5h1v5h1v-1h1v1h-1v1h1v-1h1v1h-1v3h-1v1h-2v1h-1v1h1v-1h2v-1h1v2h-1v1h-2v1h-1v-1h-1v1h-6v-1h-1v-1h-1v-2h1v1h2v1h3v1h1v-1h-1v-1h-3v-1h-4v-4h1v-2h1v-1h1v-1h1v2h1v1h1v-1h1v1h-1v1h2v-2h1v-2h1v-1h1M8 14h2v1H9v4h1v2h1v1h1v1h1v1h4v1h-6v-1H5v-1H4v-5h1v-1h1v-2h2m17 3h1v3h-1v1h-1v1h-1v2h-2v-2h2v-1h1v-1h1m1 0h1v3h-1v1h-2v-1h1v-1h1"} />
|
2023-05-05 16:41:01 +00:00
|
|
|
</svg>
|
2023-05-02 00:55:38 +00:00
|
|
|
);
|
|
|
|
}
|
|
|
|
|
|
|
|
function VencordPopoutButton() {
|
|
|
|
const [show, setShow] = useState(false);
|
|
|
|
|
|
|
|
return (
|
|
|
|
<Popout
|
|
|
|
position="bottom"
|
|
|
|
align="right"
|
|
|
|
animation={Popout.Animation.NONE}
|
|
|
|
shouldShow={show}
|
|
|
|
onRequestClose={() => setShow(false)}
|
|
|
|
renderPopout={() => VencordPopout(() => setShow(false))}
|
|
|
|
>
|
|
|
|
{(_, { isShown }) => (
|
|
|
|
<HeaderBarIcon
|
2023-05-05 16:41:01 +00:00
|
|
|
className="vc-toolbox-btn"
|
2023-05-02 00:55:38 +00:00
|
|
|
onClick={() => setShow(v => !v)}
|
|
|
|
tooltip={isShown ? null : "Vencord Toolbox"}
|
2023-10-28 00:21:35 +00:00
|
|
|
icon={() => VencordPopoutIcon(isShown)}
|
2023-05-02 00:55:38 +00:00
|
|
|
selected={isShown}
|
|
|
|
/>
|
|
|
|
)}
|
|
|
|
</Popout>
|
|
|
|
);
|
|
|
|
}
|
|
|
|
|
|
|
|
function ToolboxFragmentWrapper({ children }: { children: ReactNode[]; }) {
|
|
|
|
children.splice(
|
|
|
|
children.length - 1, 0,
|
|
|
|
<ErrorBoundary noop={true}>
|
|
|
|
<VencordPopoutButton />
|
|
|
|
</ErrorBoundary>
|
|
|
|
);
|
|
|
|
|
|
|
|
return <>{children}</>;
|
|
|
|
}
|
|
|
|
|
|
|
|
export default definePlugin({
|
|
|
|
name: "VencordToolbox",
|
|
|
|
description: "Adds a button next to the inbox button in the channel header that houses Vencord quick actions",
|
2023-05-05 16:41:01 +00:00
|
|
|
authors: [Devs.Ven, Devs.AutumnVN],
|
2023-05-02 00:55:38 +00:00
|
|
|
|
|
|
|
patches: [
|
|
|
|
{
|
2023-10-25 21:00:04 +00:00
|
|
|
find: "toolbar:function",
|
2023-05-02 00:55:38 +00:00
|
|
|
replacement: {
|
|
|
|
match: /(?<=toolbar:function.{0,100}\()\i.Fragment,/,
|
|
|
|
replace: "$self.ToolboxFragmentWrapper,"
|
|
|
|
}
|
|
|
|
}
|
|
|
|
],
|
|
|
|
|
|
|
|
ToolboxFragmentWrapper: ErrorBoundary.wrap(ToolboxFragmentWrapper, {
|
|
|
|
fallback: () => <p style={{ color: "red" }}>Failed to render :(</p>
|
|
|
|
})
|
|
|
|
});
|