165 lines
5.5 KiB
TypeScript
165 lines
5.5 KiB
TypeScript
import { useState, useEffect, useRef } from 'react';
|
|
import { useTheme } from '@emotion/react';
|
|
import { useLocation } from 'react-router-dom';
|
|
import { useTranslation } from 'react-i18next';
|
|
import dayjs from 'dayjs';
|
|
|
|
import { ToggleField, SelectField } from 'components';
|
|
|
|
import { useSettingsStore, useLocaleUpdateStore } from 'stores';
|
|
|
|
import {
|
|
OpenButton,
|
|
Modal,
|
|
Heading,
|
|
Cover,
|
|
} from './settingsStyle';
|
|
|
|
import locales from 'res/dayjs_locales';
|
|
|
|
// Language specific options
|
|
const setDefaults = (lang, store) => {
|
|
if (locales.hasOwnProperty(lang)) {
|
|
store.setWeekStart(locales[lang].weekStart);
|
|
store.setTimeFormat(locales[lang].timeFormat);
|
|
}
|
|
};
|
|
|
|
const Settings = () => {
|
|
const { pathname } = useLocation();
|
|
const theme = useTheme();
|
|
const store = useSettingsStore();
|
|
const [isOpen, _setIsOpen] = useState(false);
|
|
const { t, i18n } = useTranslation('common');
|
|
const setLocale = useLocaleUpdateStore(state => state.setLocale);
|
|
const firstControlRef = useRef();
|
|
|
|
const onEsc = e => {
|
|
if (e.key === 'Escape') {
|
|
setIsOpen(false);
|
|
}
|
|
};
|
|
|
|
const setIsOpen = open => {
|
|
_setIsOpen(open);
|
|
|
|
if (open) {
|
|
window.setTimeout(() => firstControlRef.current.focus(), 150);
|
|
document.addEventListener('keyup', onEsc, true);
|
|
} else {
|
|
document.removeEventListener('keyup', onEsc);
|
|
}
|
|
};
|
|
|
|
useEffect(() => {
|
|
if (Object.keys(locales).includes(i18n.language)) {
|
|
locales[i18n.language].import().then(() => {
|
|
dayjs.locale(i18n.language);
|
|
setLocale(dayjs.locale());
|
|
document.documentElement.setAttribute('lang', i18n.language);
|
|
});
|
|
} else {
|
|
setLocale('en');
|
|
document.documentElement.setAttribute('lang', 'en')
|
|
}
|
|
}, [i18n.language, setLocale]);
|
|
|
|
if (!i18n.options.storedLang) {
|
|
setDefaults(i18n.language, store);
|
|
i18n.options.storedLang = i18n.language;
|
|
}
|
|
|
|
i18n.on('languageChanged', lang => {
|
|
setDefaults(lang, store);
|
|
});
|
|
|
|
// Reset scroll on navigation
|
|
useEffect(() => window.scrollTo(0, 0), [pathname]);
|
|
|
|
return (
|
|
<>
|
|
<OpenButton
|
|
isOpen={isOpen}
|
|
type="button"
|
|
onClick={() => setIsOpen(!isOpen)} title={t('options.name')}
|
|
>
|
|
<svg xmlns="http://www.w3.org/2000/svg" width="24" height="24" viewBox="0 0 24 24" fill="none" stroke={theme.text} strokeWidth="2" strokeLinecap="round" strokeLinejoin="round"><circle cx="12" cy="12" r="3"></circle><path d="M19.4 15a1.65 1.65 0 0 0 .33 1.82l.06.06a2 2 0 0 1 0 2.83 2 2 0 0 1-2.83 0l-.06-.06a1.65 1.65 0 0 0-1.82-.33 1.65 1.65 0 0 0-1 1.51V21a2 2 0 0 1-2 2 2 2 0 0 1-2-2v-.09A1.65 1.65 0 0 0 9 19.4a1.65 1.65 0 0 0-1.82.33l-.06.06a2 2 0 0 1-2.83 0 2 2 0 0 1 0-2.83l.06-.06a1.65 1.65 0 0 0 .33-1.82 1.65 1.65 0 0 0-1.51-1H3a2 2 0 0 1-2-2 2 2 0 0 1 2-2h.09A1.65 1.65 0 0 0 4.6 9a1.65 1.65 0 0 0-.33-1.82l-.06-.06a2 2 0 0 1 0-2.83 2 2 0 0 1 2.83 0l.06.06a1.65 1.65 0 0 0 1.82.33H9a1.65 1.65 0 0 0 1-1.51V3a2 2 0 0 1 2-2 2 2 0 0 1 2 2v.09a1.65 1.65 0 0 0 1 1.51 1.65 1.65 0 0 0 1.82-.33l.06-.06a2 2 0 0 1 2.83 0 2 2 0 0 1 0 2.83l-.06.06a1.65 1.65 0 0 0-.33 1.82V9a1.65 1.65 0 0 0 1.51 1H21a2 2 0 0 1 2 2 2 2 0 0 1-2 2h-.09a1.65 1.65 0 0 0-1.51 1z"></path></svg>
|
|
</OpenButton>
|
|
|
|
<Cover isOpen={isOpen} onClick={() => setIsOpen(false)} />
|
|
<Modal isOpen={isOpen}>
|
|
<Heading>{t('options.name')}</Heading>
|
|
|
|
<ToggleField
|
|
label={t('options.weekStart.label')}
|
|
name="weekStart"
|
|
id="weekStart"
|
|
options={{
|
|
'Sunday': t('options.weekStart.options.Sunday'),
|
|
'Monday': t('options.weekStart.options.Monday'),
|
|
}}
|
|
value={store.weekStart === 0 ? 'Sunday' : 'Monday'}
|
|
onChange={value => store.setWeekStart(value === 'Sunday' ? 0 : 1)}
|
|
inputRef={firstControlRef}
|
|
/>
|
|
|
|
<ToggleField
|
|
label={t('options.timeFormat.label')}
|
|
name="timeFormat"
|
|
id="timeFormat"
|
|
options={{
|
|
'12h': t('options.timeFormat.options.12h'),
|
|
'24h': t('options.timeFormat.options.24h'),
|
|
}}
|
|
value={store.timeFormat}
|
|
onChange={value => store.setTimeFormat(value)}
|
|
/>
|
|
|
|
<ToggleField
|
|
label={t('options.theme.label')}
|
|
name="theme"
|
|
id="theme"
|
|
options={{
|
|
'System': t('options.theme.options.System'),
|
|
'Light': t('options.theme.options.Light'),
|
|
'Dark': t('options.theme.options.Dark'),
|
|
}}
|
|
value={store.theme}
|
|
onChange={value => store.setTheme(value)}
|
|
/>
|
|
|
|
<ToggleField
|
|
label={t('options.highlight.label')}
|
|
name="highlight"
|
|
id="highlight"
|
|
title={t('options.highlight.title')}
|
|
options={{
|
|
'Off': t('options.highlight.options.Off'),
|
|
'On': t('options.highlight.options.On'),
|
|
}}
|
|
value={store.highlight ? 'On' : 'Off'}
|
|
onChange={value => store.setHighlight(value === 'On')}
|
|
/>
|
|
|
|
<SelectField
|
|
label={t('options.language.label')}
|
|
name="language"
|
|
id="language"
|
|
options={{
|
|
...Object.keys(locales).reduce((ls, l) => {
|
|
ls[l] = locales[l].name;
|
|
return ls;
|
|
}, {}),
|
|
...process.env.NODE_ENV !== 'production' && { 'cimode': 'DEV' },
|
|
}}
|
|
small
|
|
value={i18n.language}
|
|
onChange={event => i18n.changeLanguage(event.target.value)}
|
|
/>
|
|
</Modal>
|
|
</>
|
|
);
|
|
};
|
|
|
|
export default Settings;
|