天波h5前端应用
You can not select more than 25 topics Topics must start with a letter or number, can include dashes ('-') and can be up to 35 characters long.

582 lines
12KB

  1. <template>
  2. <div
  3. class="psychologicalScale-index"
  4. v-if="topic.length"
  5. :style="`padding-top: ${tip?'10vw':showLeftArrow ? '34vw' : '24vw'};`"
  6. >
  7. <van-nav-bar
  8. title="测评"
  9. :border="true"
  10. :left-arrow="true"
  11. @click-left="onNavBack"
  12. v-if="showLeftArrow"
  13. style="position: fixed; left: 0; top: 0; width: 100vw; height: 10vw"
  14. >
  15. <template #left>
  16. <van-icon name="arrow-left" size="23" style="padding: 0" />返回
  17. </template>
  18. </van-nav-bar>
  19. <div class="tip" v-if="tip">
  20. <div class="tip-title title">{{ title }}</div>
  21. <div class="tip-title">测评说明:</div>
  22. <div class="tip-content">
  23. 请仔细阅读每一道题,根据您自己的
  24. 真实状态,评估最近7天内你最接近的感觉,而不只是你今天的感觉
  25. </div>
  26. <div class="tip-title">注意事项:</div>
  27. <div class="tip-content">
  28. 题目共10题,选项无对错之分,不用过度思考,根据第一反应选择则可。
  29. </div>
  30. <div class="tip-content">
  31. 该测评结果仅供参考,不能以此作相关诊断。如有疑问,请咨询专业人员。
  32. </div>
  33. <div class="button">
  34. <van-button
  35. round
  36. type="info"
  37. style="width: 100%; height: 100%"
  38. @click="tip = false"
  39. >开始测评</van-button
  40. >
  41. </div>
  42. </div>
  43. <div class="" v-else>
  44. <div class="top" :style="`top: ${showLeftArrow ? '10vw' : '0'};`">
  45. <div class="schedule">
  46. <div class="progress">
  47. <van-progress
  48. :percentage="
  49. (select.length == 0 ? 0 : select.length / topic.length) * 100
  50. "
  51. stroke-width="10px"
  52. color="#638ee4"
  53. :show-pivot="false"
  54. v-if="topic.length"
  55. ></van-progress>
  56. </div>
  57. <div class="text">
  58. <span class="val">{{ select.length }}</span
  59. >/<span class="total">{{ topic.length }}</span>
  60. </div>
  61. </div>
  62. <div class="title">{{ title }}</div>
  63. </div>
  64. <div class="list_box">
  65. <div class="item" v-for="(item, index) in topic" :key="index">
  66. <div
  67. :class="[
  68. 'label',
  69. {
  70. error:
  71. (isError && fromSsjl && item.Id == '6') ||
  72. (isError && fromSsjl && item.Id == '8') ||
  73. (isError && fromSsjl && item.Id == '11'),
  74. },
  75. ]"
  76. >
  77. <span>{{ item.Title }}</span>
  78. <span>({{ item.Type }})</span>
  79. </div>
  80. <van-radio-group
  81. class="radio"
  82. v-model="item.select"
  83. v-if="item.Type == '单选题'"
  84. >
  85. <van-cell-group>
  86. <van-cell
  87. clickable
  88. v-for="(ite, idx) in item.Option"
  89. :key="idx"
  90. :title="ite.label"
  91. @click="onRadio(index, ite.value)"
  92. :title-class="[
  93. {
  94. cellTitle:
  95. (isError && fromSsjl && item.Id == '6') ||
  96. (isError && fromSsjl && item.Id == '8') ||
  97. (isError && fromSsjl && item.Id == '11'),
  98. },
  99. ]"
  100. >
  101. <template #right-icon>
  102. <van-radio :name="ite.value" />
  103. </template>
  104. </van-cell>
  105. </van-cell-group>
  106. </van-radio-group>
  107. <van-checkbox-group class="checkbox" v-model="item.select" v-else>
  108. <van-cell-group>
  109. <van-cell
  110. v-for="(ite, idx) in item.Option"
  111. clickable
  112. :key="idx"
  113. :title="ite.label"
  114. @click="onCheckbox(`checkboxes${item.Id}`, idx)"
  115. >
  116. <template #right-icon>
  117. <van-checkbox
  118. :name="ite.value"
  119. :ref="`checkboxes${item.Id}`"
  120. />
  121. </template>
  122. </van-cell>
  123. </van-cell-group>
  124. </van-checkbox-group>
  125. </div>
  126. </div>
  127. <div class="button">
  128. <van-button
  129. round
  130. type="info"
  131. :disabled="topic.length != select.length"
  132. style="width: 100%; height: 100%"
  133. @click="submit"
  134. >提交测评</van-button
  135. >
  136. </div>
  137. </div>
  138. </div>
  139. </template>
  140. <script>
  141. import axios from 'axios'
  142. import APICore from '@/api/core'
  143. export default {
  144. name: 'PsychologicalScale',
  145. data() {
  146. return {
  147. uid: '',
  148. keyCode: '',
  149. title: '',
  150. tip: true,
  151. topic: [], //题目列表
  152. select: [],
  153. isError: null, //是否是答题错误
  154. fromSsjl: this.$store.getters.fromSsjl,
  155. showLeftArrow: false,
  156. isModel: false,
  157. }
  158. },
  159. mounted() {
  160. //页面标题
  161. window.document.title = '测评'
  162. //页面传参
  163. let params = { ...this.$route.query }
  164. if (params.uid) {
  165. this.uid = params.uid
  166. // 缓存从随手精灵传过来的token
  167. this.$store.commit('ssjlToken', params.accessToken || '')
  168. if (params.appType) {
  169. this.$store.commit('appType', params.appType)
  170. }
  171. // 缓存从随手精灵传过来的标识
  172. this.$store.commit('fromSsjl', params.fromSsjl)
  173. // 是否显示 返回标签
  174. if (params.showLeftArrow) {
  175. this.showLeftArrow = true
  176. }
  177. this.fromUrl = params.fromUrl
  178. console.log('fromUrl', this.fromUrl)
  179. // 是否从建模页过来
  180. if (params.isModel) {
  181. this.isModel = true
  182. }
  183. //初始化
  184. this.init()
  185. } else {
  186. this.$toast('参数错误')
  187. }
  188. },
  189. watch: {
  190. selects(val) {
  191. let temp = JSON.parse(val)
  192. let tempt = temp.filter((item) => {
  193. if (item.select >= 0) {
  194. return true
  195. }
  196. })
  197. this.select = tempt
  198. },
  199. },
  200. computed: {
  201. selects() {
  202. return JSON.stringify(this.topic)
  203. },
  204. },
  205. methods: {
  206. async init() {
  207. if (!this.$route.query.accessToken) {
  208. // 如果当前url没有 accessToken, 获取token,并且存储到本地缓存里面
  209. let authToken = await this.getAuth()
  210. this.$store.commit('ssjlToken', authToken)
  211. }
  212. let re = await this.api(
  213. '/api/Question/GetScaleQuestion',
  214. {
  215. method: 'GET',
  216. sslVerify: false,
  217. withCredentials: false,
  218. params: {
  219. // uid: this.uid,
  220. code: 'EPDS',
  221. },
  222. },
  223. this.$store.getters.ssjlToken
  224. )
  225. if (re.success) {
  226. if (
  227. re.response &&
  228. re.response.ScaleQuestionDetails &&
  229. re.response.ScaleQuestionDetails.length
  230. ) {
  231. let code = re.response.Code
  232. this.title = re.response.Name
  233. this.keyCode = re.response.Code
  234. let temp = re.response.ScaleQuestionDetails
  235. let topic = []
  236. temp.forEach((item) => {
  237. let tempt = { ...item }
  238. if (!tempt.Type) {
  239. tempt.Type = '单选题'
  240. }
  241. let a1 = tempt.Option.split('&')
  242. let option = []
  243. for (let index = 0; index < a1.length; index++) {
  244. option.push({
  245. label: a1[index],
  246. value: index,
  247. })
  248. }
  249. tempt.Option = option
  250. tempt.value = ''
  251. topic.push(tempt)
  252. })
  253. this.topic = topic
  254. } else {
  255. this.$toast('问卷不存在')
  256. }
  257. } else {
  258. this.$toast(re.msg)
  259. }
  260. },
  261. async submit() {
  262. let temp = {
  263. uid: this.uid, //客户端系统用户id
  264. keyCode: this.keyCode, //问卷编号
  265. answer: [],
  266. }
  267. this.topic.forEach((item) => {
  268. temp.answer.push({
  269. qid: item.Id, //问卷题目编号
  270. value:
  271. typeof item.select == 'object'
  272. ? item.select.join(',')
  273. : item.select, //用户所选答案的 id
  274. })
  275. })
  276. let re = await this.api(
  277. '/api/Question/ScaleRsults',
  278. {
  279. method: 'POST',
  280. sslVerify: false,
  281. withCredentials: false,
  282. data: temp,
  283. },
  284. this.$store.getters.ssjlToken
  285. )
  286. if (re.success && re.response) {
  287. this.$toast('问卷提交成功', 3000)
  288. this.isError = false
  289. setTimeout(() => {
  290. let fromSsjl = this.$store.getters.fromSsjl === 'true'
  291. if (fromSsjl) {
  292. // c端跳转按c端返回逻辑处理
  293. this.onNavBack()
  294. } else if (this.isModel) {
  295. // 建模页跳转过来的要回建模页
  296. this.$router.replace({
  297. name: 'PsychologicalModeling',
  298. query: {
  299. uid: this.uid,
  300. },
  301. })
  302. } else {
  303. this.$router.replace({
  304. name: 'PsychologicalScaleDetail',
  305. query: {
  306. id: re.response.Id,
  307. uid: this.uid,
  308. accessToken: this.$store.getters.ssjlToken,
  309. showLeftArrow: this.showLeftArrow,
  310. },
  311. })
  312. }
  313. }, 3000)
  314. } else {
  315. //this.$toast(re.msg);
  316. this.$dialog.confirm({
  317. message: re.msg,
  318. showCancelButton: false,
  319. })
  320. this.isError = true
  321. }
  322. },
  323. onRadio(index, value) {
  324. let temp = { ...this.topic[index] }
  325. temp.select = value
  326. this.$set(this.topic, index, temp)
  327. },
  328. onCheckbox(key, index) {
  329. try {
  330. this.$refs[`${key}`][index].toggle()
  331. } catch (error) {
  332. console.log(error)
  333. }
  334. },
  335. onNavBack() {
  336. let fromSsjl = this.$store.getters.fromSsjl === 'true'
  337. if (fromSsjl) {
  338. let baseUrl = this.fromUrl
  339. window.location.href = `${baseUrl}/#/${
  340. this.$route.query.fromMenu || 'device'
  341. }`
  342. } else {
  343. window.history.go(-1)
  344. }
  345. },
  346. // 获取b端token
  347. getAuth() {
  348. let manufactorId = '5bf13062-a41e-4d00-ba14-1101aad12650'
  349. let that = this
  350. return new Promise((resolve, reject) => {
  351. APICore.getAuth({ manufactorId: manufactorId }).then((res) => {
  352. let data = res.data
  353. if (data.code === 0) {
  354. resolve(res.data.data)
  355. } else {
  356. that.$toast.fail(`${data.message}`)
  357. reject('')
  358. }
  359. })
  360. })
  361. },
  362. api(url, config, token) {
  363. setTimeout(() => {
  364. this.$toast.loading({
  365. message: '',
  366. forbidClick: true,
  367. duration: 1500,
  368. })
  369. }, 100)
  370. let baseUrl =
  371. process.env.NODE_ENV === 'production'
  372. ? 'https://dbmq.rzliot.com/auth_heart'
  373. : 'https://dbmq.rzliot.com/heart'
  374. return new Promise((res) => {
  375. axios({
  376. url: `${baseUrl}${url}`,
  377. ...config,
  378. // 增加请求头部 token
  379. headers: {
  380. AccessToken: token,
  381. },
  382. })
  383. .then((re) => {
  384. if (re) {
  385. if (re.data) {
  386. console.log(re.data)
  387. res(re.data)
  388. this.$toast.clear()
  389. return
  390. }
  391. this.$toast(`信息获取失败-${re.status}`)
  392. res(true)
  393. this.$toast.clear()
  394. return
  395. }
  396. this.$toast(`信息获取失败-${url}`)
  397. res(true)
  398. this.$toast.clear()
  399. })
  400. .catch((e) => {
  401. this.$toast(`信息获取失败-${url}`)
  402. res(true)
  403. this.$toast.clear()
  404. console.log(e)
  405. })
  406. })
  407. },
  408. },
  409. }
  410. </script>
  411. <style lang="scss">
  412. .van-toast--html, .van-toast--text{
  413. padding: 2vw !important;
  414. }
  415. .van-toast--html .van-toast__text, .van-toast--text .van-toast__text{
  416. font-size: 3.6vw !important;
  417. }
  418. .psychologicalScale-index {
  419. width: 100vw;
  420. min-height: 100vh;
  421. box-sizing: border-box;
  422. font-size: 3.6vw;
  423. text-align: left;
  424. background: #f4f4f4;
  425. padding-bottom: 8vw;
  426. padding-top: 24vw;
  427. .van-nav-bar__content {
  428. height: 10vw !important;
  429. .van-nav-bar__left,
  430. .van-nav-bar__title {
  431. font-size: 4vw !important;
  432. line-height: initial !important;
  433. .van-icon-arrow-left {
  434. font-size: 4vw !important;
  435. }
  436. }
  437. }
  438. .tip{
  439. width: 90vw;
  440. margin: 0 auto;
  441. margin-top: 5vw;
  442. padding: 10vw 3vw;
  443. border-radius: 2vw;
  444. background: white;
  445. .tip-title{
  446. font-size: 4vw;
  447. font-weight: bold;
  448. margin-top: 5vw;
  449. margin-bottom: 5vw;
  450. }
  451. .title{
  452. font-size: 4.5vw;
  453. text-align: center;
  454. margin-top: 0;
  455. }
  456. .tip-content{
  457. font-size: 3.7vw;
  458. text-indent: 5vw;
  459. margin-bottom: 2vw;
  460. &:last-child{
  461. margin-bottom: 0;
  462. }
  463. }
  464. .button{
  465. position: fixed;
  466. left: 0;
  467. right: 0;
  468. bottom: 10vw;
  469. margin: auto;
  470. }
  471. }
  472. .top {
  473. box-sizing: border-box;
  474. position: fixed;
  475. top: 0;
  476. display: flex;
  477. flex-direction: column;
  478. justify-content: space-between;
  479. padding: 3vw;
  480. width: 100vw;
  481. height: 24vw;
  482. background: white;
  483. z-index: 99;
  484. .schedule {
  485. display: flex;
  486. align-items: center;
  487. .progress {
  488. width: 85%;
  489. }
  490. .text {
  491. width: 15%;
  492. font-size: 4.5vw;
  493. text-align: center;
  494. .val {
  495. font-size: 5vw;
  496. font-weight: bold;
  497. }
  498. }
  499. }
  500. .title {
  501. color: #638ee4;
  502. }
  503. }
  504. .list_box {
  505. width: 100%;
  506. margin-top: 3vw;
  507. .item {
  508. margin-top: 3vw;
  509. padding: 3vw;
  510. background: white;
  511. .label {
  512. font-size: 4vw;
  513. // &.error {
  514. // color: red;
  515. // }
  516. }
  517. .van-cell {
  518. line-height: initial !important;
  519. padding: 3vw 2vw;
  520. .van-cell__title {
  521. font-size: 3.5vw !important;
  522. // &.cellTitle {
  523. // color: red;
  524. // }
  525. }
  526. .van-radio__icon {
  527. height: 100% !important;
  528. .van-icon {
  529. font-size: 3.5vw !important;
  530. }
  531. }
  532. }
  533. }
  534. }
  535. .button {
  536. width: 75vw;
  537. height: 14vw;
  538. margin: 0 auto;
  539. margin-top: 8vw;
  540. .van-button {
  541. font-size: 4vw !important;
  542. }
  543. .van-button--info {
  544. background-color: #638ee4;
  545. border: 1px solid #638ee4;
  546. }
  547. }
  548. }
  549. </style>