936 lines
23 KiB
Vue
936 lines
23 KiB
Vue
|
|
<template>
|
||
|
|
<a-modal
|
||
|
|
v-model:open="visible"
|
||
|
|
title="新建排课"
|
||
|
|
:confirm-loading="loading"
|
||
|
|
@ok="handleSubmit"
|
||
|
|
@cancel="handleCancel"
|
||
|
|
width="700px"
|
||
|
|
destroy-on-close
|
||
|
|
>
|
||
|
|
<a-steps :current="currentStep" size="small" class="steps-navigator">
|
||
|
|
<a-step title="选择课程包" />
|
||
|
|
<a-step title="选择课程类型" />
|
||
|
|
<a-step title="选择班级" />
|
||
|
|
<a-step title="设置时间" />
|
||
|
|
</a-steps>
|
||
|
|
|
||
|
|
<div class="step-content">
|
||
|
|
<!-- 步骤1: 选择课程套餐和课程包 -->
|
||
|
|
<div v-show="currentStep === 0" class="step-panel">
|
||
|
|
<h3>选择课程套餐</h3>
|
||
|
|
<a-select
|
||
|
|
v-model:value="formData.collectionId"
|
||
|
|
placeholder="请选择课程套餐"
|
||
|
|
style="width: 100%"
|
||
|
|
show-search
|
||
|
|
:filter-option="filterCollection"
|
||
|
|
@change="handleCollectionChange"
|
||
|
|
>
|
||
|
|
<a-select-option v-for="collection in collections" :key="collection.id" :value="collection.id">
|
||
|
|
<div class="collection-option">
|
||
|
|
<div class="collection-name">{{ collection.name }}</div>
|
||
|
|
<div class="collection-info">{{ collection.packageCount }} 个课程包 · {{ collection.gradeLevels?.join(', ') }}</div>
|
||
|
|
</div>
|
||
|
|
</a-select-option>
|
||
|
|
</a-select>
|
||
|
|
|
||
|
|
<!-- 选择课程包 -->
|
||
|
|
<div v-if="selectedCollection && selectedCollection.packages" class="packages-section">
|
||
|
|
<h4>选择课程包</h4>
|
||
|
|
<div class="packages-grid">
|
||
|
|
<div
|
||
|
|
v-for="pkg in selectedCollection.packages"
|
||
|
|
:key="pkg.id"
|
||
|
|
:class="['package-card', { active: formData.packageId === pkg.id }]"
|
||
|
|
@click="selectPackage(pkg.id)"
|
||
|
|
>
|
||
|
|
<div class="package-name">{{ pkg.name }}</div>
|
||
|
|
<div class="package-grade">{{ pkg.gradeLevels?.join(', ') }}</div>
|
||
|
|
<div class="package-count">{{ pkg.courseCount }} 门课程</div>
|
||
|
|
</div>
|
||
|
|
</div>
|
||
|
|
</div>
|
||
|
|
|
||
|
|
<!-- 选择课程 -->
|
||
|
|
<div v-if="selectedPackage && selectedPackage.courses" class="courses-section">
|
||
|
|
<h4>选择课程</h4>
|
||
|
|
<div class="courses-grid">
|
||
|
|
<div
|
||
|
|
v-for="course in selectedPackage.courses"
|
||
|
|
:key="course.id"
|
||
|
|
:class="['course-card', { active: formData.courseId === course.id }]"
|
||
|
|
@click="selectCourse(course.id)"
|
||
|
|
>
|
||
|
|
<div class="course-name">{{ course.name }}</div>
|
||
|
|
<div class="course-grade">{{ course.gradeLevel }}</div>
|
||
|
|
</div>
|
||
|
|
</div>
|
||
|
|
</div>
|
||
|
|
|
||
|
|
<!-- 排课计划参考 -->
|
||
|
|
<div v-if="scheduleRefData.length > 0" class="schedule-ref-card">
|
||
|
|
<div class="ref-header">
|
||
|
|
<CalendarOutlined class="ref-icon" />
|
||
|
|
<span class="ref-title">排课计划参考</span>
|
||
|
|
</div>
|
||
|
|
<a-table
|
||
|
|
:columns="scheduleRefColumns"
|
||
|
|
:data-source="scheduleRefData"
|
||
|
|
:pagination="false"
|
||
|
|
size="small"
|
||
|
|
bordered
|
||
|
|
>
|
||
|
|
<template #bodyCell="{ column, record }">
|
||
|
|
<template v-if="column.key === 'dayOfWeek'">
|
||
|
|
{{ weekDayNames[record.dayOfWeek] || '-' }}
|
||
|
|
</template>
|
||
|
|
</template>
|
||
|
|
</a-table>
|
||
|
|
</div>
|
||
|
|
</div>
|
||
|
|
|
||
|
|
<!-- 步骤2: 选择课程类型 -->
|
||
|
|
<div v-show="currentStep === 1" class="step-panel">
|
||
|
|
<h3>选择课程类型</h3>
|
||
|
|
<a-alert
|
||
|
|
message="请选择一个课程类型进行排课"
|
||
|
|
type="info"
|
||
|
|
show-icon
|
||
|
|
style="margin-bottom: 16px"
|
||
|
|
/>
|
||
|
|
<a-spin :spinning="loadingLessonTypes">
|
||
|
|
<div class="lesson-type-grid">
|
||
|
|
<div
|
||
|
|
v-for="type in lessonTypes"
|
||
|
|
:key="type.lessonType"
|
||
|
|
:class="['lesson-type-card', { active: formData.lessonType === type.lessonType }]"
|
||
|
|
@click="selectLessonType(type.lessonType)"
|
||
|
|
>
|
||
|
|
<div class="type-icon">{{ getLessonTypeIcon(type.lessonType) }}</div>
|
||
|
|
<div class="type-name">{{ type.lessonTypeName }}</div>
|
||
|
|
<div class="type-count">{{ type.count }} 节课</div>
|
||
|
|
</div>
|
||
|
|
</div>
|
||
|
|
</a-spin>
|
||
|
|
</div>
|
||
|
|
|
||
|
|
<!-- 步骤3: 选择班级并分配教师 -->
|
||
|
|
<div v-show="currentStep === 2" class="step-panel">
|
||
|
|
<h3>选择班级并分配教师</h3>
|
||
|
|
<a-alert
|
||
|
|
message="选择班级后,为每个班级指定授课教师"
|
||
|
|
type="info"
|
||
|
|
show-icon
|
||
|
|
style="margin-bottom: 16px"
|
||
|
|
/>
|
||
|
|
<div class="grade-selector">
|
||
|
|
<a-radio-group v-model:value="selectedGrade" button-style="solid">
|
||
|
|
<a-radio-button value="">全部</a-radio-button>
|
||
|
|
<a-radio-button value="小班">小班</a-radio-button>
|
||
|
|
<a-radio-button value="中班">中班</a-radio-button>
|
||
|
|
<a-radio-button value="大班">大班</a-radio-button>
|
||
|
|
</a-radio-group>
|
||
|
|
</div>
|
||
|
|
<div class="class-teacher-grid">
|
||
|
|
<div
|
||
|
|
v-for="cls in filteredClasses"
|
||
|
|
:key="cls.id"
|
||
|
|
:class="['class-teacher-card', { selected: isClassSelected(cls.id) }]"
|
||
|
|
>
|
||
|
|
<div class="class-header" @click="toggleClass(cls.id)">
|
||
|
|
<a-checkbox :checked="isClassSelected(cls.id)" @click.stop />
|
||
|
|
<div class="class-info">
|
||
|
|
<div class="class-name">{{ cls.name }}</div>
|
||
|
|
<div class="class-detail">{{ cls.studentCount }} 名学生</div>
|
||
|
|
</div>
|
||
|
|
</div>
|
||
|
|
<div v-if="isClassSelected(cls.id)" class="teacher-selector">
|
||
|
|
<a-select
|
||
|
|
v-model:value="classTeacherMap[cls.id]"
|
||
|
|
placeholder="选择教师"
|
||
|
|
style="width: 100%"
|
||
|
|
show-search
|
||
|
|
:filter-option="filterTeacher"
|
||
|
|
>
|
||
|
|
<a-select-option v-for="teacher in teachers" :key="teacher.id" :value="teacher.id">
|
||
|
|
{{ teacher.name }}
|
||
|
|
</a-select-option>
|
||
|
|
</a-select>
|
||
|
|
</div>
|
||
|
|
</div>
|
||
|
|
</div>
|
||
|
|
<div class="selection-summary">
|
||
|
|
已选择 <strong>{{ formData.classIds.length }}</strong> 个班级
|
||
|
|
</div>
|
||
|
|
</div>
|
||
|
|
|
||
|
|
<!-- 步骤4: 设置时间 -->
|
||
|
|
<div v-show="currentStep === 3" class="step-panel">
|
||
|
|
<h3>设置时间</h3>
|
||
|
|
<a-form layout="vertical">
|
||
|
|
<a-form-item label="排课日期" required>
|
||
|
|
<a-date-picker
|
||
|
|
v-model:value="formData.scheduledDate"
|
||
|
|
style="width: 100%"
|
||
|
|
placeholder="选择日期"
|
||
|
|
/>
|
||
|
|
</a-form-item>
|
||
|
|
|
||
|
|
<a-form-item label="时间段" required>
|
||
|
|
<a-time-range-picker
|
||
|
|
v-model:value="formData.scheduledTimeRange"
|
||
|
|
format="HH:mm"
|
||
|
|
style="width: 100%"
|
||
|
|
:placeholder="['开始时间', '结束时间']"
|
||
|
|
/>
|
||
|
|
</a-form-item>
|
||
|
|
</a-form>
|
||
|
|
|
||
|
|
<!-- 确认信息 -->
|
||
|
|
<div class="confirm-info">
|
||
|
|
<a-alert type="info" show-icon>
|
||
|
|
<template #message>
|
||
|
|
<div>将为 <strong>{{ formData.classIds.length }}</strong> 个班级创建排课</div>
|
||
|
|
</template>
|
||
|
|
<template #description>
|
||
|
|
<div>课程类型: {{ getSelectedLessonTypeName() }}</div>
|
||
|
|
<div>排课日期: {{ formData.scheduledDate?.format('YYYY-MM-DD') || '-' }}</div>
|
||
|
|
<div>时间段: {{ getSelectedTimeRange() || '-' }}</div>
|
||
|
|
</template>
|
||
|
|
</a-alert>
|
||
|
|
</div>
|
||
|
|
|
||
|
|
<!-- 班级教师分配列表 -->
|
||
|
|
<div v-if="formData.classIds.length > 0" class="class-teacher-list">
|
||
|
|
<h4>班级教师分配</h4>
|
||
|
|
<a-list :data-source="getSelectedClassesWithTeachers()" size="small">
|
||
|
|
<template #renderItem="{ item }">
|
||
|
|
<a-list-item>
|
||
|
|
<a-list-item-meta>
|
||
|
|
<template #title>{{ item.className }}</template>
|
||
|
|
</a-list-item-meta>
|
||
|
|
<template #actions>
|
||
|
|
<span :class="['teacher-status', { assigned: item.teacherId }]">
|
||
|
|
{{ item.teacherName || '未分配教师' }}
|
||
|
|
</span>
|
||
|
|
</template>
|
||
|
|
</a-list-item>
|
||
|
|
</template>
|
||
|
|
</a-list>
|
||
|
|
</div>
|
||
|
|
</div>
|
||
|
|
</div>
|
||
|
|
|
||
|
|
<template #footer>
|
||
|
|
<div class="modal-footer">
|
||
|
|
<a-button v-if="currentStep > 0" @click="prevStep">上一步</a-button>
|
||
|
|
<a-button v-if="currentStep < 3" type="primary" @click="nextStep">下一步</a-button>
|
||
|
|
<a-button v-else type="primary" :loading="loading" @click="handleSubmit">创建排课</a-button>
|
||
|
|
<a-button @click="handleCancel">取消</a-button>
|
||
|
|
</div>
|
||
|
|
</template>
|
||
|
|
</a-modal>
|
||
|
|
</template>
|
||
|
|
|
||
|
|
<script setup lang="ts">
|
||
|
|
import { ref, reactive, computed, watch } from 'vue';
|
||
|
|
import { message } from 'ant-design-vue';
|
||
|
|
import dayjs, { Dayjs } from 'dayjs';
|
||
|
|
import { CalendarOutlined } from '@ant-design/icons-vue';
|
||
|
|
import {
|
||
|
|
getCourseCollections,
|
||
|
|
getCourseCollectionPackages,
|
||
|
|
getCoursePackageLessonTypes,
|
||
|
|
createSchedulesByClasses,
|
||
|
|
getClasses,
|
||
|
|
getTeachers,
|
||
|
|
type CourseCollection,
|
||
|
|
type CoursePackage,
|
||
|
|
type LessonTypeInfo,
|
||
|
|
type LessonType,
|
||
|
|
type ClassInfo,
|
||
|
|
type Teacher,
|
||
|
|
} from '@/api/school';
|
||
|
|
|
||
|
|
const emit = defineEmits<{
|
||
|
|
(e: 'success'): void;
|
||
|
|
}>();
|
||
|
|
|
||
|
|
const visible = ref(false);
|
||
|
|
const loading = ref(false);
|
||
|
|
const loadingLessonTypes = ref(false);
|
||
|
|
const currentStep = ref(0);
|
||
|
|
|
||
|
|
// 课程套餐列表
|
||
|
|
const collections = ref<CourseCollection[]>([]);
|
||
|
|
const selectedGrade = ref('');
|
||
|
|
|
||
|
|
// 课程类型列表
|
||
|
|
const lessonTypes = ref<LessonTypeInfo[]>([]);
|
||
|
|
|
||
|
|
// 班级列表
|
||
|
|
const classes = ref<ClassInfo[]>([]);
|
||
|
|
const teachers = ref<Teacher[]>([]);
|
||
|
|
|
||
|
|
// 班级教师映射
|
||
|
|
const classTeacherMap = ref<Record<number, number>>({});
|
||
|
|
|
||
|
|
// 排课计划参考数据
|
||
|
|
const scheduleRefData = ref<any[]>([]);
|
||
|
|
|
||
|
|
const scheduleRefColumns = [
|
||
|
|
{ title: '星期', dataIndex: 'dayOfWeek', key: 'dayOfWeek', width: 80 },
|
||
|
|
{ title: '活动安排', dataIndex: 'activity', key: 'activity' },
|
||
|
|
];
|
||
|
|
|
||
|
|
const weekDayNames: Record<number, string> = {
|
||
|
|
1: '周一',
|
||
|
|
2: '周二',
|
||
|
|
3: '周三',
|
||
|
|
4: '周四',
|
||
|
|
5: '周五',
|
||
|
|
6: '周六',
|
||
|
|
0: '周日',
|
||
|
|
};
|
||
|
|
|
||
|
|
// 表单数据
|
||
|
|
interface FormData {
|
||
|
|
collectionId?: number;
|
||
|
|
packageId?: number;
|
||
|
|
courseId?: number;
|
||
|
|
lessonType?: LessonType;
|
||
|
|
classIds: number[];
|
||
|
|
scheduledDate?: Dayjs;
|
||
|
|
scheduledTimeRange?: [Dayjs, Dayjs];
|
||
|
|
}
|
||
|
|
|
||
|
|
const formData = reactive<FormData>({
|
||
|
|
classIds: [],
|
||
|
|
});
|
||
|
|
|
||
|
|
// 计算属性:过滤后的班级列表
|
||
|
|
const filteredClasses = computed(() => {
|
||
|
|
if (!selectedGrade.value) return classes.value;
|
||
|
|
return classes.value.filter(cls => cls.grade === selectedGrade.value);
|
||
|
|
});
|
||
|
|
|
||
|
|
// 计算属性:选中的课程套餐
|
||
|
|
const selectedCollection = computed(() => {
|
||
|
|
if (!formData.collectionId) return null;
|
||
|
|
return collections.value.find(c => c.id === formData.collectionId) || null;
|
||
|
|
});
|
||
|
|
|
||
|
|
// 计算属性:选中的课程包
|
||
|
|
const selectedPackage = computed(() => {
|
||
|
|
if (!formData.packageId || !selectedCollection.value?.packages) return null;
|
||
|
|
return selectedCollection.value.packages.find(p => p.id === formData.packageId) || null;
|
||
|
|
});
|
||
|
|
|
||
|
|
// 打开弹窗
|
||
|
|
const open = () => {
|
||
|
|
visible.value = true;
|
||
|
|
currentStep.value = 0;
|
||
|
|
resetForm();
|
||
|
|
loadCollections();
|
||
|
|
loadClasses();
|
||
|
|
loadTeachers();
|
||
|
|
};
|
||
|
|
|
||
|
|
// 重置表单
|
||
|
|
const resetForm = () => {
|
||
|
|
formData.collectionId = undefined;
|
||
|
|
formData.packageId = undefined;
|
||
|
|
formData.courseId = undefined;
|
||
|
|
formData.lessonType = undefined;
|
||
|
|
formData.classIds = [];
|
||
|
|
formData.scheduledDate = undefined;
|
||
|
|
formData.scheduledTimeRange = undefined;
|
||
|
|
selectedGrade.value = '';
|
||
|
|
scheduleRefData.value = [];
|
||
|
|
lessonTypes.value = [];
|
||
|
|
classTeacherMap.value = {};
|
||
|
|
};
|
||
|
|
|
||
|
|
// 加载课程套餐列表
|
||
|
|
const loadCollections = async () => {
|
||
|
|
try {
|
||
|
|
collections.value = await getCourseCollections();
|
||
|
|
} catch (error) {
|
||
|
|
message.error('加载课程套餐失败');
|
||
|
|
}
|
||
|
|
};
|
||
|
|
|
||
|
|
// 加载班级列表
|
||
|
|
const loadClasses = async () => {
|
||
|
|
try {
|
||
|
|
classes.value = await getClasses();
|
||
|
|
} catch (error) {
|
||
|
|
message.error('加载班级失败');
|
||
|
|
}
|
||
|
|
};
|
||
|
|
|
||
|
|
// 加载教师列表
|
||
|
|
const loadTeachers = async () => {
|
||
|
|
try {
|
||
|
|
const data = await getTeachers({ pageNum: 1, pageSize: 100 });
|
||
|
|
teachers.value = data.list;
|
||
|
|
} catch (error) {
|
||
|
|
message.error('加载教师失败');
|
||
|
|
}
|
||
|
|
};
|
||
|
|
|
||
|
|
// 课程套餐变化 - 加载课程包列表
|
||
|
|
const handleCollectionChange = async (collectionId: number) => {
|
||
|
|
// 重置后续选择
|
||
|
|
formData.packageId = undefined;
|
||
|
|
formData.courseId = undefined;
|
||
|
|
scheduleRefData.value = [];
|
||
|
|
|
||
|
|
if (!collectionId) return;
|
||
|
|
|
||
|
|
try {
|
||
|
|
const packages = await getCourseCollectionPackages(collectionId);
|
||
|
|
// 更新当前套餐的课程包列表
|
||
|
|
const collection = collections.value.find(c => c.id === collectionId);
|
||
|
|
if (collection) {
|
||
|
|
collection.packages = packages;
|
||
|
|
}
|
||
|
|
} catch (error) {
|
||
|
|
message.error('加载课程包失败');
|
||
|
|
}
|
||
|
|
};
|
||
|
|
|
||
|
|
// 选择课程包
|
||
|
|
const selectPackage = async (packageId: number) => {
|
||
|
|
formData.packageId = packageId;
|
||
|
|
formData.courseId = undefined;
|
||
|
|
scheduleRefData.value = [];
|
||
|
|
|
||
|
|
// 加载课程类型列表
|
||
|
|
await loadLessonTypes(packageId);
|
||
|
|
};
|
||
|
|
|
||
|
|
// 选择课程
|
||
|
|
const selectCourse = (courseId: number) => {
|
||
|
|
formData.courseId = courseId;
|
||
|
|
// 加载该课程的排课计划参考
|
||
|
|
loadScheduleRefData(courseId);
|
||
|
|
};
|
||
|
|
|
||
|
|
// 加载排课计划参考数据
|
||
|
|
const loadScheduleRefData = async (courseId: number) => {
|
||
|
|
try {
|
||
|
|
// TODO: 调用获取课程详情的API
|
||
|
|
// const course = await getCourseDetail(courseId);
|
||
|
|
// if (course?.scheduleRefData) {
|
||
|
|
// try {
|
||
|
|
// const parsedData = JSON.parse(course.scheduleRefData);
|
||
|
|
// scheduleRefData.value = Array.isArray(parsedData) ? parsedData : [];
|
||
|
|
// } catch (e) {
|
||
|
|
// console.error('Failed to parse scheduleRefData:', e);
|
||
|
|
// scheduleRefData.value = [];
|
||
|
|
// }
|
||
|
|
// } else {
|
||
|
|
// scheduleRefData.value = [];
|
||
|
|
// }
|
||
|
|
scheduleRefData.value = [];
|
||
|
|
} catch (error) {
|
||
|
|
console.error('Failed to load course detail:', error);
|
||
|
|
scheduleRefData.value = [];
|
||
|
|
}
|
||
|
|
};
|
||
|
|
|
||
|
|
// 加载课程类型列表
|
||
|
|
const loadLessonTypes = async (packageId: number) => {
|
||
|
|
loadingLessonTypes.value = true;
|
||
|
|
try {
|
||
|
|
lessonTypes.value = await getCoursePackageLessonTypes(packageId);
|
||
|
|
} catch (error) {
|
||
|
|
message.error('加载课程类型失败');
|
||
|
|
} finally {
|
||
|
|
loadingLessonTypes.value = false;
|
||
|
|
}
|
||
|
|
};
|
||
|
|
|
||
|
|
// 选择课程类型
|
||
|
|
const selectLessonType = (type: LessonType) => {
|
||
|
|
formData.lessonType = type;
|
||
|
|
};
|
||
|
|
|
||
|
|
// 切换班级选择
|
||
|
|
const toggleClass = (classId: number) => {
|
||
|
|
const index = formData.classIds.indexOf(classId);
|
||
|
|
if (index > -1) {
|
||
|
|
formData.classIds.splice(index, 1);
|
||
|
|
delete classTeacherMap.value[classId];
|
||
|
|
} else {
|
||
|
|
formData.classIds.push(classId);
|
||
|
|
}
|
||
|
|
};
|
||
|
|
|
||
|
|
// 检查班级是否被选中
|
||
|
|
const isClassSelected = (classId: number): boolean => {
|
||
|
|
return formData.classIds.includes(classId);
|
||
|
|
};
|
||
|
|
|
||
|
|
// 过滤课程套餐
|
||
|
|
const filterCollection = (input: string, option: any) => {
|
||
|
|
const collection = collections.value.find(c => c.id === option.value);
|
||
|
|
return collection?.name?.toLowerCase().includes(input.toLowerCase()) || false;
|
||
|
|
};
|
||
|
|
|
||
|
|
// 过滤教师
|
||
|
|
const filterTeacher = (input: string, option: any) => {
|
||
|
|
const teacher = teachers.value.find(t => t.id === option.value);
|
||
|
|
return teacher?.name?.toLowerCase().includes(input.toLowerCase()) || false;
|
||
|
|
};
|
||
|
|
|
||
|
|
// 获取课程类型图标
|
||
|
|
const getLessonTypeIcon = (type: LessonType): string => {
|
||
|
|
const icons: Record<LessonType, string> = {
|
||
|
|
INTRODUCTION: '📖',
|
||
|
|
COLLECTIVE: '👥',
|
||
|
|
LANGUAGE: '💬',
|
||
|
|
SOCIETY: '🤝',
|
||
|
|
SCIENCE: '🔬',
|
||
|
|
ART: '🎨',
|
||
|
|
HEALTH: '❤️',
|
||
|
|
};
|
||
|
|
return icons[type] || '📚';
|
||
|
|
};
|
||
|
|
|
||
|
|
// 获取选中的课程类型名称
|
||
|
|
const getSelectedLessonTypeName = (): string => {
|
||
|
|
if (!formData.lessonType) return '-';
|
||
|
|
const type = lessonTypes.value.find(t => t.lessonType === formData.lessonType);
|
||
|
|
return type?.lessonTypeName || '-';
|
||
|
|
};
|
||
|
|
|
||
|
|
// 获取选择的时间范围
|
||
|
|
const getSelectedTimeRange = (): string => {
|
||
|
|
if (!formData.scheduledTimeRange) return '';
|
||
|
|
return `${formData.scheduledTimeRange[0].format('HH:mm')}-${formData.scheduledTimeRange[1].format('HH:mm')}`;
|
||
|
|
};
|
||
|
|
|
||
|
|
// 获取选中的班级及教师列表
|
||
|
|
const getSelectedClassesWithTeachers = () => {
|
||
|
|
return formData.classIds.map(classId => {
|
||
|
|
const cls = classes.value.find(c => c.id === classId);
|
||
|
|
const teacherId = classTeacherMap.value[classId];
|
||
|
|
const teacher = teachers.value.find(t => t.id === teacherId);
|
||
|
|
return {
|
||
|
|
classId,
|
||
|
|
className: cls?.name || '',
|
||
|
|
teacherId,
|
||
|
|
teacherName: teacher?.name || '',
|
||
|
|
};
|
||
|
|
});
|
||
|
|
};
|
||
|
|
|
||
|
|
// 验证当前步骤
|
||
|
|
const validateStep = (): boolean => {
|
||
|
|
switch (currentStep.value) {
|
||
|
|
case 0:
|
||
|
|
if (!formData.collectionId) {
|
||
|
|
message.warning('请选择课程套餐');
|
||
|
|
return false;
|
||
|
|
}
|
||
|
|
if (!formData.packageId) {
|
||
|
|
message.warning('请选择课程包');
|
||
|
|
return false;
|
||
|
|
}
|
||
|
|
if (!formData.courseId) {
|
||
|
|
message.warning('请选择课程');
|
||
|
|
return false;
|
||
|
|
}
|
||
|
|
break;
|
||
|
|
case 1:
|
||
|
|
if (!formData.lessonType) {
|
||
|
|
message.warning('请选择课程类型');
|
||
|
|
return false;
|
||
|
|
}
|
||
|
|
break;
|
||
|
|
case 2:
|
||
|
|
if (formData.classIds.length === 0) {
|
||
|
|
message.warning('请至少选择一个班级');
|
||
|
|
return false;
|
||
|
|
}
|
||
|
|
// 检查每个班级是否都分配了教师
|
||
|
|
for (const classId of formData.classIds) {
|
||
|
|
if (!classTeacherMap.value[classId]) {
|
||
|
|
const cls = classes.value.find(c => c.id === classId);
|
||
|
|
message.warning(`请为 ${cls?.name} 分配教师`);
|
||
|
|
return false;
|
||
|
|
}
|
||
|
|
}
|
||
|
|
break;
|
||
|
|
case 3:
|
||
|
|
if (!formData.scheduledDate) {
|
||
|
|
message.warning('请选择排课日期');
|
||
|
|
return false;
|
||
|
|
}
|
||
|
|
if (!formData.scheduledTimeRange) {
|
||
|
|
message.warning('请选择时间段');
|
||
|
|
return false;
|
||
|
|
}
|
||
|
|
break;
|
||
|
|
}
|
||
|
|
return true;
|
||
|
|
};
|
||
|
|
|
||
|
|
// 下一步
|
||
|
|
const nextStep = () => {
|
||
|
|
if (validateStep()) {
|
||
|
|
currentStep.value++;
|
||
|
|
}
|
||
|
|
};
|
||
|
|
|
||
|
|
// 上一步
|
||
|
|
const prevStep = () => {
|
||
|
|
currentStep.value--;
|
||
|
|
};
|
||
|
|
|
||
|
|
// 提交表单
|
||
|
|
const handleSubmit = async () => {
|
||
|
|
if (!validateStep()) return;
|
||
|
|
|
||
|
|
loading.value = true;
|
||
|
|
try {
|
||
|
|
// 格式化时间
|
||
|
|
let scheduledTime: string | undefined = undefined;
|
||
|
|
if (formData.scheduledTimeRange && formData.scheduledTimeRange.length === 2) {
|
||
|
|
scheduledTime = `${formData.scheduledTimeRange[0].format('HH:mm')}-${formData.scheduledTimeRange[1].format('HH:mm')}`;
|
||
|
|
}
|
||
|
|
|
||
|
|
// 为每个班级分别创建排课
|
||
|
|
const promises = formData.classIds.map(classId => {
|
||
|
|
return createSchedulesByClasses({
|
||
|
|
coursePackageId: formData.packageId!,
|
||
|
|
courseId: formData.courseId!,
|
||
|
|
lessonType: formData.lessonType!,
|
||
|
|
classIds: [classId],
|
||
|
|
teacherId: classTeacherMap.value[classId],
|
||
|
|
scheduledDate: formData.scheduledDate!.format('YYYY-MM-DD'),
|
||
|
|
scheduledTime,
|
||
|
|
});
|
||
|
|
});
|
||
|
|
|
||
|
|
await Promise.all(promises);
|
||
|
|
|
||
|
|
message.success(`成功创建 ${formData.classIds.length} 条排课`);
|
||
|
|
visible.value = false;
|
||
|
|
emit('success');
|
||
|
|
} catch (error) {
|
||
|
|
message.error('创建失败');
|
||
|
|
} finally {
|
||
|
|
loading.value = false;
|
||
|
|
}
|
||
|
|
};
|
||
|
|
|
||
|
|
// 取消
|
||
|
|
const handleCancel = () => {
|
||
|
|
visible.value = false;
|
||
|
|
};
|
||
|
|
|
||
|
|
defineExpose({ open });
|
||
|
|
</script>
|
||
|
|
|
||
|
|
<style scoped lang="scss">
|
||
|
|
.steps-navigator {
|
||
|
|
margin-bottom: 32px;
|
||
|
|
padding: 0 20px;
|
||
|
|
}
|
||
|
|
|
||
|
|
.step-content {
|
||
|
|
min-height: 400px;
|
||
|
|
padding: 0 20px;
|
||
|
|
}
|
||
|
|
|
||
|
|
.step-panel {
|
||
|
|
h3 {
|
||
|
|
font-size: 16px;
|
||
|
|
font-weight: 600;
|
||
|
|
margin-bottom: 16px;
|
||
|
|
color: #2D3436;
|
||
|
|
}
|
||
|
|
|
||
|
|
h4 {
|
||
|
|
font-size: 14px;
|
||
|
|
font-weight: 500;
|
||
|
|
margin-bottom: 12px;
|
||
|
|
margin-top: 20px;
|
||
|
|
color: #333;
|
||
|
|
}
|
||
|
|
}
|
||
|
|
|
||
|
|
.collection-option {
|
||
|
|
.collection-name {
|
||
|
|
font-weight: 500;
|
||
|
|
}
|
||
|
|
.collection-info {
|
||
|
|
font-size: 12px;
|
||
|
|
color: #999;
|
||
|
|
}
|
||
|
|
}
|
||
|
|
|
||
|
|
.packages-section {
|
||
|
|
margin-top: 24px;
|
||
|
|
}
|
||
|
|
|
||
|
|
.packages-grid {
|
||
|
|
display: grid;
|
||
|
|
grid-template-columns: repeat(auto-fill, minmax(180px, 1fr));
|
||
|
|
gap: 12px;
|
||
|
|
}
|
||
|
|
|
||
|
|
.package-card {
|
||
|
|
padding: 12px;
|
||
|
|
background: white;
|
||
|
|
border: 2px solid #E0E0E0;
|
||
|
|
border-radius: 8px;
|
||
|
|
cursor: pointer;
|
||
|
|
transition: all 0.2s;
|
||
|
|
|
||
|
|
&:hover {
|
||
|
|
border-color: #BDBDBD;
|
||
|
|
}
|
||
|
|
|
||
|
|
&.active {
|
||
|
|
border-color: #FF8C42;
|
||
|
|
background: #FFF0E6;
|
||
|
|
}
|
||
|
|
|
||
|
|
.package-name {
|
||
|
|
font-weight: 500;
|
||
|
|
color: #2D3436;
|
||
|
|
margin-bottom: 4px;
|
||
|
|
}
|
||
|
|
|
||
|
|
.package-grade {
|
||
|
|
font-size: 12px;
|
||
|
|
color: #999;
|
||
|
|
margin-bottom: 2px;
|
||
|
|
}
|
||
|
|
|
||
|
|
.package-count {
|
||
|
|
font-size: 11px;
|
||
|
|
color: #FF8C42;
|
||
|
|
}
|
||
|
|
}
|
||
|
|
|
||
|
|
.courses-section {
|
||
|
|
margin-top: 24px;
|
||
|
|
}
|
||
|
|
|
||
|
|
.courses-grid {
|
||
|
|
display: grid;
|
||
|
|
grid-template-columns: repeat(auto-fill, minmax(180px, 1fr));
|
||
|
|
gap: 12px;
|
||
|
|
}
|
||
|
|
|
||
|
|
.course-card {
|
||
|
|
padding: 12px;
|
||
|
|
background: white;
|
||
|
|
border: 2px solid #E0E0E0;
|
||
|
|
border-radius: 8px;
|
||
|
|
cursor: pointer;
|
||
|
|
transition: all 0.2s;
|
||
|
|
|
||
|
|
&:hover {
|
||
|
|
border-color: #BDBDBD;
|
||
|
|
}
|
||
|
|
|
||
|
|
&.active {
|
||
|
|
border-color: #FF8C42;
|
||
|
|
background: #FFF0E6;
|
||
|
|
}
|
||
|
|
|
||
|
|
.course-name {
|
||
|
|
font-weight: 500;
|
||
|
|
color: #2D3436;
|
||
|
|
margin-bottom: 4px;
|
||
|
|
}
|
||
|
|
|
||
|
|
.course-grade {
|
||
|
|
font-size: 12px;
|
||
|
|
color: #999;
|
||
|
|
}
|
||
|
|
}
|
||
|
|
|
||
|
|
.schedule-ref-card {
|
||
|
|
margin-top: 24px;
|
||
|
|
padding: 16px;
|
||
|
|
background: #FFF8F0;
|
||
|
|
border-radius: 8px;
|
||
|
|
|
||
|
|
.ref-header {
|
||
|
|
display: flex;
|
||
|
|
align-items: center;
|
||
|
|
gap: 8px;
|
||
|
|
margin-bottom: 12px;
|
||
|
|
|
||
|
|
.ref-icon {
|
||
|
|
color: #FF8C42;
|
||
|
|
}
|
||
|
|
|
||
|
|
.ref-title {
|
||
|
|
font-weight: 600;
|
||
|
|
color: #2D3436;
|
||
|
|
}
|
||
|
|
}
|
||
|
|
}
|
||
|
|
|
||
|
|
.lesson-type-grid {
|
||
|
|
display: grid;
|
||
|
|
grid-template-columns: repeat(auto-fill, minmax(140px, 1fr));
|
||
|
|
gap: 16px;
|
||
|
|
}
|
||
|
|
|
||
|
|
.lesson-type-card {
|
||
|
|
padding: 20px;
|
||
|
|
background: white;
|
||
|
|
border: 2px solid #E0E0E0;
|
||
|
|
border-radius: 12px;
|
||
|
|
text-align: center;
|
||
|
|
cursor: pointer;
|
||
|
|
transition: all 0.3s;
|
||
|
|
|
||
|
|
&:hover {
|
||
|
|
border-color: #FF8C42;
|
||
|
|
transform: translateY(-2px);
|
||
|
|
box-shadow: 0 4px 12px rgba(255, 140, 66, 0.2);
|
||
|
|
}
|
||
|
|
|
||
|
|
&.active {
|
||
|
|
border-color: #FF8C42;
|
||
|
|
background: #FFF0E6;
|
||
|
|
}
|
||
|
|
|
||
|
|
.type-icon {
|
||
|
|
font-size: 32px;
|
||
|
|
margin-bottom: 8px;
|
||
|
|
}
|
||
|
|
|
||
|
|
.type-name {
|
||
|
|
font-weight: 500;
|
||
|
|
color: #2D3436;
|
||
|
|
margin-bottom: 4px;
|
||
|
|
}
|
||
|
|
|
||
|
|
.type-count {
|
||
|
|
font-size: 12px;
|
||
|
|
color: #999;
|
||
|
|
}
|
||
|
|
}
|
||
|
|
|
||
|
|
.grade-selector {
|
||
|
|
margin-bottom: 16px;
|
||
|
|
}
|
||
|
|
|
||
|
|
.class-teacher-grid {
|
||
|
|
display: grid;
|
||
|
|
grid-template-columns: repeat(auto-fill, minmax(250px, 1fr));
|
||
|
|
gap: 12px;
|
||
|
|
max-height: 400px;
|
||
|
|
overflow-y: auto;
|
||
|
|
padding: 4px;
|
||
|
|
}
|
||
|
|
|
||
|
|
.class-teacher-card {
|
||
|
|
background: white;
|
||
|
|
border: 2px solid #E0E0E0;
|
||
|
|
border-radius: 8px;
|
||
|
|
overflow: hidden;
|
||
|
|
transition: all 0.2s;
|
||
|
|
|
||
|
|
&.selected {
|
||
|
|
border-color: #FF8C42;
|
||
|
|
}
|
||
|
|
|
||
|
|
.class-header {
|
||
|
|
display: flex;
|
||
|
|
align-items: center;
|
||
|
|
gap: 12px;
|
||
|
|
padding: 12px;
|
||
|
|
cursor: pointer;
|
||
|
|
|
||
|
|
.class-info {
|
||
|
|
flex: 1;
|
||
|
|
|
||
|
|
.class-name {
|
||
|
|
font-weight: 500;
|
||
|
|
color: #2D3436;
|
||
|
|
}
|
||
|
|
|
||
|
|
.class-detail {
|
||
|
|
font-size: 12px;
|
||
|
|
color: #999;
|
||
|
|
margin-top: 2px;
|
||
|
|
}
|
||
|
|
}
|
||
|
|
}
|
||
|
|
|
||
|
|
.teacher-selector {
|
||
|
|
padding: 12px;
|
||
|
|
border-top: 1px solid #E0E0E0;
|
||
|
|
background: #FAFAFA;
|
||
|
|
}
|
||
|
|
}
|
||
|
|
|
||
|
|
.selection-summary {
|
||
|
|
margin-top: 16px;
|
||
|
|
padding: 12px;
|
||
|
|
background: #FFF8F0;
|
||
|
|
border-radius: 8px;
|
||
|
|
text-align: center;
|
||
|
|
color: #FF8C42;
|
||
|
|
font-weight: 500;
|
||
|
|
}
|
||
|
|
|
||
|
|
.confirm-info {
|
||
|
|
margin-top: 24px;
|
||
|
|
padding: 16px;
|
||
|
|
background: #F5F5F5;
|
||
|
|
border-radius: 8px;
|
||
|
|
|
||
|
|
div {
|
||
|
|
margin-bottom: 4px;
|
||
|
|
|
||
|
|
&:last-child {
|
||
|
|
margin-bottom: 0;
|
||
|
|
}
|
||
|
|
|
||
|
|
strong {
|
||
|
|
color: #FF8C42;
|
||
|
|
}
|
||
|
|
}
|
||
|
|
}
|
||
|
|
|
||
|
|
.class-teacher-list {
|
||
|
|
margin-top: 20px;
|
||
|
|
|
||
|
|
h4 {
|
||
|
|
font-size: 14px;
|
||
|
|
font-weight: 500;
|
||
|
|
margin-bottom: 12px;
|
||
|
|
color: #333;
|
||
|
|
}
|
||
|
|
|
||
|
|
.teacher-status {
|
||
|
|
color: #999;
|
||
|
|
|
||
|
|
&.assigned {
|
||
|
|
color: #52c41a;
|
||
|
|
font-weight: 500;
|
||
|
|
}
|
||
|
|
}
|
||
|
|
}
|
||
|
|
|
||
|
|
.modal-footer {
|
||
|
|
display: flex;
|
||
|
|
justify-content: flex-end;
|
||
|
|
gap: 8px;
|
||
|
|
}
|
||
|
|
</style>
|