Tunio Desktop client
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

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;