index.tsx 34 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208209210211212213214215216217218219220221222223224225226227228229230231232233234235236237238239240241242243244245246247248249250251252253254255256257258259260261262263264265266267268269270271272273274275276277278279280281282283284285286287288289290291292293294295296297298299300301302303304305306307308309310311312313314315316317318319320321322323324325326327328329330331332333334335336337338339340341342343344345346347348349350351352353354355356357358359360361362363364365366367368369370371372373374375376377378379380381382383384385386387388389390391392393394395396397398399400401402403404405406407408409410411412413414415416417418419420421422423424425426427428429430431432433434435436437438439440441442443444445446447448449450451452453454455456457458459460461462463464465466467468469470471472473474475476477478479480481482483484485486487488489490491492493494495496497498499500501502503504505506507508509510511512513514515516517518519520521522523524525526527528529530531532533534535536537538539540541542543544545546547548549550551552553554555556557558559560561562563564565566567568569570571572573574575576577578579580581582583584585586587588589590591592593594595596597598599600601602603604605606607608609610611612613614615616617618619620621622623624625626627628629630631632633634635636637638639640641642643644645646647648649650651652653654655656657658659660661662663664665666667668669670671672673674675676677678679680681682683684685686687688689690691692693694695696697698699700701702703704705706707708709710711712713714715716717718719720721722723724725726727728729730731732733734735736737738739740741742743744745746747748749750751752753754755756757758759760761762763764765766767768769770771772773774775776777778779780781782783784785786787788789790791792793794795796797798799800801802803804805806807808809810811812813814815816817818819820821822823824825826827828829830831832833834835836837838839840841842843844845846847848849850851852853854855856857858859860861862863864865866867868869870871872873874875876877878879880881882883884885886887888889890891892893894895896897898899900901902903904905906907908909910911912913914915916917918919920921922923924925926927928929930931932933934935936937938939940941942943944945946947948949950951952953954955956957958959960961962963964965966967968969970971972973974975976977978979980981982983984985986987988989990991992993
  1. import {
  2. computed,
  3. defineComponent,
  4. nextTick,
  5. onMounted,
  6. reactive,
  7. ref
  8. } from 'vue';
  9. import styles from './index.module.less';
  10. // import infoTitle from '../images/info-title.png';
  11. import {
  12. Button,
  13. CountDown,
  14. Field,
  15. Form,
  16. Picker,
  17. Popup,
  18. Radio,
  19. RadioGroup,
  20. Tag,
  21. showToast
  22. } from 'vant';
  23. import OWxTip from '@/components/m-wx-tip';
  24. // import MProtocol from '@/components/m-protocol';
  25. import MImgCode from '@/components/m-img-code';
  26. import { browser, checkPhone } from '@/helpers/utils';
  27. import request from '@/helpers/request';
  28. import { useStudentRegisterStore } from '@/store/modules/student-register-store';
  29. // import { setLoginInit, state } from '@/state';
  30. // import iconGift from '../images/icon-gift.png';
  31. import { useRoute, useRouter } from 'vue-router';
  32. import MSticky from '@/components/m-sticky';
  33. // import registerBgIcon from '../images/register-bg.png';
  34. import vipGiftIcon from '../images/vip-gift-icon.png';
  35. // import agreeYes from '../images/agree-yes.png';
  36. // import agreeNo from '../images/agree-no.png';
  37. import MMessageTip from '@/components/m-message-tip';
  38. import SelectStudent from '../modal/select-student';
  39. import { setLoginInit } from '@/state';
  40. import dayjs from 'dayjs';
  41. const classList: any = [];
  42. for (let i = 1; i <= 40; i++) {
  43. classList.push({ text: i + '班', value: i });
  44. }
  45. export default defineComponent({
  46. name: 'register-new',
  47. emits: ['close', 'submit'],
  48. setup() {
  49. const route = useRoute();
  50. const studentRegisterStore = useStudentRegisterStore();
  51. // 初始化学校编号
  52. studentRegisterStore.setShoolId(route.query.sId as any);
  53. const countDownRef = ref();
  54. const gradeList = computed(() => {
  55. let tempList: any = [];
  56. const five = [
  57. { text: '一年级', value: 1 },
  58. { text: '二年级', value: 2 },
  59. { text: '三年级', value: 3 },
  60. { text: '四年级', value: 4 },
  61. { text: '五年级', value: 5 }
  62. ];
  63. const one = [{ text: '六年级', value: 6 }];
  64. const three = [
  65. { text: '七年级', value: 7 },
  66. { text: '八年级', value: 8 },
  67. { text: '九年级', value: 9 }
  68. ];
  69. if (forms.gradeYear === 'FIVE_YEAR_SYSTEM') {
  70. tempList.push([...five]);
  71. } else if (forms.gradeYear === 'SIX_YEAR_SYSTEM') {
  72. tempList.push([...five, ...one]);
  73. } else if (forms.gradeYear === 'THREE_YEAR_SYSTEM') {
  74. tempList.push([...three]);
  75. } else if (forms.gradeYear === 'FORE_YEAR_SYSTEM') {
  76. tempList.push([...one, ...three]);
  77. } else {
  78. tempList.push([...five, ...one, ...three]);
  79. }
  80. return tempList;
  81. });
  82. const forms = reactive({
  83. /** 参数数据 */
  84. queryParams: {
  85. classId: route.query.ci as any,
  86. currentGradeNum: route.query.cgn ? Number(route.query.cgn) : null,
  87. currentClass: route.query.cc ? Number(route.query.cc) : null,
  88. gradeYear: route.query.gy ? Number(route.query.gy) : null
  89. },
  90. countDownStatus: true,
  91. countDownTime: 1000 * 120, // 倒计时时间
  92. modelValue: false, // 是否选中协议
  93. imgCodeStatus: false,
  94. gradeNumText: '',
  95. currentClassText: '',
  96. multi_user_limit: 1, // 限制注册学生数量
  97. gradeStatus: false,
  98. classStatus: false,
  99. loading: false,
  100. schoolId: route.query.sId as any,
  101. showSelectStudent: false, // 选择学生
  102. studentList: [], // 手机号关联学生列表
  103. studentItem: {} as any, // 选择的学生
  104. isRegister: 'create' as 'create' | 'update' | '', // 是否注册学生
  105. isTipRegister: false, // 是否显示名字不一致 - 默认显示
  106. isChangeSchool: false, // 是否切换学校
  107. details: {} as any,
  108. gradeYear: null,
  109. schoolType: null,
  110. giftVipDay: null as any,
  111. showTips: false,
  112. showButton: false,
  113. showMessage: '请使用微信打开',
  114. gradePopupShow: false,
  115. gradePopupIndex: [] as any, // 年级下拉索引
  116. classPopupShow: false,
  117. classPopupIndex: [] as any // 班级下拉索引
  118. });
  119. const otherParams = reactive({
  120. showOtherSchool: false,
  121. showCloseButton: true, // 是否显示关闭按钮
  122. showOtherMessage: '',
  123. /** limit 超限制,change 更换学生,nickname 名称不一致 */
  124. otherType: '' as 'limit' | 'change' | 'nickname',
  125. showCancelButton: true,
  126. cancelButtonColor: '',
  127. cancelButtonText: '取消',
  128. showConfirmButton: true,
  129. confirmButtonColor: '',
  130. confirmButtonText: '确定',
  131. messageAlign: 'left' as 'center' | 'left' | 'right'
  132. });
  133. // 班级状态
  134. const isClassStatus = computed(() => {
  135. return {
  136. isGrade: forms.queryParams.currentGradeNum ? true : false,
  137. isClass: forms.queryParams.currentClass ? true : false
  138. };
  139. });
  140. const isAgree = ref(false);
  141. const studentInfo = reactive({
  142. autoRegister: true,
  143. multiUser: true, // 是否为多用户
  144. client_id: 'cooleshow-student',
  145. client_secret: 'cooleshow-student',
  146. extra: {
  147. nickname: '',
  148. currentGradeNum: null as any,
  149. currentClass: null as any,
  150. gender: 1,
  151. registerType: '', // 报名类型
  152. giftVipDay: 0 // 赠送会员天数
  153. },
  154. grant_type: 'password',
  155. loginType: 'SMS',
  156. password: '',
  157. username: ''
  158. });
  159. const onCodeSend = () => {
  160. forms.countDownStatus = false;
  161. nextTick(() => {
  162. countDownRef.value.start();
  163. });
  164. };
  165. // 格式化提示状态
  166. const changeTipStatus = (register: boolean, school: boolean) => {
  167. forms.isTipRegister = register;
  168. forms.isChangeSchool = school;
  169. };
  170. const onSendCode = () => {
  171. // 发送验证码
  172. if (!checkPhone(studentInfo.username)) {
  173. return showToast('请输入正确的手机号码');
  174. }
  175. forms.imgCodeStatus = true;
  176. };
  177. const getUserInfos = async () => {
  178. if (
  179. studentInfo.password.length !== 6 ||
  180. !checkPhone(studentInfo.username)
  181. ) {
  182. return;
  183. }
  184. try {
  185. const { data } = await request.get(
  186. `/edu-app/open/student/studentInfo?mobile=${studentInfo.username}&code=${studentInfo.password}&type=REGISTER&activationCodeFlag=1`
  187. );
  188. forms.studentList = data || [];
  189. if (forms.studentList.length > 0) {
  190. const firstStudent: any = forms.studentList[0];
  191. forms.studentItem = firstStudent;
  192. studentInfo.extra.nickname = firstStudent.nickname;
  193. if (!isClassStatus.value.isGrade) {
  194. const tempGrade: any = gradeList.value[0] || [];
  195. tempGrade?.forEach((i: any) => {
  196. if (i.value === firstStudent.currentGradeNum) {
  197. forms.gradeNumText = i.text;
  198. studentInfo.extra.currentGradeNum =
  199. firstStudent.currentGradeNum;
  200. }
  201. });
  202. }
  203. if (!isClassStatus.value.isClass) {
  204. classList.forEach((i: any) => {
  205. if (i.value === firstStudent.currentClass) {
  206. forms.currentClassText = i.text;
  207. studentInfo.extra.currentClass = firstStudent.currentClass;
  208. }
  209. });
  210. }
  211. studentInfo.extra.gender = firstStudent.gender;
  212. forms.isRegister = 'update';
  213. changeTipStatus(true, false);
  214. } else {
  215. forms.isRegister = 'create';
  216. changeTipStatus(false, false);
  217. forms.studentItem = [];
  218. }
  219. } catch {
  220. //
  221. }
  222. };
  223. const validatePhone = computed(() => {
  224. return checkPhone(studentInfo.username) ? true : false;
  225. });
  226. const onFinished = () => {
  227. forms.countDownStatus = true;
  228. countDownRef.value.reset();
  229. };
  230. //
  231. const checkSubmit = () => {
  232. const { extra } = studentInfo;
  233. if (
  234. forms.studentItem.nickname !== extra.nickname &&
  235. forms.isTipRegister
  236. ) {
  237. otherParams.showOtherMessage =
  238. '学生姓名与上次提交信息不一致,请确认修改学生信息或创建新的学生账号';
  239. otherParams.showOtherSchool = true;
  240. otherParams.showCancelButton = true;
  241. otherParams.showCloseButton = true;
  242. otherParams.cancelButtonColor =
  243. 'linear-gradient( 224deg, #3FE1E6 0%, #00CDD4 100%)';
  244. otherParams.cancelButtonText = '新建学生';
  245. otherParams.confirmButtonColor =
  246. 'linear-gradient( 305deg, #40C8FF 0%, #3192FF 100%)';
  247. otherParams.confirmButtonText = '修改信息';
  248. otherParams.otherType = 'nickname';
  249. otherParams.messageAlign = 'left';
  250. return true;
  251. }
  252. // 判断新建学员是否上限了
  253. if (
  254. forms.isRegister === 'create' &&
  255. forms.studentList.length >= forms.multi_user_limit
  256. ) {
  257. otherParams.showOtherMessage = `同一手机号最多创建${forms.multi_user_limit}个学生`;
  258. otherParams.showOtherSchool = true;
  259. otherParams.showCancelButton = false;
  260. otherParams.showCloseButton = true;
  261. otherParams.confirmButtonColor =
  262. 'linear-gradient( 305deg, #40C8FF 0%, #3192FF 100%)';
  263. otherParams.confirmButtonText = '我知道了';
  264. otherParams.otherType = 'limit';
  265. otherParams.messageAlign = 'center';
  266. return true;
  267. }
  268. // 判断是否为同一个学校
  269. if (
  270. forms.studentItem.schoolId &&
  271. forms.studentItem.schoolId !== forms.details.id &&
  272. !forms.isChangeSchool &&
  273. forms.isRegister === 'update'
  274. ) {
  275. otherParams.showOtherMessage = `您已绑定<span style="color: #2B85FF">【${
  276. forms.studentItem?.schoolName || ''
  277. }】</span>,提交后将更换到
  278. <span style="color: #2B85FF">【${forms.details.name || ''}】</span>
  279. ,是否确认提交?`;
  280. otherParams.showOtherSchool = true;
  281. otherParams.showCancelButton = true;
  282. otherParams.showCloseButton = false;
  283. otherParams.cancelButtonColor = '';
  284. otherParams.cancelButtonText = '取消';
  285. otherParams.confirmButtonColor = '';
  286. otherParams.confirmButtonText = '确定';
  287. otherParams.otherType = 'change';
  288. otherParams.messageAlign = 'left';
  289. return true;
  290. }
  291. return false;
  292. };
  293. const onSubmit = async () => {
  294. try {
  295. if (checkForm() || checkSubmit()) return;
  296. forms.loading = true;
  297. // await request.get('/edu-app/open/student/schoolQuery', {
  298. // params: {
  299. // schoolId: forms.schoolId,
  300. // mobile: studentInfo.username
  301. // }
  302. // });
  303. // const { extra, ...res } = studentInfo;
  304. const { extra, loginType, autoRegister, password, multiUser, ...res } =
  305. studentInfo;
  306. let tLoginType = loginType,
  307. tAutoRegister = autoRegister,
  308. tPassword = password,
  309. tMultiUser = multiUser;
  310. if (forms.isRegister === 'update') {
  311. tLoginType = 'TOKEN';
  312. tAutoRegister = false;
  313. tPassword = forms.studentItem.token;
  314. tMultiUser = false;
  315. }
  316. const result = await request.post('/edu-app/userlogin', {
  317. hideLoading: false,
  318. requestType: 'form',
  319. data: {
  320. loginType: tLoginType,
  321. autoRegister: tAutoRegister,
  322. password: tPassword,
  323. multiUser: tMultiUser,
  324. ...res,
  325. extra: JSON.stringify({
  326. ...extra,
  327. gradeYear: forms.queryParams.gradeYear,
  328. classId: forms.queryParams.classId,
  329. schoolId: forms.schoolId
  330. })
  331. }
  332. });
  333. if (result.code === 5435 || result.code === 5436) {
  334. forms.showTips = true;
  335. forms.showMessage = '报名信息更新,请刷新后重新提交';
  336. forms.showButton = true;
  337. } else {
  338. studentRegisterStore.setToken(
  339. result.data.token_type + ' ' + result.data.access_token
  340. );
  341. setLoginInit();
  342. const { extra, username } = studentInfo;
  343. const registerResult = await request.post(
  344. '/edu-app/student/register',
  345. {
  346. data: {
  347. schoolId: forms.schoolId,
  348. clientType: 'STUDENT',
  349. ...extra,
  350. gradeYear: forms.queryParams.gradeYear,
  351. classId: forms.queryParams.classId,
  352. mobile: username,
  353. newRegUser: forms.isRegister === 'create' ? true : false
  354. }
  355. }
  356. );
  357. if (registerResult.code !== 200) {
  358. if (result.code === 5435 || result.code === 5436) {
  359. forms.showTips = true;
  360. forms.showMessage = '报名信息更新,请刷新后重新提交';
  361. forms.showButton = true;
  362. }
  363. return;
  364. }
  365. setTimeout(() => {
  366. showToast('报名成功');
  367. }, 100);
  368. setTimeout(() => {
  369. if (browser().weixin) {
  370. // 关闭微信
  371. (window as any).WeixinJSBridge.call('closeWindow');
  372. }
  373. }, 1000);
  374. }
  375. } catch {
  376. // 重置信息 - 如果是新建则不提示
  377. changeTipStatus(forms.isRegister === 'create' ? false : true, false);
  378. } finally {
  379. forms.loading = false;
  380. }
  381. };
  382. const checkForm = () => {
  383. if (!checkPhone(studentInfo.username)) {
  384. showToast('请输入正确的手机号码');
  385. return true;
  386. } else if (!studentInfo.password) {
  387. showToast('请输入验证码');
  388. return true;
  389. } else if (!studentInfo.extra.nickname) {
  390. showToast('请输入学生姓名');
  391. return true;
  392. } else if (![0, 1].includes(studentInfo.extra.gender)) {
  393. showToast('请选择性别');
  394. return true;
  395. } else if (!studentInfo.extra.currentGradeNum) {
  396. showToast('请选择所在年级');
  397. return true;
  398. } else if (!studentInfo.extra.currentClass) {
  399. showToast('请选择所在班级');
  400. return true;
  401. }
  402. return false;
  403. };
  404. const getRegisterGoods = async () => {
  405. try {
  406. const { data } = await request.get('/edu-app/open/school/detail', {
  407. params: {
  408. id: forms.schoolId
  409. },
  410. noAuthorization: true // 是否请求接口的时候添加toekn
  411. });
  412. forms.giftVipDay = data.giftVipDay;
  413. forms.schoolType = data.schoolType;
  414. forms.gradeYear = data.gradeYear;
  415. forms.details = data;
  416. studentInfo.extra.giftVipDay = data.giftVipDay;
  417. studentInfo.extra.registerType = data.registerType;
  418. if (browser().weixin) {
  419. if (data.status === 0) {
  420. forms.showTips = true;
  421. forms.showMessage = '二维码已经失效,详情请咨询学校老师';
  422. forms.showButton = false;
  423. }
  424. } else {
  425. forms.showTips = true;
  426. }
  427. // 判断是否有传来的年级班级
  428. const tempGrade: any = gradeList.value[0] || [];
  429. tempGrade?.forEach((i: any) => {
  430. if (i.value === forms.queryParams.currentGradeNum) {
  431. forms.gradeNumText = i.text;
  432. studentInfo.extra.currentGradeNum =
  433. forms.queryParams.currentGradeNum;
  434. }
  435. });
  436. classList.forEach((i: any) => {
  437. if (i.value === forms.queryParams.currentClass) {
  438. forms.currentClassText = i.text;
  439. studentInfo.extra.currentClass = forms.queryParams.currentClass;
  440. }
  441. });
  442. } catch {}
  443. };
  444. onMounted(async () => {
  445. try {
  446. // 获取支付类型
  447. let expireDay = null;
  448. const { data } = await request.get(
  449. '/edu-app/open/paramConfig/queryByParamNameList',
  450. {
  451. requestType: 'form',
  452. params: {
  453. paramNames: 'multi_user_limit,qr_code_expire_time'
  454. }
  455. }
  456. );
  457. if (data && Array.isArray(data)) {
  458. data.forEach((item: any) => {
  459. if (item.paramName === 'multi_user_limit') {
  460. forms.multi_user_limit = item.paramValue
  461. ? Number(item.paramValue)
  462. : 1;
  463. } else if (item.paramName === 'qr_code_expire_time') {
  464. expireDay = item.paramValue ? Number(item.paramValue) : null;
  465. }
  466. });
  467. }
  468. const createT = route.query.t;
  469. if (createT && expireDay !== null) {
  470. if (dayjs(Number(createT)).add(expireDay, 'day').isBefore(dayjs())) {
  471. forms.showTips = true;
  472. forms.showMessage = '二维码已经失效,详情请咨询学校老师';
  473. forms.showButton = false;
  474. }
  475. }
  476. await getRegisterGoods();
  477. } catch {}
  478. });
  479. /** 手机号变更时清空验证码信息,用户信息 */
  480. const phoneChangeEmptyInfo = () => {
  481. studentInfo.password = '';
  482. studentInfo.extra.nickname = '';
  483. if (!isClassStatus.value.isGrade) {
  484. studentInfo.extra.currentGradeNum = '';
  485. forms.gradeNumText = '';
  486. }
  487. if (!isClassStatus.value.isClass) {
  488. studentInfo.extra.currentClass = '';
  489. forms.currentClassText = '';
  490. }
  491. studentInfo.extra.gender = 1;
  492. forms.studentList = []; // 手机号关联学生列表
  493. forms.studentItem = {}; // 选择的学生
  494. forms.isRegister = 'create'; // 是否注册学生
  495. forms.isTipRegister = false; // 是否显示名字不一致 - 默认显示
  496. forms.isChangeSchool = false; // 是否切换学校
  497. };
  498. return () => (
  499. <div class={styles.registerModal}>
  500. {/* {forms.giftVipDay ? (
  501. <div class={styles.memberNumer}>
  502. <img src={iconGift} class={styles.iconGift} />
  503. <p>
  504. 现在报名立即赠送乐器AI学练工具有效期{' '}
  505. <span>{forms.giftVipDay}</span> 天
  506. </p>
  507. </div>
  508. ) : (
  509. ''
  510. )} */}
  511. <div class={styles.studentRegisterContainer}>
  512. <div
  513. class={[
  514. styles.studentSection,
  515. styles.studentSectionForm,
  516. styles.noSendDay
  517. ]}>
  518. <div class={styles.title3}></div>
  519. <Form labelAlign="left" class={styles.registerForm}>
  520. <Field
  521. clearable={false}
  522. label="联系方式(直接监护人)"
  523. placeholder="请输入手机号码"
  524. type="tel"
  525. required
  526. autocomplete="off"
  527. inputAlign="right"
  528. class={styles.username}
  529. v-model={studentInfo.username}
  530. border={false}
  531. maxlength={11}
  532. onUpdate:modelValue={() => {
  533. phoneChangeEmptyInfo();
  534. }}>
  535. {{
  536. label: () => (
  537. <div>
  538. 联系方式
  539. <p class={styles.tips}>(直接监护人)</p>
  540. </div>
  541. )
  542. }}
  543. </Field>
  544. <div class={['van-hairline--bottom', styles.fieldTipsGroup]}>
  545. <div class={[styles.fieldTips]}>
  546. 手机号是音乐数字课堂的唯一登录账户
  547. </div>
  548. </div>
  549. <Field
  550. center
  551. clearable={false}
  552. required
  553. inputAlign="right"
  554. label="验证码"
  555. placeholder="请输入验证码"
  556. autocomplete="off"
  557. type="number"
  558. v-model={studentInfo.password}
  559. maxlength={6}
  560. onUpdate:modelValue={(val: any) => {
  561. getUserInfos();
  562. }}>
  563. {{
  564. button: () =>
  565. forms.countDownStatus ? (
  566. <span
  567. class={[
  568. styles.codeText,
  569. !validatePhone.value ? styles.codeTextDisabled : ''
  570. ]}
  571. onClick={onSendCode}>
  572. 获取验证码
  573. </span>
  574. ) : (
  575. <CountDown
  576. ref={(el: any) => (countDownRef.value = el)}
  577. auto-start={false}
  578. class={styles.countDown}
  579. time={forms.countDownTime}
  580. onFinish={onFinished}
  581. format="ss秒后重试"
  582. />
  583. )
  584. }}
  585. </Field>
  586. </Form>
  587. </div>
  588. <div
  589. class={[
  590. styles.studentSection,
  591. styles.studentSectionForm,
  592. Number(forms.giftVipDay) > 0 && styles.noSendDay
  593. ]}>
  594. <div class={styles.title1}></div>
  595. <Form labelAlign="left" class={styles.registerForm}>
  596. {/* 大于等于2,则可以切换学生 */}
  597. {forms.studentList.length > 1 && (
  598. <div
  599. class={[
  600. styles.selectStudentGroup,
  601. forms.showSelectStudent && styles.selectStudentGroupChecked
  602. ]}
  603. onClick={() => (forms.showSelectStudent = true)}>
  604. <i
  605. class={[
  606. styles.studentIcon,
  607. !forms.studentItem.userId && styles.studentIconAdd
  608. ]}></i>
  609. <span>
  610. {forms.studentItem.userId
  611. ? forms.studentItem.nickname
  612. : '新增学生'}
  613. </span>
  614. </div>
  615. )}
  616. <Field
  617. clearable={false}
  618. label="学生姓名"
  619. placeholder="请输入学生姓名"
  620. autocomplete="off"
  621. maxlength={14}
  622. v-model={studentInfo.extra.nickname}
  623. required
  624. input-align="right"
  625. />
  626. <Field
  627. clearable={false}
  628. label="学生性别"
  629. placeholder="请选择性别"
  630. autocomplete="off"
  631. required
  632. input-align="right"
  633. // v-model={studentInfo.extra.nickname}
  634. >
  635. {{
  636. input: () => (
  637. <RadioGroup
  638. checked-color="#ffcb75"
  639. v-model={studentInfo.extra.gender}
  640. direction="horizontal">
  641. <Tag
  642. size="large"
  643. type="primary"
  644. color={
  645. !(studentInfo.extra.gender === 1)
  646. ? '#F5F6FA'
  647. : 'linear-gradient( 135deg, #31C7FF 0%, #007AFE 100%)'
  648. }
  649. textColor={
  650. !(studentInfo.extra.gender === 1) ? '#626264' : '#fff'
  651. }
  652. class={styles.radioSection}
  653. round>
  654. <Radio class={styles.radioItem} name={1}></Radio>男
  655. </Tag>
  656. <Tag
  657. size="large"
  658. type="primary"
  659. color={
  660. !(studentInfo.extra.gender === 0)
  661. ? '#F5F6FA'
  662. : 'linear-gradient( 135deg, #31C7FF 0%, #007AFE 100%)'
  663. }
  664. textColor={
  665. !(studentInfo.extra.gender === 0) ? '#626264' : '#fff'
  666. }
  667. class={styles.radioSection}
  668. round>
  669. <Radio class={styles.radioItem} name={0}></Radio>女
  670. </Tag>
  671. </RadioGroup>
  672. )
  673. }}
  674. </Field>
  675. <Field
  676. clearable={false}
  677. label="所在年级"
  678. placeholder="请选择年级"
  679. isLink={isClassStatus.value.isGrade ? false : true}
  680. readonly
  681. clickable={false}
  682. modelValue={forms.gradeNumText}
  683. onClick={() => {
  684. // 如果传进来则不修改
  685. if (isClassStatus.value.isGrade) return;
  686. forms.gradePopupIndex = [studentInfo.extra.currentGradeNum];
  687. forms.gradeStatus = true;
  688. }}
  689. required
  690. input-align="right"
  691. />
  692. <Field
  693. clearable={false}
  694. label="所在班级"
  695. isLink={isClassStatus.value.isClass ? false : true}
  696. placeholder="请选择班级"
  697. readonly
  698. clickable={false}
  699. modelValue={forms.currentClassText}
  700. onClick={() => {
  701. // 如果传进来则不修改
  702. if (isClassStatus.value.isClass) return;
  703. forms.classPopupIndex = [studentInfo.extra.currentClass];
  704. forms.classStatus = true;
  705. }}
  706. required
  707. input-align="right"
  708. />
  709. {forms.giftVipDay && Number(forms.giftVipDay) > 0 ? (
  710. <div class={styles.giftTips}>
  711. <img src={vipGiftIcon} />
  712. <span>
  713. 注册成功即可获得乐器AI学练工具<i>{forms.giftVipDay}</i>
  714. 天有效期
  715. </span>
  716. </div>
  717. ) : null}
  718. </Form>
  719. </div>
  720. </div>
  721. {/* <div class={styles.agreeColumn}>
  722. <img src={isAgree.value ? agreeYes : agreeNo} onClick={() => (isAgree.value = !isAgree.value)} />
  723. <p onClick={(e: MouseEvent) => {
  724. e.stopPropagation();
  725. router.push('/preview-protocol');
  726. }}>我已阅读并同意<i>《音乐数字课堂学生端》</i>注册协议</p>
  727. </div> */}
  728. {/* <MProtocol
  729. center
  730. v-model:modelValue={forms.modelValue}
  731. prototcolType="REGISTER"
  732. /> */}
  733. <MSticky position="bottom">
  734. <div class={styles.paymentContainer}>
  735. <div class={styles.traditionBtn}>
  736. <Button
  737. type="primary"
  738. class={styles.submitBtn}
  739. color="linear-gradient( 135deg, #31C7FF 0%, #007AFE 100%)"
  740. round
  741. onClick={() => onSubmit()}
  742. disabled={forms.loading}
  743. loading={forms.loading}>
  744. 提交注册
  745. </Button>
  746. </div>
  747. </div>
  748. </MSticky>
  749. {forms.imgCodeStatus ? (
  750. <MImgCode
  751. v-model:value={forms.imgCodeStatus}
  752. phone={studentInfo.username}
  753. type="REGISTER"
  754. onClose={() => {
  755. forms.imgCodeStatus = false;
  756. }}
  757. onSendCode={onCodeSend}
  758. />
  759. ) : null}
  760. {/* 年级 */}
  761. <Popup
  762. v-model:show={forms.gradeStatus}
  763. position="bottom"
  764. round
  765. safeAreaInsetBottom
  766. lazyRender={false}
  767. class={'popupBottomSearch'}
  768. onOpen={() => {
  769. forms.gradePopupShow = true;
  770. }}
  771. onClosed={() => {
  772. forms.gradePopupShow = false;
  773. }}>
  774. {forms.gradePopupShow && (
  775. <Picker
  776. showToolbar
  777. v-model={forms.gradePopupIndex}
  778. columns={gradeList.value as any}
  779. onCancel={() => (forms.gradeStatus = false)}
  780. onConfirm={(val: any) => {
  781. const selectedOption = val.selectedOptions[0];
  782. studentInfo.extra.currentGradeNum = selectedOption.value;
  783. forms.gradeNumText = selectedOption.text;
  784. forms.gradeStatus = false;
  785. }}
  786. />
  787. )}
  788. </Popup>
  789. {/* 班级 */}
  790. <Popup
  791. v-model:show={forms.classStatus}
  792. position="bottom"
  793. round
  794. class={'popupBottomSearch'}
  795. onOpen={() => {
  796. forms.classPopupShow = true;
  797. }}
  798. onClosed={() => {
  799. forms.classPopupShow = false;
  800. }}>
  801. {forms.classPopupShow && (
  802. <Picker
  803. showToolbar
  804. v-model={forms.classPopupIndex}
  805. columns={classList}
  806. onCancel={() => (forms.classStatus = false)}
  807. onConfirm={(val: any) => {
  808. const selectedOption = val.selectedOptions[0];
  809. studentInfo.extra.currentClass = selectedOption.value;
  810. forms.currentClassText = selectedOption.text;
  811. forms.classStatus = false;
  812. }}
  813. />
  814. )}
  815. </Popup>
  816. {/* 是否在微信中打开 */}
  817. <OWxTip
  818. v-model:show={forms.showTips}
  819. message={forms.showMessage}
  820. showButton={forms.showButton}
  821. buttonText="刷新"
  822. onConfirm={async () => {
  823. forms.showTips = false;
  824. await getRegisterGoods();
  825. studentInfo.password = '';
  826. window.scrollTo({
  827. top: 0,
  828. behavior: 'smooth'
  829. });
  830. }}
  831. />
  832. <MMessageTip
  833. show={otherParams.showOtherSchool}
  834. // showCloseButton={otherParams.showCloseButton}
  835. messageAlign={otherParams.messageAlign}
  836. message={otherParams.showOtherMessage}
  837. showCancelButton={otherParams.showCancelButton}
  838. cancelButtonColor={otherParams.cancelButtonColor}
  839. cancelButtonText={otherParams.cancelButtonText}
  840. confirmButtonColor={otherParams.confirmButtonColor}
  841. confirmButtonText={otherParams.confirmButtonText}
  842. onClose={() => (otherParams.showOtherSchool = false)}
  843. onCancel={() => {
  844. otherParams.showOtherSchool = false;
  845. if (otherParams.otherType === 'nickname') {
  846. forms.isRegister = 'create'; // 新建
  847. changeTipStatus(false, false);
  848. onSubmit();
  849. } else if (otherParams.otherType === 'limit') {
  850. }
  851. }}
  852. onConfirm={() => {
  853. otherParams.showOtherSchool = false;
  854. // 名字
  855. if (otherParams.otherType === 'nickname') {
  856. forms.isRegister = 'update'; // 修改
  857. changeTipStatus(false, false);
  858. // 直接注册
  859. onSubmit();
  860. } else if (otherParams.otherType === 'change') {
  861. // 学校更换
  862. forms.isChangeSchool = true;
  863. // 直接注册
  864. onSubmit();
  865. } else if (otherParams.otherType === 'limit') {
  866. // 人数超限制
  867. changeTipStatus(
  868. forms.isRegister === 'create' && !forms.studentItem.userId
  869. ? false
  870. : true,
  871. false
  872. );
  873. }
  874. }}
  875. />
  876. <Popup
  877. v-model:show={forms.showSelectStudent}
  878. round
  879. position="bottom"
  880. safeAreaInsetBottom
  881. closeable
  882. class={styles.selectStudentPopup}>
  883. <SelectStudent
  884. studentItem={forms.studentItem}
  885. list={forms.studentList}
  886. onClose={() => (forms.showSelectStudent = false)}
  887. onConfirm={(val: any) => {
  888. if (val.userId) {
  889. forms.studentItem = val;
  890. const firstStudent = val;
  891. studentInfo.extra.nickname = firstStudent.nickname;
  892. const tempGrade: any = gradeList.value[0] || [];
  893. studentInfo.extra.currentGradeNum = '';
  894. forms.gradeNumText = '';
  895. tempGrade?.forEach((i: any) => {
  896. if (i.value === firstStudent.currentGradeNum) {
  897. // forms.instrumentCode = i.instrumentCode;
  898. forms.gradeNumText = i.text;
  899. studentInfo.extra.currentGradeNum =
  900. firstStudent.currentGradeNum;
  901. // if (forms.schoolInstrumentSetType === 'CLASS') {
  902. // forms.classList = i.classList;
  903. // }
  904. }
  905. });
  906. forms.currentClassText = '';
  907. studentInfo.extra.currentClass = '';
  908. classList.forEach((i: any) => {
  909. if (i.value === firstStudent.currentClass) {
  910. forms.currentClassText = i.text;
  911. studentInfo.extra.currentClass = firstStudent.currentClass;
  912. }
  913. });
  914. studentInfo.extra.gender = firstStudent.gender;
  915. forms.isRegister = 'update';
  916. changeTipStatus(true, false);
  917. forms.showSelectStudent = false;
  918. } else {
  919. // 判断新建学员是否上限了
  920. if (forms.studentList.length >= forms.multi_user_limit) {
  921. otherParams.showOtherMessage = `同一手机号最多创建${forms.multi_user_limit}个学生`;
  922. otherParams.showOtherSchool = true;
  923. otherParams.showCancelButton = false;
  924. otherParams.showCloseButton = true;
  925. otherParams.confirmButtonColor =
  926. 'linear-gradient( 305deg, #40C8FF 0%, #3192FF 100%)';
  927. otherParams.confirmButtonText = '我知道了';
  928. otherParams.otherType = 'limit';
  929. otherParams.messageAlign = 'center';
  930. return true;
  931. } else {
  932. forms.studentItem = val;
  933. forms.isRegister = 'create';
  934. changeTipStatus(false, false);
  935. studentInfo.extra.nickname = '';
  936. studentInfo.extra.currentGradeNum = '';
  937. studentInfo.extra.currentClass = '';
  938. studentInfo.extra.gender = 1;
  939. forms.currentClassText = '';
  940. forms.gradeNumText = '';
  941. forms.showSelectStudent = false;
  942. }
  943. }
  944. }}
  945. />
  946. </Popup>
  947. </div>
  948. );
  949. }
  950. });