ImportInspect.vue 4.5 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161
  1. <template>
  2. <div class="my-container">
  3. <mark-header />
  4. <div class="tw-flex tw-gap-1">
  5. <mark-body @error="renderError" />
  6. <MarkBoardInspect
  7. :tagged="isCurrentTagged"
  8. :isFirst="isFirst"
  9. :isLast="isLast"
  10. @makeTag="saveTaskToServer"
  11. @fetchTask="fetchTask"
  12. />
  13. </div>
  14. </div>
  15. </template>
  16. <script setup lang="ts">
  17. import { computed, onMounted, ref } from "vue";
  18. import {
  19. getInspectedSetting,
  20. getSingleInspectedTask,
  21. saveInspectedTask,
  22. } from "@/api/importInspectPage";
  23. import { store } from "@/features/mark/store";
  24. import MarkHeader from "./MarkHeader.vue";
  25. import { useRoute } from "vue-router";
  26. import MarkBody from "./MarkBody.vue";
  27. import MarkBoardInspect from "./MarkBoardInspect.vue";
  28. import type { Task, Setting } from "@/types";
  29. import { message } from "ant-design-vue";
  30. const route = useRoute();
  31. let isSingleStudent = !!route.query.studentId;
  32. const { studentId } = route.query as {
  33. studentId: string;
  34. };
  35. let studentIds = ref([] as Array<number>);
  36. let tagIds = ref([] as Array<number>);
  37. let currentStudentId = ref(0);
  38. async function updateSetting() {
  39. const settingRes = await getInspectedSetting(studentId);
  40. store.setting.examType = settingRes.data.examType;
  41. store.setting.fileServer = settingRes.data.fileServer;
  42. store.status.totalCount = settingRes.data.inspectCount;
  43. store.status.markedCount = 0;
  44. store.setting.uiSetting = {
  45. "answer.paper.scale": 1,
  46. "score.board.collapse": false,
  47. "normal.mode": "keyboard",
  48. } as Setting["uiSetting"];
  49. if (!settingRes.data.inspectCount) {
  50. store.message = settingRes.data.message;
  51. } else {
  52. studentIds.value = settingRes.data.studentIds;
  53. tagIds.value = settingRes.data.tagIds;
  54. }
  55. }
  56. // 要通过fetchTask调用
  57. async function updateTask() {
  58. // const mkey = "fetch_task_key";
  59. message.info({ content: "获取任务中...", duration: 1.5 });
  60. let res;
  61. if (currentStudentId.value) {
  62. res = await getSingleInspectedTask("" + currentStudentId.value);
  63. } else {
  64. return;
  65. }
  66. if (res.data.studentId) {
  67. let rawTask = res.data as Task;
  68. rawTask.sliceUrls = rawTask.sliceUrls?.map(
  69. (s) => store.setting.fileServer + s
  70. );
  71. rawTask.sheetUrls = rawTask.sheetUrls?.map(
  72. (s) => store.setting.fileServer + s
  73. );
  74. store.currentTask = res.data;
  75. store.setting.subject = res.data.subject;
  76. } else {
  77. store.message = res.data.message;
  78. }
  79. }
  80. const isCurrentTagged = computed(() =>
  81. tagIds.value.includes(currentStudentId.value)
  82. );
  83. const isFirst = computed(
  84. () => studentIds.value.indexOf(currentStudentId.value) === 0
  85. );
  86. const isLast = computed(
  87. () =>
  88. studentIds.value.indexOf(currentStudentId.value) ===
  89. studentIds.value.length - 1
  90. );
  91. async function fetchTask(next: boolean, init?: boolean) {
  92. if (init) {
  93. currentStudentId.value = studentIds.value[0];
  94. } else if (isLast.value && next) {
  95. return; // currentStudentId.value是最后一个不调用
  96. } else if (isFirst.value && !next) {
  97. return; // currentStudentId.value是第一个不调用
  98. } else {
  99. currentStudentId.value =
  100. studentIds.value[
  101. studentIds.value.indexOf(currentStudentId.value) + (next ? 1 : -1)
  102. ];
  103. }
  104. if (!currentStudentId.value) return; // 无currentStudentId不调用
  105. store.status.markedCount =
  106. studentIds.value.indexOf(currentStudentId.value) + 1;
  107. await updateTask();
  108. }
  109. onMounted(async () => {
  110. await updateSetting();
  111. await fetchTask(true, true); // mark-header 会调用 (watchEffect)
  112. });
  113. const realStudentId = computed(
  114. () => (isSingleStudent ? studentId : store.currentTask?.studentId) as string
  115. );
  116. const saveTaskToServer = async () => {
  117. const mkey = "save_task_key";
  118. message.loading({ content: "标记评卷任务...", key: mkey });
  119. const res = (await saveInspectedTask(
  120. currentStudentId.value + "",
  121. !isCurrentTagged.value + ""
  122. )) as any;
  123. if (res.data.success) {
  124. message.success({
  125. content: isCurrentTagged.value ? "取消标记成功" : "标记成功",
  126. key: mkey,
  127. duration: 2,
  128. });
  129. if (isCurrentTagged.value) {
  130. tagIds.value.splice(tagIds.value.indexOf(currentStudentId.value), 1);
  131. } else {
  132. tagIds.value.push(currentStudentId.value);
  133. }
  134. } else {
  135. console.log(res.data.message);
  136. message.error({ content: res.data.message, key: mkey, duration: 10 });
  137. }
  138. };
  139. const renderError = () => {
  140. store.currentTask = undefined;
  141. store.message = "加载失败,请重新加载。";
  142. };
  143. </script>
  144. <style scoped>
  145. .my-container {
  146. width: 100%;
  147. overflow: clip;
  148. }
  149. </style>