index.tsx 17 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208209210211212213214215216217218219220221222223224225226227228229230231232233234235236237238239240241242243244245246247248249250251252253254255256257258259260261262263264265266267268269270271272273274275276277278279280281282283284285286287288289290291292293294295296297298299300301302303304305306307308309310311312313314315316317318319320321322323324325326327328329330331332333334335336337338339340341342343344345346347348349350351352353354355356357358359360361362363364365366367368369370371372373374375376377378379380381382383384385386387388389390391392393394395396397398399400401402403404405406407408409410411412413414415416417418419420421422423424425426427428429430431432433434435436437438439440441442443444445446447448449450451452453454455456457458459460461462463464465466467468469470471472473474475476477478479480481482483484485486487488489490491492493494495496497498499500501502503504505506507508509510511512513514515516517518519520521522523524525526527528529530531532533534535536537538539540541542543544545546547548549550551552553554555556557558559560561562563564565566567568569570571572573
  1. import ColHeader from '@/components/col-header'
  2. import ColSearch from '@/components/col-search'
  3. import {
  4. Sticky,
  5. Image,
  6. List,
  7. Popup,
  8. Icon,
  9. Area,
  10. Field,
  11. Form,
  12. CellGroup,
  13. Button,
  14. Toast,
  15. Picker,
  16. DatetimePicker,
  17. Overlay,
  18. Dialog
  19. } from 'vant'
  20. import { defineComponent, onMounted, reactive } from 'vue'
  21. import styles from './index.module.less'
  22. import bg from './images/bg.png'
  23. import rejectLogo from './images/rejectLogo.png'
  24. import rejectSchool from './images/rejest-school.png'
  25. import subTitle from './images/subTitle.png'
  26. import centerLogo from './images/center.png'
  27. import studentText from './images/studentText.png'
  28. import { useRoute } from 'vue-router'
  29. import icon_arrow from './images/icon_arrow.png'
  30. import rejectBtn from './images/rejectBtn.png'
  31. import studentSuccess from './images/studentSuccess.png'
  32. import request from '@/helpers/request'
  33. import dayjs from 'dayjs'
  34. import { removeAuth } from '@/helpers/utils'
  35. export default defineComponent({
  36. name: 'tenantStudentRejest',
  37. setup() {
  38. const route = useRoute()
  39. const forms = reactive({
  40. gender: '',
  41. name: '',
  42. phone: '',
  43. subjectId: '',
  44. subjectName: '',
  45. tenantGroupName: '',
  46. tenantGroupId: '',
  47. birthdate: '',
  48. code: '',
  49. genderName: '',
  50. tenantId: route.query.tenantId
  51. })
  52. const data = reactive({
  53. birthdate: new Date(dayjs().year(), dayjs().month() + 1, dayjs().date()),
  54. schoolName: route.query.name || '',
  55. id: route.query.tenantId,
  56. tenantGroupId: route.query.tenantGroupId,
  57. userSchoolName: '',
  58. cityName: '', // 所属城市
  59. showArea: false,
  60. checked: true,
  61. success: false,
  62. areaList: {} as any,
  63. sendMsg: '获取验证码',
  64. imgCodeStatus: false,
  65. subjectList: [],
  66. searchStatus: false,
  67. openStatus: false,
  68. dateState: false,
  69. tenantGroupList: [] as any,
  70. tenantGroupStatus: false,
  71. genderState: false,
  72. genderList: [
  73. { text: '男', value: '1' },
  74. { text: '女', value: '0' }
  75. ],
  76. showSuccess: false,
  77. secondConfirm: false,
  78. minDate: new Date(1980, 0, 1),
  79. maxDate: new Date()
  80. })
  81. const handleSubmit = async () => {
  82. try {
  83. if (!forms.name) {
  84. Toast('请输入姓名')
  85. return
  86. }
  87. if (!forms.gender) {
  88. Toast('请选择性别')
  89. return
  90. }
  91. if (!forms.birthdate) {
  92. Toast('请选择出生年月')
  93. return
  94. }
  95. if (!forms.phone) {
  96. Toast('请输入手机号')
  97. return
  98. }
  99. if (!forms.code) {
  100. Toast('请输入验证码')
  101. return
  102. }
  103. if (!forms.subjectId) {
  104. Toast('请选择声部')
  105. return
  106. }
  107. if (!forms.tenantGroupId) {
  108. Toast('请选择小组')
  109. return
  110. }
  111. const res = await request.post('/api-tenant/open/student/save', {
  112. data: { ...forms },
  113. hideLoading: true
  114. })
  115. if (res.code == 200) {
  116. data.showSuccess = true
  117. }
  118. if (res.code == 5004) {
  119. data.secondConfirm = true
  120. data.userSchoolName = res.msg
  121. }
  122. } catch {
  123. //
  124. }
  125. }
  126. const getSubjectList = async () => {
  127. try {
  128. const res = await request.get('/api-tenant/open/subject/queryPage', {
  129. params: { page: 1, rows: 9999, queryType: 'list' }
  130. })
  131. const rows = res.data.rows || []
  132. const tempList: any = []
  133. rows.forEach((item: any) => {
  134. // if (item.parentSubjectId > 0) {
  135. tempList.push({
  136. text: item.name,
  137. value: item.id
  138. })
  139. // }
  140. })
  141. data.subjectList = tempList
  142. // res.data.rows.map((item: any) => {
  143. // return {
  144. // text: item.name,
  145. // value: item.id
  146. // }
  147. // }) || []
  148. } catch (e) {
  149. console.log(e)
  150. }
  151. }
  152. const confirmSubject = (val: any) => {
  153. forms.subjectName = val.text
  154. forms.subjectId = val.value
  155. data.searchStatus = false
  156. }
  157. const confirmTenant = (val: any) => {
  158. console.log(val, 'confirmTenant')
  159. forms.tenantGroupName = val.text
  160. forms.tenantGroupId = val.value
  161. data.tenantGroupStatus = false
  162. }
  163. const confirmDate = (val: any) => {
  164. forms.birthdate = dayjs(val).format('YYYY-MM-DD')
  165. data.dateState = false
  166. }
  167. const tenantGroupList = async () => {
  168. try {
  169. const res = await request.post('/api-tenant/open/tenantGroup/page', {
  170. data: {
  171. page: 1,
  172. rows: 999,
  173. tenantId: data.id
  174. }
  175. })
  176. data.tenantGroupList =
  177. res.data.rows.map((item: any) => {
  178. return {
  179. text: item.name,
  180. value: item.id
  181. }
  182. }) || []
  183. } catch {
  184. //
  185. }
  186. }
  187. onMounted(() => {
  188. removeAuth()
  189. if (data.tenantGroupId) {
  190. forms.tenantGroupId = data.tenantGroupId as any
  191. }
  192. tenantGroupList()
  193. getSubjectList()
  194. })
  195. const confirmGender = (val: any) => {
  196. if (val.value) {
  197. forms.gender = val.value
  198. forms.genderName = val.text
  199. } else {
  200. forms.gender = ''
  201. forms.genderName = ''
  202. }
  203. data.genderState = false
  204. }
  205. /** 发送验证码 */
  206. const onSendSms = async () => {
  207. if (!forms.phone) {
  208. Toast('请输入手机号码')
  209. return
  210. }
  211. if (!/^1[3456789]\d{9}$/.test(forms.phone)) {
  212. Toast('手机号码格式不正确')
  213. return
  214. }
  215. await request.post('/api-student/code/sendSmsCode', {
  216. requestType: 'form',
  217. data: {
  218. mobile: forms.phone,
  219. type: 'REGISTER'
  220. }
  221. })
  222. onCountDown()
  223. setTimeout(() => {
  224. Toast('验证码已发送')
  225. }, 100)
  226. }
  227. const onCountDown = () => {
  228. data.sendMsg = '60s'
  229. let count = 60
  230. const timer = setInterval(() => {
  231. count--
  232. data.sendMsg = `${count}s`
  233. if (count <= 0) {
  234. data.sendMsg = '获取验证码'
  235. clearInterval(timer)
  236. }
  237. }, 1000)
  238. }
  239. const downApp = () => {
  240. window.open(location.origin + '/student/#/download')
  241. data.showSuccess = false
  242. }
  243. const submitSecond = async () => {
  244. try {
  245. await request.post('/api-tenant/open/student/save', {
  246. data: { ...forms, updateTenant: true },
  247. hideLoading: true
  248. })
  249. data.showSuccess = true
  250. data.secondConfirm = false
  251. } catch (e) {
  252. console.log(e)
  253. }
  254. }
  255. return () => (
  256. <>
  257. <div class={styles.videoClass}>
  258. {/* <ColHeader
  259. class={styles.classHeader}
  260. border={false}
  261. isFixed={false}
  262. background="#fff"
  263. /> */}
  264. <div class={styles.resjetStudentWrap}>
  265. <img src={rejectLogo} class={styles.rejectLogo} alt="" />
  266. <img src={studentText} class={styles.studentText} alt="" />
  267. <img src={bg} class={styles.bgWrap} alt="" />
  268. <div class={styles.schoolNameWrap}>
  269. <img src={rejectSchool} class={styles.rejectSchool} alt="" />
  270. <p>{data.schoolName}</p>
  271. </div>
  272. <img class={styles.centerLogo} src={centerLogo} alt="" />
  273. <div class={styles.infoWrap}>
  274. <div class={styles.infoWrapCore}>
  275. <img src={subTitle} class={styles.subTitle} alt="" />
  276. <Form onSubmit={() => handleSubmit()}>
  277. <CellGroup class={styles.group} border={false}>
  278. <Field
  279. class={styles.noArrow}
  280. inputAlign="right"
  281. label="姓名"
  282. placeholder="请输入姓名"
  283. maxlength={14}
  284. v-model={forms.name}
  285. // onUpdate: modelValue={(val: string) => {
  286. // forms.nickname = val.trim();
  287. // }}
  288. />
  289. <Field
  290. readonly
  291. inputAlign="right"
  292. label="性别"
  293. placeholder="请选择性别"
  294. v-model={forms.genderName}
  295. onClick={() => {
  296. data.genderState = true
  297. }}
  298. // onUpdate: modelValue={(val: string) => {
  299. // forms.nickname = val.trim();
  300. // }}
  301. >
  302. {{
  303. button: () => (
  304. <img
  305. style={{
  306. display: 'block',
  307. width: '12px',
  308. height: '12px'
  309. }}
  310. src={icon_arrow}
  311. />
  312. )
  313. }}
  314. </Field>
  315. <Field
  316. readonly
  317. inputAlign="right"
  318. label="出生日期"
  319. placeholder="请选择出生日期"
  320. maxlength={20}
  321. v-model={forms.birthdate}
  322. onClick={() => {
  323. data.dateState = true
  324. }}
  325. // onUpdate: modelValue={(val: string) => {
  326. // forms.nickname = val.trim();
  327. // }}
  328. >
  329. {{
  330. button: () => (
  331. <img
  332. style={{
  333. display: 'block',
  334. width: '12px',
  335. height: '12px'
  336. }}
  337. src={icon_arrow}
  338. />
  339. )
  340. }}
  341. </Field>
  342. <Field
  343. inputAlign="right"
  344. label="手机号"
  345. class={styles.noArrow}
  346. maxlength={11}
  347. placeholder="手机号码"
  348. v-model={forms.phone}
  349. />
  350. <div class={styles.tips}>
  351. 手机号码为酷乐秀机构版登录账号
  352. </div>
  353. <Field
  354. class={styles.inputCode}
  355. inputAlign="left"
  356. label="验证码"
  357. labelWidth={0}
  358. v-model={forms.code}
  359. maxlength={6}
  360. >
  361. {{
  362. button: () => (
  363. <Button
  364. disabled={data.sendMsg.includes('s')}
  365. class={styles.sendBtn}
  366. onClick={() => onSendSms()}
  367. >
  368. {data.sendMsg}
  369. </Button>
  370. )
  371. }}
  372. </Field>
  373. <Field
  374. border={false}
  375. inputAlign="right"
  376. label="声部"
  377. placeholder="请选择声部"
  378. readonly
  379. v-model={forms.subjectName}
  380. onClick={() => (data.searchStatus = true)}
  381. >
  382. {{
  383. button: () => (
  384. <img
  385. style={{
  386. display: 'block',
  387. width: '12px',
  388. height: '12px'
  389. }}
  390. src={icon_arrow}
  391. />
  392. )
  393. }}
  394. </Field>
  395. {data.tenantGroupId ? (
  396. ''
  397. ) : (
  398. <Field
  399. border={false}
  400. inputAlign="right"
  401. label="小组"
  402. placeholder="请选择小组"
  403. readonly
  404. v-model={forms.tenantGroupName}
  405. onClick={() => (data.tenantGroupStatus = true)}
  406. >
  407. {{
  408. button: () => (
  409. <img
  410. style={{
  411. display: 'block',
  412. width: '12px',
  413. height: '12px'
  414. }}
  415. src={icon_arrow}
  416. />
  417. )
  418. }}
  419. </Field>
  420. )}
  421. </CellGroup>
  422. </Form>
  423. </div>
  424. <img
  425. src={rejectBtn}
  426. onClick={() => {
  427. handleSubmit()
  428. }}
  429. class={styles.rejectBtn}
  430. alt=""
  431. />
  432. </div>
  433. </div>
  434. <Popup
  435. show={data.searchStatus}
  436. position="bottom"
  437. round
  438. columns-field-names={{ text: '' }}
  439. safe-area-inset-bottom
  440. onClose={() => (data.searchStatus = false)}
  441. onClosed={() => (data.openStatus = false)}
  442. >
  443. <Picker
  444. columns={data.subjectList}
  445. onCancel={() => {
  446. data.searchStatus = false
  447. }}
  448. onConfirm={confirmSubject}
  449. ></Picker>
  450. </Popup>
  451. <Popup
  452. show={data.tenantGroupStatus}
  453. position="bottom"
  454. round
  455. columns-field-names={{ text: '' }}
  456. safe-area-inset-bottom
  457. onClose={() => (data.tenantGroupStatus = false)}
  458. onClosed={() => (data.openStatus = false)}
  459. >
  460. <Picker
  461. columns={data.tenantGroupList}
  462. onCancel={() => {
  463. data.tenantGroupStatus = false
  464. }}
  465. onConfirm={confirmTenant}
  466. ></Picker>
  467. </Popup>
  468. <Popup
  469. show={data.dateState}
  470. position="bottom"
  471. round
  472. columns-field-names={{ text: '' }}
  473. safe-area-inset-bottom
  474. onClose={() => (data.dateState = false)}
  475. onClosed={() => (data.dateState = false)}
  476. >
  477. <DatetimePicker
  478. min-date={data.minDate}
  479. max-date={data.maxDate}
  480. type="date"
  481. v-model={data.birthdate}
  482. onCancel={() => {
  483. data.dateState = false
  484. }}
  485. onConfirm={confirmDate}
  486. ></DatetimePicker>
  487. </Popup>
  488. <Popup
  489. show={data.genderState}
  490. position="bottom"
  491. round
  492. columns-field-names={{ text: '' }}
  493. safe-area-inset-bottom
  494. onClose={() => (data.genderState = false)}
  495. onClosed={() => (data.genderState = false)}
  496. >
  497. <Picker
  498. columns={data.genderList}
  499. onCancel={() => {
  500. data.genderState = false
  501. }}
  502. onConfirm={confirmGender}
  503. ></Picker>
  504. </Popup>
  505. <Overlay show={data.showSuccess} z-index={1000}>
  506. <div class={styles.showWrap}>
  507. <img class={styles.showWrapTop} src={studentSuccess} alt="" />
  508. <h2>恭喜您已成功登记为</h2>
  509. <h4>
  510. {data.schoolName} <span>【学员】</span>{' '}
  511. </h4>
  512. <p>请下载酷乐秀机构版APP进行学习</p>
  513. <div class={styles.downApp} onClick={downApp}>
  514. 立即下载
  515. </div>
  516. </div>
  517. </Overlay>
  518. <Popup
  519. show={data.secondConfirm}
  520. position="center"
  521. round
  522. onClose={() => (data.secondConfirm = false)}
  523. onClosed={() => (data.secondConfirm = false)}
  524. >
  525. <div class={styles.secondWrap}>
  526. <h2>提示</h2>
  527. <p>
  528. 当前账号已存在 <span>【{data.userSchoolName}】</span>{' '}
  529. ,是否确认更换到
  530. <span>{data.schoolName}</span>
  531. 吗?
  532. </p>
  533. <div class={styles.buttonWrap}>
  534. <div
  535. class={styles.closeBtn}
  536. onClick={() => {
  537. data.secondConfirm = false
  538. }}
  539. >
  540. {' '}
  541. 取消
  542. </div>
  543. <div class={styles.submitBtn} onClick={submitSecond}>
  544. 确定
  545. </div>
  546. </div>
  547. </div>
  548. </Popup>
  549. </div>
  550. </>
  551. )
  552. }
  553. })