/* * XmlDigitalTeaching v0.0.1 * Copyright ©Tue May 27 2025 20:10:29 GMT+0800 (中国标准时间) smile * Released under the ISC License. */ var playIcon = ""; var pauseIcon = ""; var audioIcon = ""; var coverLocal = "5d9cf0a5d487a818.png"; var playRepeat = ""; var playOnce = ""; function parseLrc(lrc_s) { if (lrc_s) { lrc_s = lrc_s.replace(/([^\]^\n])\[/g, (match, p1) => p1 + '\n['); const lyric = lrc_s.split('\n'); const lrc = []; const lyricLen = lyric.length; for (let i = 0; i < lyricLen; i++) { // match lrc time const lrcTimes = lyric[i].match(/\[(\d{2}):(\d{2})(\.(\d{2,3}))?]/g); // match lrc text const lrcText = lyric[i].replace(/.*\[(\d{2}):(\d{2})(\.(\d{2,3}))?]/g, '').replace(/<(\d{2}):(\d{2})(\.(\d{2,3}))?>/g, '').replace(/^\s+|\s+$/g, ''); if (lrcTimes) { // handle multiple time tag const timeLen = lrcTimes.length; for (let j = 0; j < timeLen; j++) { const oneTime = /\[(\d{2}):(\d{2})(\.(\d{2,3}))?]/.exec(lrcTimes[j]); const min2sec = oneTime[1] * 60; const sec2sec = parseInt(oneTime[2]); const msec2sec = oneTime[4] ? parseInt(oneTime[4]) / ((oneTime[4] + '').length === 2 ? 100 : 1000) : 0; const lrcTime = min2sec + sec2sec + msec2sec; lrc.push([lrcTime, lrcText]); } } } // sort by time lrc.sort((a, b) => a[0] - b[0]); return lrc; } else { return []; } } // var script$3 = { props: { currentMusic: { type: Object, required: true }, playStat: { type: Object, required: true } }, data() { return { displayLrc: '', currentLineIndex: 0 }; }, computed: { lrcLines() { return parseLrc(this.displayLrc); }, currentLine() { if (this.currentLineIndex > this.lrcLines.length - 1) { return null; } return this.lrcLines[this.currentLineIndex]; }, transformStyle() { // transform: translateY(0); -webkit-transform: translateY(0); return { transform: `translateY(${-this.currentLineIndex * 16}px)`, webkitTransform: `translateY(${-this.currentLineIndex * 16}px)` }; } }, methods: { applyLrc(lrc) { // if (/^https?:\/\//.test(lrc)) { // this.fetchLrc(lrc) // } else { // this.displayLrc = lrc // } this.fetchLrc(lrc); }, fetchLrc(src) { fetch(src).then(response => response.text()).then(lrc => { this.displayLrc = lrc; }); }, hideLrc() { this.displayLrc = ''; } }, watch: { currentMusic: { deep: true, immediate: true, handler(music) { this.currentLineIndex = 0; if (music.lrc) { this.applyLrc(music.lrc); } else { this.hideLrc(); } } }, 'playStat.playedTime'(playedTime) { for (let i = 0; i < this.lrcLines.length; i++) { const line = this.lrcLines[i]; const nextLine = this.lrcLines[i + 1]; if (playedTime >= line[0] && (!nextLine || playedTime < nextLine[0])) { this.currentLineIndex = i; } } } } }; function normalizeComponent(template, style, script, scopeId, isFunctionalTemplate, moduleIdentifier /* server only */, shadowMode, createInjector, createInjectorSSR, createInjectorShadow) { if (typeof shadowMode !== 'boolean') { createInjectorSSR = createInjector; createInjector = shadowMode; shadowMode = false; } // Vue.extend constructor export interop. const options = typeof script === 'function' ? script.options : script; // render functions if (template && template.render) { options.render = template.render; options.staticRenderFns = template.staticRenderFns; options._compiled = true; // functional template if (isFunctionalTemplate) { options.functional = true; } } // scopedId if (scopeId) { options._scopeId = scopeId; } let hook; if (moduleIdentifier) { // server build hook = function (context) { // 2.3 injection context = context || // cached call this.$vnode && this.$vnode.ssrContext || // stateful this.parent && this.parent.$vnode && this.parent.$vnode.ssrContext; // functional // 2.2 with runInNewContext: true if (!context && typeof __VUE_SSR_CONTEXT__ !== 'undefined') { context = __VUE_SSR_CONTEXT__; } // inject component styles if (style) { style.call(this, createInjectorSSR(context)); } // register component module identifier for async chunk inference if (context && context._registeredComponents) { context._registeredComponents.add(moduleIdentifier); } }; // used by ssr in case component is cached and beforeCreate // never gets called options._ssrRegister = hook; } else if (style) { hook = shadowMode ? function (context) { style.call(this, createInjectorShadow(context, this.$root.$options.shadowRoot)); } : function (context) { style.call(this, createInjector(context)); }; } if (hook) { if (options.functional) { // register for functional component in vue file const originalRender = options.render; options.render = function renderWithStyleInjection(h, context) { hook.call(context); return originalRender(h, context); }; } else { // inject component registration as beforeCreate hook const existing = options.beforeCreate; options.beforeCreate = existing ? [].concat(existing, hook) : [hook]; } } return script; } /* script */ const __vue_script__$3 = script$3; /* template */ var __vue_render__$3 = function () { var _vm = this; var _h = _vm.$createElement; var _c = _vm._self._c || _h; return _c("div", { staticClass: "aplayer-lrc" }, [ _c( "div", { staticClass: "aplayer-lrc-contents", style: _vm.transformStyle }, _vm._l(_vm.lrcLines, function (line, index) { return _c( "p", { key: index, class: { "aplayer-lrc-current": index === _vm.currentLineIndex }, }, [_vm._v("\n " + _vm._s(line[1]) + "\n ")] ) }), 0 ), ]) }; var __vue_staticRenderFns__$3 = []; __vue_render__$3._withStripped = true; /* style */ const __vue_inject_styles__$3 = undefined; /* scoped */ const __vue_scope_id__$3 = undefined; /* module identifier */ const __vue_module_identifier__$3 = undefined; /* functional template */ const __vue_is_functional_template__$3 = false; /* style inject */ /* style inject SSR */ /* style inject shadow dom */ const __vue_component__$3 = /*#__PURE__*/normalizeComponent( { render: __vue_render__$3, staticRenderFns: __vue_staticRenderFns__$3 }, __vue_inject_styles__$3, __vue_script__$3, __vue_scope_id__$3, __vue_is_functional_template__$3, __vue_module_identifier__$3, false, undefined, undefined, undefined ); // var script$2 = { name: 'XmlAudioModalPlayer', components: { lyrics: __vue_component__$3 }, props: { //模式:preview 预览 (默认),editor 编辑 mode: { type: String, default: 'editor' }, playStyle: { type: String, default: 'yuan-wen' }, styleMode: { type: String, default: 'A' }, url: { type: String, default: '', require: true }, title: { type: String, default: '音频文件' }, coverImageUrl: { type: String, default: '' }, //进度是否能控制 scheduleControl: { type: String, default: '' }, loop: { type: Boolean, default: true }, playMode: { type: String, default: 'xun-huan' }, content: { type: String, default: '暂无简介' }, targerLink: { type: String, default: '' }, subTitle: { type: Boolean, default: false }, subTitleSrc: { type: String, default: '' }, subTitleSrcName: { type: String, default: '' } }, computed: { videoAndMusicStop() { let playState = sessionStorage.getItem('SET_PLAY_STATE'); return playState; }, playModeIconSrc() { let srcMap = { 'xun-huan': playRepeat, 'dan-ci': playOnce }; return srcMap[this.playMode]; } }, watch: { url: { handler(newVal) { let _this = this; this.isPlay = false; let ua = navigator.userAgent; /micromessenger/.test(ua.toLowerCase()); let isIOS = !!ua.match(/\(i[^;]+;( U;)? CPU.+Mac OS X/); if (isIOS && newVal) { try { this.$nextTick(() => { _this.$refs.musicAudio.muted = true; const playPromise = _this.$refs.musicAudio.play(); _this.$refs.musicAudio.muted = false; if (playPromise) { playPromise.then(() => { _this.$refs.musicAudio.pause(); }).catch(error => { _this.$refs.musicAudio.pause(); console.log(error); }); } }); } catch (error) { console.log(error); this.$message({ type: 'warning', message: error, duration: 5000 }); } } }, immediate: true }, subTitleSrc: { handler(newVal) { this.audio.title = this.title; this.audio.src = this.url; this.audio.pic = this.coverImageUrl || coverLocal; this.audio.lrc = newVal; }, immediate: true }, videoAndMusicStop(val) { console.log('watch', val); let currentPlayStamp = val || '--'; if (`${this.url}__playTimeStamp__${this.lastOperateTimeStamp}` !== currentPlayStamp) { this.musicAudio.pause(); } } }, data() { return { playIcon, pauseIcon, audioIcon, coverLocal, isPlay: false, playTime: 0, playDuration: '00:00', playCurrentTime: '00:00', totalDuration: 0, link: { type: String, default: '' }, //链接地址 lastOperateTimeStamp: '', musicAudio: null, manualPlay: false, audio: { title: '', artist: '', src: '', pic: '', lrc: '' }, playStat: { duration: 0, loadedTime: 0, playedTime: 0 } }; }, mounted() { // 加载完毕后,先获取