gift.vue 15 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208209210211212213214215216217218219220221222223224225226227228229230231232233234235236237238239240241242243244245246247248249250251252253254255256257258259260261262263264265266267268269270271272273274275276277278279280281282283284285286287288289290291292293294295296297298299300301302303304305306307308309310311312313314315316317318319320321322323324325326327328329330331332333334335336337338339340341342343344345346347348349350351352353354355356357358359360361362363364365366367368369370371372373374375376377378379380381382383384385386387388389390391392393394395396397398399400401402403404405406407408409410411412413414415416417418419420421422423424425426427428429430431432433434435436437438439440441442443444445446447448449450451452453454455456457458459460461462463464465466467468
  1. <template>
  2. <view class="page" style="background-color: #fff;">
  3. <cu-custom :isBack="true" title="智能报课"></cu-custom>
  4. <view class="steps solid-bottom flex justify-between">
  5. <!-- <view class="text-center"><text class="text-red">点击切换下一步</text></view>
  6. <view class="cu-steps">
  7. <view class="cu-item" :class="index > cur ? '' : 'text-student'" v-for="(item, index) in numList" :key="index" @tap="changeStep(index)">
  8. <text class="num" :data-index="index + 1"></text> {{ item.name }}
  9. </view>
  10. </view> -->
  11. <view v-for="(item, index) in stepList" :key="index" class="step flex-sub text-center" :class="{ cur: cur === index }" @tap="changeStep(index)">{{ item.name }}</view>
  12. </view>
  13. <view class="main" :style="[{ height: 'calc(100vh - 40px - ' + topHeader + 'px)' }]">
  14. <swiper @change="swipeChange" :current="cur" class="swipe">
  15. <swiper-item @touchmove.stop>
  16. <view class="btn-check-group padding">
  17. <checkbox-group @change="checkboxChange" class="flex justify-between" style="flex-wrap: wrap;" id="week">
  18. <label class="btn-check-label margin" :class="{ 'bg-light-green': params.week.includes(item.key + '') }" v-for="item in days" :key="item.key">
  19. <checkbox :value="item.key" />
  20. {{ item.value }}
  21. </label>
  22. </checkbox-group>
  23. </view>
  24. <view class="flex next-step padding-lr" style="width: 100%;">
  25. <button class="cu-btn round lg bg-light-green flex-sub" @tap="changeStep(1)">下一步</button>
  26. </view>
  27. </swiper-item>
  28. <swiper-item @touchmove.stop>
  29. <view class="btn-check-group padding">
  30. <view class="margin">
  31. <input v-model="serachKey" placeholder="输入兴趣内容 如:画画" class="margin-top input-border" @confirm="search" />
  32. <view class="flex margin-top-sm">
  33. <view v-for="(h, i) in params.hobbyOther" :key="i" class="hobby"
  34. >{{ h }}
  35. <text class="cuIcon-close del-hobby text-red" @tap="delHobby(h)"></text>
  36. </view>
  37. </view>
  38. </view>
  39. <view class="interests" v-if="interests.length > 0">
  40. <img src="/static/imgs/intelligent-bg.png" mode="widthFix" alt="" />
  41. <checkbox-group id="hobby">
  42. <view
  43. v-for="(item, index) in interests"
  44. :key="index"
  45. :style="[{ top: index * 80 + 'rpx', left: (index % 2) * 320 + index * 40 + 'rpx' }]"
  46. @tap="chooseInterest(item)"
  47. class="interest"
  48. >
  49. <label
  50. class="btn-check-label margin"
  51. :style="{
  52. color: params.hobbyOther.includes(item) ? '#fff' : '#000',
  53. backgroundColor: params.hobbyOther.includes(item) ? colors[index % 4] : '#f0f0f0'
  54. }"
  55. >
  56. <checkbox :value="item" />
  57. {{ item }}
  58. </label>
  59. <text class="cuIcon-title" :style="[{ color: colors[index % 4] }]"></text>
  60. </view>
  61. </checkbox-group>
  62. </view>
  63. </view>
  64. <view class="flex next-step padding-lr" style="width: 100%;">
  65. <button class="cu-btn round lg bg-light-green flex-sub margin-right-sm" @tap="changeStep(0)">上一步</button>
  66. <button class="cu-btn round lg bg-light-green flex-sub" @tap="changeStep(2)">下一步</button>
  67. </view>
  68. </swiper-item>
  69. <swiper-item>
  70. <view v-if="attends.length === 0" class="padding text-center text-lg">
  71. 没有找到相关课程!
  72. </view>
  73. <view v-else>
  74. <view class="VerticalBox">
  75. <scroll-view class="VerticalNav nav" scroll-y scroll-with-animation :scroll-top="verticalNavTop" :style="'height:calc(100vh - 70px - 40px - ' + topHeader + 'px)'">
  76. <view class="cu-item" :class="index == tabCur ? 'text-green cur' : ''" v-for="(item, index) in attends" :key="index" @tap="TabSelect" :data-id="index">
  77. {{ weekList[index] }}
  78. </view>
  79. </scroll-view>
  80. <scroll-view
  81. class="VerticalMain padding-sm"
  82. scroll-y
  83. scroll-with-animation
  84. :style="'height:calc(100vh - 70px - 40px - ' + topHeader + 'px)'"
  85. :scroll-into-view="'main-' + mainCur"
  86. @scroll="VerticalMain"
  87. >
  88. <radio-group @change="changeClass" style="width: 100%;" v-for="(attend, i) in attends" :key="i">
  89. <view class="cu-item bg-white padding margin-bottom-sm" v-for="(item, index) in attend.list" v-show="i == mainCur" :key="index" :id="'main-' + index">
  90. <view class="cu-bar solid-bottom">
  91. <view>
  92. <label>
  93. <view class="flex justify-between align-center">
  94. <view
  95. ><text>{{ item.attend_name }}</text></view
  96. >
  97. <view>
  98. <radio :value="item.class_attend_id" :checked="checked.includes(item.class_attend_id)" class="cyan" style="transform: scale(0.7);" />
  99. </view>
  100. </view>
  101. </label>
  102. </view>
  103. </view>
  104. <view v-if="item.prop.length > 0">
  105. <checkbox-group class="checkbox-group margin-top-xs" :data-id="item.class_attend_id" @change="propChange">
  106. <label v-for="prop in item.prop" :key="prop.id">
  107. <view>
  108. <checkbox
  109. :value="prop.id"
  110. class="cyan"
  111. :checked="checked_props[mainCur].class_attend_id === item.class_attend_id && checked_props[mainCur].prop.includes(prop.id + '')"
  112. style="transform: scale(0.7);"
  113. />{{ prop.name }}<text>¥{{ prop.money }}</text>
  114. </view>
  115. </label>
  116. </checkbox-group>
  117. </view>
  118. </view>
  119. </radio-group>
  120. </scroll-view>
  121. </view>
  122. <view class="static shadow">
  123. <view class="flex justify-between" style="width: 100%;">
  124. <view>
  125. <button class="cu-btn round line-grey" @tap="changeStep(1)">
  126. <text class="cuIcon-back lg margin-right-sm"></text>
  127. 重选
  128. </button>
  129. </view>
  130. <view class="static-order flex">
  131. <shop-cart :class_attend_id="attend_id" @changeDisableBtn="changeDisableBtn"></shop-cart>
  132. <button class="cu-btn round bg-student text-white margin-left-xs" @tap="addCart">报课</button>
  133. </view>
  134. </view>
  135. </view>
  136. </view>
  137. </swiper-item>
  138. </swiper>
  139. </view>
  140. </view>
  141. </template>
  142. <script>
  143. import { _intelligentDates, _intelligentInterests, _intelligentCourses, _joinShops } from '@/api/course'
  144. import shopCart from '@/components/shop-cart.vue'
  145. import { mapGetters } from 'vuex'
  146. export default {
  147. components: {
  148. shopCart
  149. },
  150. data() {
  151. return {
  152. topHeader: this.globalCustomBarHeight,
  153. stepList: [
  154. {
  155. name: '选周期'
  156. },
  157. {
  158. name: '选兴趣'
  159. },
  160. {
  161. name: '确认选课'
  162. }
  163. ],
  164. cur: 0,
  165. serachKey: '', //关键字搜素
  166. keys: [], //兴趣选项
  167. days: [], //周期
  168. interests: [], //兴趣
  169. attends: [], //课程
  170. weekList: [], //日期
  171. params: {
  172. week: [],
  173. hobby: [],
  174. student_id: '',
  175. hobbyOther: []
  176. },
  177. tabCur: 0,
  178. mainCur: 0,
  179. verticalNavTop: 0,
  180. colors: ['#0CAAF4', '#082FCC', '#FDC534', '#09E294'],
  181. checked: [], // 选中
  182. checked_props: [], // 选中工具
  183. attend_id: 0, // 添加成功课程
  184. disableBtn: false, // 避免多次重复点击
  185. action: 'stopmove'
  186. }
  187. },
  188. computed: {
  189. ...mapGetters(['kid'])
  190. },
  191. onLoad(options) {
  192. const id = decodeURIComponent(options.package_id)
  193. this.package_id = id
  194. this.params.student_id = this.kid
  195. this.init()
  196. },
  197. onShow() {
  198. this.disableBtn = false
  199. },
  200. methods: {
  201. init() {
  202. this.getDateConfig()
  203. this.getHobbyConfig()
  204. },
  205. getDateConfig() {
  206. _intelligentDates().then(res => {
  207. this.days = res.data
  208. })
  209. },
  210. getHobbyConfig() {
  211. _intelligentInterests().then(res => {
  212. // this.interests = res.data
  213. const choose = res.data[5]
  214. this.params.hobby = [choose.id]
  215. this.keys = choose.names
  216. })
  217. },
  218. search() {
  219. this.interests = this.keys.filter(item => item.includes(this.serachKey))
  220. },
  221. get_options() {
  222. const params = Object.assign({}, this.params)
  223. params.week = params.week.join(',')
  224. params.hobby = params.hobby.join(',')
  225. params.hobbyOther = params.hobbyOther.join(',')
  226. _intelligentCourses(params).then(res => {
  227. this.attends = res.data
  228. this.checked = [] //初始化选项
  229. Object.keys(this.attends).forEach(item => {
  230. const class_attend_id = this.attends[item].list[0].class_attend_id
  231. const prop = this.attends[item].list[0].prop.length > 0 ? [this.attends[item].list[0].prop[0].id + ''] : []
  232. this.$set(this.checked, item, class_attend_id)
  233. this.$set(this.checked_props, item, { class_attend_id: class_attend_id, prop: prop })
  234. })
  235. this.weekList = this.attends.map(item => item.week)
  236. })
  237. },
  238. delHobby(key) {
  239. this.params.hobbyOther = this.params.hobbyOther.filter(item => item != key)
  240. },
  241. chooseInterest(key) {
  242. if (this.params.hobbyOther.includes(key)) {
  243. this.params.hobbyOther = this.params.hobbyOther.filter(item => item != key)
  244. } else {
  245. this.params.hobbyOther.push(key)
  246. }
  247. },
  248. checkboxChange(e) {
  249. const type = e.target.id
  250. const values = e.detail.value
  251. switch (type) {
  252. case 'week':
  253. this.params.week = values
  254. break
  255. case 'hobby':
  256. this.params.hobby = values
  257. break
  258. default:
  259. console.log('error')
  260. }
  261. },
  262. changeStep(index) {
  263. if (index === 1 && this.params.week.length < 1) {
  264. uni.showToast({ title: '请先选择周期', icon: 'none' })
  265. return false
  266. }
  267. if (index === 2 && this.params.hobby.length < 1) {
  268. uni.showToast({ title: '请先选择兴趣', icon: 'none' })
  269. return false
  270. }
  271. const step = Math.abs(index - this.cur)
  272. if (step > 1) return false
  273. this.cur = index
  274. },
  275. swipeChange(e) {
  276. this.cur = e.detail.current
  277. if (this.cur === 2) {
  278. this.get_options()
  279. }
  280. },
  281. changeDisableBtn() {
  282. this.disableBtn = true
  283. },
  284. changeClass(e) {
  285. const value = Number(e.detail.value)
  286. this.checked[this.mainCur] = value
  287. this.$set(this.checked, this.mainCur, value)
  288. this.$set(this.checked_props, this.mainCur, { class_attend_id: value, prop: [] })
  289. console.log(this.checked_props)
  290. },
  291. propChange(e) {
  292. const parentId = e.target.dataset.id
  293. const values = e.detail.value
  294. if (!this.checked.includes(parentId)) {
  295. this.$set(this.checked, this.mainCur, parentId)
  296. }
  297. const item = { class_attend_id: parentId, prop: values }
  298. this.$set(this.checked_props, this.mainCur, item)
  299. },
  300. TabSelect(e) {
  301. this.tabCur = e.currentTarget.dataset.id
  302. this.mainCur = e.currentTarget.dataset.id
  303. this.verticalNavTop = (e.currentTarget.dataset.id - 1) * 50
  304. },
  305. addCart() {
  306. if (this.disableBtn) return false
  307. this.disableBtn = true
  308. const _self = this
  309. const list = this.checked_props
  310. .filter(item => item.class_attend_id)
  311. .map(item => {
  312. return {
  313. class_attend_id: item.class_attend_id,
  314. prop: item.prop.join(','),
  315. student_id: this.kid
  316. }
  317. })
  318. if (list.length < 1) {
  319. uni.showToast({ title: '请先选择课程', icon: 'none' })
  320. this.disableBtn = false
  321. return false
  322. }
  323. _joinShops({ list }).then(res => {
  324. if (res.code === 1) {
  325. _self.globalNavigateTo('classCart')
  326. } else {
  327. this.disableBtn = false
  328. }
  329. })
  330. }
  331. }
  332. }
  333. </script>
  334. <style lang="scss" scoped>
  335. .page {
  336. height: 100vh;
  337. }
  338. .main {
  339. .swipe {
  340. height: 100%;
  341. }
  342. }
  343. .step {
  344. position: relative;
  345. padding: 10px 0;
  346. &.cur {
  347. color: #09e294;
  348. &:before {
  349. content: '';
  350. position: absolute;
  351. bottom: 0;
  352. background-color: #09e294;
  353. width: 30px;
  354. height: 2px;
  355. transform: translateX(25%);
  356. }
  357. }
  358. }
  359. .hobby {
  360. position: relative;
  361. margin-left: 20rpx;
  362. .del-hobby {
  363. position: absolute;
  364. top: -6rpx;
  365. right: -20rpx;
  366. }
  367. }
  368. .interests {
  369. position: relative;
  370. .interest {
  371. position: absolute;
  372. }
  373. }
  374. .next-step {
  375. position: fixed;
  376. bottom: 60px;
  377. }
  378. .btn-check-group {
  379. checkbox {
  380. display: none;
  381. }
  382. radio {
  383. display: none;
  384. }
  385. label {
  386. display: inline-block;
  387. margin-right: 10rpx;
  388. position: relative;
  389. padding: 10rpx 40rpx;
  390. // line-height: 30px;
  391. border-radius: 100rpx;
  392. flex-wrap: nowrap;
  393. border: 1px solid #eee;
  394. }
  395. }
  396. .input-border {
  397. height: 80rpx;
  398. border-radius: 4px;
  399. border: 1px solid #eee;
  400. &:focus {
  401. border: 1px solid #5fd0e4;
  402. }
  403. }
  404. ::v-deep radio.disabled::before,
  405. ::v-deep checkbox.disabled::before {
  406. color: #e1e1e1 !important;
  407. }
  408. .static {
  409. position: fixed;
  410. left: 0;
  411. right: 0;
  412. bottom: 0;
  413. display: flex;
  414. background: #fff;
  415. height: 70px;
  416. align-items: center;
  417. padding: 0 0.8rem;
  418. // justify-content: flex-end;
  419. }
  420. .fixed {
  421. position: fixed;
  422. z-index: 99;
  423. }
  424. .VerticalNav.nav {
  425. width: 200upx;
  426. white-space: initial;
  427. }
  428. .VerticalNav.nav .cu-item {
  429. width: 100%;
  430. text-align: center;
  431. background-color: #fff;
  432. margin: 0;
  433. border: none;
  434. height: 50px;
  435. position: relative;
  436. }
  437. .VerticalNav.nav .cu-item.cur {
  438. background-color: #f1f1f1;
  439. }
  440. .VerticalNav.nav .cu-item.cur::after {
  441. content: '';
  442. width: 8upx;
  443. height: 30upx;
  444. border-radius: 10upx 0 0 10upx;
  445. position: absolute;
  446. background-color: currentColor;
  447. top: 0;
  448. right: 0upx;
  449. bottom: 0;
  450. margin: auto;
  451. }
  452. .VerticalBox {
  453. display: flex;
  454. }
  455. .VerticalMain {
  456. background-color: #f1f1f1;
  457. flex: 1;
  458. }
  459. </style>