150 lines
6.3 KiB
TypeScript
150 lines
6.3 KiB
TypeScript
import React, { useContext, useEffect, useState } from 'react';
|
|
import { Schedule } from '@/api/server/schedules/getServerSchedules';
|
|
import Field from '@/components/elements/Field';
|
|
import { Form, Formik, FormikHelpers } from 'formik';
|
|
import FormikSwitch from '@/components/elements/FormikSwitch';
|
|
import createOrUpdateSchedule from '@/api/server/schedules/createOrUpdateSchedule';
|
|
import { ServerContext } from '@/state/server';
|
|
import { httpErrorToHuman } from '@/api/http';
|
|
import FlashMessageRender from '@/components/FlashMessageRender';
|
|
import useFlash from '@/plugins/useFlash';
|
|
import tw from 'twin.macro';
|
|
import { Button } from '@/components/elements/button/index';
|
|
import ModalContext from '@/context/ModalContext';
|
|
import asModal from '@/hoc/asModal';
|
|
import Switch from '@/components/elements/Switch';
|
|
import ScheduleCheatsheetCards from '@/components/server/schedules/ScheduleCheatsheetCards';
|
|
|
|
interface Props {
|
|
schedule?: Schedule;
|
|
}
|
|
|
|
interface Values {
|
|
name: string;
|
|
dayOfWeek: string;
|
|
month: string;
|
|
dayOfMonth: string;
|
|
hour: string;
|
|
minute: string;
|
|
enabled: boolean;
|
|
onlyWhenOnline: boolean;
|
|
}
|
|
|
|
const EditScheduleModal = ({ schedule }: Props) => {
|
|
const { addError, clearFlashes } = useFlash();
|
|
const { dismiss } = useContext(ModalContext);
|
|
|
|
const uuid = ServerContext.useStoreState((state) => state.server.data!.uuid);
|
|
const appendSchedule = ServerContext.useStoreActions((actions) => actions.schedules.appendSchedule);
|
|
const [showCheatsheet, setShowCheetsheet] = useState(false);
|
|
|
|
useEffect(() => {
|
|
return () => {
|
|
clearFlashes('schedule:edit');
|
|
};
|
|
}, []);
|
|
|
|
const submit = (values: Values, { setSubmitting }: FormikHelpers<Values>) => {
|
|
clearFlashes('schedule:edit');
|
|
createOrUpdateSchedule(uuid, {
|
|
id: schedule?.id,
|
|
name: values.name,
|
|
cron: {
|
|
minute: values.minute,
|
|
hour: values.hour,
|
|
dayOfWeek: values.dayOfWeek,
|
|
month: values.month,
|
|
dayOfMonth: values.dayOfMonth,
|
|
},
|
|
onlyWhenOnline: values.onlyWhenOnline,
|
|
isActive: values.enabled,
|
|
})
|
|
.then((schedule) => {
|
|
setSubmitting(false);
|
|
appendSchedule(schedule);
|
|
dismiss();
|
|
})
|
|
.catch((error) => {
|
|
console.error(error);
|
|
|
|
setSubmitting(false);
|
|
addError({ key: 'schedule:edit', message: httpErrorToHuman(error) });
|
|
});
|
|
};
|
|
|
|
return (
|
|
<Formik
|
|
onSubmit={submit}
|
|
initialValues={
|
|
{
|
|
name: schedule?.name || '',
|
|
minute: schedule?.cron.minute || '*/5',
|
|
hour: schedule?.cron.hour || '*',
|
|
dayOfMonth: schedule?.cron.dayOfMonth || '*',
|
|
month: schedule?.cron.month || '*',
|
|
dayOfWeek: schedule?.cron.dayOfWeek || '*',
|
|
enabled: schedule?.isActive ?? true,
|
|
onlyWhenOnline: schedule?.onlyWhenOnline ?? true,
|
|
} as Values
|
|
}
|
|
>
|
|
{({ isSubmitting }) => (
|
|
<Form>
|
|
<h3 css={tw`text-2xl mb-6`}>{schedule ? 'Edit schedule' : 'Create new schedule'}</h3>
|
|
<FlashMessageRender byKey={'schedule:edit'} css={tw`mb-6`} />
|
|
<Field
|
|
name={'name'}
|
|
label={'Schedule name'}
|
|
description={'A human readable identifier for this schedule.'}
|
|
/>
|
|
<div css={tw`grid grid-cols-2 sm:grid-cols-5 gap-4 mt-6`}>
|
|
<Field name={'minute'} label={'Minute'} />
|
|
<Field name={'hour'} label={'Hour'} />
|
|
<Field name={'dayOfMonth'} label={'Day of month'} />
|
|
<Field name={'month'} label={'Month'} />
|
|
<Field name={'dayOfWeek'} label={'Day of week'} />
|
|
</div>
|
|
<p css={tw`text-neutral-400 text-xs mt-2`}>
|
|
The schedule system supports the use of Cronjob syntax when defining when tasks should begin
|
|
running. Use the fields above to specify when these tasks should begin running.
|
|
</p>
|
|
<div css={tw`mt-6 bg-neutral-700 border border-neutral-800 shadow-inner p-4 rounded`}>
|
|
<Switch
|
|
name={'show_cheatsheet'}
|
|
description={'Show the cron cheatsheet for some examples.'}
|
|
label={'Show Cheatsheet'}
|
|
defaultChecked={showCheatsheet}
|
|
onChange={() => setShowCheetsheet((s) => !s)}
|
|
/>
|
|
{showCheatsheet && (
|
|
<div css={tw`block md:flex w-full`}>
|
|
<ScheduleCheatsheetCards />
|
|
</div>
|
|
)}
|
|
</div>
|
|
<div css={tw`mt-6 bg-neutral-700 border border-neutral-800 shadow-inner p-4 rounded`}>
|
|
<FormikSwitch
|
|
name={'onlyWhenOnline'}
|
|
description={'Only execute this schedule when the server is in a running state.'}
|
|
label={'Only When Server Is Online'}
|
|
/>
|
|
</div>
|
|
<div css={tw`mt-6 bg-neutral-700 border border-neutral-800 shadow-inner p-4 rounded`}>
|
|
<FormikSwitch
|
|
name={'enabled'}
|
|
description={'This schedule will be executed automatically if enabled.'}
|
|
label={'Schedule Enabled'}
|
|
/>
|
|
</div>
|
|
<div css={tw`mt-6 text-right`}>
|
|
<Button className={'w-full sm:w-auto'} type={'submit'} disabled={isSubmitting}>
|
|
{schedule ? 'Save changes' : 'Create schedule'}
|
|
</Button>
|
|
</div>
|
|
</Form>
|
|
)}
|
|
</Formik>
|
|
);
|
|
};
|
|
|
|
export default asModal<Props>()(EditScheduleModal);
|