index.vue 7.0 KB

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