gift.vue 11 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208209210211212213214215216217218219220221222223224225226227228229230231232233234235236237238239240241242243244245246247248249250251252253254255256257258259260261262263264265266267268269270271272273274275276277278279280281282283284285286287288289290291292293294295296297298299300301302303304305306307308309310311312313314315316317318319320321322323324325326327328329330331332333334335336337338339340341342343344345346347348349350351352353354355356357358359360361362
  1. <template>
  2. <view class="page">
  3. <cu-custom :isBack="true" title="智能报课"></cu-custom>
  4. <view class="steps padding-tb">
  5. <view class="cu-steps">
  6. <view class="cu-item" :class="index > cur ? '' : 'text-student'" v-for="(item, index) in numList" :key="index" @tap="changeStep(index)">
  7. <text class="num" :data-index="index + 1"></text> {{ item.name }}
  8. </view>
  9. </view>
  10. </view>
  11. <view class="main bg-white" :style="[{ height: 'calc(100vh - 92px - ' + topHeader + 'px)' }]">
  12. <swiper @change="swipeChange" :current="cur" class="swipe">
  13. <swiper-item @touchmove.stop>
  14. <view class="btn-check-group padding">
  15. <checkbox-group @change="checkboxChange" class="flex justify-between" style="flex-wrap: wrap;" id="week">
  16. <label class="btn-check-label margin" :class="{ 'bg-cyan': params.week.includes(item.key + '') }" v-for="item in days" :key="item.key">
  17. <checkbox :value="item.key" />
  18. {{ item.value }}
  19. </label>
  20. </checkbox-group>
  21. </view>
  22. </swiper-item>
  23. <swiper-item @touchmove.stop>
  24. <view class="btn-check-group padding">
  25. <checkbox-group @change="checkboxChange" class="flex justify-between" style="flex-wrap: wrap;" id="hobby">
  26. <label class="btn-check-label margin" :class="{ 'bg-cyan': params.hobby.includes(item.id + '') }" v-for="item in interests" :key="item.id">
  27. <checkbox :value="item.id" />
  28. {{ item.name }}
  29. </label>
  30. </checkbox-group>
  31. <view class="margin">
  32. <input v-model="params.hobbyOther" placeholder="输入其它兴趣内容(选填)" class="margin-top input-border" />
  33. </view>
  34. </view>
  35. </swiper-item>
  36. <swiper-item>
  37. <view class="VerticalBox">
  38. <scroll-view class="VerticalNav nav" scroll-y scroll-with-animation :scroll-top="verticalNavTop" :style="'height:calc(100vh - 70px - 90px - ' + topHeader + 'px)'">
  39. <view class="cu-item" :class="index == tabCur ? 'text-green cur' : ''" v-for="(item, index) in attends" :key="index" @tap="TabSelect" :data-id="index">
  40. {{ weekList[index] }}
  41. </view>
  42. </scroll-view>
  43. <scroll-view
  44. class="VerticalMain padding-sm"
  45. scroll-y
  46. scroll-with-animation
  47. :style="'height:calc(100vh - 70px - 90px - ' + topHeader + 'px)'"
  48. :scroll-into-view="'main-' + mainCur"
  49. @scroll="VerticalMain"
  50. >
  51. <radio-group @change="changeClass" style="width: 100%;" v-for="(attend, i) in attends" :key="i">
  52. <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">
  53. <view class="cu-bar solid-bottom">
  54. <view>
  55. <label>
  56. <view class="flex justify-between align-center">
  57. <view
  58. ><text>{{ item.attend_name }}</text></view
  59. >
  60. <view>
  61. <radio :value="item.class_attend_id" :checked="checked.includes(item.class_attend_id)" class="cyan" style="transform: scale(0.7);" />
  62. </view>
  63. </view>
  64. </label>
  65. </view>
  66. </view>
  67. <view v-if="item.prop.length > 0">
  68. <checkbox-group class="checkbox-group margin-top-xs" :data-id="item.class_attend_id" @change="propChange">
  69. <label v-for="prop in item.prop" :key="prop.id">
  70. <view>
  71. <checkbox
  72. :value="prop.id"
  73. class="cyan"
  74. :checked="checked_props[mainCur].class_attend_id === item.class_attend_id && checked_props[mainCur].prop.includes(prop.id + '')"
  75. style="transform: scale(0.7);"
  76. />{{ prop.name }}<text>¥{{ prop.money }}</text>
  77. </view>
  78. </label>
  79. </checkbox-group>
  80. </view>
  81. </view>
  82. </radio-group>
  83. </scroll-view>
  84. </view>
  85. <view class="static shadow">
  86. <view class="static-order flex">
  87. <shop-cart :class_attend_id="attend_id" @changeDisableBtn="changeDisableBtn"></shop-cart>
  88. <button class="cu-btn round bg-student text-white margin-left-xs" @tap="addCart">报课</button>
  89. </view>
  90. </view>
  91. </swiper-item>
  92. </swiper>
  93. </view>
  94. </view>
  95. </template>
  96. <script>
  97. import { _intelligentDates, _intelligentInterests, _intelligentCourses, _joinShops } from '@/api/course'
  98. import shopCart from '@/components/shop-cart.vue'
  99. import { mapGetters } from 'vuex'
  100. export default {
  101. components: {
  102. shopCart
  103. },
  104. data() {
  105. return {
  106. topHeader: this.globalCustomBarHeight,
  107. numList: [
  108. {
  109. name: '选周期'
  110. },
  111. {
  112. name: '选兴趣'
  113. },
  114. {
  115. name: '确认选课'
  116. }
  117. ],
  118. cur: 0,
  119. days: [], //周期
  120. interests: [], //兴趣
  121. attends: [], //课程
  122. weekList: [], //日期
  123. params: {
  124. week: [],
  125. hobby: [],
  126. student_id: '',
  127. hobbyOther: ''
  128. },
  129. tabCur: 0,
  130. mainCur: 0,
  131. verticalNavTop: 0,
  132. checked: [], // 选中
  133. checked_props: [], // 选中工具
  134. attend_id: 0, // 添加成功课程
  135. disableBtn: false, // 避免多次重复点击
  136. action: 'stopmove'
  137. }
  138. },
  139. computed: {
  140. ...mapGetters(['kid'])
  141. },
  142. onLoad(options) {
  143. const id = decodeURIComponent(options.package_id)
  144. this.package_id = id
  145. this.params.student_id = this.kid
  146. this.init()
  147. },
  148. onShow() {
  149. this.disableBtn = false
  150. },
  151. methods: {
  152. init() {
  153. this.getDateConfig()
  154. this.getHobbyConfig()
  155. },
  156. getDateConfig() {
  157. _intelligentDates().then(res => {
  158. this.days = res.data
  159. })
  160. },
  161. getHobbyConfig() {
  162. _intelligentInterests().then(res => {
  163. this.interests = res.data
  164. })
  165. },
  166. get_options() {
  167. const params = Object.assign({}, this.params)
  168. params.week = params.week.join(',')
  169. params.hobby = params.hobby.join(',')
  170. _intelligentCourses(params).then(res => {
  171. this.attends = res.data
  172. Object.keys(this.attends).forEach(item => {
  173. this.$set(this.checked_props, item, { prop: [] })
  174. })
  175. this.weekList = this.attends.map(item => item.week)
  176. })
  177. },
  178. checkboxChange(e) {
  179. const type = e.target.id
  180. const values = e.detail.value
  181. switch (type) {
  182. case 'week':
  183. this.params.week = values
  184. break
  185. case 'hobby':
  186. this.params.hobby = values
  187. break
  188. default:
  189. console.log('error')
  190. }
  191. },
  192. changeStep(index) {
  193. if (index === 1 && this.params.week.length < 1) {
  194. uni.showToast({ title: '请先选择周期', icon: 'none' })
  195. return false
  196. }
  197. if (index === 2 && this.params.hobby.length < 1) {
  198. uni.showToast({ title: '请先选择兴趣', icon: 'none' })
  199. return false
  200. }
  201. const step = Math.abs(index - this.cur)
  202. if (step > 1) return false
  203. this.cur = index
  204. },
  205. swipeChange(e) {
  206. this.cur = e.detail.current
  207. if (this.cur === 2) {
  208. this.get_options()
  209. }
  210. },
  211. changeDisableBtn() {
  212. this.disableBtn = true
  213. },
  214. changeClass(e) {
  215. const value = Number(e.detail.value)
  216. this.checked[this.mainCur] = value
  217. this.$set(this.checked, this.mainCur, value)
  218. this.$set(this.checked_props, this.mainCur, { class_attend_id: value, prop: [] })
  219. },
  220. propChange(e) {
  221. const parentId = e.target.dataset.id
  222. const values = e.detail.value
  223. if (!this.checked.includes(parentId)) {
  224. this.$set(this.checked, this.mainCur, parentId)
  225. }
  226. const item = { class_attend_id: parentId, prop: values }
  227. this.$set(this.checked_props, this.mainCur, item)
  228. },
  229. TabSelect(e) {
  230. this.tabCur = e.currentTarget.dataset.id
  231. this.mainCur = e.currentTarget.dataset.id
  232. this.verticalNavTop = (e.currentTarget.dataset.id - 1) * 50
  233. },
  234. addCart() {
  235. if (this.disableBtn) return false
  236. this.disableBtn = true
  237. const _self = this
  238. const list = this.checked_props
  239. .filter(item => item.class_attend_id)
  240. .map(item => {
  241. return {
  242. class_attend_id: item.class_attend_id,
  243. prop: item.prop.join(','),
  244. student_id: this.kid
  245. }
  246. })
  247. if (list.length < 1) {
  248. uni.showToast({ title: '请先选择课程', icon: 'none' })
  249. this.disableBtn = false
  250. return false
  251. }
  252. _joinShops({ list }).then(res => {
  253. if (res.code === 1) {
  254. _self.globalNavigateTo('classCart')
  255. } else {
  256. this.disableBtn = false
  257. }
  258. })
  259. }
  260. }
  261. }
  262. </script>
  263. <style lang="scss" scoped>
  264. .page {
  265. height: 100vh;
  266. }
  267. .main {
  268. .swipe {
  269. height: 100%;
  270. }
  271. }
  272. .btn-check-group {
  273. checkbox {
  274. display: none;
  275. }
  276. radio {
  277. display: none;
  278. }
  279. label {
  280. display: inline-block;
  281. margin-right: 10rpx;
  282. position: relative;
  283. padding: 10rpx 30rpx;
  284. line-height: 30px;
  285. border-radius: 100rpx;
  286. flex-wrap: nowrap;
  287. border: 1px solid #eee;
  288. }
  289. }
  290. .input-border {
  291. height: 80rpx;
  292. border-radius: 4px;
  293. border: 1px solid #eee;
  294. &:focus {
  295. border: 1px solid #5fd0e4;
  296. }
  297. }
  298. ::v-deep radio.disabled::before,
  299. ::v-deep checkbox.disabled::before {
  300. color: #e1e1e1 !important;
  301. }
  302. .static {
  303. position: fixed;
  304. left: 0;
  305. right: 0;
  306. bottom: 0;
  307. display: flex;
  308. background: #fff;
  309. height: 70px;
  310. align-items: center;
  311. padding: 0 0.8rem;
  312. justify-content: flex-end;
  313. }
  314. .fixed {
  315. position: fixed;
  316. z-index: 99;
  317. }
  318. .VerticalNav.nav {
  319. width: 200upx;
  320. white-space: initial;
  321. }
  322. .VerticalNav.nav .cu-item {
  323. width: 100%;
  324. text-align: center;
  325. background-color: #fff;
  326. margin: 0;
  327. border: none;
  328. height: 50px;
  329. position: relative;
  330. }
  331. .VerticalNav.nav .cu-item.cur {
  332. background-color: #f1f1f1;
  333. }
  334. .VerticalNav.nav .cu-item.cur::after {
  335. content: '';
  336. width: 8upx;
  337. height: 30upx;
  338. border-radius: 10upx 0 0 10upx;
  339. position: absolute;
  340. background-color: currentColor;
  341. top: 0;
  342. right: 0upx;
  343. bottom: 0;
  344. margin: auto;
  345. }
  346. .VerticalBox {
  347. display: flex;
  348. }
  349. .VerticalMain {
  350. background-color: #f1f1f1;
  351. flex: 1;
  352. }
  353. </style>