mirror of
https://github.com/ZeroCatDev/Classworks.git
synced 2026-02-03 23:23:09 +00:00
feat: 添加教师列表卡片组件并更新设置页面
This commit is contained in:
parent
472519ef5e
commit
1325038fa0
@ -9,7 +9,7 @@
|
||||
<v-card>
|
||||
<v-card-title class="d-flex align-center">
|
||||
<v-icon class="mr-2" icon="mdi-account-group" />
|
||||
出勤状态管理
|
||||
考勤
|
||||
<v-spacer />
|
||||
<v-chip v-if="!isMobile" class="ml-2" color="primary" size="small">
|
||||
{{ dateString }}
|
||||
|
||||
543
src/components/settings/TeacherListCard.vue
Normal file
543
src/components/settings/TeacherListCard.vue
Normal file
@ -0,0 +1,543 @@
|
||||
<template>
|
||||
<v-card
|
||||
:class="{ 'unsaved-changes': unsavedChanges }"
|
||||
:color="unsavedChanges ? 'warning-subtle' : undefined"
|
||||
border
|
||||
>
|
||||
<v-card-item>
|
||||
<template #prepend>
|
||||
<v-icon class="mr-2" icon="mdi-account-tie" size="large"/>
|
||||
</template>
|
||||
<v-card-title class="text-h6">教师列表</v-card-title>
|
||||
<template #append>
|
||||
<unsaved-warning :show="unsavedChanges" message="有未保存的更改"/>
|
||||
<v-btn
|
||||
:color="modelValue.advanced ? 'primary' : undefined"
|
||||
prepend-icon="mdi-code-braces"
|
||||
variant="text"
|
||||
@click="toggleAdvanced"
|
||||
>
|
||||
{{ modelValue.advanced ? "返回基础编辑" : "高级编辑" }}
|
||||
</v-btn>
|
||||
</template>
|
||||
</v-card-item>
|
||||
|
||||
<v-card-text>
|
||||
<v-progress-linear
|
||||
v-if="loading"
|
||||
class="mb-4"
|
||||
color="primary"
|
||||
indeterminate
|
||||
/>
|
||||
|
||||
<v-alert v-if="error" class="mb-4" closable type="error" variant="tonal">
|
||||
{{ error }}
|
||||
</v-alert>
|
||||
|
||||
<v-expand-transition>
|
||||
<!-- 普通编辑模式 -->
|
||||
<div v-if="!modelValue.advanced">
|
||||
<!-- 添加教师表单 -->
|
||||
<v-card class="mb-6" variant="outlined">
|
||||
<v-card-text>
|
||||
<v-row>
|
||||
<v-col cols="12" md="4">
|
||||
<v-text-field
|
||||
v-model="newTeacher.name"
|
||||
density="comfortable"
|
||||
hide-details
|
||||
label="教师姓名"
|
||||
placeholder="输入教师姓名"
|
||||
prepend-inner-icon="mdi-account"
|
||||
variant="outlined"
|
||||
/>
|
||||
</v-col>
|
||||
<v-col cols="12" md="5">
|
||||
<v-combobox
|
||||
v-model="newTeacher.subjects"
|
||||
:items="commonSubjects"
|
||||
chips
|
||||
clearable
|
||||
closable-chips
|
||||
density="comfortable"
|
||||
hide-details
|
||||
label="任教科目"
|
||||
multiple
|
||||
placeholder="选择或输入科目"
|
||||
prepend-inner-icon="mdi-book-open-variant"
|
||||
variant="outlined"
|
||||
/>
|
||||
</v-col>
|
||||
<v-col cols="12" md="3" class="d-flex align-center gap-2">
|
||||
<v-checkbox
|
||||
v-model="newTeacher.isHeadTeacher"
|
||||
density="comfortable"
|
||||
hide-details
|
||||
label="班主任"
|
||||
/>
|
||||
<v-btn
|
||||
:disabled="!newTeacher.name.trim() || newTeacher.subjects.length === 0"
|
||||
color="primary"
|
||||
prepend-icon="mdi-plus"
|
||||
@click="addTeacher"
|
||||
>
|
||||
添加教师
|
||||
</v-btn>
|
||||
</v-col>
|
||||
</v-row>
|
||||
</v-card-text>
|
||||
</v-card>
|
||||
|
||||
<!-- 教师列表 -->
|
||||
<v-row v-if="modelValue.list.length === 0">
|
||||
<v-col cols="12">
|
||||
<v-alert type="info" variant="tonal">
|
||||
暂无教师信息,请添加教师
|
||||
</v-alert>
|
||||
</v-col>
|
||||
</v-row>
|
||||
|
||||
<v-row v-else>
|
||||
<v-col
|
||||
v-for="(teacher, index) in modelValue.list"
|
||||
:key="index"
|
||||
cols="12"
|
||||
lg="6"
|
||||
xl="4"
|
||||
>
|
||||
<v-hover v-slot="{ isHovering, props }">
|
||||
<v-card
|
||||
:elevation="isMobile ? 1 : isHovering ? 4 : 1"
|
||||
border
|
||||
class="teacher-card"
|
||||
v-bind="props"
|
||||
>
|
||||
<v-card-text class="pa-4">
|
||||
<div class="d-flex align-start mb-3">
|
||||
<v-avatar
|
||||
:color="teacher.isHeadTeacher ? 'primary' : 'grey-lighten-1'"
|
||||
class="mr-3"
|
||||
size="48"
|
||||
>
|
||||
<v-icon
|
||||
:icon="teacher.isHeadTeacher ? 'mdi-star' : 'mdi-account'"
|
||||
size="28"
|
||||
/>
|
||||
</v-avatar>
|
||||
|
||||
<div class="flex-grow-1">
|
||||
<div class="d-flex align-center mb-1">
|
||||
<v-text-field
|
||||
v-if="editState.index === index"
|
||||
v-model="editState.teacher.name"
|
||||
autofocus
|
||||
class="flex-grow-1"
|
||||
density="compact"
|
||||
hide-details
|
||||
variant="underlined"
|
||||
/>
|
||||
<span
|
||||
v-else
|
||||
class="text-h6 font-weight-medium"
|
||||
@click="handleClick(index, teacher)"
|
||||
>
|
||||
{{ teacher.name }}
|
||||
</span>
|
||||
<v-chip
|
||||
v-if="teacher.isHeadTeacher"
|
||||
class="ml-2"
|
||||
color="primary"
|
||||
density="comfortable"
|
||||
size="small"
|
||||
variant="flat"
|
||||
>
|
||||
班主任
|
||||
</v-chip>
|
||||
</div>
|
||||
|
||||
<div v-if="editState.index === index" class="mt-2">
|
||||
<v-combobox
|
||||
v-model="editState.teacher.subjects"
|
||||
:items="commonSubjects"
|
||||
chips
|
||||
closable-chips
|
||||
density="compact"
|
||||
hide-details
|
||||
label="任教科目"
|
||||
multiple
|
||||
variant="outlined"
|
||||
/>
|
||||
<v-checkbox
|
||||
v-model="editState.teacher.isHeadTeacher"
|
||||
class="mt-2"
|
||||
density="compact"
|
||||
hide-details
|
||||
label="班主任"
|
||||
/>
|
||||
</div>
|
||||
<div v-else class="mt-1">
|
||||
<v-chip
|
||||
v-for="(subject, sIndex) in teacher.subjects"
|
||||
:key="sIndex"
|
||||
class="mr-1 mb-1"
|
||||
density="comfortable"
|
||||
size="small"
|
||||
variant="tonal"
|
||||
>
|
||||
{{ subject }}
|
||||
</v-chip>
|
||||
</div>
|
||||
</div>
|
||||
|
||||
<div
|
||||
:class="{ 'opacity-100': isHovering || isMobile || editState.index === index }"
|
||||
class="d-flex gap-1 action-buttons ml-2"
|
||||
>
|
||||
<v-btn
|
||||
v-if="editState.index === index"
|
||||
color="success"
|
||||
icon="mdi-check"
|
||||
size="small"
|
||||
variant="text"
|
||||
@click="saveEdit"
|
||||
/>
|
||||
<v-btn
|
||||
v-if="editState.index === index"
|
||||
color="grey"
|
||||
icon="mdi-close"
|
||||
size="small"
|
||||
variant="text"
|
||||
@click="cancelEdit"
|
||||
/>
|
||||
<v-btn
|
||||
v-else
|
||||
color="primary"
|
||||
icon="mdi-pencil"
|
||||
size="small"
|
||||
variant="text"
|
||||
@click="startEdit(index, teacher)"
|
||||
/>
|
||||
<v-btn
|
||||
v-if="editState.index !== index"
|
||||
color="error"
|
||||
icon="mdi-delete"
|
||||
size="small"
|
||||
variant="text"
|
||||
@click="removeTeacher(index)"
|
||||
/>
|
||||
</div>
|
||||
</div>
|
||||
</v-card-text>
|
||||
</v-card>
|
||||
</v-hover>
|
||||
</v-col>
|
||||
</v-row>
|
||||
</div>
|
||||
|
||||
<!-- 高级编辑模式 -->
|
||||
<div v-else class="pt-2">
|
||||
<v-textarea
|
||||
v-model="modelValue.text"
|
||||
hint="JSON 格式编辑教师列表。每个教师需包含 name、subjects(数组)、isHeadTeacher(布尔值)"
|
||||
label="批量编辑教师列表 (JSON)"
|
||||
persistent-hint
|
||||
placeholder='[{"name":"教师姓名","subjects":["语文","数学"],"isHeadTeacher":true}]'
|
||||
rows="15"
|
||||
variant="outlined"
|
||||
@update:model-value="handleTextInput"
|
||||
/>
|
||||
</div>
|
||||
</v-expand-transition>
|
||||
|
||||
<v-row class="mt-6">
|
||||
<v-col class="d-flex gap-2" cols="12">
|
||||
<v-btn
|
||||
:disabled="loading"
|
||||
:loading="loading"
|
||||
color="primary"
|
||||
prepend-icon="mdi-content-save"
|
||||
size="large"
|
||||
@click="saveTeachers"
|
||||
>
|
||||
保存教师列表
|
||||
</v-btn>
|
||||
<v-btn
|
||||
:disabled="loading"
|
||||
:loading="loading"
|
||||
color="error"
|
||||
prepend-icon="mdi-refresh"
|
||||
size="large"
|
||||
variant="outlined"
|
||||
@click="loadTeachers"
|
||||
>
|
||||
重载教师列表
|
||||
</v-btn>
|
||||
</v-col>
|
||||
</v-row>
|
||||
</v-card-text>
|
||||
</v-card>
|
||||
</template>
|
||||
|
||||
<script>
|
||||
import UnsavedWarning from "../common/UnsavedWarning.vue";
|
||||
import "@/styles/warnings.scss";
|
||||
import dataProvider from "@/utils/dataProvider";
|
||||
import {getSetting} from "@/utils/settings";
|
||||
|
||||
export default {
|
||||
name: "TeacherListCard",
|
||||
components: {
|
||||
UnsavedWarning,
|
||||
},
|
||||
props: {
|
||||
isMobile: Boolean,
|
||||
},
|
||||
|
||||
data() {
|
||||
return {
|
||||
newTeacher: {
|
||||
name: "",
|
||||
subjects: [],
|
||||
isHeadTeacher: false,
|
||||
},
|
||||
editState: {
|
||||
index: -1,
|
||||
teacher: null,
|
||||
},
|
||||
modelValue: {
|
||||
list: [],
|
||||
text: "",
|
||||
advanced: false,
|
||||
},
|
||||
loading: false,
|
||||
error: null,
|
||||
lastSavedData: null,
|
||||
unsavedChanges: false,
|
||||
commonSubjects: [
|
||||
"语文",
|
||||
"数学",
|
||||
"英语",
|
||||
"物理",
|
||||
"化学",
|
||||
"生物",
|
||||
"政治",
|
||||
"历史",
|
||||
"地理",
|
||||
"信息技术",
|
||||
"音乐",
|
||||
"美术",
|
||||
"体育",
|
||||
],
|
||||
};
|
||||
},
|
||||
|
||||
watch: {
|
||||
modelValue: {
|
||||
handler(newData) {
|
||||
if (this.lastSavedData) {
|
||||
this.unsavedChanges = JSON.stringify(newData.list) !== JSON.stringify(this.lastSavedData);
|
||||
}
|
||||
if (!this.modelValue.advanced) {
|
||||
this.modelValue.text = JSON.stringify(newData.list, null, 2);
|
||||
}
|
||||
},
|
||||
deep: true,
|
||||
},
|
||||
},
|
||||
|
||||
mounted() {
|
||||
this.loadTeachers();
|
||||
},
|
||||
|
||||
methods: {
|
||||
async loadTeachers() {
|
||||
this.error = null;
|
||||
try {
|
||||
this.loading = true;
|
||||
const classNum = getSetting("server.classNumber");
|
||||
|
||||
if (!classNum) {
|
||||
throw new Error("请先设置班号");
|
||||
}
|
||||
|
||||
try {
|
||||
const response = await dataProvider.loadData("classworks-list-teacher");
|
||||
|
||||
if (response.success !== false && Array.isArray(response)) {
|
||||
this.modelValue.list = response.map((item) => ({
|
||||
name: item.name || "",
|
||||
subjects: Array.isArray(item.subjects) ? item.subjects : [],
|
||||
isHeadTeacher: Boolean(item.isHeadTeacher),
|
||||
}));
|
||||
|
||||
this.modelValue.text = JSON.stringify(this.modelValue.list, null, 2);
|
||||
this.lastSavedData = JSON.parse(JSON.stringify(this.modelValue.list));
|
||||
this.unsavedChanges = false;
|
||||
}
|
||||
} catch (error) {
|
||||
console.warn("Failed to load teacher list, initializing empty list", error);
|
||||
this.modelValue.list = [];
|
||||
this.modelValue.text = "[]";
|
||||
this.lastSavedData = [];
|
||||
}
|
||||
} catch (error) {
|
||||
console.error("加载教师列表失败:", error);
|
||||
this.error = error.message || "加载失败,请检查设置";
|
||||
this.$message?.error("加载失败", this.error);
|
||||
} finally {
|
||||
this.loading = false;
|
||||
}
|
||||
},
|
||||
|
||||
async saveTeachers() {
|
||||
try {
|
||||
const classNum = getSetting("server.classNumber");
|
||||
|
||||
if (!classNum) {
|
||||
throw new Error("请先设置班号");
|
||||
}
|
||||
|
||||
const formattedList = this.modelValue.list.map((teacher) => ({
|
||||
name: teacher.name,
|
||||
subjects: Array.isArray(teacher.subjects) ? teacher.subjects : [],
|
||||
isHeadTeacher: Boolean(teacher.isHeadTeacher),
|
||||
}));
|
||||
|
||||
const response = await dataProvider.saveData(
|
||||
"classworks-list-teacher",
|
||||
formattedList
|
||||
);
|
||||
|
||||
if (response.success === false) {
|
||||
throw new Error(response.error?.message || "保存失败");
|
||||
}
|
||||
|
||||
this.modelValue.list = formattedList;
|
||||
this.lastSavedData = JSON.parse(JSON.stringify(formattedList));
|
||||
this.unsavedChanges = false;
|
||||
this.$message?.success("保存成功", "教师列表已更新");
|
||||
} catch (error) {
|
||||
console.error("保存教师列表失败:", error);
|
||||
this.$message?.error("保存失败", error.message || "请重试");
|
||||
}
|
||||
},
|
||||
|
||||
toggleAdvanced() {
|
||||
this.modelValue.advanced = !this.modelValue.advanced;
|
||||
if (this.modelValue.advanced) {
|
||||
this.modelValue.text = JSON.stringify(this.modelValue.list, null, 2);
|
||||
}
|
||||
},
|
||||
|
||||
handleTextInput(text) {
|
||||
if (!this.modelValue.advanced) return;
|
||||
|
||||
try {
|
||||
const parsed = JSON.parse(text);
|
||||
if (Array.isArray(parsed)) {
|
||||
this.modelValue.list = parsed.map((item) => ({
|
||||
name: item.name || "",
|
||||
subjects: Array.isArray(item.subjects) ? item.subjects : [],
|
||||
isHeadTeacher: Boolean(item.isHeadTeacher),
|
||||
}));
|
||||
this.error = null;
|
||||
} else {
|
||||
this.error = "JSON 必须是一个数组";
|
||||
}
|
||||
} catch (e) {
|
||||
// JSON 解析错误,不更新列表,用户可能还在输入
|
||||
this.error = "JSON 格式错误: " + e.message;
|
||||
}
|
||||
},
|
||||
|
||||
addTeacher() {
|
||||
const name = this.newTeacher.name.trim();
|
||||
if (!name) {
|
||||
this.$message?.warning("提示", "请输入教师姓名");
|
||||
return;
|
||||
}
|
||||
if (this.newTeacher.subjects.length === 0) {
|
||||
this.$message?.warning("提示", "请选择至少一个任教科目");
|
||||
return;
|
||||
}
|
||||
|
||||
this.modelValue.list.push({
|
||||
name,
|
||||
subjects: [...this.newTeacher.subjects],
|
||||
isHeadTeacher: this.newTeacher.isHeadTeacher,
|
||||
});
|
||||
|
||||
// 重置表单
|
||||
this.newTeacher = {
|
||||
name: "",
|
||||
subjects: [],
|
||||
isHeadTeacher: false,
|
||||
};
|
||||
},
|
||||
|
||||
startEdit(index, teacher) {
|
||||
this.editState.index = index;
|
||||
this.editState.teacher = {
|
||||
name: teacher.name,
|
||||
subjects: [...teacher.subjects],
|
||||
isHeadTeacher: teacher.isHeadTeacher,
|
||||
};
|
||||
},
|
||||
|
||||
saveEdit() {
|
||||
if (this.editState.index !== -1) {
|
||||
const newName = this.editState.teacher.name.trim();
|
||||
if (!newName) {
|
||||
this.$message?.warning("提示", "教师姓名不能为空");
|
||||
return;
|
||||
}
|
||||
if (this.editState.teacher.subjects.length === 0) {
|
||||
this.$message?.warning("提示", "请选择至少一个任教科目");
|
||||
return;
|
||||
}
|
||||
|
||||
this.modelValue.list[this.editState.index] = {
|
||||
name: newName,
|
||||
subjects: [...this.editState.teacher.subjects],
|
||||
isHeadTeacher: this.editState.teacher.isHeadTeacher,
|
||||
};
|
||||
|
||||
this.editState.index = -1;
|
||||
this.editState.teacher = null;
|
||||
}
|
||||
},
|
||||
|
||||
cancelEdit() {
|
||||
this.editState.index = -1;
|
||||
this.editState.teacher = null;
|
||||
},
|
||||
|
||||
removeTeacher(index) {
|
||||
if (index !== undefined) {
|
||||
this.modelValue.list.splice(index, 1);
|
||||
}
|
||||
},
|
||||
|
||||
handleClick(index, teacher) {
|
||||
if (this.isMobile) {
|
||||
this.startEdit(index, teacher);
|
||||
}
|
||||
},
|
||||
},
|
||||
};
|
||||
</script>
|
||||
|
||||
<style lang="scss" scoped>
|
||||
.teacher-card {
|
||||
transition: all 0.2s ease;
|
||||
}
|
||||
|
||||
.action-buttons {
|
||||
opacity: 0;
|
||||
transition: opacity 0.2s ease;
|
||||
}
|
||||
|
||||
.unsaved-changes {
|
||||
border-color: rgb(var(--v-theme-warning)) !important;
|
||||
}
|
||||
</style>
|
||||
@ -144,6 +144,7 @@
|
||||
|
||||
<v-tabs-window-item value="student">
|
||||
<student-list-card :is-mobile="isMobile" border/>
|
||||
<teacher-list-card :is-mobile="isMobile" border class="mt-4"/>
|
||||
</v-tabs-window-item>
|
||||
<v-tabs-window-item value="share">
|
||||
<settings-link-generator border class="mt-4"/>
|
||||
@ -272,6 +273,7 @@ import {
|
||||
import MessageLog from "@/components/MessageLog.vue";
|
||||
import SettingsCard from "@/components/SettingsCard.vue";
|
||||
import StudentListCard from "@/components/settings/StudentListCard.vue";
|
||||
import TeacherListCard from "@/components/settings/TeacherListCard.vue";
|
||||
import AboutCard from "@/components/settings/AboutCard.vue";
|
||||
import "../styles/settings.scss";
|
||||
import SettingsExplorer from "@/components/settings/SettingsExplorer.vue";
|
||||
@ -293,6 +295,7 @@ export default {
|
||||
MessageLog,
|
||||
SettingsCard,
|
||||
StudentListCard,
|
||||
TeacherListCard,
|
||||
AboutCard,
|
||||
DataProviderSettingsCard,
|
||||
ThemeSettingsCard,
|
||||
@ -397,7 +400,7 @@ export default {
|
||||
value: "subject",
|
||||
},
|
||||
{
|
||||
title: "学生列表",
|
||||
title: "花名册",
|
||||
icon: "mdi-account-group",
|
||||
value: "student",
|
||||
},
|
||||
|
||||
Loading…
x
Reference in New Issue
Block a user