You can not select more than 25 topics
Topics must start with a letter or number, can include dashes ('-') and can be up to 35 characters long.
202 lines
7.2 KiB
202 lines
7.2 KiB
import React from 'react';
|
|
import { ipcRenderer, shell } from 'electron';
|
|
import { Icon, RadioGroup } from 'rsuite';
|
|
import { Trans, useTranslation } from 'react-i18next';
|
|
import { ConfigOptionDescription, ConfigPanel } from '../styled';
|
|
import {
|
|
StyledInput,
|
|
StyledInputGroup,
|
|
StyledInputGroupButton,
|
|
StyledInputNumber,
|
|
StyledLink,
|
|
StyledRadio,
|
|
StyledToggle,
|
|
} from '../../shared/styled';
|
|
import { useAppDispatch, useAppSelector } from '../../../redux/hooks';
|
|
import { setDiscord, setOBS } from '../../../redux/configSlice';
|
|
import ConfigOption from '../ConfigOption';
|
|
import { Server } from '../../../types';
|
|
import { settings } from '../../shared/setDefaultSettings';
|
|
|
|
const ExternalConfig = ({ bordered }: any) => {
|
|
const { t } = useTranslation();
|
|
const dispatch = useAppDispatch();
|
|
const config = useAppSelector((state) => state.config);
|
|
|
|
return (
|
|
<ConfigPanel bordered={bordered} header={t('External')}>
|
|
<ConfigOptionDescription>
|
|
{t('Config for integration with external programs.')}
|
|
</ConfigOptionDescription>
|
|
<ConfigPanel header="Discord" collapsible $noBackground>
|
|
<ConfigOption
|
|
name={t('Rich Presence')}
|
|
description={t(
|
|
"Integrates with Discord's rich presence to display the currently playing song as your status."
|
|
)}
|
|
option={
|
|
<StyledToggle
|
|
defaultChecked={config.external.discord.enabled}
|
|
checked={config.external.discord.enabled}
|
|
disabled={config.external.discord.clientId.length !== 18}
|
|
onChange={(e: boolean) => {
|
|
settings.set('discord.enabled', e);
|
|
dispatch(setDiscord({ ...config.external.discord, enabled: e }));
|
|
}}
|
|
/>
|
|
}
|
|
/>
|
|
<ConfigOption
|
|
name={t('Discord Client Id')}
|
|
description={
|
|
<Trans t={t}>
|
|
The client/application Id of the Sonixd discord application. To use your own, create
|
|
one on the{' '}
|
|
<StyledLink
|
|
onClick={() => shell.openPath('https://discord.com/developers/applications')}
|
|
>
|
|
developer application portal
|
|
</StyledLink>
|
|
. The large icon uses the name "icon". Default is 923372440934055968.
|
|
</Trans>
|
|
}
|
|
option={
|
|
<StyledInput
|
|
placeholder={t('Client/Application Id')}
|
|
value={config.external.discord.clientId}
|
|
disabled={config.external.discord.enabled}
|
|
onChange={(e: boolean) => {
|
|
settings.set('discord.clientId', e);
|
|
dispatch(setDiscord({ ...config.external.discord, clientId: e }));
|
|
}}
|
|
/>
|
|
}
|
|
/>
|
|
{config.serverType === Server.Jellyfin && (
|
|
<ConfigOption
|
|
name={t('Display Song Images')}
|
|
description={
|
|
<Trans t={t}>
|
|
Uses the song image returned by your server (only works if your server is publically
|
|
accessible).
|
|
</Trans>
|
|
}
|
|
option={
|
|
<StyledToggle
|
|
defaultChecked={config.external.discord.serverImage}
|
|
checked={config.external.discord.serverImage}
|
|
onChange={(e: boolean) => {
|
|
settings.set('discord.serverImage', e);
|
|
dispatch(setDiscord({ ...config.external.discord, serverImage: e }));
|
|
}}
|
|
/>
|
|
}
|
|
/>
|
|
)}
|
|
</ConfigPanel>
|
|
<ConfigPanel header="OBS (Open Broadcaster Software)" collapsible $noBackground>
|
|
<ConfigOption
|
|
name={
|
|
<>
|
|
{t('Scrobbling')}
|
|
<RadioGroup
|
|
inline
|
|
defaultValue={config.external.obs.type}
|
|
value={config.external.obs.type}
|
|
onChange={(e: string) => {
|
|
settings.set('obs.type', e);
|
|
dispatch(setOBS({ ...config.external.obs, type: e }));
|
|
}}
|
|
>
|
|
<StyledRadio value="local">{t('Local')}</StyledRadio>
|
|
<StyledRadio value="web">{t('Web')}</StyledRadio>
|
|
</RadioGroup>
|
|
</>
|
|
}
|
|
description={t(
|
|
"If local, scrobbles the currently playing song to local .txt files. If web, scrobbles the currently playing song to Tuna plugin's webserver."
|
|
)}
|
|
option={
|
|
<>
|
|
<StyledToggle
|
|
defaultChecked={config.external.obs.enabled}
|
|
checked={config.external.obs.enabled}
|
|
onChange={(e: boolean) => {
|
|
settings.set('obs.enabled', e);
|
|
dispatch(setOBS({ ...config.external.obs, enabled: e }));
|
|
}}
|
|
/>
|
|
</>
|
|
}
|
|
/>
|
|
<ConfigOption
|
|
name={t('Polling Interval')}
|
|
description={t(
|
|
'The number in milliseconds (ms) between each poll when metadata is sent.'
|
|
)}
|
|
option={
|
|
<StyledInputNumber
|
|
defaultValue={config.external.obs.pollingInterval}
|
|
value={config.external.obs.pollingInterval}
|
|
step={1}
|
|
min={100}
|
|
max={25000}
|
|
width={125}
|
|
onChange={(e: number) => {
|
|
settings.set('obs.pollingInterval', e);
|
|
dispatch(setOBS({ ...config.external.obs, pollingInterval: e }));
|
|
}}
|
|
/>
|
|
}
|
|
/>
|
|
|
|
{config.external.obs.type === 'web' ? (
|
|
<ConfigOption
|
|
name={t('Tuna Webserver Url')}
|
|
description={t('The full URL to the Tuna webserver.')}
|
|
option={
|
|
<StyledInput
|
|
width={200}
|
|
placeholder="http://localhost:1608"
|
|
value={config.external.obs.url}
|
|
onChange={(e: string) => {
|
|
settings.set('obs.url', e);
|
|
dispatch(setOBS({ ...config.external.obs, url: e }));
|
|
}}
|
|
/>
|
|
}
|
|
/>
|
|
) : (
|
|
<ConfigOption
|
|
name={t('File Path')}
|
|
description={t('The full path to the directory where song metadata will be created.')}
|
|
option={
|
|
<StyledInputGroup>
|
|
<StyledInput disabled width={200} value={config.external.obs.path} />
|
|
<StyledInputGroupButton
|
|
onClick={() => {
|
|
ipcRenderer
|
|
.invoke('file-path')
|
|
.then((path) => {
|
|
if (path) {
|
|
settings.set('obs.path', path[0]);
|
|
dispatch(setOBS({ ...config.external.obs, path: path[0] }));
|
|
}
|
|
|
|
return null;
|
|
})
|
|
.catch((err) => console.log(err));
|
|
}}
|
|
>
|
|
<Icon icon="folder-open" />
|
|
</StyledInputGroupButton>
|
|
</StyledInputGroup>
|
|
}
|
|
/>
|
|
)}
|
|
</ConfigPanel>
|
|
</ConfigPanel>
|
|
);
|
|
};
|
|
|
|
export default ExternalConfig;
|
|
|