bazarr/frontend/src/pages/Settings/components/Layout.tsx

122 lines
3.5 KiB
TypeScript
Raw Normal View History

import { useSettingsMutation, useSystemSettings } from "@/apis/hooks";
2022-05-31 15:49:04 +00:00
import { Toolbox } from "@/components";
import { LoadingProvider } from "@/contexts";
import { useOnValueChange } from "@/utilities";
import { LOG } from "@/utilities/console";
import { useUpdateLocalStorage } from "@/utilities/storage";
2021-03-25 14:22:43 +00:00
import { faSave } from "@fortawesome/free-solid-svg-icons";
2022-05-31 15:49:04 +00:00
import { Container, Group, LoadingOverlay } from "@mantine/core";
import { useDocumentTitle, useForm } from "@mantine/hooks";
import { FunctionComponent, ReactNode, useCallback, useMemo } from "react";
2021-03-25 14:22:43 +00:00
import {
enabledLanguageKey,
languageProfileKey,
notificationsKey,
} from "../keys";
2022-05-31 15:49:04 +00:00
import { FormContext, FormValues } from "../utilities/FormValues";
import { SettingsProvider } from "../utilities/SettingsProvider";
2021-03-25 14:22:43 +00:00
function submitHooks(settings: LooseObject) {
if (languageProfileKey in settings) {
const item = settings[languageProfileKey];
settings[languageProfileKey] = JSON.stringify(item);
}
if (enabledLanguageKey in settings) {
const item = settings[enabledLanguageKey] as Language.Info[];
2021-03-25 14:22:43 +00:00
settings[enabledLanguageKey] = item.map((v) => v.code2);
}
if (notificationsKey in settings) {
const item = settings[notificationsKey] as Settings.NotificationInfo[];
settings[notificationsKey] = item.map((v) => JSON.stringify(v));
}
}
interface Props {
name: string;
2022-05-31 15:49:04 +00:00
children: ReactNode;
2021-03-25 14:22:43 +00:00
}
const Layout: FunctionComponent<Props> = (props) => {
const { children, name } = props;
2021-03-25 14:22:43 +00:00
2022-05-31 15:49:04 +00:00
const { data: settings, isLoading, isRefetching } = useSystemSettings();
const { mutate, isLoading: isMutating } = useSettingsMutation();
2022-05-31 15:49:04 +00:00
const form = useForm<FormValues>({
initialValues: {
settings: {},
storages: {},
},
2022-05-31 15:49:04 +00:00
});
2021-03-25 14:22:43 +00:00
2022-05-31 15:49:04 +00:00
useOnValueChange(isRefetching, (value) => {
if (!value) {
form.reset();
}
});
2021-03-25 14:22:43 +00:00
2022-05-31 15:49:04 +00:00
const updateStorage = useUpdateLocalStorage();
2021-03-25 14:22:43 +00:00
2022-05-31 15:49:04 +00:00
const submit = useCallback(
(values: FormValues) => {
const { settings, storages } = values;
2021-03-25 14:22:43 +00:00
2022-05-31 15:49:04 +00:00
if (Object.keys(settings).length > 0) {
const settingsToSubmit = { ...settings };
submitHooks(settingsToSubmit);
LOG("info", "submitting settings", settingsToSubmit);
mutate(settingsToSubmit);
2021-03-25 14:22:43 +00:00
}
2022-05-31 15:49:04 +00:00
if (Object.keys(storages).length > 0) {
const storagesToSubmit = { ...storages };
LOG("info", "submitting storages", storagesToSubmit);
updateStorage(storagesToSubmit);
2021-03-25 14:22:43 +00:00
}
2022-05-31 15:49:04 +00:00
},
[mutate, updateStorage]
);
2021-03-25 14:22:43 +00:00
2022-05-31 15:49:04 +00:00
const totalStagedCount = useMemo(() => {
const object = { ...form.values.settings, ...form.values.storages };
2021-03-25 14:22:43 +00:00
2022-05-31 15:49:04 +00:00
return Object.keys(object).length;
}, [form.values.settings, form.values.storages]);
useDocumentTitle(`${name} - Bazarr (Settings)`);
2021-03-25 14:22:43 +00:00
2022-05-31 15:49:04 +00:00
if (settings === undefined) {
return <LoadingOverlay visible></LoadingOverlay>;
}
2021-03-25 14:22:43 +00:00
return (
2022-05-31 15:49:04 +00:00
<SettingsProvider value={settings}>
<LoadingProvider value={isLoading || isMutating}>
<form onSubmit={form.onSubmit(submit)}>
<Toolbox>
<Group>
<Toolbox.Button
type="submit"
icon={faSave}
loading={isMutating}
disabled={totalStagedCount === 0}
>
Save
</Toolbox.Button>
</Group>
</Toolbox>
<FormContext.Provider value={form}>
<Container size="xl" mx={0}>
{children}
</Container>
</FormContext.Provider>
</form>
</LoadingProvider>
</SettingsProvider>
2021-03-25 14:22:43 +00:00
);
};
export default Layout;