index.tsx 24 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208209210211212213214215216217218219220221222223224225226227228229230231232233234235236237238239240241242243244245246247248249250251252253254255256257258259260261262263264265266267268269270271272273274275276277278279280281282283284285286287288289290291292293294295296297298299300301302303304305306307308309310311312313314315316317318319320321322323324325326327328329330331332333334335336337338339340341342343344345346347348349350351352353354355356357358359360361362363364365366367368369370371372373374375376377378379380381382383384385386387388389390391392393394395396397398399400401402403404405406407408409410411412413414415416417418419420421422423424425426427428429430431432433434435436437438439440441442443444445446447448449450451452453454455456457458459460461462463464465466467468469470471472473474475476477478479480481482483484485486487488489490491492493494495496497498499500501502503504505506507508509510511512513514515516517518519520521522523524525526527528529530531532533534535536537538539540541542543544545546547548549550551552553554555556557558559560561562563564565566567568569570571572573574575576577578579580581582583584585586587588589590591592593594595596597598599600601602603604605606607608609610611612613614615616617618619620621622623624625626627628629630631632633634635636637638639640641642643644645646647648649650651652653654655656657658659660661662663664665666667668669670671672673674675676677678679680681682683684685686687688689690691692693694695696697698699700701702703704705706707708709710711712713714715716717718719720721722723724725
  1. import {
  2. defineComponent,
  3. onMounted,
  4. onUnmounted,
  5. reactive,
  6. ref,
  7. watch,
  8. toRef
  9. } from 'vue';
  10. import styles from './index.module.less';
  11. import { postMessage } from '@/helpers/native-message';
  12. import icon_title from './images/icon-title.png';
  13. import icon_back from './images/icon-back.png';
  14. import icon_back1 from './images/icon-back1.png';
  15. import icon_setting from './images/icon-setting.png';
  16. import iconPlay from './images/icon-play.png';
  17. import iconPause from './images/icon-pause.png';
  18. import beat from './images/btn-2.png';
  19. import tempo from './images/btn-3.png';
  20. import randDom from './images/btn-1.png';
  21. import setImg from './images/setting.png';
  22. import iconPlus from './images/icon-plus.png';
  23. import iconAdd from './images/icon-add.png';
  24. import { getImage } from './images/music';
  25. import '@vant/touch-emulator';
  26. // import j2 from './images/music/j-2.png';
  27. import { Popover, Popup, showToast } from 'vant';
  28. import SettingModal from './setting-modal';
  29. import {
  30. randomScoreElement,
  31. renderScore,
  32. setting,
  33. elementDirection,
  34. setting_modal,
  35. initSelectScorePartModal,
  36. renderScoreModal
  37. } from './setting';
  38. import { handleStartTick, hendleEndTick } from './tick';
  39. import { handleStartBeat, hendleEndBeat } from './beat-tick';
  40. import { browser } from '@/helpers/utils';
  41. import { useRoute } from 'vue-router';
  42. import useDrag from '@/hooks/useDrag';
  43. import useDragGuidance from '@/hooks/useDrag/useDragGuidance';
  44. import { state as stateData } from '@/state';
  45. import SettingPcModal from './setting-pc-modal';
  46. import Draggable from 'vuedraggable';
  47. export default defineComponent({
  48. name: 'tempo-practice',
  49. props: {
  50. dataJson: {
  51. type: Object,
  52. default: () => {}
  53. },
  54. modeType: {
  55. type: String,
  56. default: ''
  57. },
  58. show: {
  59. type: Boolean,
  60. default: false
  61. }
  62. },
  63. setup(props, { expose }) {
  64. const route = useRoute();
  65. const state = reactive({
  66. modeType: '' as any,
  67. platform: route.query.platform,
  68. win: route.query.win,
  69. settingStatus: false,
  70. settingPcStatus: false,
  71. speedList: [
  72. { text: '40', value: 40, color: '#060606' },
  73. { text: '50', value: 50, color: '#060606' },
  74. { text: '60', value: 60, color: '#060606' },
  75. { text: '70', value: 70, color: '#060606' },
  76. { text: '80', value: 80, color: '#060606' },
  77. { text: '90', value: 90, color: '#060606' },
  78. { text: '100', value: 100, color: '#060606' },
  79. { text: '110', value: 110, color: '#060606' },
  80. { text: '120', value: 120, color: '#060606' },
  81. { text: '130', value: 130, color: '#060606' },
  82. { text: '140', value: 140, color: '#060606' },
  83. { text: '150', value: 150, color: '#060606' },
  84. { text: '160', value: 160, color: '#060606' },
  85. { text: '170', value: 170, color: '#060606' },
  86. { text: '180', value: 180, color: '#060606' },
  87. { text: '190', value: 190, color: '#060606' },
  88. { text: '200', value: 200, color: '#060606' }
  89. ],
  90. dataJson: {} as any,
  91. playPos: (route.query.imagePos || 'left') as 'left' | 'right' // 数字课堂老师端上课 镜像字段
  92. });
  93. // 拖拽临时数据
  94. const tempDragData = ref<any>({});
  95. // 返回
  96. const goback = (e: any) => {
  97. e.stopPropagation();
  98. if (route.query.backBtnType === 'microapp') {
  99. // microapp 老师端应用里面打开单独处理返回逻辑
  100. window.parent.postMessage(
  101. {
  102. api: 'iframe_exit'
  103. },
  104. '*'
  105. );
  106. return;
  107. }
  108. if (!browser().isApp) {
  109. window.close();
  110. return;
  111. }
  112. postMessage({ api: 'goBack' });
  113. };
  114. /* 修改设置的某个值 */
  115. const updateSettingValue = (key: string, value: any) => {
  116. if (state.settingStatus) {
  117. setting_modal[key] = value;
  118. } else {
  119. setting[key] = value;
  120. }
  121. };
  122. /* 获取设置的某一个值 */
  123. const getSettingValue = (key: string) => {
  124. let value: any = null;
  125. if (state.settingStatus) {
  126. value = setting_modal[key];
  127. } else {
  128. value = setting[key];
  129. }
  130. return value;
  131. };
  132. /** 播放切换 */
  133. const handlePlay = async () => {
  134. const playState = getSettingValue('playState');
  135. const playType = getSettingValue('playType');
  136. if (playState === 'pause') {
  137. updateSettingValue('playState', 'play');
  138. if (playType === 'beat') {
  139. await handleStartTick(state.settingStatus);
  140. } else {
  141. await handleStartBeat(state.settingStatus);
  142. }
  143. } else {
  144. handleStop();
  145. }
  146. };
  147. /** 播放切换 */
  148. const handlePlay2 = async () => {
  149. const playState = getSettingValue('playState');
  150. const playType = getSettingValue('playType');
  151. if (playState === 'pause') {
  152. updateSettingValue('playState', 'play');
  153. if (playType === 'beat') {
  154. await handleStartTick(state.settingStatus);
  155. } else {
  156. await handleStartBeat(state.settingStatus);
  157. }
  158. }
  159. };
  160. /** 播放类型 */
  161. const handlePlayType = () => {
  162. handleStop();
  163. // 判断是否有设置
  164. const playType = getSettingValue('playType');
  165. if (playType === 'beat') {
  166. updateSettingValue('playType', 'tempo');
  167. } else {
  168. updateSettingValue('playType', 'beat');
  169. }
  170. };
  171. const handleStop = () => {
  172. const playType = getSettingValue('playType');
  173. updateSettingValue('playState', 'pause');
  174. if (playType === 'beat') {
  175. hendleEndTick(state.settingStatus);
  176. } else {
  177. hendleEndBeat(state.settingStatus);
  178. }
  179. };
  180. // {"element":"jianpu","beat":"4-4","barLine":"1","tempo":["1","2","3"]}'
  181. const onIframeHandle = (ev: MessageEvent) => {
  182. // 获取配置
  183. if (ev.data.api === 'getTempoSetting') {
  184. window.parent.postMessage(
  185. {
  186. api: 'getTempoSetting',
  187. data: JSON.stringify({
  188. setting: {
  189. element: setting.element,
  190. beat: setting.beat,
  191. barLine: setting.barLine,
  192. tempo: setting.tempo,
  193. scorePart: setting.scorePart,
  194. playType: setting.playType,
  195. speed: setting.speed
  196. },
  197. coverImg: ''
  198. })
  199. },
  200. '*'
  201. );
  202. }
  203. if (ev.data.api === 'setPlayState') {
  204. if (ev.data.data) {
  205. handlePlay();
  206. } else {
  207. handleStop();
  208. }
  209. }
  210. if (ev.data.api === 'resetPlay') {
  211. resetSetting();
  212. }
  213. if (ev.data.api === 'imagePos') {
  214. if (ev.data.data === 'right') {
  215. state.playPos = 'right';
  216. } else {
  217. state.playPos = 'left';
  218. }
  219. }
  220. };
  221. const resetSetting = () => {
  222. try {
  223. let dataJson = props.dataJson;
  224. if (route.query.dataJson) {
  225. dataJson = JSON.parse(route.query.dataJson as any);
  226. }
  227. if (dataJson) {
  228. setting.element = dataJson.element;
  229. setting.beat = dataJson.beat;
  230. setting.barLine = dataJson.barLine;
  231. setting.tempo = dataJson.tempo;
  232. setting.scorePart = dataJson.scorePart;
  233. setting.playType = dataJson.playType;
  234. setting.speed = dataJson.speed;
  235. setting_modal.element = dataJson.element;
  236. setting_modal.beat = dataJson.beat;
  237. setting_modal.barLine = dataJson.barLine;
  238. setting_modal.tempo = dataJson.tempo;
  239. setting_modal.scorePart = dataJson.scorePart;
  240. setting_modal.playType = dataJson.playType;
  241. setting_modal.speed = dataJson.speed;
  242. state.dataJson = dataJson;
  243. }
  244. } catch (e) {
  245. //
  246. console.log(e, '1');
  247. }
  248. };
  249. // watch(
  250. // () => props.show,
  251. // val => {
  252. // console.log(val, props.show);
  253. // if (!val) {
  254. // // resetSetting();
  255. // handleStop();
  256. // } else {
  257. // resetSetting();
  258. // }
  259. // }
  260. // );
  261. /** 打开设置 */
  262. const onOpenSetting = () => {
  263. handleStop();
  264. // 初始化设置的数据
  265. for (let i in setting) {
  266. setting_modal[i] = JSON.parse(JSON.stringify(setting[i]));
  267. }
  268. if (state.win === 'pc' || state.platform === 'modal') {
  269. state.settingPcStatus = true;
  270. } else {
  271. state.settingStatus = true;
  272. }
  273. };
  274. onMounted(() => {
  275. // 安卓的状态栏
  276. postMessage({
  277. api: 'setStatusBarVisibility',
  278. content: {
  279. isVisibility: 0
  280. }
  281. });
  282. if (route.query.modeType) {
  283. state.modeType = route.query.modeType;
  284. }
  285. resetSetting();
  286. state.speedList.forEach((item: any) => {
  287. if (item.value === setting.speed) item.color = '#1CACF1';
  288. });
  289. if (setting?.scorePart?.length <= 0) {
  290. renderScore();
  291. }
  292. window.addEventListener('message', onIframeHandle);
  293. });
  294. onUnmounted(() => {
  295. window.removeEventListener('message', onIframeHandle);
  296. });
  297. watch(
  298. () => state.settingStatus,
  299. () => {
  300. if (!state.settingStatus) {
  301. state.speedList.forEach((item: any) => {
  302. if (item.value === getSettingValue('speed')) {
  303. item.color = '#1CACF1';
  304. } else {
  305. item.color = '#060606';
  306. }
  307. });
  308. handleStop();
  309. }
  310. }
  311. );
  312. expose({
  313. resetSetting
  314. });
  315. let settingBoxDragData: any;
  316. let settingBoxClass: string;
  317. if (state.platform === 'modal') {
  318. settingBoxClass = 'settingBoxClass_drag';
  319. settingBoxDragData = useDrag(
  320. [
  321. `${settingBoxClass} .iconTitBoxMove`,
  322. `${settingBoxClass} .bom_drag_point`,
  323. `${settingBoxClass} .bom_drag_point_right`
  324. ],
  325. settingBoxClass,
  326. toRef(state, 'settingPcStatus'),
  327. stateData.user.data.id
  328. );
  329. }
  330. // 引导页
  331. const { guidanceShow, setGuidanceShow } = useDragGuidance();
  332. return () => (
  333. <div
  334. onClick={() => {
  335. state.settingStatus = false;
  336. window.parent.postMessage(
  337. {
  338. api: 'clickTempo'
  339. },
  340. '*'
  341. );
  342. }}
  343. class={[
  344. styles.tempoPractice,
  345. state.win === 'pc' ? styles.pc : '',
  346. state.platform === 'modal' ? styles.modal : '',
  347. state.modeType === 'courseware' ? styles.courseware : ''
  348. ]}>
  349. <div
  350. class={[
  351. styles.containerLeft,
  352. state.settingStatus ? styles.leftShow : ''
  353. ]}>
  354. <div class={styles.head}>
  355. {state.modeType !== 'courseware' && (
  356. <div
  357. class={[styles.back, styles.iconBack]}
  358. onClick={goback}
  359. style={{ cursor: 'pointer' }}>
  360. <img src={icon_back} />
  361. </div>
  362. )}
  363. <div class={styles.title}>
  364. <img src={icon_title} />
  365. </div>
  366. {state.modeType !== 'courseware' &&
  367. state.platform !== 'modal' &&
  368. !state.settingStatus ? (
  369. <div
  370. class={styles.back}
  371. style={{ cursor: 'pointer' }}
  372. onClick={(e: any) => {
  373. e.stopPropagation();
  374. onOpenSetting();
  375. }}>
  376. <img src={icon_setting} />
  377. </div>
  378. ) : (
  379. <div class={styles.back}></div>
  380. )}
  381. </div>
  382. <div class={styles.conCon}>
  383. <div class={styles.container}>
  384. {getSettingValue('scorePart')?.map((item: any, i: number) => (
  385. <div
  386. class={[
  387. styles.beatSection,
  388. getSettingValue('scorePart').length >= 2 &&
  389. item.length !== 1 &&
  390. styles.small
  391. ]}>
  392. {item.map((child: any, jIndex: number) => (
  393. <Draggable
  394. modelValue={[child]}
  395. itemKey="index"
  396. // eslint-disable-next-line @typescript-eslint/ban-ts-comment
  397. // @ts-ignore
  398. disabled={!state.settingStatus}
  399. // eslint-disable-next-line @typescript-eslint/ban-ts-comment
  400. // @ts-ignore
  401. group={{
  402. name: 'description',
  403. pull: 'clone',
  404. put: true
  405. }}
  406. animation={200}
  407. sort={false}
  408. componentData={{
  409. draggable: 'row-nav',
  410. itemKey: 'index',
  411. tag: 'div',
  412. pull: 'clone',
  413. put: true,
  414. animation: 200,
  415. group: 'description'
  416. }}
  417. onChange={(evt: any) => {
  418. tempDragData.value = evt.added || '';
  419. }}
  420. onAdd={() => {
  421. const added = tempDragData.value?.element || {};
  422. // 判断是否有数据
  423. if (added.url && added.sourceFrom === 'setting-modal') {
  424. handleStop();
  425. setting_modal.scorePart.forEach(
  426. (part: Array<any>, ci: number) => {
  427. part.forEach((child: any, cj: number) => {
  428. if (i === ci && jIndex === cj) {
  429. child.url = added.url;
  430. child.index = added.index;
  431. }
  432. });
  433. }
  434. );
  435. }
  436. }}
  437. onStart={(evt: any) => {
  438. evt.from.classList.add('onstart');
  439. tempDragData.value = {};
  440. }}
  441. onEnd={(evt: any) => {
  442. evt.from.classList.remove('onstart');
  443. tempDragData.value = {};
  444. }}
  445. onClick={(e: any) => {
  446. e.stopPropagation();
  447. // 编辑时可以操作
  448. if (state.settingStatus) {
  449. handleStop();
  450. if (setting_modal.scorePart[i][jIndex].selected) {
  451. initSelectScorePartModal();
  452. } else {
  453. initSelectScorePartModal(i, jIndex);
  454. }
  455. }
  456. }}
  457. class={[
  458. styles.beat,
  459. child.selected ? styles.active : '',
  460. state.settingStatus && styles.disabledChange
  461. ]}>
  462. {{
  463. item: (element: any) => {
  464. const child = element.element;
  465. const jIndex = element.index;
  466. return (
  467. <div>
  468. {/* 编辑时不可上下切换 */}
  469. {!state.settingStatus && (
  470. <div class={styles.direction}>
  471. <div
  472. class={styles.up}
  473. style={{ cursor: 'pointer' }}
  474. onClick={() => {
  475. if (setting.playState === 'play') return;
  476. if (setting.tempo.length <= 1) {
  477. showToast(
  478. '无法切换,请选择至少2种节奏型'
  479. );
  480. return;
  481. }
  482. // const obj = randomScoreElement(child.index);
  483. const obj = elementDirection(
  484. 'up',
  485. child.index
  486. );
  487. child.index = obj.index;
  488. child.url = obj.url;
  489. }}></div>
  490. <div
  491. class={styles.down}
  492. style={{ cursor: 'pointer' }}
  493. onClick={() => {
  494. if (setting.playState === 'play') return;
  495. if (setting.tempo.length <= 1) {
  496. showToast(
  497. '无法切换,请选择至少2种节奏型'
  498. );
  499. return;
  500. }
  501. // const obj = randomScoreElement(child.index);
  502. const obj = elementDirection(
  503. 'down',
  504. child.index
  505. );
  506. child.index = obj.index;
  507. child.url = obj.url;
  508. }}></div>
  509. </div>
  510. )}
  511. <div class={styles.imgSection}>
  512. <img src={getImage(child.url)} />
  513. </div>
  514. </div>
  515. );
  516. }
  517. }}
  518. </Draggable>
  519. ))}
  520. </div>
  521. ))}
  522. </div>
  523. </div>
  524. <div
  525. class={styles.footer}
  526. onClick={(e: any) => {
  527. e.stopPropagation();
  528. }}>
  529. {/* 播放 */}
  530. {state.playPos === 'left' && (
  531. <>
  532. {route.query.back === 'show' && (
  533. <div
  534. class={[styles.play]}
  535. onClick={goback}
  536. style={{ cursor: 'pointer' }}>
  537. <img src={icon_back1} />
  538. </div>
  539. )}
  540. <div class={styles.play} onClick={handlePlay}>
  541. {getSettingValue('playState') === 'pause' ? (
  542. <img src={iconPause} />
  543. ) : (
  544. <img src={iconPlay} />
  545. )}
  546. </div>
  547. </>
  548. )}
  549. {/* 老师端来的时候的设置按钮 */}
  550. {state.platform === 'modal' && state.playPos === 'right' && (
  551. <div
  552. class={styles.setting}
  553. onClick={() => {
  554. onOpenSetting();
  555. }}>
  556. <img src={setImg} />
  557. </div>
  558. )}
  559. {/* 播放类型 */}
  560. <div class={styles.playType} onClick={handlePlayType}>
  561. {getSettingValue('playType') === 'beat' ? (
  562. <img src={beat} />
  563. ) : (
  564. <img src={tempo} />
  565. )}
  566. </div>
  567. {/* 随机生成 */}
  568. <div
  569. class={styles.randomTempo}
  570. onClick={() => {
  571. if (state.settingStatus) {
  572. renderScoreModal();
  573. } else {
  574. renderScore();
  575. }
  576. handleStop();
  577. }}>
  578. <img src={randDom} />
  579. </div>
  580. {/* 速度 */}
  581. <div class={styles.speedChange}>
  582. <img
  583. src={iconPlus}
  584. class={styles.speedPlus}
  585. onClick={() => {
  586. const speed = getSettingValue('speed');
  587. if (speed <= 40) return;
  588. updateSettingValue('speed', speed - 1);
  589. handleStop();
  590. state.speedList.forEach((item: any) => {
  591. if (item.value === getSettingValue('speed')) {
  592. item.color = '#1CACF1';
  593. updateSettingValue('speed', getSettingValue('speed'));
  594. } else {
  595. item.color = '#060606';
  596. }
  597. });
  598. }}
  599. />
  600. <Popover
  601. placement="top"
  602. class={styles.popupContainer}
  603. actions={state.speedList}
  604. onSelect={(val: any) => {
  605. const speed = getSettingValue('speed');
  606. if (val.value === speed) return;
  607. state.speedList.forEach((item: any) => {
  608. if (item.value === val.value) {
  609. item.color = '#1CACF1';
  610. updateSettingValue('speed', val.value);
  611. } else {
  612. item.color = '#060606';
  613. }
  614. });
  615. handleStop();
  616. }}>
  617. {{
  618. reference: () => (
  619. <div class={styles.speedNum}>
  620. {getSettingValue('speed')}
  621. </div>
  622. )
  623. }}
  624. </Popover>
  625. <img
  626. src={iconAdd}
  627. class={styles.speedAdd}
  628. onClick={() => {
  629. const speed = getSettingValue('speed');
  630. if (speed >= 200) return;
  631. updateSettingValue('speed', speed + 1);
  632. handleStop();
  633. state.speedList.forEach((item: any) => {
  634. if (item.value === getSettingValue('speed')) {
  635. item.color = '#1CACF1';
  636. updateSettingValue('speed', getSettingValue('speed'));
  637. } else {
  638. item.color = '#060606';
  639. }
  640. });
  641. }}
  642. />
  643. </div>
  644. {/* 播放 */}
  645. {state.playPos === 'right' && (
  646. <div class={styles.play} onClick={handlePlay}>
  647. {getSettingValue('playState') === 'pause' ? (
  648. <img src={iconPause} />
  649. ) : (
  650. <img src={iconPlay} />
  651. )}
  652. </div>
  653. )}
  654. {/* 老师端来的时候的设置按钮 */}
  655. {state.platform === 'modal' && state.playPos === 'left' && (
  656. <div
  657. class={styles.setting}
  658. onClick={() => {
  659. onOpenSetting();
  660. }}>
  661. <img src={setImg} />
  662. </div>
  663. )}
  664. </div>
  665. </div>
  666. <div
  667. class={[
  668. styles.containerRight,
  669. state.settingStatus ? '' : styles.rightHide
  670. ]}
  671. onClick={(e: any) => {
  672. e.stopPropagation();
  673. }}>
  674. <SettingModal
  675. class={styles.settingModalShow}
  676. onClose={() => (state.settingStatus = false)}
  677. />
  678. </div>
  679. <Popup
  680. style={
  681. state.platform === 'modal' ? settingBoxDragData.styleDrag.value : {}
  682. }
  683. closeOnClickOverlay={false}
  684. v-model:show={state.settingPcStatus}
  685. class={[styles.settingPopup, settingBoxClass]}>
  686. <SettingPcModal
  687. onGuideDone={setGuidanceShow}
  688. showGuide={guidanceShow.value}
  689. onClose={() => (state.settingPcStatus = false)}
  690. />
  691. </Popup>
  692. </div>
  693. );
  694. }
  695. });