createLiveClass.vue 41 KB

12345678910111213141516171819202122232425262728293031323334353637383940414243444546474849505152535455565758596061626364656667686970717273747576777879808182838485868788899091929394959697989910010110210310410510610710810911011111211311411511611711811912012112212312412512612712812913013113213313413513613713813914014114214314414514614714814915015115215315415515615715815916016116216316416516616716816917017117217317417517617717817918018118218318418518618718818919019119219319419519619719819920020120220320420520620720820921021121221321421521621721821922022122222322422522622722822923023123223323423523623723823924024124224324424524624724824925025125225325425525625725825926026126226326426526626726826927027127227327427527627727827928028128228328428528628728828929029129229329429529629729829930030130230330430530630730830931031131231331431531631731831932032132232332432532632732832933033133233333433533633733833934034134234334434534634734834935035135235335435535635735835936036136236336436536636736836937037137237337437537637737837938038138238338438538638738838939039139239339439539639739839940040140240340440540640740840941041141241341441541641741841942042142242342442542642742842943043143243343443543643743843944044144244344444544644744844945045145245345445545645745845946046146246346446546646746846947047147247347447547647747847948048148248348448548648748848949049149249349449549649749849950050150250350450550650750850951051151251351451551651751851952052152252352452552652752852953053153253353453553653753853954054154254354454554654754854955055155255355455555655755855956056156256356456556656756856957057157257357457557657757857958058158258358458558658758858959059159259359459559659759859960060160260360460560660760860961061161261361461561661761861962062162262362462562662762862963063163263363463563663763863964064164264364464564664764864965065165265365465565665765865966066166266366466566666766866967067167267367467567667767867968068168268368468568668768868969069169269369469569669769869970070170270370470570670770870971071171271371471571671771871972072172272372472572672772872973073173273373473573673773873974074174274374474574674774874975075175275375475575675775875976076176276376476576676776876977077177277377477577677777877978078178278378478578678778878979079179279379479579679779879980080180280380480580680780880981081181281381481581681781881982082182282382482582682782882983083183283383483583683783883984084184284384484584684784884985085185285385485585685785885986086186286386486586686786886987087187287387487587687787887988088188288388488588688788888989089189289389489589689789889990090190290390490590690790890991091191291391491591691791891992092192292392492592692792892993093193293393493593693793893994094194294394494594694794894995095195295395495595695795895996096196296396496596696796896997097197297397497597697797897998098198298398498598698798898999099199299399499599699799899910001001100210031004100510061007100810091010101110121013101410151016101710181019102010211022102310241025102610271028102910301031103210331034103510361037103810391040104110421043104410451046104710481049105010511052105310541055105610571058105910601061106210631064106510661067106810691070107110721073107410751076107710781079108010811082108310841085108610871088108910901091109210931094109510961097109810991100110111021103110411051106110711081109111011111112111311141115111611171118111911201121112211231124112511261127112811291130113111321133113411351136113711381139114011411142114311441145114611471148114911501151115211531154115511561157115811591160116111621163116411651166116711681169117011711172117311741175117611771178117911801181118211831184118511861187118811891190119111921193119411951196119711981199120012011202120312041205120612071208120912101211121212131214121512161217121812191220122112221223122412251226122712281229123012311232123312341235123612371238123912401241124212431244124512461247124812491250125112521253125412551256125712581259126012611262126312641265126612671268126912701271127212731274127512761277127812791280128112821283128412851286
  1. <template>
  2. <div class="m-container">
  3. <h2>
  4. <el-page-header @back="onCancel" :content="name"></el-page-header>
  5. </h2>
  6. <div class="m-core">
  7. <el-form ref="liveForm" :model="form" label-position="top">
  8. <el-alert
  9. title="课程规划"
  10. :closable="false"
  11. type="info"
  12. style="margin: 0 0 20px"
  13. />
  14. <el-row :gutter="20">
  15. <el-col :span="10">
  16. <el-form-item
  17. label="直播课标题"
  18. prop="roomTitle"
  19. :rules="[{ required: true, message: '请输入直播课标题' }]"
  20. >
  21. <el-input
  22. v-model="form.roomTitle"
  23. placeholder="请输入直播课标题"
  24. maxlength="20"
  25. show-word-limit
  26. :disabled="![0, 1].includes(status)"
  27. ></el-input>
  28. </el-form-item>
  29. </el-col>
  30. <el-col :span="10">
  31. <el-form-item
  32. label="直播课内容"
  33. prop="liveRemark"
  34. :rules="[{ required: true, message: '请输入直播课内容' }]"
  35. >
  36. <el-input
  37. type="textarea"
  38. v-model="form.liveRemark"
  39. placeholder="请输入直播课内容"
  40. maxlength="200"
  41. show-word-limit
  42. :disabled="![0, 1].includes(status)"
  43. ></el-input>
  44. </el-form-item>
  45. </el-col>
  46. <el-col :span="10">
  47. <el-form-item
  48. label="分部"
  49. prop="organIds"
  50. :rules="[{ required: true, message: '请选择分部' }]"
  51. >
  52. <select-all
  53. v-model.trim="form.organIds"
  54. filterable
  55. placeholder="请选择分部"
  56. multiple
  57. clearable
  58. :disabled="isDisabled"
  59. >
  60. <el-option
  61. v-for="(item, index) in selects.branchs"
  62. :key="index"
  63. :label="item.name"
  64. :value="item.id"
  65. ></el-option>
  66. </select-all>
  67. </el-form-item>
  68. </el-col>
  69. <el-col :span="10">
  70. <el-form-item
  71. label="声部"
  72. prop="subjectIdList"
  73. :rules="[{ required: true, message: '请选择声部' }]"
  74. >
  75. <el-select
  76. v-model.trim="form.subjectIdList"
  77. filterable
  78. clearable
  79. @change="onChangeSubject"
  80. placeholder="请选择声部"
  81. style="width: 100% !important"
  82. :disabled="isDisabled"
  83. >
  84. <el-option
  85. v-for="(item, index) in subjectList"
  86. :key="index"
  87. :value="item.id"
  88. :label="item.name"
  89. />
  90. </el-select>
  91. </el-form-item>
  92. </el-col>
  93. <el-col :span="10">
  94. <el-form-item
  95. label="指导老师"
  96. prop="teacher"
  97. :rules="[{ required: true, message: '请选择指导老师' }]"
  98. >
  99. <el-select
  100. v-model.trim="form.teacher"
  101. filterable
  102. clearable
  103. placeholder="请选择指导老师"
  104. style="width: 100% !important"
  105. :disabled="!form.subjectIdList"
  106. >
  107. <el-option
  108. v-for="(item, index) in teacherList"
  109. :key="index"
  110. :label="item.realName"
  111. :value="item.id"
  112. />
  113. </el-select>
  114. </el-form-item>
  115. </el-col>
  116. <el-col :span="10">
  117. <el-form-item
  118. label="乐团主管"
  119. prop="educationalTeacherId"
  120. :rules="[{ required: true, message: '请选择乐团主管' }]"
  121. >
  122. <el-select
  123. v-model.trim="form.educationalTeacherId"
  124. filterable
  125. clearable
  126. style="width: 100% !important"
  127. :rules="[{ required: true, message: '请选择乐团主管' }]"
  128. >
  129. <el-option
  130. v-for="(item, key) in educationList"
  131. :key="key"
  132. :label="item.userName"
  133. :value="item.userId"
  134. />
  135. </el-select>
  136. </el-form-item>
  137. </el-col>
  138. <el-col :span="10">
  139. <el-form-item
  140. label="课程购买开始时间"
  141. prop="signUpStart"
  142. :rules="[{ required: true, message: '请选择课程购买开始时间' }]"
  143. >
  144. <!-- <el-date-picker style="width: 100%" v-model="form.signUpTimeList" :picker-options="pickerOptions"
  145. type="daterange" :default-time="['00:00:00', '23:59:59']" range-separator="-" start-placeholder="购买开始日期"
  146. end-placeholder="购买结束日期">
  147. </el-date-picker> -->
  148. <el-date-picker
  149. v-model="form.signUpStart"
  150. :picker-options="startBigin()"
  151. type="date"
  152. style="width: 100% !important"
  153. placeholder="购买开始日期"
  154. @change="
  155. () => {
  156. form.signUpEnd = '';
  157. form.timeTable = []; // 课表重置
  158. }
  159. "
  160. :disabled="isDisabled"
  161. >
  162. </el-date-picker>
  163. </el-form-item>
  164. </el-col>
  165. <el-col :span="10">
  166. <el-form-item
  167. label="课程购买结束时间"
  168. prop="signUpEnd"
  169. :rules="[{ required: true, message: '请选择课程购买结束时间' }]"
  170. >
  171. <el-date-picker
  172. v-model="form.signUpEnd"
  173. type="date"
  174. :picker-options="beginDate()"
  175. style="width: 100% !important"
  176. placeholder="购买结束日期"
  177. @change="
  178. () => {
  179. if (status == 0) {
  180. form.timeTable = []; // 课表重置
  181. }
  182. }
  183. "
  184. :disabled="type === 'update' && status === 2"
  185. >
  186. </el-date-picker>
  187. </el-form-item>
  188. </el-col>
  189. <el-col :span="10">
  190. <el-form-item
  191. label="课时数"
  192. prop="onlineClassesNum"
  193. :rules="[
  194. {
  195. required: true,
  196. validator: validStock,
  197. trigger: 'blur'
  198. }
  199. ]"
  200. >
  201. <el-input
  202. v-model="form.onlineClassesNum"
  203. placeholder="请输入课时数"
  204. maxlength="2"
  205. @input="
  206. val => {
  207. form.onlineClassesNum = val.replace(/[^\d]/g, '');
  208. }
  209. "
  210. @change="
  211. () => {
  212. form.timeTable = []; // 课表重置
  213. }
  214. "
  215. :disabled="isDisabled"
  216. ></el-input>
  217. </el-form-item>
  218. </el-col>
  219. <el-col :span="10">
  220. <el-form-item
  221. label="课程时长"
  222. prop="singleClassMinuteId"
  223. :rules="[{ required: true, message: '请选择课程时长' }]"
  224. >
  225. <el-select
  226. v-model.trim="form.singleClassMinuteId"
  227. filterable
  228. clearable
  229. style="width: 100% !important"
  230. placeholder="请选择课程时长"
  231. @change="onSingleClassChange"
  232. :disabled="isDisabled"
  233. >
  234. <el-option
  235. v-for="(item, key) in liveGroupList"
  236. :key="key"
  237. :label="item.singleClassMinutes"
  238. :value="item.id"
  239. />
  240. </el-select>
  241. </el-form-item>
  242. </el-col>
  243. <el-col :span="10">
  244. <el-form-item
  245. label="现单价"
  246. prop="onlineClassesUnitPrice"
  247. :rules="[
  248. {
  249. required: true,
  250. message: '请输入正确的现单价',
  251. trigger: 'blur',
  252. pattern: /(^[1-9]([0-9]+)?(\.[0-9]{1,2})?$)|(^(0){1}$)|(^[0-9]\.[0-9]([0-9])?$)/
  253. }
  254. ]"
  255. >
  256. <el-input
  257. v-model="form.onlineClassesUnitPrice"
  258. placeholder="请输入现单价"
  259. maxlength="9"
  260. :disabled="isDisabled"
  261. ></el-input>
  262. </el-form-item>
  263. </el-col>
  264. <el-col :span="10">
  265. <el-form-item
  266. label="原单价"
  267. prop="offlineClassesUnitPrice"
  268. :rules="[
  269. {
  270. required: true,
  271. message: '请输入正确的原单价',
  272. trigger: 'blur',
  273. pattern: /(^[1-9]([0-9]+)?(\.[0-9]{1,2})?$)|(^(0){1}$)|(^[0-9]\.[0-9]([0-9])?$)/
  274. }
  275. ]"
  276. >
  277. <el-input
  278. v-model="form.offlineClassesUnitPrice"
  279. placeholder="请输入原单价"
  280. maxlength="9"
  281. :disabled="isDisabled"
  282. ></el-input>
  283. </el-form-item>
  284. </el-col>
  285. <el-col :span="10">
  286. <el-form-item label="总现价" prop="countOnlineClassesUnitPrice">
  287. <el-input
  288. v-model="countOnlineClassesUnitPrice"
  289. placeholder="请输入总现价"
  290. maxlength="9"
  291. disabled
  292. ></el-input>
  293. </el-form-item>
  294. </el-col>
  295. <el-col :span="10">
  296. <el-form-item label="总原价" prop="countOfflineClassesUnitPrice">
  297. <el-input
  298. v-model="countOfflineClassesUnitPrice"
  299. placeholder="请输入总原价"
  300. maxlength="9"
  301. disabled
  302. ></el-input>
  303. </el-form-item>
  304. </el-col>
  305. </el-row>
  306. <el-button type="danger" @click="onTimeTable" :disabled="isDisabled"
  307. >点击排课</el-button
  308. >
  309. <el-table
  310. :key="studentRefKey"
  311. v-if="showTable"
  312. style="width: 100%; margin-top: 20px"
  313. :header-cell-style="{ background: '#EDEEF0', color: '#444' }"
  314. :data="form.timeTable"
  315. >
  316. <el-table-column align="center" label="课时">
  317. <template slot-scope="scope">
  318. 第{{ scope.$index + 1 }}课
  319. </template>
  320. </el-table-column>
  321. <el-table-column
  322. align="center"
  323. label="内容"
  324. width="300px"
  325. prop="teachingContent"
  326. key="teachingContent"
  327. >
  328. <template slot-scope="scope">
  329. <el-form-item
  330. :prop="'timeTable.' + scope.$index + '.teachingContent'"
  331. :rules="[{ required: true, message: '请输入内容' }]"
  332. style="margin-bottom: 0"
  333. >
  334. <el-input
  335. v-model="scope.row.teachingContent"
  336. placeholder="请输入内容"
  337. maxlength="100"
  338. show-word-limit
  339. type="textarea"
  340. :rows="3"
  341. >
  342. </el-input>
  343. </el-form-item>
  344. </template>
  345. </el-table-column>
  346. <el-table-column
  347. align="center"
  348. label="技能/知识点掌握"
  349. width="300px"
  350. prop="teachingPoint"
  351. key="teachingPoint"
  352. >
  353. <template slot-scope="scope">
  354. <el-form-item
  355. :prop="'timeTable.' + scope.$index + '.teachingPoint'"
  356. :rules="[{ required: true, message: '请输入技能/知识点掌握' }]"
  357. style="margin-bottom: 0"
  358. >
  359. <el-input
  360. v-model="form.timeTable[scope.$index].teachingPoint"
  361. placeholder="请输入技能/知识点掌握"
  362. show-word-limit
  363. maxlength="100"
  364. type="textarea"
  365. :rows="3"
  366. >
  367. </el-input>
  368. </el-form-item>
  369. </template>
  370. </el-table-column>
  371. <el-table-column
  372. align="center"
  373. prop="singleClassMinutes"
  374. label="时长"
  375. ></el-table-column>
  376. <el-table-column align="center" label="课程日期">
  377. <template slot-scope="scope">
  378. <div>{{ scope.row.classDate | formatTimer }}</div>
  379. </template>
  380. </el-table-column>
  381. <el-table-column
  382. align="center"
  383. prop="startClassTimeStr"
  384. label="开始时间"
  385. ></el-table-column>
  386. <el-table-column
  387. align="center"
  388. prop="endClassTimeStr"
  389. label="结束时间"
  390. ></el-table-column>
  391. <el-table-column align="center" label="操作">
  392. <template slot-scope="scope">
  393. <el-button
  394. type="text"
  395. @click="resetCourse(scope.row, scope.$index)"
  396. :disabled="isDisabled"
  397. >调整</el-button
  398. >
  399. </template>
  400. </el-table-column>
  401. </el-table>
  402. <el-alert
  403. title="直播课信息"
  404. :closable="false"
  405. type="info"
  406. style="margin: 20px 0"
  407. />
  408. <el-row :gutter="20">
  409. <el-col :span="6">
  410. <el-form-item
  411. label="直播设备"
  412. prop="os"
  413. :rules="[{ required: true, message: '请选择推广类型' }]"
  414. >
  415. <el-radio-group v-model="form.os" :disabled="isDisabled">
  416. <!-- 根据不同的模式,显示不同的直播设备 -->
  417. <el-radio v-if="serviceProvider === 'rongCloud'" label="pc"
  418. >web</el-radio
  419. >
  420. <el-radio
  421. v-if="serviceProvider === 'tencentCloud'"
  422. label="client"
  423. >乐直播</el-radio
  424. >
  425. <el-radio label="mobile">手机</el-radio>
  426. </el-radio-group>
  427. </el-form-item>
  428. </el-col>
  429. <el-col :span="6">
  430. <el-form-item
  431. label="直播场景"
  432. prop="useScene"
  433. :rules="[{ required: true, message: '请选择直播场景' }]"
  434. >
  435. <el-radio-group v-model="form.useScene" :disabled="isDisabled">
  436. <el-radio label="NORMAL">普通场景</el-radio>
  437. <el-radio label="MUSIC">音乐场景</el-radio>
  438. </el-radio-group>
  439. </el-form-item>
  440. </el-col>
  441. <el-col :span="6">
  442. <el-form-item
  443. prop="roomConfig.whether_video"
  444. label="保存直播回放"
  445. :rules="[{ required: true, message: '是否保存直播回放' }]"
  446. >
  447. <el-radio-group
  448. v-model="form.roomConfig.whether_video"
  449. :disabled="isDisabled"
  450. >
  451. <el-radio :label="0">是</el-radio>
  452. <el-radio :label="1">否</el-radio>
  453. </el-radio-group>
  454. </el-form-item>
  455. </el-col>
  456. <el-col :span="6">
  457. <el-form-item
  458. prop="roomConfig.whether_view_shop_cart"
  459. label="是否展示购物车"
  460. :rules="[{ required: true, message: '是否展示购物车' }]"
  461. >
  462. <el-radio-group
  463. v-model="form.roomConfig.whether_view_shop_cart"
  464. :disabled="isDisabled"
  465. >
  466. <el-radio :label="0">是</el-radio>
  467. <el-radio :label="1">否</el-radio>
  468. </el-radio-group>
  469. </el-form-item>
  470. </el-col>
  471. <el-col :span="24">
  472. <el-form-item
  473. label="预热模板(模板使用于分享宣传图片)"
  474. prop="preTemplate"
  475. :rules="[{ required: true, message: '请选择预热模板' }]"
  476. >
  477. <el-radio-group v-model="form.preTemplate" :disabled="isDisabled">
  478. <div class="chioseWrap">
  479. <div class="chioseItem" @click="setPreTemplate(1)">
  480. <img src="./images/img1.png" alt="" />
  481. <i
  482. class="dotWrap"
  483. :class="[
  484. form.preTemplate == 1 ? 'checked' : '',
  485. isDisabled ? 'disabled' : ''
  486. ]"
  487. ></i>
  488. </div>
  489. <div class="chioseItem" @click="setPreTemplate(2)">
  490. <img src="./images/img2.png" alt="" />
  491. <i
  492. class="dotWrap"
  493. :class="[
  494. form.preTemplate == 2 ? 'checked' : '',
  495. isDisabled ? 'disabled' : ''
  496. ]"
  497. ></i>
  498. </div>
  499. <div class="chioseItem" @click="setPreTemplate(3)">
  500. <img src="./images/img3.png" alt="" />
  501. <i
  502. class="dotWrap"
  503. :class="[
  504. form.preTemplate == 3 ? 'checked' : '',
  505. isDisabled ? 'disabled' : ''
  506. ]"
  507. ></i>
  508. </div>
  509. </div>
  510. </el-radio-group>
  511. </el-form-item>
  512. </el-col>
  513. </el-row>
  514. <el-row>
  515. <el-col :span="24">
  516. <el-button
  517. type="primary"
  518. @click="onReset"
  519. v-if="[0, 1].includes(status)"
  520. >重置</el-button
  521. >
  522. <el-button type="primary" @click="onSubmit">确定</el-button>
  523. </el-col>
  524. </el-row>
  525. </el-form>
  526. </div>
  527. <el-dialog
  528. title="排课"
  529. ref="maskForm"
  530. width="500px"
  531. :visible.sync="dialogFormVisible"
  532. >
  533. <addLiveCourse
  534. :singleClassMinutes="form.singleClassMinutes"
  535. :signUpEnd="form.signUpEnd"
  536. :status="status"
  537. :onlineCourseNum="form.onlineClassesNum"
  538. @close="dialogFormVisible = false"
  539. @confirm="onConfirm"
  540. />
  541. </el-dialog>
  542. <el-dialog
  543. title="课程调整"
  544. width="400px"
  545. :before-close="handleClose"
  546. :visible.sync="courseVisible"
  547. >
  548. <el-form
  549. :model="maskForm"
  550. class="maskForm"
  551. ref="maskForm"
  552. :rules="maskRules"
  553. label-position="right"
  554. label-width="80px"
  555. :inline="true"
  556. >
  557. <el-form-item label="上课日期" prop="date">
  558. <el-date-picker
  559. v-model.trim="maskForm.date"
  560. type="date"
  561. :picker-options="coursesDate()"
  562. value-format="yyyy-MM-dd HH:mm:ss"
  563. placeholder="选择日期"
  564. ></el-date-picker>
  565. </el-form-item>
  566. <el-form-item label="开始时间" prop="startTime">
  567. <el-time-picker
  568. placeholder="起始时间"
  569. v-model.trim="startTime"
  570. @change="changeStartTime"
  571. format="HH:mm"
  572. value-format="HH:mm"
  573. :picker-options="{
  574. selectableRange: `${nowTime} - 23:30:00`
  575. }"
  576. ></el-time-picker>
  577. </el-form-item>
  578. <el-form-item label="结束时间" prop="endTime">
  579. <el-time-picker
  580. placeholder="结束时间"
  581. format="HH:mm"
  582. value-format="HH:mm"
  583. v-model="maskForm.endTime"
  584. disabled
  585. >
  586. </el-time-picker>
  587. </el-form-item>
  588. </el-form>
  589. <div slot="footer" class="dialog-footer">
  590. <el-button @click="courseVisible = false">取 消</el-button>
  591. <el-button type="primary" @click="submitResetClass">确 定</el-button>
  592. </div>
  593. </el-dialog>
  594. </div>
  595. </template>
  596. <script>
  597. import dayjs from "dayjs";
  598. import deepClone from "@/helpers/deep-clone";
  599. import preview from "./modals/preview.vue";
  600. import addLiveCourse from "./modals/addLiveCourse.vue";
  601. import { diffTimerFormMinute, addTimerFormMinute } from "@/utils/date";
  602. import {
  603. sysTenantConfigAll,
  604. findTeacherByTenantId,
  605. liveGroupDetail,
  606. updateLiveGroup
  607. } from "./api";
  608. import { getSubject, getOrganRole } from "@/api/buildTeam";
  609. import { vipGroupCategory, createVip } from "@/api/vipSeting";
  610. export default {
  611. components: { preview, addLiveCourse },
  612. data() {
  613. const query = this.$route.query;
  614. return {
  615. name: query.type == "update" ? "修改直播课" : "新建直播课",
  616. id: query.id,
  617. type: query.type,
  618. dialogFormVisible: false,
  619. form: {
  620. roomTitle: "", //
  621. liveRemark: "", // 内容
  622. organIds: [],
  623. subjectIdList: null, // 声部
  624. teacher: "", // 指导老师列表
  625. educationalTeacherId: null, // 乐团主管
  626. preTemplate: 1, // 模板
  627. signUpStart: null, // 开始时间
  628. signUpEnd: null, // 结束时间
  629. signUpTimeList: [], // 课程购买时间
  630. onlineClassesNum: null,
  631. singleClassMinuteId: null, //时长编号
  632. singleClassMinutes: null, // 时长
  633. onlineClassesUnitPrice: null, // 售价
  634. offlineClassesUnitPrice: null, // 原价
  635. os: "client", // 直播设备
  636. useScene: "NORMAL", // 直播场景
  637. popularizeType: "ALL", // 观看权限信息
  638. viewMode: "LOGIN",
  639. roomConfig: {
  640. whether_like: 0,
  641. whether_chat: 0,
  642. whether_video: 0,
  643. whether_mic: 0,
  644. whether_view_shop_cart: 1
  645. },
  646. timeTable: [], // 排课
  647. clientType: "TEACHER" // 主讲人身份 默认[老师]
  648. },
  649. status: 0, // 直播课状态
  650. auditStatus: null, // 审核状态
  651. courseStartDate: null, // 课程开始时间
  652. serviceProvider: "tencentCloud", // 直播模式
  653. subjectList: [], // 声部列表
  654. teacherList: [], // 指导老师
  655. educationList: [], // 乐团主管
  656. liveGroupList: [], // 课时列表
  657. activeRow: null,
  658. actvieIndex: 0,
  659. maskForm: {
  660. date: "",
  661. startTime: "",
  662. endTime: "",
  663. id: "",
  664. teachMode: ""
  665. },
  666. maskRules: {
  667. date: [{ required: true, message: "请选择上课时间", trigger: "blur" }]
  668. },
  669. courseVisible: false,
  670. startTime: "",
  671. courseTime: "",
  672. studentRefKey: "12321321",
  673. showTable: true
  674. };
  675. },
  676. async mounted() {
  677. this.$store.dispatch("setBranchs");
  678. await this.__init();
  679. },
  680. methods: {
  681. async onChangeSubject(val) {
  682. try {
  683. // 判断声部,如果为MUSIC_THEORY则为乐理,显示普通
  684. let subjectItem = {};
  685. this.subjectList.forEach(item => {
  686. if (val == item.id) {
  687. subjectItem = item;
  688. }
  689. });
  690. if (subjectItem.code !== "MUSIC_THEORY") {
  691. this.form.useScene = "MUSIC";
  692. } else {
  693. this.form.useScene = "NORMAL";
  694. }
  695. this.form.teacher = ""; // 重置指导老师
  696. // 根据科目id获取相应的老师
  697. await findTeacherByTenantId({
  698. subjectIds: subjectItem.code == "MUSIC_THEORY" ? null : val
  699. }).then(res => {
  700. this.teacherList = res.data;
  701. });
  702. } catch {}
  703. },
  704. onCancel() {
  705. this.$store.dispatch("delVisitedViews", this.$route);
  706. this.$router.push("/liveClassManager?tabrouter=2");
  707. },
  708. coursesDate(dateStr) {
  709. let self = this;
  710. return {
  711. firstDayOfWeek: 1,
  712. disabledDate: time => {
  713. return time.getTime() + 86400000 <= new Date().getTime();
  714. }
  715. };
  716. },
  717. pickerOptions(dateStr) {
  718. return {
  719. firstDayOfWeek: 1,
  720. disabledDate(time) {
  721. return time.getTime() + 86400000 <= new Date().getTime();
  722. }
  723. };
  724. },
  725. resetCourse(row, index) {
  726. /**
  727. * maskForm.startTime
  728. */
  729. this.activeRow = row;
  730. this.actvieIndex = index;
  731. this.maskForm.date = row.classDate;
  732. this.startTime = row.startClassTimeStr;
  733. this.maskForm.endTime = row.endClassTimeStr;
  734. this.courseTime = row.singleClassMinutes;
  735. // this.maskForm.endTime = row.endClassTimeStr;
  736. console.log(this.startTime, "this.startTime");
  737. this.maskForm.id = index;
  738. this.maskForm.teachMode = row.teachMode;
  739. // 修改课时
  740. this.courseVisible = true;
  741. },
  742. handleClose() {
  743. this.courseVisible = false;
  744. this.startTime = "";
  745. this.maskForm = {
  746. date: "",
  747. startTime: "",
  748. endTime: "",
  749. id: "",
  750. address: "",
  751. teachMode: ""
  752. };
  753. this.actvieRow = null;
  754. this.actvieIndex = 0;
  755. this.$refs["maskForm"].resetFields();
  756. },
  757. submitResetClass() {
  758. console.log(this.activeRow, "activeRow");
  759. console.log(this.maskForm, "maskForm");
  760. this.form.timeTable[
  761. this.actvieIndex
  762. ].actualTeacherId = this.activeRow.actualTeacherId;
  763. this.form.timeTable[this.actvieIndex].classDate = this.maskForm.date;
  764. this.form.timeTable[
  765. this.actvieIndex
  766. ].endClassTimeStr = this.maskForm.endTime;
  767. this.form.timeTable[
  768. this.actvieIndex
  769. ].singleClassMinutes = this.courseTime;
  770. this.form.timeTable[this.actvieIndex].startClassTimeStr = this.startTime;
  771. this.form.timeTable[
  772. this.actvieIndex
  773. ].teachMode = this.activeRow.teachMode;
  774. this.form.timeTable[
  775. this.actvieIndex
  776. ].teachingContent = this.activeRow.teachingContent;
  777. this.form.timeTable[
  778. this.actvieIndex
  779. ].teachingPoint = this.activeRow.teachingPoint;
  780. console.log(
  781. " this.form.timeTable[this.actvieIndex]",
  782. this.form.timeTable[this.actvieIndex]
  783. );
  784. this.form.timeTable.sort((a, b) => {
  785. const aTime = dayjs(
  786. dayjs(a.classDate).format("YYYY-MM-DD") +
  787. " " +
  788. a.startClassTimeStr +
  789. ":00"
  790. ).valueOf();
  791. const bTime = dayjs(
  792. dayjs(b.classDate).format("YYYY-MM-DD") +
  793. " " +
  794. b.startClassTimeStr +
  795. ":00"
  796. ).valueOf();
  797. return aTime - bTime;
  798. });
  799. this.showTable = false;
  800. this.handleClose();
  801. this.studentRefKey = Math.random();
  802. this.$nextTick(() => {
  803. this.showTable = true;
  804. });
  805. },
  806. changeStartTime(val) {
  807. this.$nextTick(res => {
  808. if (val) {
  809. this.$set(
  810. this.maskForm,
  811. "endTime",
  812. addTimerFormMinute(
  813. dayjs(this.maskForm.date).format("YYYY-MM-DD"),
  814. val,
  815. this.courseTime
  816. )
  817. );
  818. console.log(
  819. addTimerFormMinute(
  820. dayjs(this.maskForm.date).format("YYYY-MM-DD"),
  821. val,
  822. this.courseTime
  823. ),
  824. "addTimerFormMinute"
  825. );
  826. } else {
  827. this.$set(this.maskForm, "endTime", "");
  828. }
  829. if (!this.maskForm.endTime) {
  830. this.$set(this, "startTime", "");
  831. }
  832. });
  833. },
  834. validStock(rule, value, callback) {
  835. if ((value == "" && typeof value == "string") || value == null) {
  836. callback(new Error("请输入课时数"));
  837. } else if (value <= 0) {
  838. callback(new Error("课时数必须大于0"));
  839. } else {
  840. callback();
  841. }
  842. },
  843. setPreTemplate(index) {
  844. if (this.isDisabled) return;
  845. this.form.preTemplate = index;
  846. },
  847. async onSubmit() {
  848. this.$refs.liveForm.validate(async flag => {
  849. if (!flag) {
  850. this.onScrollError();
  851. return false;
  852. }
  853. try {
  854. const form = this.form;
  855. if (form.timeTable.length <= 0) {
  856. this.$message.error("请点击排课");
  857. return;
  858. }
  859. const timeTable = [];
  860. form.timeTable.forEach(item => {
  861. timeTable.push({
  862. classDate: item.classDate,
  863. actualTeacherId: form.teacher,
  864. startClassTimeStr: item.startClassTimeStr,
  865. endClassTimeStr: item.endClassTimeStr,
  866. teachMode: item.teachMode,
  867. id: item.id,
  868. teachingContent: item.teachingContent,
  869. teachingPoint: item.teachingPoint
  870. });
  871. });
  872. let obj = {
  873. courseSchedules: timeTable,
  874. vipGroupApplyBaseInfo: {
  875. groupType: "LIVE",
  876. vipGroupStudentCoursePrices: [],
  877. // coursesExpireDate: this.leftForm.courseEnd,
  878. // teacherSchoolId: this.leftForm.section,
  879. studentIdList: "",
  880. offlineClassesNum: 0,
  881. onlineClassesNum: form.onlineClassesNum || 0,
  882. offlineClassesUnitPrice: form.offlineClassesUnitPrice || 0,
  883. onlineClassesUnitPrice: form.onlineClassesUnitPrice || 0,
  884. registrationStartTime: dayjs(form.signUpStart).format(
  885. "YYYY-MM-DD"
  886. ),
  887. paymentExpireDate: dayjs(form.signUpEnd).format("YYYY-MM-DD"),
  888. singleClassMinutes: form.singleClassMinutes,
  889. userId: form.teacher,
  890. // vipGroupActivityId: form.singleClassMinuteId,
  891. vipGroupCategoryId: form.singleClassMinuteId,
  892. onlineTeacherSalary: 0,
  893. offlineTeacherSalary: 0,
  894. giveTeachMode: "ONLINE",
  895. subjectIdList: form.subjectIdList,
  896. educationalTeacherId: form.educationalTeacherId,
  897. organId: -1,
  898. organIdList: form.organIds.join(",")
  899. },
  900. liveBroadcastRoom: {
  901. speakerId: form.teacher,
  902. clientType: "TEACHER",
  903. roomTitle: form.roomTitle,
  904. liveRemark: form.liveRemark,
  905. preTemplate: form.preTemplate,
  906. useScene: form.useScene,
  907. os: form.os,
  908. serviceProvider: form.serviceProvider,
  909. viewMode: form.viewMode,
  910. popularizeType: form.popularizeType,
  911. roomConfig: {
  912. ...form.roomConfig,
  913. subjectId: form.subjectIdList,
  914. groupType: "LIVE"
  915. }
  916. }
  917. };
  918. console.log(obj, "obj");
  919. if (this.type === "update") {
  920. obj.vipGroupApplyBaseInfo.id = this.id;
  921. obj.vipGroupApplyBaseInfo.auditStatus = this.auditStatus;
  922. await updateLiveGroup(obj);
  923. this.$message.success("修改成功");
  924. this.$store.dispatch("delVisitedViews", this.$route);
  925. this.$router.push({
  926. path: "/liveClassManager",
  927. query: {
  928. tabrouter: 2
  929. }
  930. });
  931. } else {
  932. createVip(obj).then(res => {
  933. if (res.code == 200) {
  934. this.$message.success("创建成功");
  935. this.$store.dispatch("delVisitedViews", this.$route);
  936. this.$router.push({
  937. path: "/liveClassManager",
  938. query: {
  939. tabrouter: 2
  940. }
  941. });
  942. }
  943. });
  944. }
  945. } catch (e) {
  946. console.log(e);
  947. }
  948. });
  949. },
  950. onReset() {
  951. // 重置
  952. this.form.timeTable = [];
  953. this.$refs.liveForm.resetFields();
  954. this.$nextTick(() => {
  955. let isError = document.getElementsByClassName("el-alert");
  956. isError[0].scrollIntoView({
  957. block: "center",
  958. behavior: "smooth"
  959. });
  960. });
  961. },
  962. // 点击排课
  963. async onTimeTable() {
  964. let count = 0;
  965. this.$refs.liveForm.validateField(
  966. ["signUpStart", "signUPEnd", "onlineClassesNum", "singleClassMinuteId"],
  967. valid => {
  968. count += 1;
  969. if (valid) {
  970. this.onScrollError();
  971. return;
  972. }
  973. if (count >= 3) {
  974. this.dialogFormVisible = true;
  975. }
  976. }
  977. );
  978. },
  979. beginDate() {
  980. const timer = this.form.signUpStart || "";
  981. const courseTimer = this.courseStartDate
  982. ? dayjs(this.courseStartDate).format("YYYY-MM-DD")
  983. : "";
  984. const type = this.type;
  985. const status = this.status;
  986. // console.log(type, courseTimer, 'courseTimer')
  987. return {
  988. firstDayOfWeek: 1,
  989. disabledDate(time) {
  990. if (timer) {
  991. if (courseTimer && type === "update" && status != 0) {
  992. // console.log(new Date(timer).getTime(), time.getTime(), new Date(timer).getTime() >= time.getTime(), dayjs(time).format('YYYY-MM-DD'), new Date(courseTimer + ' 00:00:00').getTime(), time.getTime() >= new Date(courseTimer + ' 00:00:00').getTime(), courseTimer)
  993. return (
  994. new Date(timer).getTime() > time.getTime() ||
  995. time.getTime() >= new Date(courseTimer + " 00:00:00").getTime()
  996. );
  997. } else {
  998. return new Date(timer).getTime() > time.getTime();
  999. }
  1000. } else {
  1001. return time.getTime() + 86400000 <= new Date().getTime();
  1002. }
  1003. }
  1004. };
  1005. },
  1006. startBigin() {
  1007. return {
  1008. firstDayOfWeek: 1,
  1009. disabledDate(time) {
  1010. // return time.getTime() >= Date.now();
  1011. return time.getTime() + 86400000 <= new Date().getTime();
  1012. }
  1013. };
  1014. },
  1015. onScrollError() {
  1016. this.$nextTick(() => {
  1017. let isError = document.getElementsByClassName("is-error");
  1018. isError[0].scrollIntoView({
  1019. block: "center",
  1020. behavior: "smooth"
  1021. });
  1022. });
  1023. },
  1024. onSingleClassChange(val) {
  1025. // 设置 - 课程时长切换时
  1026. let onlinePrice = null;
  1027. let offLinePrice = null;
  1028. let minus = null;
  1029. this.liveGroupList.forEach(item => {
  1030. if (item.id === val) {
  1031. onlinePrice = item.onlineClassesUnitPrice;
  1032. offLinePrice = item.offlineClassesUnitPrice;
  1033. minus = item.singleClassMinutes;
  1034. }
  1035. });
  1036. this.form.onlineClassesUnitPrice = onlinePrice;
  1037. this.form.offlineClassesUnitPrice = offLinePrice;
  1038. this.form.singleClassMinutes = minus;
  1039. this.form.timeTable = []; // 课表重置
  1040. },
  1041. onConfirm(val) {
  1042. let tempVal = deepClone(val || []);
  1043. tempVal.forEach(item => {
  1044. item.teachingContent = "";
  1045. item.teachingPoint = "";
  1046. item.singleClassMinutes = this.form.singleClassMinutes;
  1047. });
  1048. this.form.timeTable = tempVal;
  1049. console.log(this.form.timeTable, "time table");
  1050. this.$forceUpdate();
  1051. },
  1052. async __init() {
  1053. try {
  1054. const findName = await sysTenantConfigAll({
  1055. group: "LIVE_CLIENT"
  1056. });
  1057. if (findName.data && findName.data.length > 0) {
  1058. findName.data.forEach(item => {
  1059. if (item.paramName == "live_client") {
  1060. this.serviceProvider = item.paranValue;
  1061. this.form.os =
  1062. this.serviceProvider == "tencentCloud" ? "client" : "pc";
  1063. }
  1064. });
  1065. }
  1066. // 获取 指导老师列表
  1067. await getSubject({
  1068. tenantId: 1
  1069. }).then(res => {
  1070. if (res.code == 200) {
  1071. this.subjectList = [...res.data];
  1072. }
  1073. });
  1074. // 获取乐团主管
  1075. await getOrganRole({ all: true }).then(ruselt => {
  1076. this.educationList = ruselt?.data?.EDUCATION;
  1077. });
  1078. // 获取课时数
  1079. // 获取默认左边参数
  1080. await vipGroupCategory({
  1081. groupType: "live"
  1082. }).then(res => {
  1083. if (res.code == 200) {
  1084. this.liveGroupList = res.data;
  1085. }
  1086. });
  1087. // 获取详情
  1088. if (this.id) {
  1089. const query = this.$route.query;
  1090. const { data } = await liveGroupDetail({ id: query.id });
  1091. const liveBroadcastRoom = data.liveBroadcastRoom || {};
  1092. const vipGroupApplyBaseInfo = data.vipGroupApplyBaseInfo || {};
  1093. const courseSchedules = data.courseSchedules || [];
  1094. const tempCourse = [];
  1095. courseSchedules.forEach(item => {
  1096. tempCourse.push({
  1097. classDate: item.classDate,
  1098. actualTeacherId: item.teacherId,
  1099. startClassTimeStr: dayjs(item.startClassTime).format("HH:mm"),
  1100. endClassTimeStr: dayjs(item.endClassTime).format("HH:mm"),
  1101. teachMode: "ONLINE",
  1102. id: item.id,
  1103. singleClassMinutes: vipGroupApplyBaseInfo.singleClassMinutes,
  1104. teachingContent: item.teachingContent,
  1105. teachingPoint: item.teachingPoint
  1106. });
  1107. });
  1108. const subjectId = vipGroupApplyBaseInfo.subjectIdList
  1109. ? Number(vipGroupApplyBaseInfo.subjectIdList)
  1110. : "";
  1111. await this.onChangeSubject(subjectId);
  1112. // 初始化数据
  1113. this.form = {
  1114. roomTitle: liveBroadcastRoom.roomTitle, //
  1115. liveRemark: liveBroadcastRoom.liveRemark, // 内容
  1116. organIds: vipGroupApplyBaseInfo.organIdList
  1117. ? vipGroupApplyBaseInfo.organIdList
  1118. .split(",")
  1119. .map(item => Number(item))
  1120. : [],
  1121. subjectIdList: subjectId, // 声部
  1122. teacher: liveBroadcastRoom.speakerId, // 指导老师列表
  1123. educationalTeacherId: vipGroupApplyBaseInfo.educationalTeacherId, // 乐团主管
  1124. preTemplate: liveBroadcastRoom.preTemplate, // 模板
  1125. signUpStart: vipGroupApplyBaseInfo.registrationStartTime, // 开始时间
  1126. signUpEnd: vipGroupApplyBaseInfo.paymentExpireDate, // 结束时间
  1127. // signUpTimeList: [
  1128. // vipGroupApplyBaseInfo.registrationStartTime,
  1129. // vipGroupApplyBaseInfo.paymentExpireDate
  1130. // ], // 课程购买时间
  1131. onlineClassesNum: vipGroupApplyBaseInfo.onlineClassesNum,
  1132. singleClassMinuteId: vipGroupApplyBaseInfo.vipGroupCategoryId, //时长编号
  1133. singleClassMinutes: vipGroupApplyBaseInfo.singleClassMinutes, // 时长
  1134. onlineClassesUnitPrice:
  1135. vipGroupApplyBaseInfo.onlineClassesUnitPrice, // 售价
  1136. offlineClassesUnitPrice:
  1137. vipGroupApplyBaseInfo.offlineClassesUnitPrice, // 原价
  1138. os: liveBroadcastRoom.os, // 直播设备
  1139. useScene: liveBroadcastRoom.useScene, // 直播场景
  1140. popularizeType: liveBroadcastRoom.popularizeType, // 观看权限信息
  1141. viewMode: liveBroadcastRoom.viewMode,
  1142. roomConfig: liveBroadcastRoom.roomConfig || {
  1143. whether_like: 0,
  1144. whether_chat: 0,
  1145. whether_video: 0,
  1146. whether_mic: 0,
  1147. whether_view_shop_cart: 0
  1148. },
  1149. timeTable: tempCourse, // 排课
  1150. clientType: "TEACHER" // 主讲人身份 默认[老师]
  1151. };
  1152. this.status = data.vipGroupApplyBaseInfo.status;
  1153. this.courseStartDate = data.vipGroupApplyBaseInfo.courseStartDate;
  1154. this.auditStatus = data.vipGroupApplyBaseInfo.auditStatus;
  1155. }
  1156. } catch (e) {
  1157. //
  1158. console.log(e, "e info");
  1159. }
  1160. }
  1161. },
  1162. computed: {
  1163. nowTime() {
  1164. // console.log(that.maskForm.date)
  1165. let str = "06:00:00";
  1166. if (this.maskForm.date == dayjs(new Date()).format("YYYY-MM-DD")) {
  1167. str = dayjs(new Date()).format("HH:mm:ss");
  1168. }
  1169. return str;
  1170. },
  1171. countOnlineClassesUnitPrice() {
  1172. return (
  1173. Math.ceil(
  1174. this.form.onlineClassesNum * this.form.onlineClassesUnitPrice
  1175. ) || 0
  1176. );
  1177. },
  1178. countOfflineClassesUnitPrice() {
  1179. return (
  1180. Math.ceil(
  1181. this.form.onlineClassesNum * this.form.offlineClassesUnitPrice
  1182. ) || 0
  1183. );
  1184. },
  1185. isDisabled() {
  1186. console.log(this.status, this.type, "isDisabled");
  1187. return this.type === "update" && [1, 2].includes(this.status)
  1188. ? true
  1189. : false;
  1190. }
  1191. }
  1192. };
  1193. </script>
  1194. <style lang="scss" scoped>
  1195. .chioseWrap {
  1196. display: flex;
  1197. flex-direction: row;
  1198. justify-content: flex-start;
  1199. .chioseItem {
  1200. border-radius: 4px;
  1201. overflow: hidden;
  1202. position: relative;
  1203. margin-right: 10px;
  1204. width: 188px;
  1205. height: 188px;
  1206. cursor: pointer;
  1207. .dotWrap {
  1208. width: 21px;
  1209. height: 21px;
  1210. background: url("../../assets/images/icon_checkbox_default.png") no-repeat
  1211. center;
  1212. background-size: contain;
  1213. display: block;
  1214. position: absolute;
  1215. top: 10px;
  1216. right: 12px;
  1217. overflow: hidden;
  1218. &.checked {
  1219. background: url("../../assets/images/icon_checkbox.png") no-repeat
  1220. center;
  1221. background-size: contain;
  1222. }
  1223. &.disabled {
  1224. opacity: 0.6;
  1225. }
  1226. }
  1227. }
  1228. }
  1229. ::v-deep .el-select > .el-input {
  1230. height: 36px !important;
  1231. }
  1232. ::v-deep .el-input__inner {
  1233. height: 36px !important;
  1234. }
  1235. ::v-deep .el-col-10 .el-form-item__content {
  1236. height: 36px !important;
  1237. }
  1238. ::v-deep .select-all {
  1239. .select {
  1240. .el-input__inner {
  1241. height: 36px !important;
  1242. min-height: 36px !important;
  1243. }
  1244. }
  1245. .btn {
  1246. height: 36px !important;
  1247. min-height: 36px !important;
  1248. }
  1249. }
  1250. </style>