vip.tsx 19 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208209210211212213214215216217218219220221222223224225226227228229230231232233234235236237238239240241242243244245246247248249250251252253254255256257258259260261262263264265266267268269270271272273274275276277278279280281282283284285286287288289290291292293294295296297298299300301302303304305306307308309310311312313314315316317318319320321322323324325326327328329330331332333334335336337338339340341342343344345346347348349350351352353354355356357358359360361362363364365366367368369370371372373374375376377378379380381382383384385386387388389390391392393394395396397398399400401402403404405406407408409410411412413414415416417418419420421422423424425426427428429430431432433434435436437438439440441442443444445446447448449450451452453454455456457458459460461462463464465466467468469470471472473474475476477478479480481482483484485486487488489490491492493494495496497498499500501502503504505506507508509510511512513514515516517518519520521522523524525526527528529530531532533534535536537538539540541542543544545546547548549550551552553554555556557558559560561562563564565566567568569570571572573574575576577578579580581
  1. import Calendar from '@/business-components/calendar'
  2. import request from '@/helpers/request'
  3. import { state } from '@/state'
  4. import dayjs from 'dayjs'
  5. import {
  6. ActionSheet,
  7. Button,
  8. Cell,
  9. CellGroup,
  10. Dialog,
  11. Stepper,
  12. Sticky,
  13. Tag,
  14. Popup,
  15. Toast,
  16. Icon
  17. } from 'vant'
  18. import { defineComponent } from 'vue'
  19. import { getWeekCh } from '@/helpers/utils'
  20. import styles from './practice.module.less'
  21. import { orderStatus } from '@/views/order-detail/orderStatus'
  22. import ColResult from '@/components/col-result'
  23. import { tradeOrder } from '@/student/trade/tradeOrder'
  24. import icon3 from '../images/icon3.png'
  25. import Tips from './tips'
  26. export default defineComponent({
  27. name: 'vip',
  28. props: {
  29. userInfo: {
  30. type: Object,
  31. default: {}
  32. }
  33. },
  34. data() {
  35. const query = this.$route.query
  36. return {
  37. teacherId: query.teacherId,
  38. subjectId: query.subjectId,
  39. teacherSubjectList: [],
  40. subjectStatus: false,
  41. subjectInfo: {
  42. subjectPrice: 0,
  43. courseMinutes: 0,
  44. id: null,
  45. subjectName: '',
  46. subjectId: 0
  47. },
  48. courseNum: 4,
  49. calendarStatus: false,
  50. calendarList: [] as any,
  51. selectCourseList: [] as any,
  52. coursePlanStatus: false,
  53. selectStatus: false,
  54. coursePlanList: [] as any,
  55. calendarDate: dayjs().add(1, 'day').toDate() as Date, // 日历当前时间
  56. settingStatus: true, // 是否设置VIP定制课
  57. loadDataStatus: true // 是否加载数据
  58. }
  59. },
  60. async mounted() {
  61. try {
  62. this.loadDataStatus = true
  63. const res = await request.get(
  64. '/api-student/courseSchedule/getTeacherSubjectPrice',
  65. {
  66. params: {
  67. teacherId: this.teacherId
  68. }
  69. }
  70. )
  71. const result = res.data || []
  72. if (result.length > 0) {
  73. const userSubjectId = this.subjectId || state.user.data?.subjectId
  74. const findItem = result.find((item: any) => {
  75. return item.subjectId === Number(userSubjectId)
  76. })
  77. // 判断是否有跟学生相同的科目,如果没有则默认取第一个
  78. const tempRes = findItem || result[0]
  79. const { subjectName, subjectPrice, courseMinutes, subjectId, id } = tempRes
  80. this.subjectInfo = {
  81. subjectPrice,
  82. id,
  83. courseMinutes,
  84. subjectName,
  85. subjectId
  86. }
  87. result.forEach((item: any) => {
  88. item.name = item.subjectName
  89. })
  90. this.teacherSubjectList = result
  91. this.getList()
  92. this.onBuy(true)
  93. this.settingStatus = true
  94. } else {
  95. this.settingStatus = false
  96. }
  97. // 判断如果是审核的则不显示
  98. const resVersion = await request.post('/api-teacher/open/appVersion', {
  99. data: {
  100. platform:
  101. state.platformType === 'STUDENT' ? 'ios-student' : 'ios-teacher',
  102. version: state.version
  103. }
  104. })
  105. this.settingStatus = resVersion.data.check ? false : true
  106. this.loadDataStatus = false
  107. } catch {
  108. this.loadDataStatus = false
  109. }
  110. },
  111. computed: {
  112. showSelectList() {
  113. const arr: any = this.selectCourseList
  114. let list = [...arr]
  115. list.forEach((item: any) => {
  116. item.title =
  117. dayjs(item.startTime).format('YYYY-MM-DD') +
  118. ' ' +
  119. getWeekCh(dayjs(item.startTime).day()) +
  120. ' ' +
  121. item.start +
  122. '~' +
  123. item.end
  124. })
  125. return list
  126. },
  127. selectType() {
  128. // 循环次数是否足够
  129. return this.selectCourseList.length < this.courseNum
  130. ? 'noEnough'
  131. : 'enough'
  132. }
  133. },
  134. methods: {
  135. async onSubmit() {
  136. if (this.selectCourseList.length <= 0) {
  137. Toast('请选择课程时间')
  138. return
  139. }
  140. if (this.selectCourseList.length < this.courseNum) {
  141. this.selectStatus = true
  142. return
  143. }
  144. await this._lookCourse()
  145. },
  146. async getList(date?: Date) {
  147. try {
  148. const tempDate = date || dayjs().add(1, 'day').toDate()
  149. let params = {
  150. day: dayjs(tempDate).format('DD'),
  151. month: dayjs(tempDate).format('MM'),
  152. year: dayjs(tempDate).format('YYYY')
  153. }
  154. let res = await request.post(
  155. '/api-student/courseSchedule/createPracticeCourseCalendar',
  156. {
  157. data: {
  158. ...params,
  159. teacherSubjectPriceId: this.subjectInfo.id,
  160. studentId: state.user.data?.userId,
  161. teacherId: this.teacherId
  162. }
  163. }
  164. )
  165. const result = res.data || []
  166. let tempObj = {}
  167. result.forEach((item: any) => {
  168. tempObj[item.date] = item
  169. })
  170. this.calendarList = tempObj
  171. this.calendarStatus = result.length > 0
  172. } catch {}
  173. },
  174. onSelectDay(obj: any) {
  175. const result = obj || []
  176. let list = [...this.selectCourseList] as any
  177. result.forEach((item: any) => {
  178. const isExist = list.some(
  179. (course: any) => course.startTime === item.startTime
  180. )
  181. !isExist && list.push({ ...item })
  182. })
  183. // 去掉不在
  184. let tempList: any[] = []
  185. list.forEach((item: any) => {
  186. const isExist = result.some(
  187. (course: any) => course.startTime === item.startTime
  188. )
  189. isExist && tempList.push(item)
  190. })
  191. // 对数组进行排序
  192. tempList.sort((first: any, second: any) => {
  193. if (first.startTime > second.startTime) return 1
  194. if (first.startTime < second.startTime) return -1
  195. return 0
  196. })
  197. console.log(tempList, 'list')
  198. this.selectCourseList = [...tempList] as any
  199. },
  200. onCloseTag(item: any) {
  201. Dialog.confirm({
  202. title: '提示',
  203. message: '您是否要删除该选择的课程?',
  204. confirmButtonColor: 'var(--van-primary)'
  205. }).then(() => {
  206. const index = this.selectCourseList.findIndex(
  207. (course: any) => course.startTime === item.startTime
  208. )
  209. this.selectCourseList.splice(index, 1)
  210. })
  211. },
  212. async _lookCourse(callBack?: Function) {
  213. try {
  214. let times = [] as any
  215. this.selectCourseList.forEach((item: any) => {
  216. times.push({
  217. startTime: item.startTime,
  218. endTime: item.endTime
  219. })
  220. })
  221. const res = await request.post(
  222. '/api-student/courseGroup/lockCourseToCache',
  223. {
  224. data: {
  225. courseNum: this.courseNum,
  226. courseType: 'PRACTICE',
  227. loop: this.selectType === 'noEnough' ? 1 : 0,
  228. teacherId: this.teacherId,
  229. timeList: [...times]
  230. }
  231. }
  232. )
  233. const result = res.data || []
  234. result.forEach((item: any, index: number) => {
  235. this.coursePlanList[index] = {
  236. ...this.coursePlanList[index],
  237. startTime: item.startTime,
  238. endTime: item.endTime,
  239. classNum: index + 1
  240. }
  241. })
  242. this.coursePlanStatus = true
  243. this.selectStatus = true
  244. callBack && callBack()
  245. } catch (e: any) {
  246. // 报错时需要重置日历表的数据
  247. const message = e.message
  248. Dialog.alert({
  249. title: '提示',
  250. confirmButtonColor: 'var(--van-primary)',
  251. message
  252. }).then(() => {
  253. this.getList(this.calendarDate || new Date())
  254. this.selectCourseList = []
  255. this.selectStatus = false
  256. })
  257. }
  258. },
  259. async onReset() {
  260. // 是否有锁课状态 或 是锁课类型的
  261. if (this.coursePlanStatus || this.selectType === 'enough') {
  262. this.selectStatus = false
  263. setTimeout(() => {
  264. this.coursePlanList = []
  265. }, 500)
  266. } else if (this.selectType === 'noEnough') {
  267. this.selectStatus = false
  268. }
  269. setTimeout(() => {
  270. this.coursePlanStatus = false
  271. }, 500)
  272. },
  273. async onSure() {
  274. const status = this.coursePlanStatus
  275. await this._lookCourse(() => {
  276. if (status) {
  277. this.selectStatus = false
  278. this.onBuy()
  279. }
  280. })
  281. },
  282. async onBuy(goTo?: boolean) {
  283. try {
  284. const res = await request.post(
  285. '/api-student/userOrder/getPendingOrder',
  286. {
  287. data: {
  288. goodType: 'VIP',
  289. bizId: this.teacherId
  290. }
  291. }
  292. )
  293. const subjectInfo = this.subjectInfo
  294. const tempCourseList = [...this.coursePlanList]
  295. // console.log(this.coursePlanList)
  296. tempCourseList.forEach((item: any) => {
  297. item.classDate = dayjs(item.startTime).format('YYYY-MM-DD')
  298. item.title = `${dayjs(item.startTime).format(
  299. 'YYYY-MM-DD'
  300. )} ${getWeekCh(dayjs(item.startTime).day())} ${dayjs(
  301. item.startTime
  302. ).format('HH:mm')}~${dayjs(item.endTime).format('HH:mm')}`
  303. })
  304. orderStatus.orderObject.orderType = 'VIP'
  305. orderStatus.orderObject.orderName = subjectInfo.subjectName + 'VIP定制课'
  306. orderStatus.orderObject.orderDesc = subjectInfo.subjectName + 'VIP定制课'
  307. orderStatus.orderObject.actualPrice = Number(
  308. (this.courseNum * subjectInfo.subjectPrice).toFixed(2)
  309. )
  310. orderStatus.orderObject.orderNo = ''
  311. orderStatus.orderObject.orderList = [
  312. {
  313. orderType: 'VIP',
  314. goodsName: subjectInfo.subjectName + 'VIP定制课',
  315. courseGroupName: subjectInfo.subjectName + 'VIP定制课',
  316. courseIntroduce: subjectInfo.subjectName + 'VIP定制课',
  317. subjectId: subjectInfo.subjectId,
  318. singleCourseMinutes: subjectInfo.courseMinutes,
  319. courseNum: this.courseNum,
  320. coursePrice: (this.courseNum * subjectInfo.subjectPrice).toFixed(2),
  321. teacherName:
  322. this.userInfo.username || `游客${this.userInfo.userId || ''}`,
  323. teacherId: this.userInfo.userId,
  324. starGrade: this.userInfo.starGrade,
  325. avatar: this.userInfo.heardUrl,
  326. classTime: tempCourseList
  327. }
  328. ]
  329. const result = res.data
  330. if (result) {
  331. Dialog.confirm({
  332. title: '提示',
  333. message: '您有一个未支付的订单,是否继续支付?',
  334. confirmButtonColor: '#269a93',
  335. cancelButtonText: '取消订单',
  336. confirmButtonText: '继续支付'
  337. })
  338. .then(async () => {
  339. tradeOrder(result, this.routerTo)
  340. // this.routerTo()
  341. })
  342. .catch(() => {
  343. Dialog.close()
  344. // 只用取消订单,不用做其它处理
  345. this.cancelPayment(result.orderNo)
  346. })
  347. } else {
  348. !goTo && this.routerTo()
  349. }
  350. } catch {}
  351. },
  352. routerTo() {
  353. this.$router.push({
  354. path: '/orderDetail',
  355. query: {
  356. orderType: 'VIP'
  357. }
  358. })
  359. },
  360. async cancelPayment(orderNo: string) {
  361. try {
  362. await request.post('/api-student/userOrder/orderCancel', {
  363. data: {
  364. orderNo
  365. }
  366. })
  367. // this.routerTo()
  368. } catch {}
  369. }
  370. },
  371. render() {
  372. return (
  373. <>
  374. {!this.loadDataStatus &&
  375. (this.settingStatus ? (
  376. <>
  377. <div class={styles.practice}>
  378. <Tips title='什么是VIP定制课?' content='VIP定制课程采用一对一专属授课模式,每节课时长为45分钟。课程内容根据学生的具体需求量身打造,旨在全面提升学生的个人技能与表现。不论是希望在乐器演奏技巧上取得突破,如提高指法精准度、气息控制能力或节奏掌握等;还是为即将到来的重要活动、比赛或考级做充分准备,我们都能提供高度匹配的教学方案。此外,教学进度将根据每位学员的学习吸收情况灵活调整,确保每个人都能在最适合自己的节奏中稳步前进,扎实提升个人能力。' />
  379. <CellGroup class={styles.group} border={false}>
  380. <Cell
  381. title="选择专业"
  382. isLink
  383. value={this.subjectInfo.subjectName}
  384. onClick={() => (this.subjectStatus = true)}
  385. />
  386. {this.subjectInfo.subjectPrice > 0 && (
  387. <Cell
  388. title="VIP定制课收费"
  389. v-slots={{
  390. default: () => (
  391. <div class={styles.price}>
  392. <span>
  393. ¥
  394. {(this as any).$filters.moneyFormat(
  395. this.subjectInfo.subjectPrice
  396. )}
  397. </span>
  398. /{this.subjectInfo.courseMinutes}分钟
  399. </div>
  400. )
  401. }}
  402. />
  403. )}
  404. <Cell
  405. title="课时数"
  406. v-slots={{
  407. default: () => (
  408. <Stepper
  409. v-model={this.courseNum}
  410. theme="round"
  411. max={12}
  412. min={1}
  413. buttonSize={22}
  414. onChange={() => {
  415. this.selectCourseList = []
  416. }}
  417. />
  418. )
  419. }}
  420. />
  421. </CellGroup>
  422. {this.calendarStatus && (
  423. <div class={styles.group}>
  424. <Calendar
  425. selectList={this.selectCourseList}
  426. list={this.calendarList}
  427. maxDays={this.courseNum}
  428. nextMonth={(date: Date) => this.getList(date)}
  429. prevMonth={(date: Date) => this.getList(date)}
  430. selectDay={this.onSelectDay}
  431. v-model:calendarDate={this.calendarDate}
  432. />
  433. </div>
  434. )}
  435. {this.showSelectList.length > 0 && <Cell
  436. class={[styles.arrangeCell]}
  437. v-slots={{
  438. title: () => (
  439. <div class={styles.rTitle}>
  440. <span>已选择课程时间</span>
  441. </div>
  442. ),
  443. label: () => (
  444. <div class={styles.rTag}>
  445. {this.showSelectList.map((item: any) => (
  446. <>
  447. <Tag
  448. plain
  449. round
  450. closeable
  451. size="large"
  452. type="primary"
  453. class={styles.tag}
  454. onClose={() => this.onCloseTag(item)}
  455. >
  456. {item.title}
  457. </Tag>
  458. <br />
  459. </>
  460. ))}
  461. </div>
  462. )
  463. }}
  464. ></Cell>}
  465. <Popup show={this.selectStatus} class={styles.selectPopup}>
  466. <div class={styles.selectContainer}>
  467. <div class={styles.rTitle}>
  468. <span>提示</span>
  469. </div>
  470. <div class={styles.selectPopupContent}>
  471. <p class={styles.desc}>
  472. {this.selectType === 'noEnough' &&
  473. !this.coursePlanStatus
  474. ? '您所选择的上课时间未达到您输入的课时数,系统根据已选时间将自动按周顺延排课。'
  475. : '您已选择以下上课时间段,时间段会暂时锁定,锁定期间学员不可购买该时间段课程。'}
  476. </p>
  477. {this.coursePlanList &&
  478. this.coursePlanList.length > 0 &&
  479. this.coursePlanStatus && (
  480. <p class={styles.times}>
  481. {this.coursePlanList.map((item: any) => (
  482. <span>
  483. {dayjs(item.startTime || new Date()).format(
  484. 'YYYY-MM-DD'
  485. )}{' '}
  486. {dayjs(item.startTime || new Date()).format(
  487. 'HH:mm'
  488. )}
  489. ~
  490. {dayjs(item.endTime || new Date()).format(
  491. 'HH:mm'
  492. )}
  493. </span>
  494. ))}
  495. </p>
  496. )}
  497. </div>
  498. <div class={styles.selectBtn}>
  499. <Button
  500. class={styles.btn}
  501. type="primary"
  502. round
  503. block
  504. plain
  505. onClick={this.onReset}
  506. >
  507. {this.selectType === 'noEnough'
  508. ? '继续选择'
  509. : '重新选择'}
  510. </Button>
  511. <Button
  512. class={styles.btn}
  513. type="primary"
  514. round
  515. block
  516. onClick={this.onSure}
  517. >
  518. 确认
  519. </Button>
  520. </div>
  521. </div>
  522. </Popup>
  523. <ActionSheet
  524. show={this.subjectStatus}
  525. actions={this.teacherSubjectList}
  526. cancelText="取消"
  527. closeOnClickAction
  528. onCancel={() => (this.subjectStatus = false)}
  529. onSelect={(item: any) => {
  530. const {
  531. subjectName,
  532. subjectPrice,
  533. courseMinutes,
  534. id,
  535. subjectId
  536. } = item
  537. this.subjectInfo = {
  538. subjectPrice,
  539. id,
  540. courseMinutes,
  541. subjectName,
  542. subjectId
  543. }
  544. this.subjectStatus = false
  545. }}
  546. />
  547. </div>
  548. <div
  549. class={['btnGroup', styles.fixedBtn]}
  550. style={{ background: '#fff', paddingTop: '10px' }}
  551. >
  552. <Button block round type="primary" onClick={this.onSubmit}>
  553. 确认约课
  554. </Button>
  555. </div>
  556. </>
  557. ) : (
  558. <ColResult
  559. btnStatus={false}
  560. classImgSize="SMALL"
  561. tips="老师暂未开放VIP定制课"
  562. />
  563. ))}
  564. </>
  565. )
  566. }
  567. })