index.vue 6.6 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208209210211212213214215216217218219220221222223224225226227228229230231232233234235
  1. <script setup lang="ts">
  2. import {useUserStore} from "@/store/modules/user";
  3. import {GradeList} from "@/api/grade/types";
  4. import {TrainingItem, TrainingParams} from "@/api/training/types";
  5. import {getGradeSelect} from "@/api/grade";
  6. import {getTrainingLists} from "@/api/training";
  7. import {trimInput} from "@/utils";
  8. const userStore = useUserStore();
  9. defineOptions({
  10. name: "TrainingIndex",
  11. inheritAttrs: false,
  12. });
  13. const pageParams: TrainingParams = reactive({
  14. school_id: userStore.schoolId,
  15. is_formal: 1, // 1正式,2非正式
  16. grade_id: 0,
  17. page: 1,
  18. page_size: 10,
  19. // 学生名称/手机号
  20. search: "",
  21. });
  22. const isFormal = ref(1);
  23. /**
  24. * 班级数据
  25. */
  26. const gradeData = ref<GradeList[]>();
  27. async function getGradeData(schoolId: number) {
  28. getGradeSelect(schoolId)
  29. .then(({data}) => {
  30. gradeData.value = data;
  31. gradeData.value?.unshift({id: 0, name: "全部班级"});
  32. })
  33. .catch((error) => {
  34. gradeData.value = [];
  35. gradeData.value?.unshift({id: 0, name: "全部班级"});
  36. console.log(error.message);
  37. });
  38. }
  39. const dataMessage = ref("加载中...");
  40. const trainingData = ref<TrainingItem[]>();
  41. const trainingCount = ref(0);
  42. async function getTrainingData(schoolId: number) {
  43. pageParams.school_id = schoolId;
  44. isFormal.value = pageParams.is_formal;
  45. getTrainingLists(pageParams)
  46. .then(({data}) => {
  47. const {count, lists} = data;
  48. trainingData.value = lists;
  49. trainingCount.value = count;
  50. if (!(count && count > 0 && lists.length > 0)) {
  51. dataMessage.value = "没有符合搜索条件的记录!";
  52. if (pageParams.is_formal == 2 && pageParams.search == "") {
  53. dataMessage.value = "还没有任何体验用户训练记录!";
  54. }
  55. if (pageParams.is_formal == 1 && pageParams.grade_id == 0 && pageParams.search == "") {
  56. dataMessage.value = "还没有任何学生训练记录!";
  57. }
  58. }
  59. })
  60. .catch((error) => {
  61. dataMessage.value = error.message;
  62. console.log(error.message);
  63. });
  64. }
  65. function getTrainingSearch() {
  66. getTrainingData(userStore.schoolId);
  67. }
  68. function alertError() {
  69. ElMessage.error("本次训练采集的数据不足,无法分析并生成有效报告!");
  70. }
  71. onMounted(() => {
  72. getGradeData(userStore.schoolId);
  73. getTrainingData(userStore.schoolId);
  74. });
  75. // watch(
  76. // () => userStore.schoolId,
  77. // (newValue) => {
  78. // // 学校切换后重新获取
  79. // getGradeData(newValue);
  80. // getTrainingData(newValue);
  81. // }
  82. // );
  83. </script>
  84. <template>
  85. <div class="training-container">
  86. <!-- 学生查找 -->
  87. <div class="training-search">
  88. <el-select v-model="pageParams.is_formal" placeholder="正式学生/体验用户" size="large">
  89. <el-option key="1" :value="Number(1)" label="正式学生"></el-option>
  90. <el-option key="2" :value="Number(2)" label="体验用户"></el-option>
  91. </el-select>
  92. <el-select v-if="pageParams.is_formal == 1" v-model="pageParams.grade_id" placeholder="请选择班级" size="large">
  93. <el-option v-for="item in gradeData" :key="item.id" :label="item.name" :value="item.id" />
  94. </el-select>
  95. <el-input
  96. v-model="pageParams.search"
  97. placeholder="请输入学生名称或手机号码"
  98. size="large"
  99. @input="(value:string) => (pageParams.search = trimInput(value))" />
  100. <el-button size="large" type="primary" @click="getTrainingSearch()">查找</el-button>
  101. </div>
  102. <!-- 学生数据 -->
  103. <div class="list-table">
  104. <el-table :data="trainingData" style="width: 100%">
  105. <el-table-column align="center" label="序号" max-width="120" type="index" />
  106. <el-table-column prop="name" label="学生名称" align="center" />
  107. <el-table-column v-if="isFormal == 1" prop="phone" label="手机号码" align="center" />
  108. <el-table-column prop="device_name" label="训练方式" align="center" />
  109. <el-table-column prop="create_time" label="训练时间" align="center" />
  110. <el-table-column prop="play_time" label="训练时长" align="center" />
  111. <el-table-column v-if="isFormal == 1" prop="grade_name" label="所在班级" align="center" />
  112. <el-table-column label="操作" align="center" min-width="120">
  113. <template #default="scope">
  114. <template v-if="scope.row.report_status == 1">
  115. <router-link
  116. :to="
  117. '/training/result?id=' +
  118. scope.row.game_record_id +
  119. '&type=' +
  120. scope.row.device_id +
  121. '&formal=' +
  122. isFormal
  123. "
  124. class="table-btn"
  125. >报告详情</router-link
  126. >
  127. </template>
  128. <button v-else class="table-btn disabled" @click="alertError()">报告详情</button>
  129. </template>
  130. </el-table-column>
  131. <!-- 无数据插槽 -->
  132. <template #empty>
  133. <div class="empty">
  134. <img src="../../assets/empty.png" alt="数据为空" />
  135. <p>{{ dataMessage }}</p>
  136. </div>
  137. </template>
  138. </el-table>
  139. </div>
  140. <pagination
  141. v-if="trainingCount > 0"
  142. v-model:total="trainingCount"
  143. v-model:page="pageParams.page"
  144. v-model:limit="pageParams.page_size"
  145. @pagination="getTrainingSearch()" />
  146. </div>
  147. </template>
  148. <style lang="scss" scoped>
  149. .training-container {
  150. position: relative;
  151. padding: 20px 30px;
  152. }
  153. .training-search {
  154. margin-bottom: 20px;
  155. font-size: 16px;
  156. .el-select {
  157. width: 200px;
  158. margin: 0 20px 0 0;
  159. }
  160. .el-input {
  161. width: 250px;
  162. margin: 0;
  163. }
  164. :deep(.el-input__inner) {
  165. font-size: 16px;
  166. }
  167. .el-button {
  168. padding: 0 26px;
  169. margin: 0 20px;
  170. font-size: 16px;
  171. background: #4284f2;
  172. border-radius: 10px;
  173. }
  174. }
  175. :deep(.el-input__wrapper) {
  176. background: #fff;
  177. border-radius: 12px;
  178. box-shadow: none !important;
  179. }
  180. :deep(.el-select) {
  181. --el-select-input-focus-border-color: none !important;
  182. }
  183. :deep(.el-select .el-input__wrapper.is-focus) {
  184. box-shadow: none !important;
  185. }
  186. :deep(.el-select:hover:not(.el-select--disabled) .el-input__wrapper) {
  187. box-shadow: none !important;
  188. }
  189. :deep(.el-table .el-table__header .el-table__cell .cell) {
  190. overflow: visible;
  191. white-space: nowrap;
  192. }
  193. :deep(.el-table th.el-table__cell) {
  194. background: #e9ebee;
  195. }
  196. .list-table {
  197. overflow: hidden;
  198. background: #fff;
  199. border-radius: 25px;
  200. .table-btn {
  201. display: inline-block;
  202. height: 38px;
  203. padding: 0 15px;
  204. line-height: 38px;
  205. color: #fff;
  206. background: #4284f2;
  207. border-radius: 10px;
  208. &.disabled {
  209. background: #bfbfbf;
  210. }
  211. }
  212. }
  213. .empty {
  214. padding: 200px 0;
  215. }
  216. </style>