1
0
Fork 1
mirror of https://github.com/Vendicated/Vencord.git synced 2025-01-25 08:46:25 +00:00

rework UI

This commit is contained in:
Elvy 2025-01-04 18:10:30 +01:00
parent b5c041282a
commit 84952943c6
3 changed files with 219 additions and 52 deletions

View file

@ -4,18 +4,27 @@
* SPDX-License-Identifier: GPL-3.0-or-later * SPDX-License-Identifier: GPL-3.0-or-later
*/ */
import { classNameFactory } from "@api/Styles";
import ErrorBoundary from "@components/ErrorBoundary"; import ErrorBoundary from "@components/ErrorBoundary";
import { Flex } from "@components/Flex"; import { Flex } from "@components/Flex";
import { Margins } from "@utils/margins"; import { Margins } from "@utils/margins";
import { wordsFromCamel, wordsToTitle } from "@utils/text"; import { wordsFromCamel, wordsToTitle } from "@utils/text";
import { OptionType, PluginOptionList } from "@utils/types"; import { OptionType, PluginOptionList } from "@utils/types";
import { findComponentByCodeLazy } from "@webpack"; import { findByCodeLazy, findComponentByCodeLazy } from "@webpack";
import { Button, ChannelStore, Forms, GuildStore, React, TextInput, useState } from "@webpack/common"; import { Avatar, Button, ChannelStore, Forms, GuildStore, IconUtils, React, Text, TextInput, useState } from "@webpack/common";
import { Channel } from "discord-types/general"; import { Channel, Guild } from "discord-types/general";
import { JSX } from "react";
import { ISettingElementProps } from "."; import { ISettingElementProps } from ".";
const cl = classNameFactory("vc-plugin-modal-");
const UserMentionComponent = findComponentByCodeLazy(".USER_MENTION)"); const UserMentionComponent = findComponentByCodeLazy(".USER_MENTION)");
const getDMChannelIcon = findByCodeLazy(".getChannelIconURL({");
const GroupDMAvatars = findComponentByCodeLazy(".AvatarSizeSpecs[", "getAvatarURL");
// FIXME saving is broken, so are indexes apparently?
const CloseIcon = () => { const CloseIcon = () => {
return <svg viewBox="0 0 20 20" fill="currentColor" aria-hidden="true" width="18" height="18"> return <svg viewBox="0 0 20 20" fill="currentColor" aria-hidden="true" width="18" height="18">
@ -52,6 +61,30 @@ export function SettingArrayComponent({
setItems(pluginSettings[id]); setItems(pluginSettings[id]);
} }
const removeButton = (index: number) => {
return (
<Button
size={Button.Sizes.MIN}
onClick={() => removeItem(index)}
style={
{ background: "none", }
}
>
<CloseIcon />
</Button>
);
};
const guildIcon = (guild: Guild) => {
const icon = guild?.icon == null ? undefined : IconUtils.getGuildIconURL({
id: guild.id,
icon: guild.icon,
size: 16,
});
return icon != null && <img className={cl("guild-icon")} src={icon} alt="" />;
};
const removeItem = (index: number) => { const removeItem = (index: number) => {
if (items.length === 1) { if (items.length === 1) {
setItems([]); setItems([]);
@ -62,13 +95,149 @@ export function SettingArrayComponent({
pluginSettings[id] = items; pluginSettings[id] = items;
}; };
function wrapChannel(id: string) { function renderGuildView() {
const channel = ChannelStore.getChannel(id) as Channel; return items.map(item => GuildStore.getGuild(item))
.map((guild, index) => (
<Flex
flexDirection="row"
key={index}
style={{
gap: "1px",
}}
>
{guild ? (
<div className={cl("name")} style={{ color: "var(--text-normal)" }}>
<span style={{ display: "inline-flex", alignItems: "center" }}>
{guildIcon(guild)}
<Text variant="text-sm/semibold" style={{ marginLeft: "4px" }}>{guild.name}</Text>
</span>
</div>
) : <Text variant="text-sm/semibold">{"Unknown Guild"}</Text>}
{removeButton(index)}
</Flex>
));
}
function renderChannelView() {
const getChannelSymbol = (type: number) => {
switch (type) {
case 2:
return <svg style={{ color: " var(--channel-icon)" }} className={cl("icon")} aria-hidden="true" role="img" width="16" height="16" fill="none" viewBox="0 0 24 24">
<path fill="currentColor" d="M12 3a1 1 0 0 0-1-1h-.06a1 1 0 0 0-.74.32L5.92 7H3a1 1 0 0 0-1 1v8a1 1 0 0 0 1 1h2.92l4.28 4.68a1 1 0 0 0 .74.32H11a1 1 0 0 0 1-1V3ZM15.1 20.75c-.58.14-1.1-.33-1.1-.92v-.03c0-.5.37-.92.85-1.05a7 7 0 0 0 0-13.5A1.11 1.11 0 0 1 14 4.2v-.03c0-.6.52-1.06 1.1-.92a9 9 0 0 1 0 17.5Z"></path>
<path fill="currentColor" d="M15.16 16.51c-.57.28-1.16-.2-1.16-.83v-.14c0-.43.28-.8.63-1.02a3 3 0 0 0 0-5.04c-.35-.23-.63-.6-.63-1.02v-.14c0-.63.59-1.1 1.16-.83a5 5 0 0 1 0 9.02Z"></path>
</svg>;
case 5:
return <svg style={{ color: " var(--channel-icon)" }} className={cl("icon")} aria-hidden="true" role="img" width="16" height="16" fill="none" viewBox="0 0 24 24">
<path fill="currentColor" fillRule="evenodd" d="M19.56 2a3 3 0 0 0-2.46 1.28 3.85 3.85 0 0 1-1.86 1.42l-8.9 3.18a.5.5 0 0 0-.34.47v10.09a3 3 0 0 0 2.27 2.9l.62.16c1.57.4 3.15-.56 3.55-2.12a.92.92 0 0 1 1.23-.63l2.36.94c.42.27.79.62 1.07 1.03A3 3 0 0 0 19.56 22h.94c.83 0 1.5-.67 1.5-1.5v-17c0-.83-.67-1.5-1.5-1.5h-.94Zm-8.53 15.8L8 16.7v1.73a1 1 0 0 0 .76.97l.62.15c.5.13 1-.17 1.12-.67.1-.41.29-.78.53-1.1Z" clipRule="evenodd"></path>
<path fill="currentColor" d="M2 10c0-1.1.9-2 2-2h.5c.28 0 .5.22.5.5v7a.5.5 0 0 1-.5.5H4a2 2 0 0 1-2-2v-4Z"></path>
</svg>;
case 13:
return <svg style={{ color: " var(--channel-icon)" }} className={cl("icon")} aria-hidden="true" role="img" width="16" height="16" fill="none" viewBox="0 0 24 24">
<path fill="currentColor" d="M19.61 18.25a1.08 1.08 0 0 1-.07-1.33 9 9 0 1 0-15.07 0c.26.42.25.97-.08 1.33l-.02.02c-.41.44-1.12.43-1.46-.07a11 11 0 1 1 18.17 0c-.33.5-1.04.51-1.45.07l-.02-.02Z"></path>
<path fill="currentColor" d="M16.83 15.23c.43.47 1.18.42 1.45-.14a7 7 0 1 0-12.57 0c.28.56 1.03.6 1.46.14l.05-.06c.3-.33.35-.81.17-1.23A4.98 4.98 0 0 1 12 7a5 5 0 0 1 4.6 6.94c-.17.42-.13.9.18 1.23l.05.06Z"></path>
<path fill="currentColor" d="M15 12a3 3 0 1 1-6 0 3 3 0 0 1 6 0ZM6.33 20.03c-.25.72.12 1.5.8 1.84a10.96 10.96 0 0 0 9.73 0 1.52 1.52 0 0 0 .8-1.84 6 6 0 0 0-11.33 0Z"></path>
</svg>;
case 15:
return <svg style={{ color: " var(--channel-icon)" }} className={cl("icon")} aria-hidden="true" role="img" width="16" height="16" fill="none" viewBox="0 0 24 24">
<path fill="currentColor" d="M18.91 12.98a5.45 5.45 0 0 1 2.18 6.2c-.1.33-.09.68.1.96l.83 1.32a1 1 0 0 1-.84 1.54h-5.5A5.6 5.6 0 0 1 10 17.5a5.6 5.6 0 0 1 5.68-5.5c1.2 0 2.32.36 3.23.98Z"></path>
<path fill="currentColor" d="M19.24 10.86c.32.16.72-.02.74-.38L20 10c0-4.42-4.03-8-9-8s-9 3.58-9 8c0 1.5.47 2.91 1.28 4.11.14.21.12.49-.06.67l-1.51 1.51A1 1 0 0 0 2.4 18h5.1a.5.5 0 0 0 .49-.5c0-4.2 3.5-7.5 7.68-7.5 1.28 0 2.5.3 3.56.86Z"></path>
</svg>;
default: // Text channel icon
return <svg style={{ color: " var(--channel-icon)" }} className={cl("icon")} aria-hidden="true" role="img" width="16" height="16" fill="none" viewBox="0 0 24 24">
<path fill="currentColor" fillRule="evenodd" d="M10.99 3.16A1 1 0 1 0 9 2.84L8.15 8H4a1 1 0 0 0 0 2h3.82l-.67 4H3a1 1 0 1 0 0 2h3.82l-.8 4.84a1 1 0 0 0 1.97.32L8.85 16h4.97l-.8 4.84a1 1 0 0 0 1.97.32l.86-5.16H20a1 1 0 1 0 0-2h-3.82l.67-4H21a1 1 0 1 0 0-2h-3.82l.8-4.84a1 1 0 1 0-1.97-.32L15.15 8h-4.97l.8-4.84ZM14.15 14l.67-4H9.85l-.67 4h4.97Z" clipRule="evenodd"></path>
</svg>;
}
};
// collapsible guild list with channels in it
const channels: Record<string, Channel[]> = {};
const dmChannels: Channel[] = [];
const elements: JSX.Element[] = [];
for (const item of items) {
const channel = ChannelStore.getChannel(item);
if (!channel) { if (!channel) {
return "Unknown Channel"; continue;
} }
return (GuildStore.getGuild(channel.guild_id)?.name ?? "Unknown Guild") + " - " + channel.name; if (channel.isDM() || channel.isGroupDM()) {
dmChannels.push(channel);
continue;
} }
if (!channels[channel.guild_id]) {
channels[channel.guild_id] = [];
}
channels[channel.guild_id].push(channel);
}
if (dmChannels.length > 0) {
elements.push(
<details>
<summary style={{ color: "var(--text-normal)", marginBottom: "8px" }}>DMs</summary>
<div style={{ paddingLeft: "16px" }}>
{dmChannels.map((channel, index) => (
<Flex
flexDirection="row"
key={index}
style={{
gap: "1px",
marginBottom: "8px",
}}
>
<span style={{ display: "inline-flex", alignItems: "center" }}>
{channel.recipients.length >= 2 && channel.icon == null ? (
<GroupDMAvatars recipients={channel.recipients} size="SIZE_16" />
) : (
<Avatar src={getDMChannelIcon(channel)} size="SIZE_16" />
)}
<Text variant="text-sm/semibold" style={{ marginLeft: "4px" }}>{channel.name}</Text>
{removeButton(index)}
</span>
</Flex>
))}
</div>
</details>
);
}
Object.keys(channels).forEach(guildId => {
const guild = GuildStore.getGuild(guildId);
elements.push(
<details>
{!guild ? <summary style={{ color: "var(--text-normal)", marginBottom: "8px" }}>Unknown Guild</summary> : (
<summary style={{ color: "var(--text-normal)", marginBottom: "8px" }}>
<span style={{ display: "inline-flex", alignItems: "center" }}>
{guildIcon(guild)}
<Text variant="text-sm/semibold" style={{ marginLeft: "4px" }}>{guild.name}</Text>
</span>
</summary>
)}
<div style={{ paddingLeft: "16px", color: "var(--text-normal)" }}>
{channels[guildId].map((channel, index) => (
<Flex
flexDirection="row"
key={index}
style={{
gap: "1px",
marginBottom: "8px"
}}
>
<span style={{ display: "inline-flex", alignItems: "center" }}>
{getChannelSymbol(channel.type)}
<Text variant="text-sm/semibold" style={{ marginLeft: "4px" }}>{channel.name}</Text>
{removeButton(index)}
</span>
</Flex>
))}
</div>
</details>
);
});
return elements;
}
function handleSubmit() { function handleSubmit() {
const inputElement = document.getElementById(`vc-plugin-modal-input-${option.type === OptionType.CHANNELS ? "channel" : option.type === OptionType.GUILDS ? "guild" : option.type === OptionType.USERS ? "user" : "string"}`) as HTMLInputElement; const inputElement = document.getElementById(`vc-plugin-modal-input-${option.type === OptionType.CHANNELS ? "channel" : option.type === OptionType.GUILDS ? "guild" : option.type === OptionType.USERS ? "user" : "string"}`) as HTMLInputElement;
if (!inputElement || inputElement.value === "") { if (!inputElement || inputElement.value === "") {
@ -79,8 +248,17 @@ export function SettingArrayComponent({
setError("Value is not a valid snowflake ID"); setError("Value is not a valid snowflake ID");
return; return;
} }
if (items.includes(inputElement.value)) {
setError("This item is already added");
inputElement.value = "";
return;
}
setItems([...items, inputElement.value]); setItems([...items, inputElement.value]);
console.log(pluginSettings[id]);
pluginSettings[id] = items; pluginSettings[id] = items;
console.log(pluginSettings[id]);
inputElement.value = ""; inputElement.value = "";
} }
@ -90,10 +268,11 @@ export function SettingArrayComponent({
<Forms.FormTitle>{wordsToTitle(wordsFromCamel(id))}</Forms.FormTitle> <Forms.FormTitle>{wordsToTitle(wordsFromCamel(id))}</Forms.FormTitle>
<Forms.FormText className={Margins.bottom16} type="description">{option.description}</Forms.FormText> <Forms.FormText className={Margins.bottom16} type="description">{option.description}</Forms.FormText>
<ErrorBoundary noop> <ErrorBoundary noop>
<React.Fragment> {option.type === OptionType.ARRAY || option.type === OptionType.USERS ?
{items.map((item, index) => ( items.map((item, index) => (
<Flex <Flex
flexDirection="row" flexDirection="row"
key={index}
style={{ style={{
gap: "1px", gap: "1px",
}} }}
@ -103,27 +282,14 @@ export function SettingArrayComponent({
userId={item} userId={item}
className="mention" className="mention"
/> />
) : option.type === OptionType.CHANNELS ? (
<span style={{ color: "white" }}>{wrapChannel(item)}</span>
) : option.type === OptionType.GUILDS ? (
<span style={{ color: "white" }}>
{GuildStore.getGuild(item)?.name || "Unknown Guild"}
</span>
// TODO add logo to guild and channel?
) : ( ) : (
<span style={{ color: "white" }}>{item}</span> <span style={{ color: "var(--text-normal)" }}>{item}</span>
)} )}
<Button {removeButton(index)}
size={Button.Sizes.MIN}
onClick={() => removeItem(index)}
style={
{ background: "none", }
}
>
<CloseIcon />
</Button>
</Flex> </Flex>
))} )) : option.type === OptionType.CHANNELS ?
renderChannelView() : renderGuildView()
}
<Flex <Flex
flexDirection="row" flexDirection="row"
style={{ style={{
@ -144,10 +310,7 @@ export function SettingArrayComponent({
<CheckMarkIcon /> <CheckMarkIcon />
</Button> </Button>
</Flex> </Flex>
</React.Fragment>
</ErrorBoundary> </ErrorBoundary>
{error && <Forms.FormText style={{ color: "var(--text-danger)" }}>{error}</Forms.FormText>} {error && <Forms.FormText style={{ color: "var(--text-danger)" }}>{error}</Forms.FormText>}
</Forms.FormSection> </Forms.FormSection>
); );

View file

@ -53,10 +53,12 @@ function renderRegisteredPlugins(name: string, value: any) {
<Menu.MenuItem <Menu.MenuItem
id={`vc-plugin-settings-${plugin}`} id={`vc-plugin-settings-${plugin}`}
label={plugin} label={plugin}
key={`vc-plugin-settings-${plugin}`}
> >
{plugins[plugin].map(setting => ( {plugins[plugin].map(setting => (
<Menu.MenuCheckboxItem <Menu.MenuCheckboxItem
id={`vc-plugin-settings-${plugin}-${setting}`} id={`vc-plugin-settings-${plugin}-${setting}`}
key={`vc-plugin-settings-${plugin}-${setting}`}
// @ts-ignore popoutText exists due to this being a list option type // @ts-ignore popoutText exists due to this being a list option type
label={Vencord.Plugins.plugins[plugin].settings?.def[setting].popoutText ?? setting} label={Vencord.Plugins.plugins[plugin].settings?.def[setting].popoutText ?? setting}
action={() => handleCheckboxClick(plugin, setting)} action={() => handleCheckboxClick(plugin, setting)}
@ -104,9 +106,11 @@ export default definePlugin({
contextMenus: { contextMenus: {
"channel-context": MakeContextCallback("Channel"), "channel-context": MakeContextCallback("Channel"),
"thread-context": MakeContextCallback("Channel"), "thread-context": MakeContextCallback("Channel"),
"gdm-context": MakeContextCallback("Channel"), // TODO make this work
"guild-context": MakeContextCallback("Guild"), "guild-context": MakeContextCallback("Guild"),
"user-context": MakeContextCallback("User") "user-context": MakeContextCallback("User")
}, },
required: true,
start() { start() {
for (const plugin of Object.values(Vencord.Plugins.plugins)) { for (const plugin of Object.values(Vencord.Plugins.plugins)) {

View file

@ -205,7 +205,7 @@ export default definePlugin({
name: "MessageLogger", name: "MessageLogger",
description: "Temporarily logs deleted and edited messages.", description: "Temporarily logs deleted and edited messages.",
authors: [Devs.rushii, Devs.Ven, Devs.AutumnVN, Devs.Nickyux, Devs.Kyuuhachi], authors: [Devs.rushii, Devs.Ven, Devs.AutumnVN, Devs.Nickyux, Devs.Kyuuhachi],
dependencies: ["MessageUpdaterAPI", "SettingArraysAPI"], dependencies: ["MessageUpdaterAPI"],
contextMenus: { contextMenus: {
"message": patchMessageContextMenu, "message": patchMessageContextMenu,