detail.vue 17 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208209210211212213214215216217218219220221222223224225226227228229230231232233234235236237238239240241242243244245246247248249250251252253254255256257258259260261262263264265266267268269270271272273274275276277278279280281282283284285286287288289290291292293294295296297298299300301302303304305306307308309310311312313314315316317318319320321322323324325326327328329330331332333334335336337338339340341342343344345346347348349350351352353354355356357358359360361362363364365366367368369370371372373374375376377378379380381382383384385386387388389390391392393394395396397398399400401402403404405406407408409410411412413414415416417418419420421422423424425426427428429430431432433434435436437438439440441442443444445446447448449450451452453454455456457458459460461462463464465466467468469470471472473474475476477478479480481482483484485486487488489490491492493494495496497498499500501502503504505506507508509510511512513514515516517518519520521522523524525526527528529530531532533534535536537538539540541542543544545546547548549550551552553554555556557558559560561562563564565566567568569570571572573574575576577578579580581582583584585586587588589590591592593594595596597598599600601602603604605606607608609610611612613614615616617618619620621622623624625626627628629630631632633634635636637638639640641642643644645646647648649650651652653654655656657658659660661662663664665666667668669670671672673674675676677678679680681682683684685686687688689690691692693694695696697698699700701702703704705706707708
  1. <template>
  2. <view class="web_box">
  3. <!-- <u-navbar height="44" leftIcon="arrow-left" leftIconColor="#000" leftText="课程详情" @leftClick="leftClick" :autoBack="true" bgColor="#fff">
  4. </u-navbar> -->
  5. <view class="banner">
  6. <image v-if="!show" :src="course.img" mode="widthFix" style="width:750rpx"></image>
  7. <video v-else id="popup_video" :src="courseClassDetail.video" object-fit="contain" :show-center-play-btn="false" :controls="true" @fullscreenchange="fullscreenchange"
  8. @timeupdate="timeupdate" @ended="ended" :enable-progress-gesture="user.is_kj?true:false" :initial-time="initial_time"></video>
  9. </view>
  10. <u-tabs :list="tabs" @click="changeTabs" :scrollable="false"
  11. :inactiveStyle="{
  12. color: '#333333'
  13. }"
  14. :activeStyle="{
  15. color: '#2988FE'
  16. }"
  17. ></u-tabs>
  18. <view class="content" v-if="active == 0">
  19. <view class="box1">
  20. <view class="hflex acenter jbetween">
  21. <view class="title" style="font-size: 36rpx;">{{course.name}}</view>
  22. <u-icon v-if="course.is_collection == 1" name="star-fill" color="#FA6400" size="28" @click="collect"></u-icon>
  23. <u-icon v-else name="star" color="#FA6400" size="28" @click="collect"></u-icon>
  24. </view>
  25. <view class="hflex acenter" style="padding: 14rpx 0 30rpx;">
  26. <view class="text_style1" style="padding-right: 40rpx;" v-if="!course.is_buy == 1">
  27. 总学时:{{course.period?course.period:''}}
  28. </view>
  29. <view class="text_style1">
  30. 讲师:{{course.lecturer?course.lecturer:''}}
  31. </view>
  32. </view>
  33. <view class="vflex" v-if="course.is_buy == 1">
  34. <view class="text_style1">
  35. 进度:{{course.study_class_count}}/{{course.period }}时
  36. </view>
  37. <view class="progress">
  38. <u-line-progress class="line_progress" :percentage="progress" :showText="false" height="8" activeColor="#2988FE"></u-line-progress>
  39. </view>
  40. </view>
  41. <view class="hflex acenter jbetween" v-else>
  42. <!-- <view class="price">
  43. ¥{{course.price}}
  44. </view> -->
  45. <view class="price" v-if="!course.buy&&course.type == 1">
  46. ¥{{course.price}}
  47. </view>
  48. <view class="price hflex acenter" v-if="!course.buy&&course.type == 2">
  49. <image src="/static/images/integral/integral.png" mode="widthFix" style="width: 44rpx;"></image>
  50. <view class="price">{{course.price}}</view>
  51. </view>
  52. <view></view>
  53. <view class="text_style1">
  54. 已有{{course.collection_count?course.collection_count:'0'}}人收藏
  55. </view>
  56. </view>
  57. </view>
  58. <view class="box2" v-if="study.length>0">
  59. <view class="title">
  60. 最近学习
  61. </view>
  62. <scroll-view scroll-x="true" class="hflex acenter scroll-view_H" @scroll="scroll">
  63. <view v-for="(item,index) in study" :key="index" class="study_item">
  64. <view class="hflex acenter">
  65. <u-avatar :src="item.headimg"></u-avatar>
  66. <view class="vflex" style="padding-left: 12rpx;">
  67. <view class="name">
  68. {{item.user_name}}
  69. </view>
  70. <view class="name">
  71. 已学习{{item.is_accomplish}}课
  72. </view>
  73. </view>
  74. </view>
  75. </view>
  76. </scroll-view>
  77. </view>
  78. <view class="box1">
  79. <view class="title">
  80. 课程概述
  81. </view>
  82. <u-parse :content="course.content"></u-parse>
  83. <!-- <image :src="course.img" mode="widthFix" style="width: 100%;"></image> -->
  84. </view>
  85. </view>
  86. <view class="content" v-else-if="active == 1">
  87. <u-checkbox-group shape="circle" placement="column">
  88. <block v-for="(item,index) in catalog" :key="index">
  89. <view class="box3 hflex jbetween" @click="startLearn(index)">
  90. <view class="hflex">
  91. <view class="" v-if="course.is_buy == 1">
  92. <!-- <u-checkbox :checked="item.selected"></u-checkbox> -->
  93. </view>
  94. <view class="text_style2" style="margin-right: 15rpx;">
  95. {{index+1}}:
  96. </view>
  97. <view class="text_style2">
  98. {{item.name}}
  99. </view>
  100. </view>
  101. <view class="text_style3">
  102. ({{item.playtime}})
  103. </view>
  104. </view>
  105. </block>
  106. <view class="box3 hflex acenter jbetween text_style2" v-if="course.is_questions == 1" @click="openExam">
  107. 在线考试
  108. </view>
  109. </u-checkbox-group>
  110. </view>
  111. <view class="content" v-else-if="active == 2">
  112. <block v-for="(item,index) in commentList" :key="index">
  113. <view class="box4 hflex">
  114. <u-avatar :src="item.headimg"></u-avatar>
  115. <view class="vflex box4_right">
  116. <view class="hflex acenter jbetween">
  117. <view class="box4_name">
  118. {{item.user_name}}
  119. </view>
  120. <view class="box4_date">
  121. {{item.create_at}}
  122. </view>
  123. </view>
  124. <view class="box4_content hflex fwrap">
  125. {{item.content}}
  126. </view>
  127. </view>
  128. </view>
  129. </block>
  130. </view>
  131. <view class="bottom hflex acenter jcenter">
  132. <u-button v-if="course.is_buy == 0" @click="buyCourse">立即参加</u-button>
  133. <view class="hflex acenter" v-else-if="active == 2">
  134. <u-input placeholder="说点什么吧" v-model="commentValue" :adjustPosition="false">
  135. <template slot="suffix">
  136. <image src="/static/images/index/commentIcon.png" mode="widthFix" style="width: 44rpx;padding-top: 10rpx;"></image>
  137. </template>
  138. </u-input>
  139. <view class="btn" @click="sendComment">发送</view>
  140. </view>
  141. <view class="" v-else-if="active == 0">
  142. <u-button v-if="course.study_class_count == course.period && course.is_questions == 1" @click="openExam">开始考试</u-button>
  143. <!-- <u-button v-else-if="course.study_class_count !== 0">继续学习</u-button> -->
  144. <u-button v-else-if="course.is_buy == 1" @click="startStudy(catalog[0].id)">开始学习</u-button>
  145. </view>
  146. </view>
  147. <!-- 考试须知 -->
  148. <u-modal :show="exam.show" title="考试须知" confirmText="开始考试" @confirm="startExam" :closeOnClickOverlay="true" @close="close">
  149. <view class="slot-content">
  150. <rich-text :nodes="exam.content"></rich-text>
  151. </view>
  152. </u-modal>
  153. </view>
  154. </template>
  155. <script>
  156. import $api from '@/static/js/api.js'
  157. export default {
  158. data() {
  159. return {
  160. show: false,
  161. course: {},
  162. progress: '',
  163. tabs: [
  164. {
  165. name: '课程详情'
  166. },
  167. {
  168. name: '课程目录'
  169. },
  170. {
  171. name: '学生评价'
  172. }
  173. ],
  174. active: 0,
  175. study: [],
  176. catalog: [],
  177. class_id: 1,
  178. courseClassDetail: [],
  179. videoContext: '',
  180. commentList: [],
  181. commentValue: '',
  182. exam: {
  183. show: false,
  184. content: '考试须知。'
  185. },
  186. currentTime: 0,
  187. page: 1,
  188. limit: 10,
  189. user: {},
  190. initial_time: 0
  191. }
  192. },
  193. onLoad(e) {
  194. var id = e.id
  195. this.getUser()
  196. this.getData(id)
  197. this.getClass(id)
  198. this.getCommit(id)
  199. this.getStudy(id)
  200. },
  201. onHide() {
  202. this.submitTime(this.currentTime)
  203. },
  204. onshow() {
  205. this.getData(id)
  206. },
  207. onUnload() {
  208. this.submitTime(this.currentTime)
  209. },
  210. methods: {
  211. // 获取用户信息
  212. getUser() {
  213. var that = this
  214. $api.req({
  215. url: '/api/User/userinfo'
  216. }, function(res) {
  217. console.log(res);
  218. if (res.code == 1) {
  219. that.user = res.data
  220. }
  221. })
  222. },
  223. // 根据id获取课程信息
  224. getData(id) {
  225. var that = this
  226. $api.req({
  227. url: '/api/Index/courseDetail',
  228. data: {
  229. id: id
  230. }
  231. }, function(res) {
  232. console.log("课程信息=",res);
  233. if (res.code === 1) {
  234. that.course = res.data
  235. that.totalProgress()
  236. }
  237. })
  238. },
  239. // 根据id获取课程目录信息
  240. getClass(id) {
  241. var that = this
  242. $api.req({
  243. url: '/api/Index/courseClassList',
  244. data: {
  245. id: id
  246. }
  247. }, function(res) {
  248. console.log("课程目录=",res);
  249. if (res.code === 1) {
  250. that.catalog = res.data.list
  251. }
  252. })
  253. },
  254. // 根据id获取课程评论信息
  255. getCommit(id) {
  256. var that = this
  257. $api.req({
  258. url: '/api/Index/courseCommentsList',
  259. data: {
  260. id: id
  261. }
  262. }, function(res) {
  263. console.log(res);
  264. if(res.code == 1) {
  265. that.commentList = res.data.list
  266. }
  267. })
  268. },
  269. // 获取学院榜样
  270. getStudy(id) {
  271. var that = this
  272. $api.req({
  273. url: '/api/Index/studentsDynamic',
  274. data: {
  275. page: that.page,
  276. limit: that.limit,
  277. id: id
  278. }
  279. }, function(res) {
  280. console.log(res);
  281. if (res.code == 1) {
  282. that.study = res.data.list
  283. }
  284. })
  285. },
  286. // 计算进度条
  287. totalProgress() {
  288. var course = this.course
  289. var progress = course.study_class_count / (course.period )* 100
  290. this.progress = progress
  291. console.log(this.course)
  292. },
  293. // 切换tabs
  294. changeTabs(e) {
  295. var that = this
  296. that.active = e.index
  297. },
  298. // 购买课程
  299. buyCourse() {
  300. var that = this
  301. var order_no = ''
  302. console.log(that.course.type)
  303. if (that.course.type == 1) {
  304. $api.req({
  305. url: '/api/Index/createOrder',
  306. data: {
  307. id: that.course.id
  308. }
  309. }, function(res) {
  310. console.log(res);
  311. if (res.code == 1) {
  312. order_no = res.data.order_no
  313. var order_id = res.data.id
  314. $api.jump('/pages/index/course/buy?order_no=' + order_no + '&id=' + that.course.id + '&order_id=' + order_id)
  315. } else {
  316. $api.info(res.msg)
  317. }
  318. })
  319. } else {
  320. $api.req({
  321. url: '/api/Index/createIntergalOrder',
  322. data: {
  323. id: that.course.id
  324. }
  325. }, function(res) {
  326. console.log(res);
  327. if(res.code == 1) {
  328. order_no = res.data.order_no
  329. $api.jump('/pages/index/course/buy?order_no=' + order_no + '&id=' + that.course.id)
  330. } else {
  331. $api.info(res.msg)
  332. }
  333. })
  334. }
  335. // $api.jump('/pages/index/course/buy?id=' + this.course.id)
  336. },
  337. // 收藏
  338. collect() {
  339. var that = this
  340. if (that.course.is_collection == 0) {
  341. that.course.is_collection = 1
  342. } else {
  343. that.course.is_collection = 0
  344. }
  345. $api.req({
  346. url: '/api/Index/collectionCancelCourse',
  347. data: {
  348. id: that.course.id,
  349. type: that.course.is_collection
  350. }
  351. }, function(res) {
  352. console.log("收藏=",res);
  353. if (res.code == 1) {
  354. $api.info(res.msg)
  355. }
  356. })
  357. },
  358. scroll(e) {
  359. },
  360. // 开始学习
  361. startStudy(id) {
  362. var that = this
  363. that.class_id = id
  364. $api.req({
  365. url: '/api/Index/courseClassDetail',
  366. data: {
  367. id: that.course.id,
  368. class_id: that.class_id
  369. }
  370. }, function(res) {
  371. if (res.code == 1) {
  372. that.courseClassDetail = res.data
  373. that.show = true
  374. that.videoContext = uni.createVideoContext('popup_video', that);
  375. // that.videoContext.requestFullScreen();
  376. }
  377. })
  378. },
  379. // 退出全屏的时候暂停视频
  380. fullscreenchange(e) {
  381. if (!e.detail.fullScreen) {
  382. this.videoContext.pause()
  383. }
  384. },
  385. // 播放进度发生改变
  386. timeupdate(e) {
  387. let that = this
  388. // this.submitTime(this.currentTime)
  389. var currentTime1 = parseInt(e.detail.currentTime)
  390. console.log("播放进度",currentTime1)
  391. var isReady = that.user.is_kj
  392. if (!isReady) {
  393. if (currentTime1 > that.initial_time && currentTime1 - that.initial_time > 2) {
  394. let videoContext = uni.createVideoContext('popup_video');
  395. videoContext.seek(that.initial_time);
  396. uni.showToast({
  397. title: '抱歉,您不能快进',
  398. icon: 'none',
  399. duration: 2000
  400. });
  401. }
  402. that.initial_time = currentTime1
  403. }
  404. this.currentTime = e.detail.currentTime
  405. },
  406. // 视频播放完成
  407. ended() {
  408. var that = this
  409. const index = uni.getStorageSync('index')
  410. $api.req({
  411. url: '/api/Index/endStudy',
  412. data: {
  413. id: that.course.id,
  414. class_id: that.class_id
  415. }
  416. }, function(res) {
  417. console.log(res)
  418. if(res.code == 1) {
  419. that.catalog[index].selected = true
  420. that.getData(that.course.id)
  421. }
  422. })
  423. },
  424. // 学习
  425. startLearn(index) {
  426. // this.catalog[index].selected = true
  427. uni.setStorageSync('index',index)
  428. var that = this
  429. that.class_id = that.catalog[index].id
  430. $api.req({
  431. url: '/api/Index/startStudy',
  432. data: {
  433. id: that.course.id,
  434. class_id: that.class_id
  435. }
  436. }, function(res) {
  437. if (res.code == 1) {
  438. that.startStudy(that.class_id)
  439. }
  440. })
  441. },
  442. // 发送评论
  443. sendComment() {
  444. var that = this
  445. $api.req({
  446. url: '/api/Index/submitComments',
  447. data: {
  448. id: that.course.id,
  449. class_id: that.class_id,
  450. content: that.commentValue
  451. }
  452. },function(res) {
  453. console.log("评论:",res);
  454. that.commentValue = ""
  455. that.getCommit(that.course.id)
  456. })
  457. },
  458. // 开始考试
  459. startExam() {
  460. if (this.progress == 100) {
  461. if (this.course.can_test == 1) {
  462. if (this.course.pass_history) {
  463. $api.info('您已经考试及格了,无需再参加考试了')
  464. }else {
  465. $api.jump('/pages/index/exam/exam?id=' + this.course.id)
  466. this.exam.show = false
  467. }
  468. } else {
  469. $api.info('未到时间')
  470. }
  471. } else {
  472. $api.info("请先学习所有课程")
  473. }
  474. /* var that = this
  475. $api.req({
  476. url: '/api/User/startQuestions',
  477. data: {
  478. id: that.course.id
  479. }
  480. }, function(res) {
  481. console.log(res);
  482. if (res.code == 1) {
  483. }
  484. }) */
  485. },
  486. // 提交观看时间
  487. submitTime(time) {
  488. var that = this
  489. $api.req({
  490. url: '/api/Index/nowTime',
  491. data: {
  492. id: that.course.id,
  493. class_id: that.class_id,
  494. now: time
  495. }
  496. }, function(res) {
  497. console.log(res)
  498. })
  499. },
  500. // 打开考试须知
  501. openExam() {
  502. var that = this
  503. if(that.course.is_buy == 1) {
  504. $api.req({
  505. url: '/api/Publics/config_info'
  506. }, function(res) {
  507. that.exam.show = true
  508. that.exam.content = res.data.test_instructions
  509. })
  510. } else {
  511. $api.info('请先购买课程')
  512. }
  513. },
  514. // 关闭
  515. close() {
  516. this.exam.show = false
  517. }
  518. }
  519. }
  520. </script>
  521. <style lang="scss" scoped>
  522. .web_box::v-deep {
  523. position: relative;
  524. .u-navbar {
  525. width: 100%;
  526. box-sizing: border-box;
  527. padding: 36px 16px 0 0;
  528. }
  529. .banner {
  530. width: 100%;
  531. // margin-top: 63rpx;
  532. }
  533. video {
  534. width: 100%;
  535. }
  536. .content {
  537. width: 100%;
  538. margin: 20rpx 0 130rpx;
  539. }
  540. .box1 {
  541. width: 690rpx;
  542. box-sizing: border-box;
  543. padding: 16rpx 26rpx;
  544. background-color: #fff;
  545. margin: 0 auto;
  546. }
  547. .box2 {
  548. width: auto;
  549. height: 200rpx;
  550. box-sizing: border-box;
  551. padding: 20rpx 30rpx;
  552. background-color: #fff;
  553. margin: 20rpx 0;
  554. overflow: auto;
  555. }
  556. .box3 {
  557. width: 690rpx;
  558. box-sizing: border-box;
  559. padding: 14rpx 24rpx;
  560. background-color: #fff;
  561. margin: 10rpx auto;
  562. }
  563. .box4 {
  564. width: 690rpx;
  565. background: #FFFFFF;
  566. border-radius: 4px;
  567. box-sizing: border-box;
  568. margin: 4rpx auto;
  569. padding: 34rpx 36rpx;
  570. }
  571. .title {
  572. font-size: 32rpx;
  573. font-weight: 600;
  574. color: #333333;
  575. line-height: 25px;
  576. }
  577. .text_style1 {
  578. font-size: 11px;
  579. font-weight: 500;
  580. color: #999999;
  581. line-height: 16px;
  582. }
  583. .progress {
  584. width: 100%;
  585. height: 28rpx;
  586. padding: 16rpx 0 28rpx;
  587. }
  588. .line_progress {
  589. width: 100% !important;
  590. height: 16rpx !important;
  591. }
  592. .price {
  593. font-size: 24px;
  594. font-weight: 600;
  595. color: #FA6400;
  596. line-height: 33px;
  597. }
  598. .scroll-view_H {
  599. white-space: nowrap;
  600. width: 100%;
  601. margin-top: 14rpx;
  602. }
  603. .study_item {
  604. display: inline-block;
  605. width: 252rpx;
  606. box-sizing: border-box;
  607. padding: 10rpx 14rpx;
  608. background: #FCF3E4;
  609. box-shadow: 1px 1px 4px 0px rgba(170,170,170,0.5);
  610. border-radius: 8px;
  611. margin-right: 40rpx;
  612. }
  613. .name {
  614. font-size: 11px;
  615. font-weight: 400;
  616. color: #333333;
  617. line-height: 16px;
  618. }
  619. .text_style2 {
  620. font-size: 15px;
  621. font-weight: 500;
  622. color: #333333;
  623. line-height: 21px;
  624. }
  625. .text_style3 {
  626. font-size: 15px;
  627. font-weight: 400;
  628. color: #333333;
  629. line-height: 21px;
  630. }
  631. .box4_right {
  632. width: calc(100% - 80rpx);
  633. padding-left: 28rpx;
  634. }
  635. .box4_name {
  636. font-size: 14px;
  637. font-weight: 600;
  638. color: #333333;
  639. line-height: 20px;
  640. }
  641. .box4_date {
  642. font-size: 14px;
  643. font-weight: 400;
  644. color: #999999;
  645. line-height: 20px;
  646. }
  647. .box4_content {
  648. padding-top: 16rpx;
  649. font-size: 14px;
  650. font-weight: 400;
  651. color: #333333;
  652. line-height: 40rpx;
  653. }
  654. .bottom {
  655. width: 100%;
  656. height: 120rpx;
  657. position: fixed;
  658. bottom: 0;
  659. background: #fff;
  660. }
  661. .u-button {
  662. width: 506rpx;
  663. height: 70rpx;
  664. background: linear-gradient(135deg, #53BDFF 0%, #2988FE 100%);
  665. border-radius: 20px;
  666. font-size: 36rpx;
  667. font-weight: 400;
  668. color: #FFFFFF;
  669. }
  670. .u-checkbox__icon-wrap--disabled--checked {
  671. background-color: #2988FE !important;
  672. border-color: #2988FE !important;
  673. }
  674. .u-input {
  675. width: 574rpx;
  676. height: 29rpx;
  677. background: #EEEEEE;
  678. border-radius: 9px;
  679. }
  680. .btn {
  681. width: 100rpx;
  682. height: 48rpx;
  683. background: #2988FE;
  684. border-radius: 10px;
  685. margin-left: 12rpx;
  686. font-size: 12px;
  687. font-weight: 500;
  688. text-align: center;
  689. color: #FFFFFF;
  690. line-height: 48rpx;
  691. }
  692. }
  693. </style>