Stroop.vue 18 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208209210211212213214215216217218219220221222223224225226227228229230231232233234235236237238239240241242243244245246247248249250251252253254255256257258259260261262263264265266267268269270271272273274275276277278279280281282283284285286287288289290291292293294295296297298299300301302303304305306307308309310311312313314315316317318319320321322323324325326327328329330331332333334335336337338339340341342343344345346347348349350351352353354355356357358359360361362363364365366367368369370371372373374375376377378379380381382383384385386387388389390391392393394395396397398399400401402403404405406407408409410411412413414415416417418419420421422423424425426427428429430431432433434435436437438439440441442443444445446447448449450451452453454455456457458459460461462463464465466467468469470471472473474475476477478479480481482483484485486487488489490491492493494495496497498499500501502503504505506507508509510511512513514515516517518519520521522523524525526527528529530531532533534535536537538539540541542543544545546547548549550551552553554555556557558559560561562563564565566567568569570571572573574575576577578579580581582583584585586587588589590591592593594595596597598599600601602603604605606607608609610611612613614615616617618619620621622623624625626627628629630631632633634635636637638639640641642643644645646647648649650651652653654655656657658659660661662663664665666667668669670671672673674675676677678679680681682683684685686687688689690691692693694695696697698699700701
  1. <script lang="ts" setup>
  2. import { onMounted, onUnmounted, ref } from 'vue'
  3. import CpdmMessage from '../components/CpdmMessage.vue'
  4. import { useRoute, useRouter } from 'vue-router';
  5. import { userInfoStore } from '@/stores'
  6. import { arMA } from 'date-fns/locale';
  7. import { getCurrentTimeApi, saveStroopApi } from '@/api/plan';
  8. import { userPlanDetailApi } from '@/api/home';
  9. import { getUnread, planNumGet } from '@/utils/test';
  10. import { ElMessage } from 'element-plus';
  11. const route = useRoute()
  12. const router = useRouter()
  13. //获取个人信息
  14. const userInfo = userInfoStore()
  15. const testFlag = ref<boolean>(true)
  16. const planId = ref<string>('');
  17. const planName = ref<string>('');
  18. const flag = ref<string>('');
  19. const flagName = ref<string>('');
  20. //是否是练习测试
  21. const formalTest = ref<string>('')
  22. //测试总轮数
  23. const allCount = ref<number>(20)
  24. //数据列表
  25. const list = ref<any>({
  26. })
  27. //是否显示倒计时
  28. const timingShow = ref<boolean>(false)
  29. //倒计时字段
  30. const countDownStr = ref<string>('开始测试')
  31. //倒计时计时器1
  32. const time1 = ref<any>()
  33. //倒计时计时器2
  34. const time2 = ref<any>()
  35. //倒计时计时器3
  36. const time3 = ref<any>()
  37. const addFlag = ref<boolean>(false)
  38. //
  39. const fontFlag = ref<boolean>(false)
  40. //计时器
  41. const timeNum1 = ref<number>(5)
  42. //开始测试 测试到第几轮的标志
  43. const numberFlag = ref<number>(0)
  44. const numberFlagTem = ref<number>(0)
  45. //保存数据的格式
  46. const recordResult = ref<any>([])
  47. const startTime1 = ref<any>()
  48. const cpdmMe = ref<any>()
  49. const saveStart = ref<boolean>(false)
  50. onMounted(() => {
  51. //接收参数
  52. planId.value = route.params.planId as string;
  53. planName.value = route.params.planName as string;
  54. flag.value = route.params.flag as string
  55. flagName.value = route.params.flagName as string
  56. formalTest.value = route.params.formalTest as string
  57. //组合数据后
  58. if (formalTest.value == '0') {
  59. allCount.value = 20
  60. } else {
  61. allCount.value = 90
  62. }
  63. formatterData();
  64. //开始随机数据进行
  65. //开始显示倒计时
  66. countdown()
  67. //监听电脑按键
  68. document.addEventListener("keydown", listenFun)
  69. getStartTime()
  70. })
  71. const getStartTime = async () => {
  72. let res: any = await getCurrentTimeApi();
  73. // let date = new Date()
  74. startTime1.value = res.data;
  75. // startTime.value = format(date, "yyyy-MM-dd HH:mm:ss");
  76. }
  77. const listenFun = (e: any) => {
  78. if (list.value.length > 0 && !timingShow.value && !addFlag.value) {
  79. if (e.key == 'f') {
  80. clickColor(e.key)
  81. }
  82. if (e.key == 'g') {
  83. clickColor(e.key)
  84. }
  85. if (e.key == 'h') {
  86. clickColor(e.key)
  87. }
  88. if (e.key == 'j') {
  89. clickColor(e.key)
  90. }
  91. }
  92. }
  93. onUnmounted(() => {
  94. //点击测试
  95. clearInterval(time1.value)
  96. clearTimeout(time2.value)
  97. clearTimeout(time3.value)
  98. document.removeEventListener("keydown", listenFun)
  99. })
  100. const countdown = () => {
  101. // countDownStr.value = '开始测试'
  102. timingShow.value = true;
  103. time1.value = setInterval(() => {
  104. if (timeNum1.value == 0) {
  105. countDownStr.value = `开始`;
  106. //当等于0的时候
  107. //开始调用开始数据
  108. setTimeout(() => {
  109. timingShow.value = false;
  110. //调用开始方法
  111. startFun()
  112. }, 1000);
  113. if (time1.value) clearInterval(time1.value);
  114. } else {
  115. countDownStr.value = `${timeNum1.value}`;
  116. //时间倒计时计数器
  117. timeNum1.value--
  118. }
  119. //倒计时
  120. }, 1000)
  121. }
  122. //开始测试方法
  123. const startFun = () => {
  124. fontFlag.value = true;
  125. showFont()
  126. }
  127. //开始显示变颜色的文字
  128. const currentDate = ref<any>()
  129. const showFont = () => {
  130. currentDate.value = new Date().getTime()
  131. time2.value = setTimeout(() => {
  132. clickColor('none')
  133. //开始调用方法
  134. }, 2000);
  135. }
  136. //开始显示 + 号
  137. const showAdd = () => {
  138. addFlag.value = true;
  139. time3.value = setTimeout(() => {
  140. addFlag.value = false;
  141. clearInterval(time3.value)
  142. showFont()
  143. }, 2000)
  144. }
  145. //调用当前方法
  146. const clickColor = async (val: string) => {
  147. if (saveStart.value) {
  148. return
  149. }
  150. let useTime = 0
  151. //
  152. if (val == 'none') {
  153. useTime = 2000
  154. } else {
  155. // 当主动点击的时候
  156. let tt = new Date().getTime()
  157. useTime = tt - currentDate.value
  158. //清空文件
  159. clearTimeout(time2.value)
  160. }
  161. //当数是19时----就去保存了
  162. //修改文件
  163. let currentP = false
  164. if (list.value[numberFlag.value].color == 'red') {
  165. if (val == 'f') {
  166. currentP = true
  167. }
  168. } else if (list.value[numberFlag.value].color == 'yellow') {
  169. if (val == 'g') {
  170. currentP = true
  171. }
  172. } else if (list.value[numberFlag.value].color == 'blue') {
  173. if (val == 'h') {
  174. currentP = true
  175. }
  176. } else if (list.value[numberFlag.value].color == 'green') {
  177. if (val == 'j') {
  178. currentP = true
  179. }
  180. }
  181. let obj = {
  182. //按键
  183. key: val,
  184. //反应时长
  185. duration: useTime,
  186. //是否超时 --超时算是错误的
  187. isTimeOut: useTime > 700 ? true : false,
  188. //颜色和按键是否是对上了
  189. isColorCorrent: currentP,
  190. //出现的选项
  191. showOptions: JSON.parse(JSON.stringify(list.value[numberFlag.value])),
  192. //已选的选项
  193. }
  194. //测试结果的集合
  195. recordResult.value.push(obj)
  196. //修改文本
  197. if (numberFlag.value < allCount.value - 1) {
  198. showAdd()
  199. numberFlag.value++
  200. numberFlagTem.value++
  201. } else {
  202. numberFlagTem.value++
  203. //开始保存了
  204. saveStart.value = true
  205. //开始调用-----
  206. //判断是测试还是正式测试
  207. if (formalTest.value == '0') {
  208. //判断测试是否通过
  209. //如果正确率大于80% 就跳转到
  210. //如果是测试计划的话
  211. let count = 0;
  212. for (let i = 0; i < recordResult.value.length; i++) {
  213. // recordResult.value[i].isColorCorrent =
  214. if (recordResult.value[i].isColorCorrent) {
  215. count++
  216. }
  217. }
  218. let score = parseInt(((count / recordResult.value.length) * 100).toFixed(2));
  219. ElMessage({
  220. type: 'info',
  221. //正确率
  222. message: `正确率为${score}%`
  223. })
  224. if (score > 60) {
  225. //正确率大于60 时
  226. //设置通过的标志
  227. userInfo.saveIspass(flag.value)
  228. setTimeout(() => {
  229. router.go(-1)
  230. }, 1000)
  231. } else {
  232. setTimeout(() => {
  233. router.go(-1)
  234. }, 1000)
  235. }
  236. } else {
  237. let testResult = {
  238. totalScore: '',
  239. }
  240. let count = 0
  241. for (let i = 0; i < recordResult.value.length; i++) {
  242. if (recordResult.value[i].isTimeOut) {
  243. count++
  244. }
  245. }
  246. testResult.totalScore = (count / allCount.value * 100).toFixed(2)
  247. let params = {
  248. beginTime: startTime1.value,
  249. orgName: userInfo.userInfo.orgName,
  250. orgNo: userInfo.userInfo.orgNo,
  251. planId: planId.value,
  252. planName: planName.value,
  253. taskFlag: flag.value,
  254. taskName: flagName.value,
  255. // testRecord: JSON.stringify(productData()),
  256. testRecord: JSON.stringify(recordResult.value),
  257. testResult: JSON.stringify(testResult),
  258. type: '1',
  259. userName: userInfo.userInfo.userName,
  260. userNo: userInfo.userInfo.userNo,
  261. }
  262. let res: any = await saveStroopApi(params)
  263. if (res.code == 200) {
  264. let paramsCu = {
  265. planId: planId.value,
  266. userNo: userInfo.userInfo.userNo
  267. }
  268. let temp: any = await userPlanDetailApi(paramsCu)
  269. let listT = temp.data;
  270. let listP = []
  271. listP = listT.filter((item: any) => {
  272. return item.isCompleted == '0' && item.contentType == '1'
  273. })
  274. //等--认知任务如果有未测试的话--需要跳转到测试计划页面
  275. //否则--跳转到测试计划页面
  276. if (listP.length > 0) {
  277. router.push({ name: 'plan' })
  278. } else {
  279. cpdmMe.value.open()
  280. planNumGet()
  281. getUnread()
  282. }
  283. } else {
  284. ElMessage({
  285. type: 'error',
  286. message: res.msg
  287. })
  288. }
  289. }
  290. //-----------------------------------
  291. //开始保存方法--保存成功后判断
  292. }
  293. }
  294. //组合数据
  295. const formatterData = () => {
  296. const listGrouop = [
  297. { type: 'depressed', name: '沮丧' },
  298. { type: 'depressed', name: '哀伤' },
  299. { type: 'depressed', name: '绝望' },
  300. { type: 'depressed', name: '忧郁' },
  301. { type: 'depressed', name: '孤寂' },
  302. { type: 'depressed', name: '沉重' },
  303. { type: 'depressed', name: '压抑' },
  304. { type: 'depressed', name: '焦虑' },
  305. { type: 'depressed', name: '落寞' },
  306. { type: 'depressed', name: '悲观' },
  307. { type: 'depressed', name: '愁苦' },
  308. { type: 'depressed', name: '无力' },
  309. { type: 'depressed', name: '低落' },
  310. { type: 'depressed', name: '迷惘' },
  311. { type: 'depressed', name: '消沉' },
  312. { type: 'depressed', name: '灰心' },
  313. { type: 'depressed', name: '苦楚' },
  314. { type: 'depressed', name: '厌世' },
  315. { type: 'depressed', name: '冷漠' },
  316. { type: 'depressed', name: '苦楚' },
  317. { type: 'anxiety', name: '紧张' },
  318. { type: 'anxiety', name: '不安' },
  319. { type: 'anxiety', name: '恐慌' },
  320. { type: 'anxiety', name: '担忧' },
  321. { type: 'anxiety', name: '烦躁' },
  322. { type: 'anxiety', name: '急躁' },
  323. { type: 'anxiety', name: '易怒' },
  324. { type: 'anxiety', name: '焦虑' },
  325. { type: 'anxiety', name: '心悸' },
  326. { type: 'anxiety', name: '忐忑' },
  327. { type: 'anxiety', name: '惊悸' },
  328. { type: 'anxiety', name: '紧迫感' },
  329. { type: 'anxiety', name: '压力' },
  330. { type: 'anxiety', name: '忧虑' },
  331. { type: 'anxiety', name: '惊惧' },
  332. { type: 'anxiety', name: '惶恐' },
  333. { type: 'anxiety', name: '恐惧' },
  334. { type: 'anxiety', name: '惊慌' },
  335. { type: 'anxiety', name: '畏惧' },
  336. { type: 'anxiety', name: '忧烦' },
  337. { type: 'neutral', name: '天空' },
  338. { type: 'neutral', name: '大地' },
  339. { type: 'neutral', name: '河流' },
  340. { type: 'neutral', name: '树木' },
  341. { type: 'neutral', name: '花朵' },
  342. { type: 'neutral', name: '阳光' },
  343. { type: 'neutral', name: '月亮' },
  344. { type: 'neutral', name: '星星' },
  345. { type: 'neutral', name: '山峰' },
  346. { type: 'neutral', name: '草原' },
  347. { type: 'neutral', name: '房屋' },
  348. { type: 'neutral', name: '道路' },
  349. { type: 'neutral', name: '书本' },
  350. { type: 'neutral', name: '笔记' },
  351. { type: 'neutral', name: '时间' },
  352. { type: 'neutral', name: '空间' },
  353. { type: 'neutral', name: '音乐' },
  354. { type: 'neutral', name: '电影' },
  355. { type: 'neutral', name: '旅行' },
  356. { type: 'neutral', name: '工作' },
  357. { type: 'neutral', name: '学习' },
  358. { type: 'neutral', name: '朋友' },
  359. { type: 'neutral', name: '家人' },
  360. { type: 'neutral', name: '生活' },
  361. { type: 'neutral', name: '动物' },
  362. { type: 'neutral', name: '植物' },
  363. { type: 'neutral', name: '季节' },
  364. { type: 'neutral', name: '天气' },
  365. { type: 'neutral', name: '日常' },
  366. { type: 'neutral', name: '平静' },
  367. ]
  368. let listNew = []
  369. // let colorList = ['红色', "黄色", "蓝色", "绿色"]
  370. let colorList = ['red', "yellow", "blue", "green"]
  371. for (let i = 0; i < listGrouop.length; i++) {
  372. for (let j = 0; j < colorList.length; j++) {
  373. let obj = {
  374. type: listGrouop[i].type,
  375. name: listGrouop[i].name,
  376. color: colorList[j]
  377. }
  378. listNew.push(obj)
  379. }
  380. }
  381. //随机数
  382. let listRandom = []
  383. for (let i = 0; i < allCount.value; i++) {
  384. let a = Math.floor(Math.random() * 280);
  385. listRandom.push(listNew[a])
  386. }
  387. // 修改文本
  388. // let listRandon =[]
  389. // for(let i=0;i<listNew.length;i++){
  390. // }
  391. //将组合好的数据
  392. list.value = listRandom;
  393. //长度280
  394. }
  395. </script>
  396. <template>
  397. <div class="activeTask">
  398. <div style="width: 50%; margin-left: 10%; margin-top: 24px; z-index: 10">
  399. <el-progress class="main_progress" color="linear-gradient(to right, hwb(17 58% 9%), #99ebee)"
  400. :text-inside="true" :stroke-width="48" :percentage="((numberFlagTem) / allCount) * 100">
  401. <span style="color: #000;font-size: 20px;">{{ numberFlagTem }}/{{ allCount }}</span>
  402. </el-progress>
  403. </div>
  404. <p v-if="timingShow" class="timingBox">{{ countDownStr }}</p>
  405. <div v-if="list.length > 0 && !timingShow" class="timingBox">
  406. <div v-if="(formalTest == '0' && numberFlag <= 19 && !addFlag) || (formalTest == '1' && numberFlag <= 89 && !addFlag)"
  407. :style="{ color: list[numberFlag].color }">{{
  408. list[numberFlag].name }}</div>
  409. </div>
  410. <div v-if="addFlag" style="text-align: center" class="timingBox">
  411. <img src="../assets/cognize/whiteFlag-new.png" alt="" style="width: 50px; height: 50px; margin-top: 20%" />
  412. </div>
  413. </div>
  414. <CpdmMessage ref="cpdmMe" />
  415. </template>
  416. <style lang="scss">
  417. .butt {
  418. width: 100px;
  419. text-align: center;
  420. border: 1px solid rgb(87, 172, 187);
  421. opacity: 1;
  422. border-radius: 10px;
  423. }
  424. .bgAdd {
  425. width: 150px;
  426. height: 400px;
  427. background: url("../assets/cognize/whiteFlag-new.png") no-repeat center;
  428. position: absolute;
  429. top: 25%;
  430. left: 50%;
  431. margin-left: -75px;
  432. }
  433. .activeTask {
  434. width: 100%;
  435. height: 100%;
  436. background: url("../assets/cognize/rememberAbilityTask.png") no-repeat center;
  437. background-size: cover;
  438. position: fixed;
  439. top: 0;
  440. left: 0;
  441. right: 0;
  442. bottom: 0;
  443. }
  444. .timingBox {
  445. font-size: 50px;
  446. font-weight: 700;
  447. position: absolute;
  448. top: 50%;
  449. left: 50%;
  450. transform: translate(-50%, -50%);
  451. color: black;
  452. font-size: 70px;
  453. text-align: center;
  454. line-height: 100px;
  455. margin: 0 auto;
  456. z-index: 10;
  457. }
  458. /* .active-img {
  459. position: absolute;
  460. width: 450px;
  461. height: 450px;
  462. top: 50%;
  463. left: 50%;
  464. transform: translate(-50%, -50%);
  465. z-index: 10;
  466. } */
  467. .btn-content {
  468. position: absolute;
  469. width: 60%;
  470. left: 50%;
  471. bottom: 10%;
  472. transform: translate(-50%, 0);
  473. text-align: center;
  474. z-index: 1000;
  475. }
  476. .scaleName {
  477. margin-top: 70px;
  478. background-size: cover;
  479. }
  480. .scaleButton {
  481. margin-top: 20px;
  482. margin-bottom: 20px;
  483. background-size: cover;
  484. background: rgb(87, 172, 187);
  485. border: 2px solid rgb(255, 255, 255);
  486. box-shadow: 0px 3px 6px rgba(0, 0, 0, 0.16);
  487. opacity: 1;
  488. border-radius: 39px;
  489. color: #ffffff;
  490. }
  491. .steering {
  492. width: 50%;
  493. position: absolute;
  494. bottom: 20%;
  495. left: 50%;
  496. transform: translateX(-50%);
  497. display: flex;
  498. justify-content: space-between;
  499. }
  500. .steering .left {
  501. width: 60px;
  502. height: 60px;
  503. background: url(../../assets/left.png) no-repeat center;
  504. }
  505. .steering .right {
  506. width: 60px;
  507. height: 60px;
  508. background: url(../../assets/right.png) no-repeat center;
  509. margin-left: 15px;
  510. }
  511. .guider {
  512. position: absolute;
  513. top: 0;
  514. bottom: 0;
  515. left: 0;
  516. right: 0;
  517. z-index: 999;
  518. background: rgba(0, 0, 0, 0.75);
  519. }
  520. .addImg {
  521. width: 50%;
  522. height: auto;
  523. z-index: 1000;
  524. }
  525. .shape_glass {
  526. display: flex;
  527. justify-content: center;
  528. align-items: center;
  529. width: 50%;
  530. height: auto;
  531. margin: 5vh auto 0;
  532. }
  533. .active {
  534. position: relative;
  535. }
  536. .active:after {
  537. background-color: #2ea598;
  538. }
  539. /* 设置动画后颜色 */
  540. .active:before {
  541. background-color: rgba(0, 168, 253, 0.2);
  542. }
  543. /* 设置动画 */
  544. .active:before,
  545. .active:after {
  546. content: "";
  547. width: 80px;
  548. height: 40px;
  549. position: absolute;
  550. left: 50%;
  551. top: 50%;
  552. margin-left: -40px;
  553. margin-top: -20px;
  554. border-radius: 10%;
  555. animation: warn 1.5s ease-out 0s infinite;
  556. }
  557. @keyframes warn {
  558. 0% {
  559. transform: scale(0.5);
  560. opacity: 1;
  561. }
  562. 30% {
  563. opacity: 1;
  564. }
  565. 100% {
  566. transform: scale(1.4);
  567. opacity: 0;
  568. }
  569. }
  570. .txt {
  571. width: 300px;
  572. font-weight: PingFang-SC-Medium;
  573. margin: 0 auto;
  574. font-size: 16px;
  575. color: #ffffff;
  576. position: absolute;
  577. left: 50%;
  578. top: 70%;
  579. transform: translateX(-50%);
  580. z-index: 1000;
  581. }
  582. .explain {
  583. padding: 6px 8px;
  584. border: 2px solid #0abdc4;
  585. border-radius: 5px;
  586. }
  587. .main_progress {
  588. padding-top: 40px;
  589. padding-left: 40px;
  590. color: linear-gradient(to right, #ebb19b, #99ebee);
  591. }
  592. .activeTask {
  593. width: 100%;
  594. height: 100%;
  595. background: url("../assets/cognize/Stroop.png") no-repeat center;
  596. background-size: cover;
  597. position: fixed;
  598. top: 0;
  599. left: 0;
  600. right: 0;
  601. bottom: 0;
  602. }
  603. </style>