index.tsx 6.8 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208209210211212213
  1. import { defineComponent, onMounted, onUnmounted, reactive, nextTick } from "vue";
  2. import state, { getMusicDetail, handleSetSpeed, addNoteBBox, getNote, gotoNext, fillWordColor, moveSvgDom } from "/src/state";
  3. import MusicScore from "../../view/music-score";
  4. import styles from "./index.module.less";
  5. import { getQuery } from "/src/utils/queryString";
  6. import { closeToast, showLoadingToast } from "vant";
  7. import store from "store";
  8. import { formateTimes } from "../../helpers/formateMusic";
  9. import { setCustomNoteRealValue } from "/src/helpers/customMusicScore"
  10. import { initSmoothAnimation, smoothAnimationState, destroySmoothAnimation, moveSmoothAnimationByPlayTime } from "../view-detail/smoothAnimation";
  11. import { api_cloudLoading, simple_musicPage } from "/src/helpers/communication";
  12. export default defineComponent({
  13. name: "simple-detail",
  14. setup() {
  15. const query: any = getQuery();
  16. const detailData = reactive({
  17. isLoading: true,
  18. currentTime: 0, // 当前播放的时间
  19. totalTime: 0, // 音视频总时长
  20. backgroundRendMode: "" as "video" | "audio", // 嵌入的时音频还是视频
  21. });
  22. const communicateCb = (res: any) => {
  23. let resInfo: any = {};
  24. try {
  25. resInfo = typeof res?.data === 'string' ? JSON.parse(res.data) : res.data;
  26. // console.log('Received data:', resInfo);
  27. } catch (error) {
  28. console.error('parse_error: Invalid JSON data received');
  29. }
  30. // 开始播放
  31. if (resInfo?.api === "api_play") {
  32. console.log('h5开始播放')
  33. state.playState = 'play';
  34. setStep();
  35. }
  36. // 暂停播放
  37. if (resInfo?.api === "api_paused") {
  38. const currentTime = resInfo?.content?.currentTime
  39. console.log('暂停播放',currentTime)
  40. state.playState = 'paused';
  41. if (currentTime === 0) {
  42. // 坐标和小节都改为初始值
  43. setTimeout(() => {
  44. detailData.currentTime = 0
  45. state.activeNoteIndex = 0
  46. state.activeMeasureIndex = state.times[0].MeasureNumberXML;
  47. handlePlaying(true);
  48. }, 200);
  49. }
  50. }
  51. // 暂停状态下,拖动进度
  52. if (resInfo?.api === "api_updateProgress") {
  53. console.log('拖动的进度',state.playState)
  54. if (state.playState === 'paused') {
  55. detailData.currentTime = resInfo?.content?.currentTime ?? detailData.currentTime;
  56. // 坐标和小节都改为初始值
  57. state.activeNoteIndex = 0
  58. state.activeMeasureIndex = state.times[0].MeasureNumberXML;
  59. handlePlaying(true);
  60. }
  61. }
  62. // 播放进度
  63. if (resInfo?.api === "api_playProgress") {
  64. // console.log('播放进度',resInfo)
  65. const currentTime = resInfo?.content?.currentTime
  66. if (currentTime) {
  67. if (currentTime < detailData.currentTime) {
  68. // 坐标和小节都改为初始值
  69. state.activeNoteIndex = 0
  70. let item = getNote(currentTime) || state.times[0];
  71. if(item.i === state.activeNoteIndex){
  72. state.activeMeasureIndex = item.MeasureNumberXML;
  73. }
  74. // 当拖动到MP3节拍器时候 需要手动移动到当前为止
  75. const fixtime = state.times[0].fixtime
  76. if(currentTime <= fixtime){
  77. moveSvgDom(true)
  78. }
  79. }
  80. detailData.currentTime = currentTime
  81. }
  82. }
  83. };
  84. // 监听评测曲谱音频播放进度,返回
  85. const progress = (res: any) => {
  86. console.log('app播放进度',res)
  87. const currentTime = res?.currentTime || res?.content?.currentTime;
  88. if (currentTime) {
  89. if (currentTime < detailData.currentTime) {
  90. state.activeNoteIndex = 0
  91. }
  92. detailData.currentTime = currentTime
  93. }
  94. };
  95. onMounted(async () => {
  96. const id = query.id || '';
  97. // 作品id
  98. const userMusicId = query.userMusicId || '';
  99. detailData.backgroundRendMode = query.backgroundRendMode;
  100. state.isSimplePage = true;
  101. state.partIndex = query["part-index"] ? Number(query["part-index"]) : 0;
  102. await getMusicDetail(id, 'open', userMusicId);
  103. detailData.isLoading = false;
  104. state.isSingleLine = true;
  105. window.addEventListener("message", communicateCb);
  106. });
  107. onUnmounted(() => {
  108. state.isSimplePage = false;
  109. window.removeEventListener("message", communicateCb);
  110. });
  111. /** 渲染完成 */
  112. const handleRendered = async (osmd: any) => {
  113. const svgHeight = document.getElementById('scrollContainer')?.getBoundingClientRect().height;
  114. state.osmd = osmd;
  115. // 没有设置速度使用读取的速度
  116. if (state.originSpeed === 0) {
  117. state.originSpeed = state.speed = (osmd as any).bpm || osmd.Sheet.userStartTempoInBPM || 100;
  118. }
  119. const saveSpeed = (store.get("speeds") || {})[state.examSongId] || state.speed || (osmd as any).bpm || osmd.Sheet.userStartTempoInBPM;
  120. // 加载本地缓存的速度
  121. if (saveSpeed) {
  122. handleSetSpeed(saveSpeed);
  123. }
  124. // setCustomNoteRealValue();
  125. state.times = formateTimes(osmd);
  126. console.log("🚀 ~ state.times:", state.times, state);
  127. nextTick(() => {
  128. state.activeMeasureIndex = state.times[0].MeasureNumberXML;
  129. // fillWordColor();
  130. })
  131. // 音符添加位置信息bbox
  132. addNoteBBox(state.times);
  133. // 一行谱创建 动画
  134. initSmoothAnimation();
  135. //destroySmoothAnimation();
  136. //smoothAnimationState.isShow.value = false;
  137. api_cloudLoading();
  138. console.log('渲染完成',svgHeight)
  139. window.parent.postMessage(
  140. {
  141. api: "api_musicPage",
  142. height: svgHeight
  143. },
  144. "*"
  145. );
  146. simple_musicPage({
  147. height: svgHeight
  148. })
  149. // state.playState = 'play';
  150. // setStep();
  151. };
  152. /**
  153. * 播放一直触发的事件
  154. */
  155. const handlePlaying = (skipNote?: boolean) => {
  156. //detailData.currentTime += 0.03
  157. const currentTime = detailData.currentTime;
  158. // console.log('👀~播放进度',currentTime)
  159. let item = getNote(currentTime) || state.times[0];
  160. if (item) {
  161. gotoNext(item, skipNote);
  162. }
  163. state.activeNoteIndex = item?.i || 0
  164. // 一行谱,需要滚动小节
  165. if (state.isSingleLine) {
  166. moveSmoothAnimationByPlayTime(currentTime, true)
  167. }
  168. };
  169. /** 在渲染前后计算光标应该走到的音符 */
  170. const setStep = () => {
  171. // console.log('播放状态',state.playState)
  172. if (state.playState !== "play") {
  173. console.log("暂停播放");
  174. return;
  175. }
  176. let startTime = Date.now();
  177. requestAnimationFrame(() => {
  178. const endTime = Date.now();
  179. // 渲染时间大于16.6,就会让页面卡顿, 如果渲染时间大与16.6就下一个渲染帧去计算
  180. if (endTime - startTime < 16.7) {
  181. handlePlaying();
  182. setStep();
  183. } else {
  184. setTimeout(() => {
  185. handlePlaying();
  186. setStep();
  187. }, 16.7);
  188. }
  189. });
  190. };
  191. return () => (
  192. <div class={[styles.detail, detailData.backgroundRendMode === 'video' && styles.whiteBg]}>
  193. <div class={styles.mask}></div>
  194. <div id="scrollContainer" class={[styles.container]}>
  195. {/* 曲谱渲染 */}
  196. {!detailData.isLoading &&
  197. <MusicScore
  198. onRendered={handleRendered}
  199. musicColor={'#000000'}
  200. />}
  201. </div>
  202. </div>
  203. );
  204. },
  205. });