vcPanelSettings/index.tsx

252 lines
9.2 KiB
TypeScript
Raw Permalink Normal View History

2024-12-25 20:04:51 -05:00
/*
* Vencord, a Discord client mod
* Copyright (c) 2024 Vendicated and contributors
* SPDX-License-Identifier: GPL-3.0-or-later
*/
2024-12-25 17:06:39 -05:00
import "./style.css";
2024-12-25 13:00:52 -05:00
import { definePluginSettings } from "@api/Settings";
2024-12-25 20:04:51 -05:00
import { Link } from "@components/Link";
2024-12-25 13:00:52 -05:00
import { Devs } from "@utils/constants";
import { identity } from "@utils/misc";
2024-12-25 20:04:51 -05:00
import definePlugin, { OptionType } from "@utils/types";
import { findByPropsLazy } from "@webpack";
import { FluxDispatcher, Forms, Select, Slider, Text, useEffect, useState } from "@webpack/common";
2024-12-25 13:00:52 -05:00
import { Settings } from "Vencord";
2024-12-25 20:04:51 -05:00
const configModule = findByPropsLazy("getOutputVolume");
2024-12-25 13:00:52 -05:00
function OutputVolumeComponent() {
2024-12-25 20:04:51 -05:00
const [outputVolume, setOutputVolume] = useState(configModule.getOutputVolume());
2024-12-25 13:00:52 -05:00
useEffect(() => {
2024-12-25 20:04:51 -05:00
const listener = () => setOutputVolume(configModule.getOutputVolume());
2024-12-25 13:00:52 -05:00
FluxDispatcher.subscribe("AUDIO_SET_OUTPUT_VOLUME", listener);
});
return (
<>
2024-12-25 20:13:34 -05:00
{Settings.plugins.VCPanelSettings.showOutputVolumeHeader && <Forms.FormTitle>Output volume</Forms.FormTitle>}
<Slider maxValue={200} minValue={0} onValueRender={v => `${v.toFixed(0)}%`} initialValue={outputVolume} asValueChanges={volume => {
2024-12-25 13:00:52 -05:00
FluxDispatcher.dispatch({
type: "AUDIO_SET_OUTPUT_VOLUME",
volume
});
}} />
</>
);
}
function InputVolumeComponent() {
2024-12-25 20:04:51 -05:00
const [inputVolume, setInputVolume] = useState(configModule.getInputVolume());
2024-12-25 13:00:52 -05:00
useEffect(() => {
2024-12-25 20:04:51 -05:00
const listener = () => setInputVolume(configModule.getInputVolume());
2024-12-25 13:00:52 -05:00
FluxDispatcher.subscribe("AUDIO_SET_INPUT_VOLUME", listener);
});
return (
<>
2024-12-25 20:13:34 -05:00
{Settings.plugins.VCPanelSettings.showInputVolumeHeader && <Forms.FormTitle>Input volume</Forms.FormTitle>}
<Slider maxValue={100} minValue={0} initialValue={inputVolume} asValueChanges={volume => {
2024-12-25 13:00:52 -05:00
FluxDispatcher.dispatch({
type: "AUDIO_SET_INPUT_VOLUME",
volume
});
}} />
</>
);
}
function OutputDeviceComponent() {
2024-12-25 20:04:51 -05:00
const [outputDevice, setOutputDevice] = useState(configModule.getOutputDeviceId());
2024-12-25 13:00:52 -05:00
useEffect(() => {
2024-12-25 20:04:51 -05:00
const listener = () => setOutputDevice(configModule.getOutputDeviceId());
2024-12-25 13:00:52 -05:00
FluxDispatcher.subscribe("AUDIO_SET_OUTPUT_DEVICE", listener);
});
return (
<>
2024-12-25 16:36:12 -05:00
{Settings.plugins.VCPanelSettings.showOutputDeviceHeader && <Forms.FormTitle>Output device</Forms.FormTitle>}
2024-12-25 20:04:51 -05:00
<Select options={Object.values(configModule.getOutputDevices()).map((device: any /* i am NOT typing this*/) => {
2024-12-25 16:36:12 -05:00
return { value: device.id, label: Settings.plugins.VCPanelSettings.showOutputDeviceHeader ? device.name : `🔊 ${device.name}` };
2024-12-25 13:00:52 -05:00
})}
serialize={identity}
2024-12-25 20:04:51 -05:00
isSelected={value => value === outputDevice}
select={id => {
2024-12-25 13:00:52 -05:00
FluxDispatcher.dispatch({
type: "AUDIO_SET_OUTPUT_DEVICE",
id
});
}}>
</Select>
</>
);
}
function InputDeviceComponent() {
2024-12-25 20:04:51 -05:00
const [inputDevice, setInputDevice] = useState(configModule.getInputDeviceId());
2024-12-25 13:00:52 -05:00
useEffect(() => {
2024-12-25 20:04:51 -05:00
const listener = () => setInputDevice(configModule.getInputDeviceId());
2024-12-25 13:00:52 -05:00
FluxDispatcher.subscribe("AUDIO_SET_INPUT_DEVICE", listener);
});
return (
<div style={{ marginTop: "10px" }}>
2024-12-25 16:36:12 -05:00
{Settings.plugins.VCPanelSettings.showInputDeviceHeader && <Forms.FormTitle>Input device</Forms.FormTitle>}
2024-12-25 20:04:51 -05:00
<Select options={Object.values(configModule.getInputDevices()).map((device: any /* i am NOT typing this*/) => {
2024-12-25 16:36:12 -05:00
return { value: device.id, label: Settings.plugins.VCPanelSettings.showInputDeviceHeader ? device.name : `🎤 ${device.name}` };
2024-12-25 13:00:52 -05:00
})}
serialize={identity}
2024-12-25 20:04:51 -05:00
isSelected={value => value === inputDevice}
select={id => {
2024-12-25 13:00:52 -05:00
FluxDispatcher.dispatch({
type: "AUDIO_SET_INPUT_DEVICE",
id
});
}}>
</Select>
</div>
);
}
2024-12-25 20:04:51 -05:00
function VideoDeviceComponent() {
const [videoDevice, setVideoDevice] = useState(configModule.getVideoDeviceId());
useEffect(() => {
const listener = () => setVideoDevice(configModule.getVideoDeviceId());
FluxDispatcher.subscribe("MEDIA_ENGINE_SET_VIDEO_DEVICE", listener);
});
return (
<div style={{ marginTop: "10px" }}>
{Settings.plugins.VCPanelSettings.showVideoDeviceHeader && <Forms.FormTitle>Camera</Forms.FormTitle>}
<Select options={Object.values(configModule.getVideoDevices()).map((device: any /* i am NOT typing this*/) => {
return { value: device.id, label: Settings.plugins.VCPanelSettings.showVideoDeviceHeader ? device.name : `📷 ${device.name}` };
})}
serialize={identity}
isSelected={value => value === videoDevice}
select={id => {
FluxDispatcher.dispatch({
type: "MEDIA_ENGINE_SET_VIDEO_DEVICE",
id
});
}}>
</Select>
</div>
);
}
2024-12-25 13:00:52 -05:00
function VoiceSettings() {
2024-12-25 17:06:39 -05:00
const [showSettings, setShowSettings] = useState(Settings.plugins.VCPanelSettings.uncollapseSettingsByDefault);
2024-12-25 16:36:12 -05:00
return <div style={{ marginTop: "20px" }}>
2024-12-25 17:06:39 -05:00
<div style={{ marginBottom: "10px" }}>
<Link className="vc-panelsettings-underline-on-hover" style={{ color: "var(--header-secondary)" }} onClick={() => { setShowSettings(!showSettings); }}>{!showSettings ? "► Settings" : "▼ Hide"}</Link>
</div>
{
showSettings && <>
{Settings.plugins.VCPanelSettings.outputVolume && <OutputVolumeComponent />}
{Settings.plugins.VCPanelSettings.inputVolume && <InputVolumeComponent />}
{Settings.plugins.VCPanelSettings.outputDevice && <OutputDeviceComponent />}
{Settings.plugins.VCPanelSettings.inputDevice && <InputDeviceComponent />}
2024-12-25 20:04:51 -05:00
{Settings.plugins.VCPanelSettings.camera && <VideoDeviceComponent />}
2024-12-25 17:06:39 -05:00
</>
}
2024-12-25 13:00:52 -05:00
</div>;
}
export default definePlugin({
name: "VCPanelSettings",
description: "Control voice settings right from the voice panel",
authors: [Devs.nin0dev],
settings: definePluginSettings({
2024-12-25 17:06:39 -05:00
title1: {
type: OptionType.COMPONENT,
component: () => <Text style={{ fontWeight: "bold", fontSize: "1.27rem" }}>Appearance</Text>,
description: ""
},
uncollapseSettingsByDefault: {
2024-12-25 13:00:52 -05:00
type: OptionType.BOOLEAN,
2024-12-25 17:06:39 -05:00
default: false,
description: "Automatically uncollapse voice settings by default"
},
title2: {
type: OptionType.COMPONENT,
component: () => <Text style={{ fontWeight: "bold", fontSize: "1.27rem" }}>Settings to show</Text>,
description: ""
2024-12-25 13:00:52 -05:00
},
outputVolume: {
type: OptionType.BOOLEAN,
default: true,
description: "Show an output volume slider"
},
2024-12-25 17:06:39 -05:00
inputVolume: {
2024-12-25 13:00:52 -05:00
type: OptionType.BOOLEAN,
default: true,
2024-12-25 17:06:39 -05:00
description: "Show an input volume slider"
2024-12-25 13:00:52 -05:00
},
2024-12-25 17:06:39 -05:00
outputDevice: {
2024-12-25 13:00:52 -05:00
type: OptionType.BOOLEAN,
default: true,
2024-12-25 17:06:39 -05:00
description: "Show an output device selector"
2024-12-25 13:00:52 -05:00
},
2024-12-25 17:06:39 -05:00
inputDevice: {
2024-12-25 13:00:52 -05:00
type: OptionType.BOOLEAN,
2024-12-25 17:06:39 -05:00
default: true,
description: "Show an input device selector"
2024-12-25 13:00:52 -05:00
},
2024-12-25 20:04:51 -05:00
camera: {
type: OptionType.BOOLEAN,
default: false,
description: "Show a camera selector"
},
2024-12-25 17:06:39 -05:00
title3: {
type: OptionType.COMPONENT,
component: () => <Text style={{ fontWeight: "bold", fontSize: "1.27rem" }}>Headers to show</Text>,
description: ""
},
showOutputVolumeHeader: {
2024-12-25 13:00:52 -05:00
type: OptionType.BOOLEAN,
default: true,
2024-12-25 17:06:39 -05:00
description: "Show header above output volume slider"
},
showInputVolumeHeader: {
type: OptionType.BOOLEAN,
default: true,
description: "Show header above input volume slider"
},
showOutputDeviceHeader: {
type: OptionType.BOOLEAN,
default: false,
2024-12-25 20:04:51 -05:00
description: "Show header above output device selector"
2024-12-25 13:00:52 -05:00
},
showInputDeviceHeader: {
type: OptionType.BOOLEAN,
default: false,
2024-12-25 20:04:51 -05:00
description: "Show header above input device selector"
},
showVideoDeviceHeader: {
type: OptionType.BOOLEAN,
default: false,
description: "Show header above camera selector"
2024-12-25 13:00:52 -05:00
},
}),
renderVoiceSettings() { return <VoiceSettings />; },
patches: [
{
find: "this.renderChannelButtons()",
replacement: {
match: /this.renderChannelButtons\(\)/,
replace: "this.renderChannelButtons(), $self.renderVoiceSettings()"
}
}
]
});