VIPApply.vue 43 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208209210211212213214215216217218219220221222223224225226227228229230231232233234235236237238239240241242243244245246247248249250251252253254255256257258259260261262263264265266267268269270271272273274275276277278279280281282283284285286287288289290291292293294295296297298299300301302303304305306307308309310311312313314315316317318319320321322323324325326327328329330331332333334335336337338339340341342343344345346347348349350351352353354355356357358359360361362363364365366367368369370371372373374375376377378379380381382383384385386387388389390391392393394395396397398399400401402403404405406407408409410411412413414415416417418419420421422423424425426427428429430431432433434435436437438439440441442443444445446447448449450451452453454455456457458459460461462463464465466467468469470471472473474475476477478479480481482483484485486487488489490491492493494495496497498499500501502503504505506507508509510511512513514515516517518519520521522523524525526527528529530531532533534535536537538539540541542543544545546547548549550551552553554555556557558559560561562563564565566567568569570571572573574575576577578579580581582583584585586587588589590591592593594595596597598599600601602603604605606607608609610611612613614615616617618619620621622623624625626627628629630631632633634635636637638639640641642643644645646647648649650651652653654655656657658659660661662663664665666667668669670671672673674675676677678679680681682683684685686687688689690691692693694695696697698699700701702703704705706707708709710711712713714715716717718719720721722723724725726727728729730731732733734735736737738739740741742743744745746747748749750751752753754755756757758759760761762763764765766767768769770771772773774775776777778779780781782783784785786787788789790791792793794795796797798799800801802803804805806807808809810811812813814815816817818819820821822823824825826827828829830831832833834835836837838839840841842843844845846847848849850851852853854855856857858859860861862863864865866867868869870871872873874875876877878879880881882883884885886887888889890891892893894895896897898899900901902903904905906907908909910911912913914915916917918919920921922923924925926927928929930931932933934935936937938939940941942943944945946947948949950951952953954955956957958959960961962963964965966967968969970971972973974975976977978979980981982983984985986987988989990991992993994995996997998999100010011002100310041005
  1. <template>
  2. <div class="vipapply">
  3. <m-header v-if="statusList.headerStatus" />
  4. <div class="vip-title">课程信息</div>
  5. <van-cell-group>
  6. <van-field v-model="form.name" label="课程班名称" input-align="right" size="large"
  7. placeholder="请输入班级名称" />
  8. <van-field v-model="formName.subjectListName" @click="onGetSheetList('subjectList')"
  9. label="科目名称" :readonly="true" input-align="right" is-link size="large" placeholder="请选择" />
  10. <van-field v-model="formName.vipGroupCategoryName" @click="onGetSheetList('vipGroupCategory')"
  11. label="课程形式" :readonly="true" input-align="right" is-link size="large" placeholder="请选择" />
  12. <van-field v-model="formName.vipGroupActivityName" @click="onGetSheetList('vipGroupActivity')"
  13. label="活动方案" :readonly="true" input-align="right" is-link size="large" placeholder="请选择" />
  14. <van-field v-model="formName.teacherSchoolName" @click="onGetSheetList('teacherSchool')"
  15. label="线下课地址" :readonly="true" input-align="right" is-link size="large" placeholder="请选择" />
  16. </van-cell-group>
  17. <div class="vip-title">课时组成</div>
  18. <van-cell-group>
  19. <van-field v-model="form.studentNum" label="每班人数" disabled input-align="right" size="large"
  20. placeholder="每班预计招收人数" />
  21. <van-field v-model="form.singleClassMinutes" @click="onClickSingleClass" label="每课时长" :readonly="true" input-align="right" is-link size="large" placeholder="请输入每课时长" />
  22. <!-- 判断是否选择活动方案 -->
  23. <van-field v-if="formName.vipGroupActivityName || statusList.hasOnline" v-model="form.onlineClassesNums" @keyup="onClassKeyUp" label="线上课" input-align="right"
  24. size="large" placeholder="请输入次数" type="number" />
  25. <van-field v-if="formName.vipGroupActivityName || statusList.hasOffline" v-model="form.offlineClassesNums" @keyup="onClassKeyUp" label="线下课" input-align="right"
  26. size="large" placeholder="请输入次数" type="number" />
  27. </van-cell-group>
  28. <div class="vip-title">时间安排</div>
  29. <van-cell-group>
  30. <van-field v-model="form.registrationStartTime" label="报名开始时间"
  31. disabled @click="onEnListShow('start', form.registrationStartTime)" input-align="right" is-link size="large" placeholder="请选择" />
  32. <van-field v-model="form.coursesExpireDate" label="报名截止时间"
  33. disabled @click="onEnListShow('end', form.coursesExpireDate)" input-align="right" is-link size="large" placeholder="请选择" />
  34. </van-cell-group>
  35. <div class="vip-title">课时安排</div>
  36. <van-cell-group>
  37. <van-field v-model="form.totalClassTime" label="课时总数" disabled input-align="right" size="large"
  38. placeholder="请输入次数" />
  39. <van-field v-if="statusList.teachOnOrOff" v-model="formName.giveTeachModeName" @click="onGiveMode" label="赠课类型"
  40. :readonly="true" input-align="right" is-link size="large" placeholder="请选择" />
  41. <van-field @click="dataForm.status = true" v-model="form.courseStart" label="排课开始时间" :readonly="true" input-align="right"
  42. is-link size="large" placeholder="请选择" />
  43. <van-cell title-class="title-time" v-for="(item, index) in scheduleList" :key="index">
  44. <template slot="title">
  45. <span class="online">{{ item.type }}</span>
  46. <span class="week">{{ item.weekStr }}</span>
  47. <span class="timer">{{ item.startTime + '-' + item.endTime }}</span>
  48. </template>
  49. <template slot="default">
  50. <van-button type="warning" @click="onScheduleRemove(item)" round size="small" plain >删除</van-button>
  51. </template>
  52. </van-cell>
  53. <div class="add-plan van-cell" @click="onCourseShedule">
  54. <van-icon name="add-o" />课时安排
  55. </div>
  56. <van-field label="排课列表" v-if="scheduleList.length > 0"
  57. disabled input-align="right" @click="onShowTimeTable" is-link size="large" />
  58. </van-cell-group>
  59. <div class="vip-title">课酬设置</div>
  60. <van-cell-group>
  61. <van-field v-model="form.onlineClassesUnitPrice" label="线上课单价" type="number" input-align="right"
  62. size="large" @keyup="getCalcClass" :disabled="loadData.vipGroupActivitySelect.salaryReadonlyFlag == 0" placeholder="金额(每课时)"
  63. v-if="statusList.hasOnline" />
  64. <van-field v-model="form.offlineClassesUnitPrice" label="线下课单价" type="number" input-align="right"
  65. size="large" @keyup="getCalcClass" :disabled="loadData.vipGroupActivitySelect.salaryReadonlyFlag == 0" placeholder="金额(每课时)"
  66. v-if="statusList.hasOffline" />
  67. <van-field v-model="form.onlineTeacherSalary" label="线上课课酬" input-align="right" size="large"
  68. :disabled="loadData.vipGroupActivitySelect.salaryReadonlyFlag == 0" placeholder="金额(每课时)"
  69. v-if="statusList.hasOnline" type="number" />
  70. <!-- || other.onlineSalary !='TEACHER_DEFAULT' -->
  71. <van-field v-model="form.offlineTeacherSalary" label="线下课课酬" input-align="right" size="large"
  72. :disabled="loadData.vipGroupActivitySelect.salaryReadonlyFlag == 0" placeholder="金额(每课时)"
  73. v-if="statusList.hasOffline" type="number" />
  74. <!-- || other.offlineSalary !='TEACHER_DEFAULT' -->
  75. <van-field v-model="form.totalCount" disabled label="课程总价" input-align="right" size="large"
  76. placeholder="金额(每课时)" />
  77. </van-cell-group>
  78. <div class="button-group">
  79. <van-button type="primary" @click="onSubmit" round size="large">确认</van-button>
  80. </div>
  81. <!-- 每课时长 -->
  82. <van-action-sheet v-model="statusList.classTimerStatus" :actions="loadData.classTimer" cancel-text="取消" @cancel="statusList.classTimerStatus = false" @select="onClassTimerSelect" />
  83. <!-- 报名开始时间&报名结束时间 -->
  84. <van-popup v-model="enlistForm.status" position="bottom">
  85. <van-datetime-picker v-model="enlistForm.currentDate" type="date" :min-date="enlistForm.minDate"
  86. :max-date="enlistForm.maxDate" :formatter="formatter" @cancel="enlistForm.status = false"
  87. @confirm="onEnlistConfirm" />
  88. </van-popup>
  89. <!-- 赠课类型 -->
  90. <van-action-sheet v-model="statusList.giveTeachModeStatus" :actions="loadData.giveTeachMode" cancel-text="取消" @cancel="statusList.giveTeachModeStatus = false" @select="onModeSelect" />
  91. <!-- 课程信息所用 :close-on-click-overlay="false" -->
  92. <van-popup v-model="sheetForm.sheetStatus" position="bottom">
  93. <van-picker :loading="sheetForm.loading" :default-index="sheetForm.index" :columns="sheetForm.columns"
  94. show-toolbar @cancel="sheetForm.sheetStatus = false" @confirm="onSheetConfirm" />
  95. </van-popup>
  96. <!-- 课时安排 -->
  97. <van-popup v-model="dataForm.status" position="bottom">
  98. <van-datetime-picker v-model="dataForm.currentDate" type="date" :min-date="dataForm.minDate"
  99. :max-date="dataForm.maxDate" :formatter="formatter" @cancel="dataForm.status = false"
  100. @confirm="onCurrentConfirm" />
  101. </van-popup>
  102. <!-- 课时安排 -->
  103. <van-popup v-model="courseForm.teachingStatus" position="bottom">
  104. <van-picker :columns="courseForm.columns" show-toolbar @cancel="courseForm.teachingStatus = false"
  105. @confirm="onTeachinConfirm" />
  106. </van-popup>
  107. <!-- 课表展示 -->
  108. <van-popup v-model="statusList.classTime" position="bottom">
  109. <van-row>
  110. <van-col span="12">上课类型</van-col>
  111. <van-col span="12">上课时间</van-col>
  112. </van-row>
  113. <div class="tableContainer">
  114. <van-row v-for="(item, index) in timeTable" :key="index">
  115. <van-col span="12">
  116. {{ item.teachMode == 'ONLINE' ? '线上' : '线下' }}
  117. </van-col>
  118. <van-col span="12">
  119. {{ item.classDate }} {{ item.startClassTimeStr }}
  120. </van-col>
  121. </van-row>
  122. </div>
  123. </van-popup>
  124. </div>
  125. </template>
  126. <script>
  127. import MHeader from '@/components/MHeader'
  128. import { browser } from '@/common/common'
  129. import { findSubSubjects,
  130. vipGroupCategory,
  131. findByVipGroupCategory,
  132. findVipSchoolByTeacher,
  133. findByTeacherAndCategory,
  134. vipGroupApply } from '@/api/teacher'
  135. let minutes = [] // 分钟数
  136. for(let i = 0; i < 60; i++) {
  137. let mi = i < 10 ? '0' + i : i
  138. minutes.push(mi + '分')
  139. }
  140. export default {
  141. name: 'vipapply',
  142. components: { MHeader },
  143. data() {
  144. return {
  145. dataForm: { // 时间下拉框
  146. status: false,
  147. minDate: new Date(),
  148. maxDate: new Date(2025, 10, 1),
  149. currentDate: new Date()
  150. },
  151. enlistForm: { // 时间下拉框
  152. updateStatus: '', // 修改哪个状态
  153. status: false,
  154. minDate: new Date(),
  155. maxDate: new Date(2025, 10, 1),
  156. currentDate: new Date()
  157. },
  158. statusList: { // 散状态集合
  159. giveTeachModeStatus: false, // 赠课弹窗状态
  160. teachOnOrOff: false, // 是否显示赠课
  161. hasOnline: false, // 是否显示线上
  162. hasOffline: false, // 是否显示线下
  163. classTime: false, // 课表展示
  164. headerStatus: false, // 头部是否展示
  165. classTimerStatus: false, // 每课时长状态
  166. },
  167. loadData: { // 下拉加载数据
  168. subjectList: [], // 声部列表
  169. subjectListSelect: [], // 选中的声部JSON
  170. vipGroupCategory: [], // 课程形式
  171. vipGroupCategorySelect: [], // 选中的课程形式JSON
  172. vipGroupActivity: [], // 活动文案
  173. vipGroupActivitySelect: [], // 选中的活动文案JSON
  174. teacherSchool: [], // 线下课地址
  175. teacherSchoolSelect: [], // 选中的线下课地址JSON
  176. giveTeachMode: [{ name: '线上课', value: 'ONLINE' }, { name: '线下课', value: 'OFFLINE' }], // 赠课
  177. giveTeachModeSelect: [], // 选中的赠课JSON
  178. teacherCategory: [], // 老师课酬信息
  179. classTimer: [], // 每课时长
  180. },
  181. sheetForm: { // 上拉弹窗
  182. currentType: null, // 当前选择的类型
  183. sheetStatus: false,
  184. loading: true, // 加载数据
  185. index: 0, // 选中的索引值
  186. columns: []
  187. },
  188. courseForm: { // 排课弹窗
  189. teachingStatus: false, // 课时安排状态
  190. columns: [{ // 课程选项
  191. values: ['线上', '线下'],
  192. className: 'type'
  193. }, {
  194. values: ['周一', '周二', '周三', '周四', '周五', '周六', '周日'],
  195. className: 'week'
  196. }, {
  197. values: [1, 2, 3, 4, 5, 6, 7, 8, 9, 10, 11, 12, 13, 14, 15, 16, 17, 18, 19, 20, 21, 22, 23],
  198. className: 'hours',
  199. defaultIndex: 7
  200. }, {
  201. values: minutes,
  202. className: 'minutes'
  203. }]
  204. },
  205. form: {
  206. name: null,
  207. subjectIdList: null,
  208. vipGroupCategoryId: null,
  209. vipGroupActivityId: null,
  210. teacherSchoolId: null,
  211. studentNum: null,
  212. singleClassMinutes: null,
  213. onlineClassesNums: null,
  214. offlineClassesNums: null,
  215. registrationStartTime: null,
  216. coursesExpireDate: null,
  217. totalClassTime: null,
  218. courseStart: null,
  219. giveTeachMode: null,
  220. onlineClassesUnitPrice: null,
  221. offlineClassesUnitPrice: null,
  222. onlineTeacherSalary: null,
  223. offlineTeacherSalary: null,
  224. totalCount: null,
  225. },
  226. formName: {
  227. subjectListName: null, // 声部名称
  228. subjectListIndex: 0, // 声部名称
  229. vipGroupCategoryName: null, // 课程形式
  230. vipGroupCategoryIndex: 0, // 课程形式
  231. vipGroupActivityName: null, // 活动文案
  232. vipGroupActivityIndex: 0, // 活动文案
  233. teacherSchoolName: null, // 线下课地址
  234. teacherSchoolIndex: 0, // 线下课地址
  235. giveTeachModeName: null, // 赠课类型
  236. },
  237. other: {
  238. onlineSalary: null, // 线上课课酬结算方式
  239. offlineSalary: null, // 线下课课酬结算方式
  240. giveNum: 0, // 赠送课时
  241. },
  242. scheduleList: [], // 课时安排
  243. timeTable: [], // 生成的课表
  244. }
  245. },
  246. mounted() {
  247. let params = this.$route.query
  248. if(params.Authorization) {
  249. localStorage.setItem('Authorization', decodeURI(params.Authorization))
  250. localStorage.setItem('userInfo', decodeURI(params.Authorization))
  251. }
  252. document.title = 'VIP课程班申请'
  253. if(browser().android) {
  254. this.headerStatus = true
  255. }
  256. },
  257. methods: {
  258. onGetSheetList(name) { // 获取科目列表
  259. let sheetForm = this.sheetForm
  260. sheetForm.columns = []
  261. if(!this.form.vipGroupCategoryId && name == 'vipGroupActivity') { // 判断是否选择了课程形式
  262. this.$toast('请选择课程形式')
  263. return
  264. }
  265. sheetForm.sheetStatus = true
  266. sheetForm.loading = true
  267. sheetForm.currentType = name
  268. sheetForm.index = 0
  269. let arr = this.loadData[name]
  270. if(arr.length > 0) {
  271. sheetForm.columns = arr
  272. sheetForm.index = this.formName[name + 'Index']
  273. sheetForm.loading = false
  274. } else {
  275. this.onLoadingData(name)
  276. }
  277. },
  278. onLoadingData() { // 加载数据
  279. let sheetForm = this.sheetForm
  280. if(sheetForm.currentType == 'subjectList') { // 声部列表
  281. findSubSubjects().then(res => {
  282. let result = res.data
  283. if(result.code == 200 && result.data.length > 0) {
  284. let tempArr = []
  285. result.data.forEach(item => {
  286. item.value = item.id
  287. item.text = item.name
  288. tempArr.push(item)
  289. })
  290. this.loadData.subjectList = tempArr
  291. sheetForm.columns = tempArr
  292. sheetForm.loading = false
  293. } else {
  294. this.$toast('暂无科目列表')
  295. sheetForm.loading = false
  296. }
  297. })
  298. } else if(sheetForm.currentType == 'vipGroupCategory') { // 课程形式
  299. vipGroupCategory().then(res => {
  300. let result = res.data
  301. if(result.code == 200 && result.data.length > 0) {
  302. let tempArr = []
  303. result.data.forEach(item => {
  304. item.value = item.id
  305. item.text = item.name
  306. tempArr.push(item)
  307. })
  308. this.loadData.vipGroupCategory = tempArr
  309. sheetForm.columns = tempArr
  310. sheetForm.loading = false
  311. } else {
  312. this.$toast('暂无课程形式')
  313. sheetForm.loading = false
  314. }
  315. })
  316. } else if(sheetForm.currentType == 'vipGroupActivity') { // 活动文案
  317. findByVipGroupCategory({ categoryId: this.form.vipGroupCategoryId }).then(res => {
  318. let result = res.data
  319. if(result.code == 200 && result.data.length > 0) {
  320. let tempArr = []
  321. result.data.forEach(item => {
  322. item.value = item.id
  323. item.text = item.name
  324. item.startTime = item.startTime ? item.startTime.split(' ')[0] : null, // 报名开始时间
  325. item.endTime = item.endTime ? item.endTime.split(' ')[0] : null // 报名结束时间
  326. tempArr.push(item)
  327. })
  328. this.loadData.vipGroupActivity = tempArr
  329. sheetForm.columns = tempArr
  330. sheetForm.loading = false
  331. } else {
  332. this.$toast('暂无活动文案')
  333. sheetForm.loading = false
  334. }
  335. })
  336. } else if(sheetForm.currentType == 'teacherSchool') { // 教师教学点
  337. findVipSchoolByTeacher().then(res => {
  338. let result = res.data
  339. if(result.code == 200 && result.data.length > 0) {
  340. let tempArr = []
  341. result.data.forEach(item => {
  342. item.value = item.id
  343. item.text = item.name
  344. tempArr.push(item)
  345. })
  346. this.loadData.teacherSchool = tempArr
  347. sheetForm.columns = tempArr
  348. sheetForm.loading = false
  349. } else {
  350. this.$toast('暂无教学点')
  351. sheetForm.loading = false
  352. }
  353. })
  354. }
  355. },
  356. findTeacherCategory(id) { // 获取教师课酬
  357. findByTeacherAndCategory({ categoryId: id }).then(res => {
  358. let result = res.data
  359. if(result.code == 200) {
  360. this.loadData.teacherCategory = result.data
  361. }
  362. })
  363. },
  364. onSheetConfirm(value, index) { // 上拉弹窗
  365. let sheetForm = this.sheetForm,
  366. form = this.form,
  367. formName = this.formName,
  368. loadData = this.loadData
  369. if(sheetForm.currentType == 'subjectList') { // 科目名称赋值
  370. form.subjectIdList = value.value
  371. formName.subjectListName = value.text
  372. formName.subjectListIndex = index
  373. loadData.subjectListSelect = value
  374. } else if(sheetForm.currentType == 'vipGroupCategory') { // 课程形式赋值
  375. if(loadData.vipGroupCategorySelect.id != value.id) {
  376. // 获取教师课酬
  377. this.findTeacherCategory(value.id)
  378. }
  379. form.vipGroupCategoryId = value.value
  380. formName.vipGroupCategoryName = value.text
  381. formName.vipGroupCategoryIndex = index
  382. loadData.vipGroupCategorySelect = value
  383. form.studentNum = value.studentNum // 每班人数
  384. // form.singleClassMinutes = value.singleClassMinutes // 每课时长
  385. form.onlineClassesUnitPrice = value.onlineClassesUnitPrice
  386. form.offlineClassesUnitPrice = value.offlineClassesUnitPrice
  387. // 每课时长赋值
  388. form.singleClassMinutes = null
  389. loadData.classTimer = []
  390. let tempSingle = value.singleClassMinutes.split(",")
  391. tempSingle.forEach(item => {
  392. this.loadData.classTimer.push({
  393. name: item,
  394. value: item
  395. })
  396. })
  397. // 重置活动文案
  398. form.vipGroupActivityId = null
  399. formName.vipGroupActivityName = null
  400. formName.vipGroupActivityIndex = 0
  401. loadData.vipGroupActivity = []
  402. loadData.vipGroupActivitySelect = []
  403. } else if(sheetForm.currentType == 'vipGroupActivity') { // 活动方案赋值
  404. form.vipGroupActivityId = value.value
  405. formName.vipGroupActivityName = value.text
  406. formName.vipGroupActivityIndex = index
  407. loadData.vipGroupActivitySelect = value
  408. this.onCalcClassTimes(value) // 计算时间等.........
  409. this.getCalcClass() // 课酬计算
  410. this.setTimeTable() // 重新排课
  411. // 时间安排
  412. // form.registrationStartTime = value.startTime
  413. // form.coursesExpireDate = value.endTime
  414. } else if(sheetForm.currentType == 'teacherSchool') { // 线下课地址
  415. form.teacherSchoolId = value.value
  416. formName.teacherSchoolName = value.text
  417. formName.teacherSchoolIndex = index
  418. }
  419. sheetForm.sheetStatus = false
  420. },
  421. onClassKeyUp() { // 线上课&线下课修改时
  422. let vas = this.loadData.vipGroupActivitySelect
  423. this.onCalcClassTimes(vas)
  424. this.getCalcClass()
  425. this.setTimeTable()
  426. },
  427. onCalcClassTimes(vas) { // 计算课时总数
  428. let form = this.form,
  429. other = this.other,
  430. statusList = this.statusList
  431. let totalCount = Number(form.onlineClassesNums) + Number(form.offlineClassesNums)
  432. // ...
  433. if(vas.salarySettlementJson) {
  434. let obj = JSON.parse(vas.salarySettlementJson)
  435. if (obj && obj.onlineSalarySettlement) {
  436. // 有线上课
  437. statusList.hasOnline = true
  438. } else {
  439. statusList.hasOnline = false
  440. }
  441. if (obj && obj.offlineSalarySettlement) {
  442. // 有线下课
  443. statusList.hasOffline = true
  444. } else {
  445. statusList.hasOffline = false
  446. }
  447. }
  448. if(vas.type == "GIVE_CLASS") { // 买赠活动
  449. this.statusList.teachOnOrOff = true // 显示赠课
  450. if(totalCount >= Number(vas.attribute1)) {
  451. form.totalClassTime = totalCount + '+' + vas.attribute2
  452. other.giveNum = vas.attribute2
  453. } else {
  454. form.totalClassTime = totalCount
  455. other.giveNum = 0
  456. }
  457. } else { // 折扣活动
  458. form.totalClassTime = totalCount
  459. other.giveNum = 0
  460. this.statusList.teachOnOrOff = false // 隐藏赠课
  461. }
  462. },
  463. onModeSelect(value) { // 赠课确认
  464. this.form.giveTeachMode = value.value
  465. this.formName.giveTeachModeName = value.name
  466. this.statusList.giveTeachModeStatus = false
  467. },
  468. onClickSingleClass() {
  469. if(!this.formName.vipGroupCategoryName) {
  470. this.$toast('请选择课程形式')
  471. return
  472. }
  473. this.statusList.classTimerStatus = true
  474. },
  475. onClassTimerSelect(value) { // 每课时长设置
  476. this.form.singleClassMinutes = value.value
  477. this.statusList.classTimerStatus = false
  478. this.getCalcClass()
  479. },
  480. onCurrentConfirm(value) { // 排课开始时间
  481. let selectDate = new Date(value)
  482. let tempMonth = selectDate.getMonth() + 1 >= 10 ? selectDate.getMonth() + 1 : '0' + (selectDate.getMonth() + 1)
  483. let tempDay = selectDate.getDate() >= 10 ? selectDate.getDate() : '0' + selectDate.getDate()
  484. this.form.courseStart = selectDate.getFullYear() + '-' + tempMonth + '-' + tempDay
  485. this.dataForm.status = false
  486. },
  487. onEnlistConfirm(value) { // 报名开始/结束时间
  488. let selectDate = new Date(value)
  489. let tempMonth = selectDate.getMonth() + 1 >= 10 ? selectDate.getMonth() + 1 : '0' + (selectDate.getMonth() + 1)
  490. let tempDay = selectDate.getDate() >= 10 ? selectDate.getDate() : '0' + selectDate.getDate()
  491. let enlistForm = this.enlistForm
  492. if(enlistForm.updateStatus == 'start') {
  493. this.form.registrationStartTime = selectDate.getFullYear() + '-' + tempMonth + '-' + tempDay
  494. } else if(enlistForm.updateStatus == 'end') {
  495. this.form.coursesExpireDate = selectDate.getFullYear() + '-' + tempMonth + '-' + tempDay
  496. }
  497. enlistForm.status = false
  498. },
  499. onEnListShow(type, value) { // 报名开始/截止时间
  500. let enlistForm = this.enlistForm
  501. if(type == 'end') {
  502. if(this.form.registrationStartTime) {
  503. enlistForm.minDate = new Date(this.form.registrationStartTime.replace(/-/ig, '/'))
  504. } else {
  505. enlistForm.minDate = new Date()
  506. }
  507. }
  508. if(value) {
  509. enlistForm.currentDate = new Date(value.replace(/-/ig, '/'))
  510. } else {
  511. enlistForm.currentDate = new Date()
  512. }
  513. enlistForm.updateStatus = type
  514. enlistForm.status = true
  515. },
  516. onCourseShedule() { // 课时安排
  517. if(!this.form.singleClassMinutes) {
  518. this.$toast('请选每课时长')
  519. return
  520. }
  521. if(this.other.giveNum > 0 && !this.form.giveTeachMode) {
  522. this.$toast('请选择赠课类型')
  523. return
  524. }
  525. if(!this.form.courseStart) {
  526. this.$toast('请选择排课开始时间')
  527. return
  528. }
  529. this.courseForm.teachingStatus = true
  530. },
  531. onScheduleRemove(item) { // 删除课程安排
  532. let index = this.scheduleList.indexOf(item)
  533. if(index !== -1) {
  534. this.scheduleList.splice(index, 1)
  535. }
  536. },
  537. onTeachinConfirm(value) { // 添加课程
  538. let scheduleList = this.scheduleList
  539. let startTime = (value[2] >= 10 ? value[2] : '0' + value[2]) + ':' + value[3].split('分')[0]
  540. let endTime = this.MinutesTest(value[2], value[3], this.form.singleClassMinutes)
  541. let isAdd = true
  542. scheduleList.forEach(item => {
  543. let isStartTime = this.timeIsrange(startTime, endTime, item.startTime);
  544. let isEndTime = this.timeIsrange(startTime, endTime, item.endTime);
  545. if(isAdd) { //
  546. if(value[1] == item.weekStr) {
  547. if(isStartTime || isEndTime) {
  548. isAdd = false
  549. } else {
  550. isAdd = true
  551. }
  552. } else if(value[1] != item.weekStr) {
  553. isAdd = true
  554. }
  555. }
  556. })
  557. if(isAdd) { // 判断时间范围是否有重复
  558. scheduleList.push({
  559. type: value[0], // 线上还是线下
  560. weekStr: value[1],
  561. weekIndex: this.getWeek(value[1]),
  562. startTime: startTime,
  563. endTime: endTime,
  564. id: Date.now()
  565. })
  566. // scheduleList.sort((a, b) => {
  567. // return a.weekStr
  568. // })
  569. this.courseForm.teachingStatus = false
  570. this.setTimeTable()
  571. } else {
  572. this.$toast('该时间段已排课请重选时间')
  573. return
  574. }
  575. },
  576. onShowTimeTable() { // 显示排课列表
  577. this.statusList.classTime = true
  578. this.setTimeTable()
  579. },
  580. setTimeTable () {
  581. // 重置排课列表
  582. this.timeTable = []
  583. let form = this.form,
  584. scheduleList = this.scheduleList
  585. console.log(JSON.stringify(scheduleList))
  586. // if(!form.courseStart) {
  587. // this.$toast('请选择排课开始时间')
  588. // return
  589. // }
  590. // 拿到线上课数与线下课数 以及
  591. let online = parseInt(form.onlineClassesNums ? form.onlineClassesNums : 0)
  592. let offline = parseInt(form.offlineClassesNums ? form.offlineClassesNums : 0)
  593. let giveNum = parseInt(this.other.giveNum)
  594. let giveClassType = form.giveTeachMode
  595. if (giveClassType == 'ONLINE') {
  596. // 线上
  597. online += giveNum
  598. } else if (giveClassType == 'OFFLINE') {
  599. offline += giveNum
  600. }
  601. // 判断是否有课程安排
  602. if (scheduleList.length <= 0) {
  603. return
  604. }
  605. let totalCount = Number(online) + Number(offline)
  606. let tempCourseStart = form.courseStart.replace(/-/ig, '/')
  607. let dateOperation = new Date(tempCourseStart)
  608. let forMark = 0
  609. while(totalCount && totalCount > 0) {
  610. for(let i = 0; i < scheduleList.length; i++) {
  611. if (online == 0 && offline == 0) break
  612. let num = scheduleList[i].weekIndex - dateOperation.getDay()
  613. // 如果是同一天一个周期会出现排课都排到一天
  614. if(forMark > 0 && num == 0 && i == 0) {
  615. num = num + 7
  616. }
  617. if(num < 0) { // 如果为负数则为下周
  618. num = num + 7
  619. }
  620. let dataStr = this.getThinkDate(dateOperation, num)
  621. let tempArr = {
  622. 'classDate': dataStr,
  623. 'startClassTimeStr': scheduleList[i].startTime,
  624. "endClassTimeStr": scheduleList[i].endTime
  625. }
  626. if(scheduleList[i].type == '线上' && online > 0) {
  627. tempArr.teachMode = 'ONLINE'
  628. this.timeTable.push(tempArr)
  629. online--
  630. } else if(scheduleList[i].type == '线下' && offline > 0) {
  631. tempArr.teachMode = 'OFFLINE'
  632. this.timeTable.push(tempArr)
  633. offline--
  634. } else {
  635. if(online > 0) {
  636. tempArr.teachMode = 'ONLINE'
  637. this.timeTable.push(tempArr)
  638. online--
  639. } else if(offline > 0) {
  640. tempArr.teachMode = 'OFFLINE'
  641. this.timeTable.push(tempArr)
  642. offline--
  643. }
  644. }
  645. totalCount--
  646. }
  647. // 加一周
  648. if(scheduleList.length == 1) {
  649. dateOperation.setDate(dateOperation.getDate() + 7)
  650. } else if(scheduleList.every( item => item.weekStr === scheduleList[0].weekStr)) {
  651. // 标记循环次数(标记判断课程安排是不是同一天)
  652. forMark++
  653. }
  654. // forMark++
  655. }
  656. },
  657. onGiveMode() { // 赠送课改变时
  658. this.statusList.giveTeachModeStatus = true
  659. this.getCalcClass()
  660. },
  661. getCalcClass() { // 计算课酬
  662. let loadData = this.loadData
  663. let form = this.form
  664. let vas = loadData.vipGroupActivitySelect, // 活动
  665. vcs = loadData.vipGroupCategorySelect, // 课程形式
  666. tc = loadData.teacherCategory, // 老师基本信息
  667. oncn = form.onlineClassesNums ? form.onlineClassesNums : 0, // 线上课次数
  668. offcn = form.offlineClassesNums ? form.offlineClassesNums : 0 // 线下课次数
  669. // giveTeachMode = form.giveTeachMode // 线下或线下
  670. // 优惠活动
  671. // ...
  672. if(vas.salarySettlementJson) {
  673. let obj = JSON.parse(vas.salarySettlementJson)
  674. // 获取每课时长
  675. // 每45Min计算一些课酬
  676. // let b = form.singleClassMinutes ? (form.singleClassMinutes / 45) : 0
  677. // 课程按课来计算
  678. if (obj && obj.onlineSalarySettlement) {
  679. let onss = obj.onlineSalarySettlement
  680. // 有线上课
  681. if(onss.salarySettlementType == 'TEACHER_DEFAULT') {
  682. form.onlineTeacherSalary = tc.onlineClassesSalary // 线上课酬
  683. } else if(onss.salarySettlementType == 'RATIO_DISCOUNT') {
  684. // 老师课酬的折扣 如果有则不打折
  685. form.onlineTeacherSalary = (vcs.onlineClassesUnitPrice * (onss.settlementValue ? onss.settlementValue : 100) / 100).toFixed(2)
  686. } else if(onss.salarySettlementType == 'FIXED_SALARY') {
  687. form.onlineTeacherSalary = (onss.settlementValue).toFixed(2)
  688. }
  689. form.onlineTeacherSalary = Number(form.onlineTeacherSalary)
  690. }
  691. if (obj && obj.offlineSalarySettlement) {
  692. let offss = obj.offlineSalarySettlement
  693. // 有线下课
  694. if(offss.salarySettlementType == 'TEACHER_DEFAULT') {
  695. form.offlineTeacherSalary = tc.offlineClassesSalary // 线上课酬
  696. } else if(offss.salarySettlementType == 'RATIO_DISCOUNT') {
  697. // 老师课酬的折扣
  698. form.offlineTeacherSalary = (vcs.offlineClassesUnitPrice * (offss.settlementValue ? offss.settlementValue : 100) / 100).toFixed(2)
  699. } else if(offss.salarySettlementType == 'FIXED_SALARY') {
  700. form.offlineTeacherSalary = (offss.settlementValue).toFixed(2)
  701. }
  702. form.offlineTeacherSalary = Number(form.offlineTeacherSalary)
  703. }
  704. }
  705. if(vas.type == 'GIVE_CLASS' || vas.type == 'BASE_ACTIVITY') {
  706. form.totalCount = ((oncn * form.onlineClassesUnitPrice) + (offcn * form.offlineClassesUnitPrice)).toFixed(2)
  707. } else if(vas.type == 'DISCOUNT') {
  708. form.totalCount = (((oncn * form.onlineClassesUnitPrice) + (offcn * form.offlineClassesUnitPrice)) * Number(vas.attribute1) / 100).toFixed(2)
  709. }
  710. form.totalCount = Number(form.totalCount)
  711. },
  712. getThinkDate (date, num) {
  713. let Stamp = date
  714. Stamp.setDate(date.getDate() + num) // 获取当前月数的第几天
  715. var year = Stamp.getFullYear(); //获取完整的年份(4位,1970-????)
  716. var month = Stamp.getMonth() + 1; //获取当前月份(0-11,0代表1月)
  717. var mvar = '';
  718. if (month < 10) {
  719. mvar = '0' + month;
  720. } else {
  721. mvar = month + '';
  722. }
  723. var day = Stamp.getDate();
  724. var dvar = '';
  725. if (day < 10) {
  726. dvar = '0' + day;
  727. } else {
  728. dvar = day + '';
  729. }
  730. return year + "-" + mvar + '-' + dvar;
  731. },
  732. // 分钟小时相加减
  733. MinutesTest (houer, mins, interval) {
  734. let min = mins.split('分')[0]
  735. let sdate1 = new Date(1900, 1, 1, houer, min)
  736. sdate1.setMinutes(sdate1.getMinutes() + parseInt(interval))
  737. let H = sdate1.getHours()
  738. let M = sdate1.getMinutes()
  739. if (H < 10) H = '0' + H
  740. if (M < 10) M = '0' + M
  741. return H + ':' + M
  742. },
  743. // 判断时间是否在时间段内
  744. timeIsrange (beginTime, endTime, nowTime) {
  745. var strb = beginTime.split(":");
  746. if (strb.length != 2) {
  747. return false;
  748. }
  749. var stre = endTime.split(":");
  750. if (stre.length != 2) {
  751. return false;
  752. }
  753. var strn = nowTime.split(":");
  754. if (stre.length != 2) {
  755. return false;
  756. }
  757. var b = new Date();
  758. var e = new Date();
  759. var n = new Date();
  760. b.setHours(strb[0]);
  761. b.setMinutes(strb[1]);
  762. e.setHours(stre[0]);
  763. e.setMinutes(stre[1]);
  764. n.setHours(strn[0]);
  765. n.setMinutes(strn[1]);
  766. if (n.getTime() - b.getTime() >= 0 && n.getTime() - e.getTime() <= 0) {
  767. // 在时间范围内
  768. return true
  769. } else {
  770. // 不在时间范围内
  771. return false
  772. }
  773. },
  774. getWeek(str) { // 获取周几索引值
  775. let template = {
  776. '周一': 1,
  777. '周二': 2,
  778. '周三': 3,
  779. '周四': 4,
  780. '周五': 5,
  781. '周六': 6,
  782. '周日': 0
  783. }
  784. return template[str]
  785. },
  786. formatter(type, value) {
  787. if (type === 'year') {
  788. return `${value}年`;
  789. } else if (type === 'month') {
  790. return `${value}月`
  791. } else if (type === 'day') {
  792. return `${value}日`
  793. }
  794. return value;
  795. },
  796. onSubmit() {
  797. let form = this.form
  798. if(!(form.name)) {
  799. this.$toast('请输入课程班名称')
  800. return false
  801. }
  802. if(!form.subjectIdList) {
  803. this.$toast('请选择科目')
  804. return false
  805. }
  806. if(!form.vipGroupCategoryId) {
  807. this.$toast('请选择课程形式')
  808. return false
  809. }
  810. if(!form.vipGroupActivityId) {
  811. this.$toast('请选择活动方案')
  812. return false
  813. }
  814. if(!form.teacherSchoolId) {
  815. this.$toast('请选择线下课地址')
  816. return false
  817. }
  818. if(!form.singleClassMinutes) {
  819. this.$toast('请选择每课时长')
  820. return false
  821. }
  822. if(!form.onlineClassesNums && form.onlineClassesNums <= 0) {
  823. this.$toast('上课次数有误')
  824. return false
  825. }
  826. if(!form.offlineClassesNums && form.offlineClassesNums <= 0) {
  827. this.$toast('请输入线下课次数')
  828. return false
  829. }
  830. if(!form.registrationStartTime) {
  831. this.$toast('请选择报名开始时间')
  832. return false
  833. }
  834. if(!form.coursesExpireDate) {
  835. this.$toast('请选择报名结束时间')
  836. return false
  837. }
  838. let vipGroupActivitySelect = this.loadData.vipGroupActivitySelect
  839. if(vipGroupActivitySelect.type == 'GIVE_CLASS' && !form.giveTeachMode) {
  840. this.$toast('请选择赠课类型')
  841. return false
  842. }
  843. if(this.scheduleList.length <= 0) {
  844. this.$toast('课时安排不能为空')
  845. return false
  846. }
  847. if(!form.onlineClassesUnitPrice) {
  848. this.$toast('请输入线上课单价')
  849. return false
  850. }
  851. if(!form.offlineClassesUnitPrice) {
  852. this.$toast('请输入线下课单价')
  853. return false
  854. }
  855. if(!form.onlineTeacherSalary) {
  856. this.$toast('请输入线上课课酬')
  857. return false
  858. }
  859. if(!form.offlineTeacherSalary) {
  860. this.$toast('请输入线下课课酬')
  861. return false
  862. }
  863. if(form.giveTeachMode == 'ONLINE') {
  864. form.onlineClassesNum = Number(form.onlineClassesNums) + Number(this.other.giveNum)
  865. form.offlineClassesNum = Number(form.offlineClassesNums)
  866. } else {
  867. form.onlineClassesNum = Number(form.onlineClassesNums)
  868. form.offlineClassesNum = Number(form.offlineClassesNums) + Number(this.other.giveNum)
  869. }
  870. form.totalClassTimes = Number(form.onlineClassesNums) + Number(form.offlineClassesNums) + Number(this.other.giveNum)
  871. form.paymentExpireDate = form.coursesExpireDate
  872. let params = {
  873. courseSchedules: this.timeTable,
  874. vipGroupApplyBaseInfo: form
  875. }
  876. vipGroupApply(params).then(res => {
  877. let result =res.data
  878. if(result.code == 200) {
  879. this.$toast('申请成功')
  880. setTimeout(() => {
  881. if(browser().iPhone) {
  882. window.webkit.messageHandlers.DAYA.postMessage(JSON.stringify({api: 'back'}))
  883. } else if(browser().android) {
  884. DAYA.postMessage(JSON.stringify({api: 'back'}))
  885. } else {
  886. this.$router.push('/business')
  887. }
  888. }, 500)
  889. } else {
  890. this.$toast(result.msg)
  891. }
  892. })
  893. }
  894. }
  895. }
  896. </script>
  897. <style lang='less' scoped>
  898. @import url("../../assets/commonLess/variable.less");
  899. .vip-title {
  900. padding: .06rem 0 .04rem;
  901. font-size: .12rem;
  902. color: @mFontColor;
  903. text-align: center;
  904. }
  905. .add-plan {
  906. display: flex;
  907. align-items: center;
  908. justify-content: center;
  909. padding: .2rem 0;
  910. font-size: .16rem;
  911. color: @tFontColor;
  912. .van-icon {
  913. margin-right: .05rem;
  914. font-size: .20rem;
  915. }
  916. }
  917. .title-time {
  918. display: flex;
  919. align-items: center;
  920. flex: 1 auto;
  921. color: #4A4A4A;
  922. .online {
  923. color: @tFontColor;
  924. }
  925. .week {
  926. padding-left: .4rem;
  927. padding-right: .15rem;
  928. }
  929. }
  930. /deep/.van-field__label, /deep/.van-cell__value {
  931. flex: 1 auto;
  932. }
  933. /deep/.van-field__control:disabled {
  934. color: #6a6969;
  935. }
  936. .button-group {
  937. margin: .3rem .26rem .2rem;
  938. .van-button--primary {
  939. background: @mColor;
  940. font-size: .18rem;
  941. }
  942. }
  943. .van-row {
  944. line-height: .4rem;
  945. border-top: 1px solid #edeef0;
  946. text-align: center;
  947. font-size: .14rem;
  948. &:first-child {
  949. border-top: 0;
  950. background: #edeef0;
  951. color: #444;
  952. font-size: .15rem;
  953. }
  954. }
  955. .tableContainer {
  956. max-height: 2.44rem;
  957. overflow: auto;
  958. .van-row {
  959. color: #444;
  960. &:first-child {
  961. border-top: 0;
  962. background: #fff;
  963. font-size: .14rem;
  964. }
  965. }
  966. }
  967. </style>