136 lines
		
	
	
		
			3.8 KiB
		
	
	
	
		
			JavaScript
		
	
	
	
	
	
			
		
		
	
	
			136 lines
		
	
	
		
			3.8 KiB
		
	
	
	
		
			JavaScript
		
	
	
	
	
	
| import {
 | ||
|     ref,
 | ||
|     onBeforeUnmount,
 | ||
|     onMounted
 | ||
| } from 'vue'
 | ||
| import {
 | ||
|     onHide,
 | ||
|     onUnload
 | ||
| } from '@dcloudio/uni-app'
 | ||
| 
 | ||
| 
 | ||
| 
 | ||
| export function useSpeechReader() {
 | ||
|     const isSpeaking = ref(false)
 | ||
|     const isPaused = ref(false)
 | ||
|     let utterance = null
 | ||
| 
 | ||
|     const cleanMarkdown = (text) => {
 | ||
|         return formatTextForSpeech(text)
 | ||
|     }
 | ||
| 
 | ||
|     const speak = (text, options = {
 | ||
|         lang: 'zh-CN',
 | ||
|         rate: 0.9,
 | ||
|         pitch: 1.2
 | ||
|     }) => {
 | ||
|         cancelAudio() // 重置之前的
 | ||
|         // const voices = speechSynthesis.getVoices()
 | ||
|         // const chineseVoices = voices.filter(v => v.lang.includes('zh'))
 | ||
|         const speechText = extractSpeechText(text);
 | ||
|         utterance = new SpeechSynthesisUtterance(speechText)
 | ||
|         // utterance.lang = options.lang || 'zh'
 | ||
|         utterance.rate = options.rate || 1
 | ||
|         utterance.pitch = options.pitch || 1.1 // 音调(0 - 2,偏高比较柔和)
 | ||
| 
 | ||
|         utterance.onend = () => {
 | ||
|             isSpeaking.value = false
 | ||
|             isPaused.value = false
 | ||
|         }
 | ||
| 
 | ||
|         speechSynthesis.speak(utterance)
 | ||
|         isSpeaking.value = true
 | ||
|         isPaused.value = false
 | ||
|     }
 | ||
| 
 | ||
|     const pause = () => {
 | ||
|         if (isSpeaking.value && !isPaused.value) {
 | ||
|             speechSynthesis.pause()
 | ||
|             isPaused.value = true
 | ||
|         }
 | ||
|     }
 | ||
| 
 | ||
|     const resume = () => {
 | ||
|         if (isSpeaking.value && isPaused.value) {
 | ||
|             speechSynthesis.resume()
 | ||
|             isPaused.value = false
 | ||
|         }
 | ||
|     }
 | ||
| 
 | ||
|     const cancelAudio = () => {
 | ||
|         speechSynthesis.cancel()
 | ||
|         isSpeaking.value = false
 | ||
|         isPaused.value = false
 | ||
|     }
 | ||
|     // 页面刷新/关闭时
 | ||
|     onMounted(() => {
 | ||
|         if (typeof window !== 'undefined') {
 | ||
|             window.addEventListener('beforeunload', cancelAudio)
 | ||
|         }
 | ||
|     })
 | ||
| 
 | ||
|     onBeforeUnmount(() => {
 | ||
|         cancelAudio()
 | ||
|         if (typeof window !== 'undefined') {
 | ||
|             window.removeEventListener('beforeunload', cancelAudio)
 | ||
|         }
 | ||
|     })
 | ||
| 
 | ||
|     onHide(cancelAudio)
 | ||
|     onUnload(cancelAudio)
 | ||
| 
 | ||
|     return {
 | ||
|         speak,
 | ||
|         pause,
 | ||
|         resume,
 | ||
|         cancelAudio,
 | ||
|         isSpeaking,
 | ||
|         isPaused,
 | ||
|     }
 | ||
| }
 | ||
| 
 | ||
| function extractSpeechText(markdown) {
 | ||
|     const jobRegex = /``` job-json\s*({[\s\S]*?})\s*```/g;
 | ||
|     const jobs = [];
 | ||
|     let match;
 | ||
|     let lastJobEndIndex = 0;
 | ||
|     let firstJobStartIndex = -1;
 | ||
| 
 | ||
|     // 提取岗位 json 数据及前后位置
 | ||
|     while ((match = jobRegex.exec(markdown)) !== null) {
 | ||
|         const jobStr = match[1];
 | ||
|         try {
 | ||
|             const job = JSON.parse(jobStr);
 | ||
|             jobs.push(job);
 | ||
|             if (firstJobStartIndex === -1) {
 | ||
|                 firstJobStartIndex = match.index;
 | ||
|             }
 | ||
|             lastJobEndIndex = jobRegex.lastIndex;
 | ||
|         } catch (e) {
 | ||
|             console.warn('JSON 解析失败', e);
 | ||
|         }
 | ||
|     }
 | ||
| 
 | ||
|     // 提取引导语(第一个 job-json 之前的文字)
 | ||
|     const guideText = firstJobStartIndex > 0 ?
 | ||
|         markdown.slice(0, firstJobStartIndex).trim() :
 | ||
|         '';
 | ||
| 
 | ||
|     // 提取结束语(最后一个 job-json 之后的文字)
 | ||
|     const endingText = lastJobEndIndex < markdown.length ?
 | ||
|         markdown.slice(lastJobEndIndex).trim() :
 | ||
|         '';
 | ||
| 
 | ||
|     // 岗位信息格式化为语音文本
 | ||
|     const jobTexts = jobs.map((job, index) => {
 | ||
|         return `第 ${index + 1} 个岗位,岗位名称是:${job.jobTitle},公司是:${job.companyName},薪资:${job.salary},地点:${job.location},学历要求:${job.education},经验要求:${job.experience}。`;
 | ||
|     });
 | ||
| 
 | ||
|     // 拼接总语音内容
 | ||
|     const finalTextParts = [];
 | ||
|     if (guideText) finalTextParts.push(guideText);
 | ||
|     finalTextParts.push(...jobTexts);
 | ||
|     if (endingText) finalTextParts.push(endingText);
 | ||
| 
 | ||
|     return finalTextParts.join('\n');
 | ||
| } | 
