group-detail.tsx 13 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208209210211212213214215216217218219220221222223224225226227228229230231232233234235236237238239240241242243244245246247248249250251252253254255256257258259260261262263264265266267268269270271272273274275276277278279280281282283284285286287288289290291292293294295296297298299300301302303304305306307308309310311312313314315316317318319320321322323324325326327328329330331332333334335336337338339340341342343344345346347348349350351352353354355356357358359360361362363364365366367368369370371372373374375376377378379380381382383384385386387388389390391392393394395396397398399400401402403404405406407408
  1. import CoursePlanStep from '@/business-components/course-plan-step'
  2. import SectionDetail from '@/business-components/section-detail'
  3. import UserDetail from '@/business-components/user-detail'
  4. import request from '@/helpers/request'
  5. import dayjs from 'dayjs'
  6. import { Icon, Sticky, Button, Dialog, Toast, Popup } from 'vant'
  7. import { defineComponent } from 'vue'
  8. import styles from './group-detail.module.less'
  9. import iconTips from '@common/images/icon_tips.png'
  10. import { onSubmitZero, orderStatus } from '@/views/order-detail/orderStatus'
  11. import ColHeader from '@/components/col-header'
  12. import { postMessage } from '@/helpers/native-message'
  13. import ColSticky from '@/components/col-sticky'
  14. import ColShare from '@/components/col-share'
  15. import LiveItem from '@/views/live-class/live-item'
  16. import iconShare from '@/views/shop-mall/images/icon-share.svg'
  17. import { state } from '@/state'
  18. import { browser } from '@/helpers/utils'
  19. import { tradeOrder } from '../trade/tradeOrder'
  20. import { courseType } from '@/constant'
  21. import GroupPlanStep from '@/business-components/group-plan-step'
  22. interface IProps {
  23. courseTime: string
  24. coursePlan: string
  25. roomUid?: string
  26. teacherName: string
  27. subjectName: string
  28. liveState?: number
  29. id?: number | string
  30. }
  31. export default defineComponent({
  32. name: 'LiveDetail',
  33. data() {
  34. const query = this.$route.query
  35. return {
  36. joinRoom: query.joinRoom, // 原生传递过来的参数,判断是否进入直播间
  37. recomUserId: query.recomUserId, // 推荐人id
  38. groupId: query.groupId,
  39. courseId: query.classId,
  40. platform: query.p, // 属于哪个平台,//机构老师 tenant,平台老师 无
  41. live: {} as any,
  42. shareStatus: false,
  43. shareUrl: ''
  44. }
  45. },
  46. computed: {
  47. userInfo() {
  48. const live = this.live as any
  49. // console.log('live', live)
  50. const planList = live.planList || []
  51. const startTime = planList[0]?.startTime || new Date()
  52. const endTime = planList[0]?.endTime || new Date()
  53. const studentNum = live.maxStudentNum || 0 - live.studentCount || 0
  54. return {
  55. avatar: live.avatar,
  56. headUrl: live.avatar,
  57. username: live.userName || `游客${live.teacherId || ''}`,
  58. id: live.teacherId,
  59. startTime:
  60. `${dayjs(startTime).format('YYYY-MM-DD')} ${dayjs(startTime).format(
  61. 'HH:mm'
  62. )}~${dayjs(endTime).format('HH:mm')}` || '',
  63. buyNum: live.studentCount,
  64. type: 'group',
  65. lessonId: live.courseGroupId,
  66. lessonPrice: live.coursePrice,
  67. lessonNum: live.courseNum,
  68. mixStudentNum: studentNum > 0 ? studentNum : 0,
  69. lessonDesc: live.courseIntroduce,
  70. lessonCoverUrl: live.backgroundPic || live.backgroundPicTemplate,
  71. lessonName: live.courseGroupName,
  72. subjectName: live.subjectName,
  73. courseStartTime: live.courseStartTime,
  74. auditVersion: live.auditVersion || 0,
  75. isDegree: live.degreeFlag ? true : false,
  76. isTeacher: live.teacherFlag ? true : false
  77. }
  78. },
  79. platformStatus() {
  80. const userInfo = state.user.data as any
  81. // 是机构学生 并且 是机构老师分享
  82. const query = this.$route.query
  83. return userInfo.tenantId > 0 && query.p == 'tenant'
  84. },
  85. courseInfo() {
  86. const tempArr = [] as IProps[]
  87. const live = this.live
  88. const coursePlanList = this.live.planList || []
  89. coursePlanList.forEach((item: any) => {
  90. const startTime = item.startTime || new Date()
  91. const endTime = item.endTime || new Date()
  92. tempArr.push({
  93. courseTime: `${dayjs(startTime).format('YYYY-MM-DD')} ${dayjs(
  94. startTime
  95. ).format('HH:mm')}~${dayjs(endTime).format('HH:mm')}`,
  96. coursePlan: item.plan,
  97. roomUid: item.roomUid,
  98. teacherName: live.teacherName,
  99. subjectName: live.subjectName,
  100. liveState: item.liveState,
  101. id: item.courseId
  102. })
  103. })
  104. return tempArr || []
  105. },
  106. salesEndDate() {
  107. const live = this.live as any
  108. return dayjs(live.salesEndDate || new Date()).format('YYYY-MM-DD')
  109. },
  110. liveStatus() {
  111. const coursePlanList = this.live.planList || []
  112. const tempObj = {
  113. status: false,
  114. liveStatus: 0,
  115. roomUid: ''
  116. }
  117. coursePlanList.forEach((item: any) => {
  118. if (item.courseId === Number(this.courseId)) {
  119. tempObj.status = true
  120. tempObj.liveStatus = item.liveStatus
  121. tempObj.roomUid = item.roomUid
  122. }
  123. })
  124. return tempObj
  125. }
  126. },
  127. async mounted() {
  128. await this._init()
  129. if (/(localhost|192)/g.test(location.origin)) {
  130. this.shareUrl = `https://dev.colexiu.com/teacher/#/shareGroup?recomUserId=${state.user.data?.userId}&groupId=${this.groupId}&userType=${state.platformType}&p=tenant`
  131. } else {
  132. this.shareUrl = `${location.origin}/teacher/#/shareGroup?recomUserId=${state.user.data?.userId}&groupId=${this.groupId}&userType=${state.platformType}&p=tenant`
  133. }
  134. },
  135. methods: {
  136. async _init() {
  137. try {
  138. const res = await request.get(
  139. '/api-student/courseGroup/queryLiveCourseInfo',
  140. {
  141. params: {
  142. groupId: this.groupId
  143. }
  144. }
  145. )
  146. this.live = res.data || {}
  147. console.log(this.live, 'list')
  148. } catch {}
  149. },
  150. async onJoinRoom() {
  151. try {
  152. const res = await request.get(
  153. '/api-student/courseGroup/queryLiveCourseInfo',
  154. {
  155. params: {
  156. courseType: 'GROUP',
  157. groupId: this.groupId
  158. }
  159. }
  160. )
  161. const result = res.data || {}
  162. const coursePlanList = result.planList || []
  163. let tempObj: any = {}
  164. coursePlanList.forEach((item: any) => {
  165. if (item.courseId === Number(this.courseId)) {
  166. tempObj = item
  167. }
  168. })
  169. if (tempObj && tempObj.liveState === 1) {
  170. postMessage({
  171. api: 'joinLiveRoom',
  172. content: {
  173. roomId: tempObj.roomUid,
  174. teacherId: this.live.teacherId
  175. }
  176. })
  177. } else if (tempObj && tempObj.liveState === 2) {
  178. setTimeout(() => {
  179. Toast('课程已结束')
  180. }, 100)
  181. } else {
  182. setTimeout(() => {
  183. Toast('课程尚未开始,请耐心等候')
  184. }, 100)
  185. }
  186. } catch {}
  187. },
  188. initLive() {
  189. const live = this.live
  190. orderStatus.orderObject.orderType = 'GROUP'
  191. orderStatus.orderObject.orderName = '小组课购买'
  192. orderStatus.orderObject.orderDesc = '小组课购买'
  193. orderStatus.orderObject.actualPrice = live.coursePrice
  194. orderStatus.orderObject.recomUserId = this.recomUserId
  195. orderStatus.orderObject.orderNo = ''
  196. orderStatus.orderObject.orderList = [
  197. {
  198. orderType: 'GROUP',
  199. goodsName: '小组课购买',
  200. courseGroupId: live.courseGroupId,
  201. courseGroupName: live.courseGroupName,
  202. coursePrice: live.coursePrice,
  203. price: live.coursePrice,
  204. teacherName: live.userName || `游客${live.teacherId || ''}`,
  205. teacherId: live.teacherId,
  206. avatar: live.avatar,
  207. courseInfo: this.courseInfo,
  208. recomUserId: this.recomUserId
  209. }
  210. ]
  211. },
  212. async onBuy() {
  213. try {
  214. const live = this.live
  215. // 判断是否是0无订单
  216. if (live.coursePrice <= 0) {
  217. this.initLive()
  218. await onSubmitZero(() => {
  219. Dialog.alert({
  220. message: '领取成功',
  221. confirmButtonText: '确定',
  222. confirmButtonColor: '#2dc7aa'
  223. }).then(() => {
  224. this._init()
  225. })
  226. })
  227. return
  228. }
  229. const res = await request.post(
  230. '/api-student/userOrder/getPendingOrder',
  231. {
  232. data: {
  233. goodType: 'GROUP',
  234. bizId: this.groupId
  235. }
  236. }
  237. )
  238. const result = res.data
  239. if (result) {
  240. Dialog.confirm({
  241. title: '提示',
  242. message: '您有一个未支付的订单,是否继续支付?',
  243. confirmButtonColor: '#269a93',
  244. cancelButtonText: '取消订单',
  245. confirmButtonText: '继续支付'
  246. })
  247. .then(async () => {
  248. tradeOrder(result, this.routerTo)
  249. })
  250. .catch(() => {
  251. Dialog.close()
  252. // 只用取消订单,不用做其它处理
  253. this.cancelPayment(result.orderNo)
  254. })
  255. } else {
  256. this.initLive()
  257. this.routerTo()
  258. }
  259. } catch {
  260. //
  261. }
  262. },
  263. routerTo() {
  264. const live = this.live
  265. this.$router.push({
  266. path: '/orderDetail',
  267. query: {
  268. orderType: 'GROUP',
  269. courseGroupId: live.courseGroupId
  270. }
  271. })
  272. },
  273. async cancelPayment(orderNo: string) {
  274. try {
  275. await request.post('/api-student/userOrder/orderCancel', {
  276. data: {
  277. orderNo
  278. }
  279. })
  280. // this.routerTo()
  281. } catch {}
  282. }
  283. },
  284. render() {
  285. return (
  286. <div class={[styles['live-detail'], 'mb12']}>
  287. <ColHeader
  288. v-slots={{
  289. right: () => (
  290. <img src={iconShare} onClick={() => (this.shareStatus = true)} />
  291. )
  292. }}
  293. />
  294. <UserDetail
  295. userInfo={this.userInfo}
  296. showBuy={true}
  297. onUserDetail={(item: any) => {
  298. if (state.platformType === 'STUDENT' && browser().isApp) {
  299. this.$router.push({
  300. path: '/teacherHome',
  301. query: {
  302. teacherId: item.id,
  303. tabs: 'live'
  304. }
  305. })
  306. }
  307. }}
  308. />
  309. <SectionDetail border>
  310. <p class={styles.introduction}>{this.userInfo.lessonDesc}</p>
  311. </SectionDetail>
  312. {this.courseInfo.length > 0 && (
  313. <GroupPlanStep
  314. courseInfo={this.courseInfo}
  315. courseId={Number(this.courseId) || 0}
  316. />
  317. )}
  318. {/* <SectionDetail
  319. title="课程列表"
  320. icon="courseList"
  321. border
  322. contentStyle={{ paddingTop: '0' }}
  323. >
  324. {this.courseInfo.length > 0 && (
  325. <CoursePlanStep
  326. courseInfo={this.courseInfo}
  327. courseId={Number(this.courseId) || 0}
  328. />
  329. )}
  330. </SectionDetail> */}
  331. {/* <div class={styles.tips}>
  332. <h3>
  333. <Icon name={iconTips} size={15} />
  334. 温馨提示
  335. </h3>
  336. <p>
  337. 1、该小组课程销售截止后,报名人数若少于
  338. {this.live.mixStudentNum || 0}
  339. 人将取消开课,已购买学员付费金额将自动返还,请您放心购买;
  340. <br />
  341. 2、小组课教学计划中的上课时间为老师预计时间,实际上课时间以老师开启直播时间为准;
  342. <br />
  343. 3、若您错过老师直播,可通过视频回放观看完整课程。
  344. </p>
  345. </div> */}
  346. {this.courseInfo.length > 0 &&
  347. (this.live.existBuy !== 1 && this.live.existBuy !== 2) && (
  348. <ColSticky position="bottom" background="white">
  349. <div class={['btnGroup', styles.btnMore]}>
  350. <Button
  351. block
  352. round
  353. type="primary"
  354. onClick={this.onBuy}
  355. disabled={this.platformStatus}
  356. >
  357. {this.live.coursePrice <= 0 ? '免费领取' : `立即购买`}
  358. </Button>
  359. </div>
  360. </ColSticky>
  361. )}
  362. {this.joinRoom == '1' && this.liveStatus.liveStatus !== 2 && (
  363. <ColSticky position="bottom" background="white">
  364. <div class={['btnGroup']} style={{ paddingTop: '12px' }}>
  365. <Button block round type="primary" onClick={this.onJoinRoom}>
  366. 进入直播间
  367. </Button>
  368. </div>
  369. </ColSticky>
  370. )}
  371. <Popup
  372. v-model:show={this.shareStatus}
  373. style={{ background: 'transparent' }}
  374. >
  375. <ColShare
  376. teacherId={this.userInfo.id}
  377. shareUrl={this.shareUrl}
  378. shareType="live"
  379. >
  380. <LiveItem
  381. class={styles.shareCourse}
  382. liveInfo={{
  383. backgroundPic: this.userInfo.lessonCoverUrl,
  384. courseGroupId: this.userInfo.lessonId,
  385. courseGroupName: this.userInfo.lessonName,
  386. courseNum: this.userInfo.lessonNum,
  387. coursePrice: this.userInfo.lessonPrice,
  388. teacherName: this.userInfo.username,
  389. teacherId: this.userInfo.id,
  390. avatar: this.userInfo.avatar,
  391. studentCount: this.userInfo.buyNum,
  392. courseStartTime: this.userInfo.courseStartTime,
  393. existBuy: 0,
  394. subjectName: this.userInfo.subjectName
  395. }}
  396. />
  397. </ColShare>
  398. </Popup>
  399. </div>
  400. )
  401. }
  402. })