2026-04-11 15:59:14 +08:00
|
|
|
<script setup lang="ts">
|
|
|
|
|
import { ref, onMounted, computed } from 'vue'
|
|
|
|
|
import { NModal, NForm, NFormItem, NInput, NButton, NSelect, NInputNumber, useMessage } from 'naive-ui'
|
|
|
|
|
import { useJobsStore } from '@/stores/jobs'
|
2026-04-13 15:15:14 +08:00
|
|
|
import { useI18n } from 'vue-i18n'
|
|
|
|
|
|
|
|
|
|
const { t } = useI18n()
|
2026-04-11 15:59:14 +08:00
|
|
|
|
|
|
|
|
const props = defineProps<{
|
|
|
|
|
jobId: string | null
|
|
|
|
|
}>()
|
|
|
|
|
|
|
|
|
|
const emit = defineEmits<{
|
|
|
|
|
close: []
|
|
|
|
|
saved: []
|
|
|
|
|
}>()
|
|
|
|
|
|
|
|
|
|
const jobsStore = useJobsStore()
|
|
|
|
|
const message = useMessage()
|
|
|
|
|
|
|
|
|
|
const showModal = ref(true)
|
|
|
|
|
const loading = ref(false)
|
|
|
|
|
|
|
|
|
|
const formData = ref({
|
|
|
|
|
name: '',
|
|
|
|
|
schedule: '',
|
|
|
|
|
prompt: '',
|
|
|
|
|
deliver: 'origin',
|
|
|
|
|
repeat_times: null as number | null,
|
|
|
|
|
})
|
|
|
|
|
|
|
|
|
|
const presetValue = ref<string | null>(null)
|
|
|
|
|
|
|
|
|
|
const isEdit = computed(() => !!props.jobId)
|
|
|
|
|
|
2026-04-13 15:15:14 +08:00
|
|
|
const schedulePresets = computed(() => [
|
|
|
|
|
{ label: t('jobs.presetEveryMinute'), value: '* * * * *' },
|
|
|
|
|
{ label: t('jobs.presetEvery5Min'), value: '*/5 * * * *' },
|
|
|
|
|
{ label: t('jobs.presetEveryHour'), value: '0 * * * *' },
|
|
|
|
|
{ label: t('jobs.presetEveryDay'), value: '0 0 * * *' },
|
|
|
|
|
{ label: t('jobs.presetEveryDay9'), value: '0 9 * * *' },
|
|
|
|
|
{ label: t('jobs.presetEveryMonday'), value: '0 9 * * 1' },
|
|
|
|
|
{ label: t('jobs.presetEveryMonth'), value: '0 9 1 * *' },
|
|
|
|
|
])
|
|
|
|
|
|
|
|
|
|
const targetOptions = computed(() => [
|
|
|
|
|
{ label: t('jobs.origin'), value: 'origin' },
|
|
|
|
|
{ label: t('jobs.local'), value: 'local' },
|
|
|
|
|
])
|
2026-04-11 15:59:14 +08:00
|
|
|
|
|
|
|
|
onMounted(async () => {
|
|
|
|
|
if (props.jobId) {
|
|
|
|
|
try {
|
|
|
|
|
const { getJob } = await import('@/api/jobs')
|
|
|
|
|
const job = await getJob(props.jobId)
|
|
|
|
|
formData.value = {
|
|
|
|
|
name: job.name,
|
|
|
|
|
schedule: typeof job.schedule === 'string' ? job.schedule : (job.schedule?.expr || job.schedule_display || ''),
|
|
|
|
|
prompt: job.prompt,
|
|
|
|
|
deliver: job.deliver || 'origin',
|
|
|
|
|
repeat_times: typeof job.repeat === 'number' ? job.repeat : (typeof job.repeat === 'object' ? job.repeat.times : null),
|
|
|
|
|
}
|
|
|
|
|
} catch (e: any) {
|
2026-04-13 15:15:14 +08:00
|
|
|
message.error(t('jobs.loadFailed') + ': ' + e.message)
|
2026-04-11 15:59:14 +08:00
|
|
|
}
|
|
|
|
|
}
|
|
|
|
|
})
|
|
|
|
|
|
|
|
|
|
async function handleSave() {
|
|
|
|
|
if (!formData.value.name.trim()) {
|
2026-04-13 15:15:14 +08:00
|
|
|
message.warning(t('jobs.nameRequired'))
|
2026-04-11 15:59:14 +08:00
|
|
|
return
|
|
|
|
|
}
|
|
|
|
|
if (!formData.value.schedule.trim()) {
|
2026-04-13 15:15:14 +08:00
|
|
|
message.warning(t('jobs.scheduleRequired'))
|
2026-04-11 15:59:14 +08:00
|
|
|
return
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
loading.value = true
|
|
|
|
|
try {
|
|
|
|
|
const payload = {
|
|
|
|
|
name: formData.value.name,
|
|
|
|
|
schedule: formData.value.schedule,
|
|
|
|
|
prompt: formData.value.prompt,
|
|
|
|
|
deliver: formData.value.deliver,
|
|
|
|
|
repeat: formData.value.repeat_times ?? undefined,
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
if (isEdit.value) {
|
|
|
|
|
await jobsStore.updateJob(props.jobId!, payload)
|
2026-04-13 15:15:14 +08:00
|
|
|
message.success(t('jobs.jobUpdated'))
|
2026-04-11 15:59:14 +08:00
|
|
|
} else {
|
|
|
|
|
await jobsStore.createJob(payload)
|
2026-04-13 15:15:14 +08:00
|
|
|
message.success(t('jobs.jobCreated'))
|
2026-04-11 15:59:14 +08:00
|
|
|
}
|
|
|
|
|
emit('saved')
|
|
|
|
|
} catch (e: any) {
|
|
|
|
|
message.error(e.message)
|
|
|
|
|
} finally {
|
|
|
|
|
loading.value = false
|
|
|
|
|
}
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
function handleClose() {
|
|
|
|
|
showModal.value = false
|
|
|
|
|
setTimeout(() => emit('close'), 200)
|
|
|
|
|
}
|
|
|
|
|
</script>
|
|
|
|
|
|
|
|
|
|
<template>
|
|
|
|
|
<NModal
|
|
|
|
|
v-model:show="showModal"
|
|
|
|
|
preset="card"
|
2026-04-13 15:15:14 +08:00
|
|
|
:title="isEdit ? t('jobs.editJob') : t('jobs.createJob')"
|
2026-04-15 09:12:54 +08:00
|
|
|
:style="{ width: 'min(520px, calc(100vw - 32px))' }"
|
2026-04-11 15:59:14 +08:00
|
|
|
:mask-closable="!loading"
|
|
|
|
|
@after-leave="emit('close')"
|
|
|
|
|
>
|
|
|
|
|
<NForm label-placement="top">
|
2026-04-13 15:15:14 +08:00
|
|
|
<NFormItem :label="t('jobs.name')" required>
|
2026-04-11 15:59:14 +08:00
|
|
|
<NInput
|
|
|
|
|
v-model:value="formData.name"
|
2026-04-13 15:15:14 +08:00
|
|
|
:placeholder="t('jobs.namePlaceholder')"
|
2026-04-11 15:59:14 +08:00
|
|
|
maxlength="200"
|
|
|
|
|
show-count
|
|
|
|
|
/>
|
|
|
|
|
</NFormItem>
|
|
|
|
|
|
2026-04-13 15:15:14 +08:00
|
|
|
<NFormItem :label="t('jobs.schedule')" required>
|
2026-04-11 15:59:14 +08:00
|
|
|
<NInput
|
|
|
|
|
v-model:value="formData.schedule"
|
2026-04-13 15:15:14 +08:00
|
|
|
:placeholder="t('jobs.schedulePlaceholder')"
|
2026-04-11 15:59:14 +08:00
|
|
|
/>
|
|
|
|
|
</NFormItem>
|
|
|
|
|
|
2026-04-13 15:15:14 +08:00
|
|
|
<NFormItem :label="t('jobs.quickPresets')">
|
2026-04-11 15:59:14 +08:00
|
|
|
<NSelect
|
|
|
|
|
v-model:value="presetValue"
|
|
|
|
|
:options="schedulePresets"
|
2026-04-13 15:15:14 +08:00
|
|
|
:placeholder="t('jobs.selectPreset')"
|
2026-04-11 15:59:14 +08:00
|
|
|
@update:value="v => formData.schedule = v"
|
|
|
|
|
/>
|
|
|
|
|
</NFormItem>
|
|
|
|
|
|
2026-04-13 15:15:14 +08:00
|
|
|
<NFormItem :label="t('jobs.prompt')" required>
|
2026-04-11 15:59:14 +08:00
|
|
|
<NInput
|
|
|
|
|
v-model:value="formData.prompt"
|
|
|
|
|
type="textarea"
|
2026-04-13 15:15:14 +08:00
|
|
|
:placeholder="t('jobs.promptPlaceholder')"
|
2026-04-11 15:59:14 +08:00
|
|
|
:rows="4"
|
|
|
|
|
maxlength="5000"
|
|
|
|
|
show-count
|
|
|
|
|
/>
|
|
|
|
|
</NFormItem>
|
|
|
|
|
|
2026-04-13 15:15:14 +08:00
|
|
|
<NFormItem :label="t('jobs.deliverTarget')">
|
2026-04-11 15:59:14 +08:00
|
|
|
<NSelect
|
|
|
|
|
v-model:value="formData.deliver"
|
|
|
|
|
:options="targetOptions"
|
|
|
|
|
/>
|
|
|
|
|
</NFormItem>
|
|
|
|
|
|
2026-04-13 15:15:14 +08:00
|
|
|
<NFormItem :label="t('jobs.repeatCount')">
|
2026-04-11 15:59:14 +08:00
|
|
|
<NInputNumber
|
|
|
|
|
v-model:value="formData.repeat_times"
|
|
|
|
|
:min="1"
|
2026-04-13 15:15:14 +08:00
|
|
|
:placeholder="t('jobs.repeatPlaceholder')"
|
2026-04-11 15:59:14 +08:00
|
|
|
clearable
|
|
|
|
|
style="width: 100%"
|
|
|
|
|
/>
|
|
|
|
|
</NFormItem>
|
|
|
|
|
</NForm>
|
|
|
|
|
|
|
|
|
|
<template #footer>
|
|
|
|
|
<div class="modal-footer">
|
2026-04-13 15:15:14 +08:00
|
|
|
<NButton @click="handleClose">{{ t('common.cancel') }}</NButton>
|
2026-04-11 15:59:14 +08:00
|
|
|
<NButton type="primary" :loading="loading" @click="handleSave">
|
2026-04-13 15:15:14 +08:00
|
|
|
{{ isEdit ? t('common.update') : t('common.create') }}
|
2026-04-11 15:59:14 +08:00
|
|
|
</NButton>
|
|
|
|
|
</div>
|
|
|
|
|
</template>
|
|
|
|
|
</NModal>
|
|
|
|
|
</template>
|
|
|
|
|
|
|
|
|
|
<style scoped lang="scss">
|
|
|
|
|
.modal-footer {
|
|
|
|
|
display: flex;
|
|
|
|
|
justify-content: flex-end;
|
|
|
|
|
gap: 8px;
|
|
|
|
|
}
|
|
|
|
|
</style>
|