fmode-ng 0.0.111 → 0.0.113

This diff represents the content of publicly available package versions that have been released to one of the supported registries. The information contained in this diff is provided for informational purposes only and reflects changes between package versions as they appear in their respective public registries.
@@ -59,8 +59,8 @@ export class ModalChatVoiceInputComponent {
59
59
  // ASR - 唤醒词
60
60
  async startASRAwake() {
61
61
  await this.voiceServ.openWithPriviledge();
62
- let speech = new this.voiceServ.webSpeech();
63
- speech.startRecognition("Nihao | Hello");
62
+ // let speech = new this.voiceServ.webSpeech();
63
+ // speech.startRecognition("Nihao | Hello");
64
64
  }
65
65
  playMusic(action) {
66
66
  this.player.src = `/assets/avatar/voice/${action}.mp3`;
@@ -144,8 +144,8 @@ export class ModalChatVoiceInputComponent {
144
144
  testTTS(sentence) {
145
145
  console.log(sentence);
146
146
  sentence = sentence || "你好呀,我是飞马小智!很高兴为您介绍脑控科技的发展历程。我们成立于2019年";
147
- let speech = new this.voiceServ.webSpeech();
148
- speech.speak(sentence);
147
+ // let speech = new this.voiceServ.webSpeech();
148
+ // speech.speak(sentence)
149
149
  }
150
150
  testXunfeiTTS() {
151
151
  // this.voiceServ.ttsXunfei.connectWebSocket()
@@ -163,4 +163,4 @@ i0.ɵɵngDeclareClassMetadata({ minVersion: "12.0.0", version: "17.3.12", ngImpo
163
163
  }], talkMode: [{
164
164
  type: Input
165
165
  }] } });
166
- //# sourceMappingURL=data:application/json;base64,{"version":3,"file":"modal-chat-voice-input.component.js","sourceRoot":"","sources":["../../../../../../../projects/fmode-ng/src/lib/aigc/avatar/modal-chat-voice-input/modal-chat-voice-input.component.ts","../../../../../../../projects/fmode-ng/src/lib/aigc/avatar/modal-chat-voice-input/modal-chat-voice-input.component.html"],"names":[],"mappings":"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","sourcesContent":["import { Component,OnInit,Input } from '@angular/core';\r\nimport { CommonModule } from '@angular/common';\r\nimport { ToastController } from \"@ionic/angular\"\r\n\r\nimport { Router, RouterModule } from '@angular/router';\r\nimport { IonicModule, Platform } from \"@ionic/angular\"\r\n\r\nimport { FmodeVoiceService } from '../../voice';\r\nimport { FmodeChat,ChatService } from '../../service-fmai/service-chat';\r\nimport { NovaCloudService } from '../../../nova-cloud/nova-cloud.service';\r\nimport { Diagnostic } from '@awesome-cordova-plugins/diagnostic/ngx';\r\n\r\n\r\n@Component({\r\n  selector: 'fm-modal-chat-voice-input',\r\n  standalone: true,\r\n  imports: [CommonModule,\r\n    IonicModule,RouterModule,\r\n  ],\r\n  providers:[\r\n  ],\r\n  templateUrl: './modal-chat-voice-input.component.html',\r\n  styleUrls: ['./modal-chat-voice-input.component.scss']\r\n})\r\nexport class ModalChatVoiceInputComponent implements OnInit{\r\n\r\n  @Input() fmodeChat:FmodeChat|undefined\r\n\r\n   /**\r\n     * 开始讲话方式：click点击开始/点击结束 press按住讲话/松开结束\r\n     * @default click\r\n    */\r\n  @Input() talkMode:\"click\"|\"press\"=\"click\"\r\n  talkTips = \"点击话筒开始讲话\"\r\n\r\n  errorText:string = ``\r\n\r\n  voiceServ:FmodeVoiceService\r\n  \r\n  constructor(\r\n    private platform:Platform,\r\n    private router:Router,\r\n    private toastCtrl:ToastController,\r\n    private ncloud:NovaCloudService,\r\n    public chatServ:ChatService,\r\n    private diagnostic:Diagnostic\r\n    \r\n\r\n  ){\r\n    this.voiceServ = new FmodeVoiceService(this.platform,this.diagnostic);\r\n\r\n  }\r\n  ngOnInit(){\r\n    if(this.talkMode==\"press\"){\r\n      this.talkTips = \"轻触底部开始讲话\"\r\n    }\r\n\r\n    // 开启录音唤醒功能\r\n    setTimeout(() => {\r\n      this.initVoiceSevice()\r\n      this.initVoiceASR()\r\n    }, 500);\r\n  }\r\n\r\n  /**\r\n   * ASR唤醒功能\r\n   */\r\n  async initVoiceASR(){\r\n    // 开启录音权限\r\n    this.voiceServ.requestPermission().then(()=>{\r\n      this.voiceServ.openWithPriviledge()\r\n      // 开始监听唤醒词 Nihao Hello\r\n      this.startASRAwake()\r\n    })\r\n  }\r\n  // ASR - 唤醒词\r\n  async startASRAwake(){\r\n    await this.voiceServ.openWithPriviledge();\r\n    let speech = new this.voiceServ.webSpeech();\r\n    speech.startRecognition(\"Nihao | Hello\");\r\n  }\r\n\r\n  /**\r\n   * 音频提示音播放\r\n   */\r\n   player = new Audio();\r\n\r\n   playMusic(action){\r\n     this.player.src = `/assets/avatar/voice/${action}.mp3`\r\n     this.player.play();\r\n   }\r\n\r\n   /**\r\n     * 初始化录音转录服务\r\n     * @desc\r\n     * 根据数字角色对话状态，设计语音转录各事件处理过程\r\n     * 嵌入提示音\r\n     * 嵌入动画\r\n     * 嵌入唤醒词切换逻辑\r\n     */\r\n    initVoiceSevice(){\r\n      this.fmodeChat.userInput = this.voiceServ.resultText\r\n      this.voiceServ.requestPermission().then(()=>{\r\n        this.voiceServ.openWithPriviledge()\r\n      })\r\n\r\n      // 开始录音前 播放倾听动画\r\n      // 开始录音前 播放提示音\r\n      this.voiceServ.onBeforeStartTalk = ()=>{\r\n        this.fmodeChat.playAnimation(\"listening\")\r\n        this.playMusic(\"start-talk\")\r\n      }\r\n\r\n      // 用户取消录音前 播放提示音\r\n      this.voiceServ.onBeforeCancelTalk = ()=>{\r\n        this.playMusic(\"interupt-talk\")\r\n        this.fmodeChat.playAnimation(\"waiting\")\r\n      }\r\n      // 用户取消录音后 麦克风实时监听唤醒词\r\n      this.voiceServ.onAfterCancelTalk = ()=>{\r\n        this.startASRAwake() // 监听与麦克风冲突，需要重启\r\n      }\r\n      // 完成录音前 播放提示音\r\n      // 完成录音后 执行处理过程\r\n      this.voiceServ.onBeforeFinishTalk = ()=>{\r\n        this.fmodeChat.playAnimation(\"thinking\")\r\n        this.playMusic(\"stop-talk\")\r\n      }\r\n      this.voiceServ.onAfterFinishTalk = ()=>{\r\n        console.log(\"onAfterFinishTalk\")\r\n        this.fmodeChat.userInput = this.voiceServ?.resultText\r\n        this.sendMessage() // 发送消息\r\n        this.startASRAwake() // 监听与麦克风冲突，需要重启\r\n      }\r\n    }\r\n\r\n    /** \r\n     * 发送语音消息\r\n     * \r\n     */\r\n     async sendMessage(){\r\n\r\n      // 检测用户登录情况\r\n      // let isLoginLock = await this.authServ.checkLoginLock()\r\n      // if(!isLoginLock) return false\r\n  \r\n      // 检测余额及模型付费限制\r\n      // let payCheck = await this.checkBalance()\r\n      // if(!payCheck) return false\r\n  \r\n      // 检测用户输入内容空值\r\n      if(!this.fmodeChat.userInput){\r\n        this.errorText = `内容不能为空`\r\n        let toast = await this.toastCtrl.create({\r\n          message:this.errorText,\r\n          position:\"top\",\r\n          icon:'alert',\r\n          color:\"warning-circle\",\r\n          duration:1000\r\n        })\r\n        toast.present();\r\n        return\r\n      }\r\n  \r\n      // 正常发送消息\r\n      this.fmodeChat?.sendMessage(this.voiceServ.resultText,null,(msg:any)=>{\r\n        \r\n      },{\r\n        onSSMLComplete:(voice:any)=>{\r\n          console.log(voice)\r\n        }\r\n      });\r\n      this.fmodeChat.userInput = ``\r\n      this.fmodeChat.userImage = ``\r\n    }\r\n\r\n      testTTS(sentence?){\r\n        console.log(sentence)\r\n        sentence = sentence || \"你好呀，我是飞马小智！很高兴为您介绍脑控科技的发展历程。我们成立于2019年\"\r\n        let speech = new this.voiceServ.webSpeech();\r\n        speech.speak(sentence)\r\n      }\r\n      testXunfeiTTS(){\r\n        // this.voiceServ.ttsXunfei.connectWebSocket()\r\n      }\r\n}\r\n\r\n","    \r\n    <ng-container *ngIf=\"fmodeChat\">\r\n\r\n    <!-- 用户输入 提示区域 -->\r\n    <div class=\"user-asr-input\" style=\"text-align: center;\" *ngIf=\"!fmodeChat?.userInput && voiceServ.btnStatus!='OPEN'\">{{talkTips}}</div>\r\n    <div class=\"user-asr-input\">{{fmodeChat?.userInput}}</div>\r\n    \r\n    <!-- 测试按钮 -->\r\n    <div class=\"test-button-group\" *ngIf=\"false\">\r\n        <button class=\"button-record\" (click)=\"voiceServ.toggleRecord()\">开始录制 {{voiceServ.connStatus}} {{voiceServ.btnStatus}}</button>\r\n        <br>\r\n        <button class=\"button-record\" (click)=\"voiceServ.playRecord()\">播放录制结果</button>\r\n        <br>\r\n        <button class=\"button-record\" (click)=\"voiceServ.playBuffers()\">播放Buffers结果</button>\r\n        <button (click)=\"testTTS()\">测试TTS纯WEB</button>\r\n        <button (click)=\"startASR()\">测试ASR</button> \r\n        <button (click)=\"testXunfeiTTS()\">测试合成</button> \r\n    </div>\r\n\r\n    \r\n    \r\n    <!-- 交互按钮 -->\r\n    <ion-fab slot=\"fixed\" horizontal=\"center\" vertical=\"bottom\">\r\n        <ng-container *ngIf=\"talkMode=='click'\">\r\n          <!-- 默认按钮：开始讲话 -->\r\n          <ion-fab-button color=\"primary\" closeIcon=\"checkmark\" (click)=\"voiceServ.toggleRecord()\">\r\n            <ion-icon name=\"mic-outline\"></ion-icon>\r\n          </ion-fab-button>\r\n\r\n          <!-- 讲话中：取消发送 -->\r\n          <ion-fab-list side=\"end\">\r\n            <ion-fab-button [class]=\"'loading'\" (click)=\"voiceServ.cancelTalk()\">\r\n              <ion-icon name=\"pause-outline\"></ion-icon>\r\n            </ion-fab-button>\r\n          </ion-fab-list>\r\n        </ng-container>\r\n        \r\n        <ng-container *ngIf=\"talkMode=='press'\">\r\n          <ion-fab-button color=\"primary\" closeIcon=\"mic-outline\" (touchstart)=\"voiceServ.toggleRecord()\"  (touchend)=\"voiceServ.cancelTalk()\">\r\n            <ion-icon name=\"mic-outline\"></ion-icon>\r\n          </ion-fab-button>\r\n        </ng-container>\r\n    </ion-fab>\r\n\r\n\r\n    <!-- 音频波动 -->\r\n    <div class=\"record-wave\">\r\n    </div>\r\n</ng-container>\r\n"]}
166
+ //# sourceMappingURL=data:application/json;base64,{"version":3,"file":"modal-chat-voice-input.component.js","sourceRoot":"","sources":["../../../../../../../projects/fmode-ng/src/lib/aigc/avatar/modal-chat-voice-input/modal-chat-voice-input.component.ts","../../../../../../../projects/fmode-ng/src/lib/aigc/avatar/modal-chat-voice-input/modal-chat-voice-input.component.html"],"names":[],"mappings":"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","sourcesContent":["import { Component,OnInit,Input } from '@angular/core';\r\nimport { CommonModule } from '@angular/common';\r\nimport { ToastController } from \"@ionic/angular\"\r\n\r\nimport { Router, RouterModule } from '@angular/router';\r\nimport { IonicModule, Platform } from \"@ionic/angular\"\r\n\r\nimport { FmodeVoiceService } from '../../voice';\r\nimport { FmodeChat,ChatService } from '../../service-fmai/service-chat';\r\nimport { NovaCloudService } from '../../../nova-cloud/nova-cloud.service';\r\nimport { Diagnostic } from '@awesome-cordova-plugins/diagnostic/ngx';\r\n\r\n\r\n@Component({\r\n  selector: 'fm-modal-chat-voice-input',\r\n  standalone: true,\r\n  imports: [CommonModule,\r\n    IonicModule,RouterModule,\r\n  ],\r\n  providers:[\r\n  ],\r\n  templateUrl: './modal-chat-voice-input.component.html',\r\n  styleUrls: ['./modal-chat-voice-input.component.scss']\r\n})\r\nexport class ModalChatVoiceInputComponent implements OnInit{\r\n\r\n  @Input() fmodeChat:FmodeChat|undefined\r\n\r\n   /**\r\n     * 开始讲话方式：click点击开始/点击结束 press按住讲话/松开结束\r\n     * @default click\r\n    */\r\n  @Input() talkMode:\"click\"|\"press\"=\"click\"\r\n  talkTips = \"点击话筒开始讲话\"\r\n\r\n  errorText:string = ``\r\n\r\n  voiceServ:FmodeVoiceService\r\n  \r\n  constructor(\r\n    private platform:Platform,\r\n    private router:Router,\r\n    private toastCtrl:ToastController,\r\n    private ncloud:NovaCloudService,\r\n    public chatServ:ChatService,\r\n    private diagnostic:Diagnostic\r\n    \r\n\r\n  ){\r\n    this.voiceServ = new FmodeVoiceService(this.platform,this.diagnostic);\r\n\r\n  }\r\n  ngOnInit(){\r\n    if(this.talkMode==\"press\"){\r\n      this.talkTips = \"轻触底部开始讲话\"\r\n    }\r\n\r\n    // 开启录音唤醒功能\r\n    setTimeout(() => {\r\n      this.initVoiceSevice()\r\n      this.initVoiceASR()\r\n    }, 500);\r\n  }\r\n\r\n  /**\r\n   * ASR唤醒功能\r\n   */\r\n  async initVoiceASR(){\r\n    // 开启录音权限\r\n    this.voiceServ.requestPermission().then(()=>{\r\n      this.voiceServ.openWithPriviledge()\r\n      // 开始监听唤醒词 Nihao Hello\r\n      this.startASRAwake()\r\n    })\r\n  }\r\n  // ASR - 唤醒词\r\n  async startASRAwake(){\r\n    await this.voiceServ.openWithPriviledge();\r\n    // let speech = new this.voiceServ.webSpeech();\r\n    // speech.startRecognition(\"Nihao | Hello\");\r\n  }\r\n\r\n  /**\r\n   * 音频提示音播放\r\n   */\r\n   player = new Audio();\r\n\r\n   playMusic(action){\r\n     this.player.src = `/assets/avatar/voice/${action}.mp3`\r\n     this.player.play();\r\n   }\r\n\r\n   /**\r\n     * 初始化录音转录服务\r\n     * @desc\r\n     * 根据数字角色对话状态，设计语音转录各事件处理过程\r\n     * 嵌入提示音\r\n     * 嵌入动画\r\n     * 嵌入唤醒词切换逻辑\r\n     */\r\n    initVoiceSevice(){\r\n      this.fmodeChat.userInput = this.voiceServ.resultText\r\n      this.voiceServ.requestPermission().then(()=>{\r\n        this.voiceServ.openWithPriviledge()\r\n      })\r\n\r\n      // 开始录音前 播放倾听动画\r\n      // 开始录音前 播放提示音\r\n      this.voiceServ.onBeforeStartTalk = ()=>{\r\n        this.fmodeChat.playAnimation(\"listening\")\r\n        this.playMusic(\"start-talk\")\r\n      }\r\n\r\n      // 用户取消录音前 播放提示音\r\n      this.voiceServ.onBeforeCancelTalk = ()=>{\r\n        this.playMusic(\"interupt-talk\")\r\n        this.fmodeChat.playAnimation(\"waiting\")\r\n      }\r\n      // 用户取消录音后 麦克风实时监听唤醒词\r\n      this.voiceServ.onAfterCancelTalk = ()=>{\r\n        this.startASRAwake() // 监听与麦克风冲突，需要重启\r\n      }\r\n      // 完成录音前 播放提示音\r\n      // 完成录音后 执行处理过程\r\n      this.voiceServ.onBeforeFinishTalk = ()=>{\r\n        this.fmodeChat.playAnimation(\"thinking\")\r\n        this.playMusic(\"stop-talk\")\r\n      }\r\n      this.voiceServ.onAfterFinishTalk = ()=>{\r\n        console.log(\"onAfterFinishTalk\")\r\n        this.fmodeChat.userInput = this.voiceServ?.resultText\r\n        this.sendMessage() // 发送消息\r\n        this.startASRAwake() // 监听与麦克风冲突，需要重启\r\n      }\r\n    }\r\n\r\n    /** \r\n     * 发送语音消息\r\n     * \r\n     */\r\n     async sendMessage(){\r\n\r\n      // 检测用户登录情况\r\n      // let isLoginLock = await this.authServ.checkLoginLock()\r\n      // if(!isLoginLock) return false\r\n  \r\n      // 检测余额及模型付费限制\r\n      // let payCheck = await this.checkBalance()\r\n      // if(!payCheck) return false\r\n  \r\n      // 检测用户输入内容空值\r\n      if(!this.fmodeChat.userInput){\r\n        this.errorText = `内容不能为空`\r\n        let toast = await this.toastCtrl.create({\r\n          message:this.errorText,\r\n          position:\"top\",\r\n          icon:'alert',\r\n          color:\"warning-circle\",\r\n          duration:1000\r\n        })\r\n        toast.present();\r\n        return\r\n      }\r\n  \r\n      // 正常发送消息\r\n      this.fmodeChat?.sendMessage(this.voiceServ.resultText,null,(msg:any)=>{\r\n        \r\n      },{\r\n        onSSMLComplete:(voice:any)=>{\r\n          console.log(voice)\r\n        }\r\n      });\r\n      this.fmodeChat.userInput = ``\r\n      this.fmodeChat.userImage = ``\r\n    }\r\n\r\n      testTTS(sentence?){\r\n        console.log(sentence)\r\n        sentence = sentence || \"你好呀，我是飞马小智！很高兴为您介绍脑控科技的发展历程。我们成立于2019年\"\r\n        // let speech = new this.voiceServ.webSpeech();\r\n        // speech.speak(sentence)\r\n      }\r\n      testXunfeiTTS(){\r\n        // this.voiceServ.ttsXunfei.connectWebSocket()\r\n      }\r\n}\r\n\r\n","    \r\n    <ng-container *ngIf=\"fmodeChat\">\r\n\r\n    <!-- 用户输入 提示区域 -->\r\n    <div class=\"user-asr-input\" style=\"text-align: center;\" *ngIf=\"!fmodeChat?.userInput && voiceServ.btnStatus!='OPEN'\">{{talkTips}}</div>\r\n    <div class=\"user-asr-input\">{{fmodeChat?.userInput}}</div>\r\n    \r\n    <!-- 测试按钮 -->\r\n    <div class=\"test-button-group\" *ngIf=\"false\">\r\n        <button class=\"button-record\" (click)=\"voiceServ.toggleRecord()\">开始录制 {{voiceServ.connStatus}} {{voiceServ.btnStatus}}</button>\r\n        <br>\r\n        <button class=\"button-record\" (click)=\"voiceServ.playRecord()\">播放录制结果</button>\r\n        <br>\r\n        <button class=\"button-record\" (click)=\"voiceServ.playBuffers()\">播放Buffers结果</button>\r\n        <button (click)=\"testTTS()\">测试TTS纯WEB</button>\r\n        <button (click)=\"startASR()\">测试ASR</button> \r\n        <button (click)=\"testXunfeiTTS()\">测试合成</button> \r\n    </div>\r\n\r\n    \r\n    \r\n    <!-- 交互按钮 -->\r\n    <ion-fab slot=\"fixed\" horizontal=\"center\" vertical=\"bottom\">\r\n        <ng-container *ngIf=\"talkMode=='click'\">\r\n          <!-- 默认按钮：开始讲话 -->\r\n          <ion-fab-button color=\"primary\" closeIcon=\"checkmark\" (click)=\"voiceServ.toggleRecord()\">\r\n            <ion-icon name=\"mic-outline\"></ion-icon>\r\n          </ion-fab-button>\r\n\r\n          <!-- 讲话中：取消发送 -->\r\n          <ion-fab-list side=\"end\">\r\n            <ion-fab-button [class]=\"'loading'\" (click)=\"voiceServ.cancelTalk()\">\r\n              <ion-icon name=\"pause-outline\"></ion-icon>\r\n            </ion-fab-button>\r\n          </ion-fab-list>\r\n        </ng-container>\r\n        \r\n        <ng-container *ngIf=\"talkMode=='press'\">\r\n          <ion-fab-button color=\"primary\" closeIcon=\"mic-outline\" (touchstart)=\"voiceServ.toggleRecord()\"  (touchend)=\"voiceServ.cancelTalk()\">\r\n            <ion-icon name=\"mic-outline\"></ion-icon>\r\n          </ion-fab-button>\r\n        </ng-container>\r\n    </ion-fab>\r\n\r\n\r\n    <!-- 音频波动 -->\r\n    <div class=\"record-wave\">\r\n    </div>\r\n</ng-container>\r\n"]}
@@ -23,14 +23,13 @@ import 'recorder-core/src/extensions/waveview';
23
23
  import CryptoJS from "crypto-js";
24
24
  import { pcmtoWav } from './lib/pcm2wav';
25
25
  import { convertFrameBufferToBase64, resampleBuffer } from './lib/resample';
26
- import { WebSpeech } from './class-asr';
27
26
  // import { FmodeTTSXunfei } fro./class-tts-xunfei.ts.bakfei'
28
27
  export class FmodeVoiceService {
29
28
  constructor(platform, diagnostic) {
30
29
  this.platform = platform;
31
30
  this.diagnostic = diagnostic;
32
31
  this.disableASR = false;
33
- this.webSpeech = WebSpeech;
32
+ // webSpeech = WebSpeech;
34
33
  // 状态管理
35
34
  this.isRecording = false;
36
35
  this.isUserFinish = false;
@@ -558,6 +557,8 @@ export class FmodeVoiceService {
558
557
  }
559
558
  // 移动端权限方法
560
559
  isCapacitor() {
560
+ if (!this.platform?.is)
561
+ return false;
561
562
  return this.platform.is("capacitor") || this.platform.is("cordova");
562
563
  }
563
564
  async requestPermission() {
@@ -632,4 +633,4 @@ export class FmodeVoiceService {
632
633
  return new Blob([audioBuffer], { type: 'audio/pcm' });
633
634
  }
634
635
  }
635
- //# sourceMappingURL=data:application/json;base64,{"version":3,"file":"fmode-voice.service.js","sourceRoot":"","sources":["../../../../../../projects/fmode-ng/src/lib/aigc/voice/fmode-voice.service.ts"],"names":[],"mappings":"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","sourcesContent":["// import RecorderManager from \"./lib/xunfei-recorder\"\r\n// import {RecorderManager} from \"./lib/recorder/recorder-manager\"\r\nimport Recorder from 'recorder-core'\r\n\r\n// 检测浏览器支持情况\r\nfunction checkWorkletSupport() {\r\n  try {\r\n    return (\r\n      window.AudioWorkletNode &&\r\n      window.AudioContext && \r\n      window.AudioContext.audioWorklet &&\r\n      !/Android|webOS|iPhone|iPad|iPod|BlackBerry|IEMobile|Opera Mini/i.test(navigator.userAgent)\r\n    );\r\n  } catch (e) {\r\n    return false;\r\n  }\r\n}\r\nRecorder.ConnectEnableWorklet = checkWorkletSupport(); // 明确禁用AudioWorklet 兼容不支持的场景\r\n\r\nimport 'recorder-core/src/engine/pcm'\r\nimport 'recorder-core/src/engine/wav'\r\nimport 'recorder-core/src/extensions/waveview'\r\n// import './lib/recorder/engine-pcm'\r\n// import './lib/recorder/engine-wav'\r\n// import './lib/recorder/extension-waveview'\r\nimport CryptoJS from \"crypto-js\"\r\nimport { pcmtoWav } from './lib/pcm2wav';\r\nimport { convertFrameBufferToBase64, resampleAudio, resampleBuffer } from './lib/resample';\r\nimport { WebSpeech } from './class-asr';\r\nimport { Platform } from '@ionic/angular';\r\n// import { Diagnostic } from '@awesome-cordova-plugins/diagnostic/ngx';\r\nimport { Diagnostic } from '@awesome-cordova-plugins/diagnostic/ngx';\r\n// import { FmodeTTSXunfei } fro./class-tts-xunfei.ts.bakfei'\r\n\r\nexport class FmodeVoiceService {\r\n  disableASR: boolean = false;\r\n  webSpeech = WebSpeech;\r\n\r\n  // 状态管理\r\n  private isRecording: boolean = false;\r\n  private isUserFinish: boolean = false;\r\n  private shouldReconnect: boolean = true;\r\n  private reconnectAttempts: number = 0;\r\n  private maxReconnectAttempts: number = 3;\r\n  private reconnectDelay: number = 1000;\r\n\r\n  // 录制相关 - 独立于WebSocket\r\n  recordWavBlob: Blob = null;\r\n  recordPcmBlob: Blob = null;\r\n  recordDuration: number = 0;\r\n  private allRecordedBuffers: Array<any> = []; // 存储所有录制的buffer\r\n  private currentRecorder: any = null; // 当前的录制器实例\r\n  \r\n  // WebSocket相关\r\n  iatWS: WebSocket;\r\n  btnStatus = \"UNDEFINED\";\r\n  connStatus = \"\";\r\n  \r\n  // 结果相关\r\n  resultText = \"\";\r\n  resultTextTemp = \"\";\r\n  \r\n  // 计时相关\r\n  countdownInterval: any;\r\n  durationInterval: any;\r\n  durationStr: string = \"00:00\";\r\n  duration: number = 0;\r\n  now: Date;\r\n  \r\n  // 波形相关\r\n  waveClient: any;\r\n  recordType = \"pcm\";\r\n  encodingType = \"raw\";\r\n\r\n  // 回调函数\r\n  onResultTextChanged: Function;\r\n  onBeforeFinishTalk: Function;\r\n  onAfterFinishTalk: Function;\r\n  onInputChange: Function;\r\n  onBeforeStartTalk: Function;\r\n  onAfterStartTalk: Function;\r\n  onBeforeCancelTalk: Function;\r\n  onAfterCancelTalk: Function;\r\n  onAfterRecordStart: Function;\r\n  onDurationStrChange: Function;\r\n\r\n  // API配置\r\n  APPID = \"50f4a46c\";\r\n  API_SECRET = \"NzFlNmFhZDJjMDNkZGM3NzI0Mzg2OGNm\";\r\n  API_KEY = \"106ddc40dfd4b9ca6d7b47c70fada749\";\r\n\r\n  constructor(\r\n    private platform: Platform,\r\n    private diagnostic: Diagnostic,\r\n  ) {\r\n    this.requestPermission();\r\n  }\r\n\r\n  // 用户操作方法\r\n  toggleRecord() {\r\n    console.log('toggleRecord', this.btnStatus);\r\n    if (this.btnStatus === \"UNDEFINED\" || this.btnStatus === \"CLOSED\") {\r\n      this.startTalk();\r\n    } else if (this.btnStatus === \"CONNECTING\" || this.btnStatus === \"OPEN\") {\r\n      this.finishTalk();\r\n    }\r\n  }\r\n\r\n  async startTalk(event?) {\r\n    console.log('startTalk called');\r\n    this.resetState();\r\n    this.onBeforeStartTalk && this.onBeforeStartTalk();\r\n    \r\n    event?.preventDefault();\r\n    \r\n    try {\r\n      await this.openWithPriviledge();\r\n      \r\n      // 1. 首先启动独立录制\r\n      await this.startIndependentRecording();\r\n      \r\n      // 2. 然后启动WebSocket连接\r\n      this.isRecording = true;\r\n      this.shouldReconnect = true;\r\n      this.reconnectAttempts = 0;\r\n      \r\n      setTimeout(() => {\r\n        this.connectWebSocket();\r\n      }, 100);\r\n\r\n      this.startCountdown();\r\n      this.onAfterStartTalk && this.onAfterStartTalk();\r\n    } catch (error) {\r\n      console.error('Failed to start talk:', error);\r\n      this.resetState();\r\n    }\r\n  }\r\n\r\n  async finishTalk() {\r\n    console.log('finishTalk called');\r\n    this.isUserFinish = true;\r\n    this.shouldReconnect = false;\r\n    this.onBeforeFinishTalk && this.onBeforeFinishTalk();\r\n    \r\n    try {\r\n      // 1. 先关闭WebSocket\r\n      this.closeWebSocket();\r\n      \r\n      // 2. 停止独立录制并获取完整音频\r\n      await this.stopIndependentRecording();\r\n      \r\n      // 3. 清理状态\r\n      this.isRecording = false;\r\n      this.clearTimers();\r\n      this.changeBtnStatus(\"CLOSED\");\r\n      \r\n      // 4. 延迟回调确保所有处理完成\r\n      setTimeout(() => {\r\n        if (this.isUserFinish) {\r\n          this.onAfterFinishTalk && this.onAfterFinishTalk();\r\n          this.isUserFinish = false;\r\n        }\r\n      }, 500);\r\n      \r\n    } catch (error) {\r\n      console.error('Error finishing talk:', error);\r\n    }\r\n  }\r\n\r\n  cancelTalk() {\r\n    console.log('cancelTalk called');\r\n    this.onBeforeCancelTalk && this.onBeforeCancelTalk();\r\n    \r\n    this.isUserFinish = false;\r\n    this.shouldReconnect = false;\r\n    this.isRecording = false;\r\n    \r\n    // 关闭WebSocket\r\n    this.closeWebSocket();\r\n    \r\n    // 停止独立录制但不保存结果\r\n    this.cancelIndependentRecording();\r\n    \r\n    this.resetState();\r\n    this.onAfterCancelTalk && this.onAfterCancelTalk();\r\n  }\r\n\r\n  // 独立录制方法 - 与WebSocket完全解耦\r\n  async startIndependentRecording(): Promise<void> {\r\n    console.log('Starting independent recording');\r\n    \r\n    this.allRecordedBuffers = [];\r\n    this.createIndependentRecorder();\r\n    \r\n    return new Promise((resolve, reject) => {\r\n      this.currentRecorder.open(() => {\r\n        console.log('Independent recorder opened');\r\n        \r\n        // 创建可视化\r\n        let waveDiv = document.querySelector(\".record-wave\");\r\n        if (waveDiv && Recorder.WaveView) {\r\n          this.waveClient = Recorder.WaveView({ elem: \".record-wave\" });\r\n        }\r\n        \r\n        this.currentRecorder.start();\r\n        this.onAfterRecordStart && this.onAfterRecordStart();\r\n        resolve();\r\n      }, (msg, isUserNotAllow) => {\r\n        console.error('Failed to open independent recorder:', msg);\r\n        reject(new Error(msg));\r\n      });\r\n    });\r\n  }\r\n\r\n  async stopIndependentRecording(): Promise<void> {\r\n    console.log('Stopping independent recording');\r\n    \r\n    if (!this.currentRecorder) {\r\n      console.log('No independent recorder to stop');\r\n      return;\r\n    }\r\n\r\n    return new Promise((resolve) => {\r\n      this.currentRecorder.stop(\r\n        async (blob: Blob, duration: number) => {\r\n          console.log('Independent recording stopped successfully', blob, duration);\r\n          \r\n          // 保存完整的录制结果\r\n          this.recordPcmBlob = blob;\r\n          this.recordWavBlob = await this.pcmBlobToWavBlob(blob, 44100);\r\n          \r\n          // 清理录制器\r\n          this.cleanupIndependentRecorder();\r\n          resolve();\r\n        },\r\n        (msg) => {\r\n          console.error(\"独立录音停止失败:\" + msg);\r\n          this.cleanupIndependentRecorder();\r\n          resolve();\r\n        }\r\n      );\r\n    });\r\n  }\r\n\r\n  cancelIndependentRecording(): void {\r\n    console.log('Cancelling independent recording');\r\n    \r\n    if (this.currentRecorder) {\r\n      try {\r\n        // 直接关闭，不保存结果\r\n        this.currentRecorder.close();\r\n      } catch (error) {\r\n        console.error('Error closing independent recorder:', error);\r\n      }\r\n    }\r\n    \r\n    this.cleanupIndependentRecorder();\r\n    this.recordPcmBlob = null;\r\n    this.recordWavBlob = null;\r\n  }\r\n\r\n  private createIndependentRecorder(): void {\r\n    if (this.currentRecorder) {\r\n      this.cleanupIndependentRecorder();\r\n    }\r\n\r\n    this.currentRecorder = Recorder({\r\n      type: this.recordType,\r\n      sampleRate: 44100,\r\n      bitRate: 16,\r\n      onProcess: (buffers, powerLevel, bufferDuration, bufferSampleRate, newBufferIdx, asyncEnd) => {\r\n        if (!this.isRecording) return;\r\n\r\n        // 保存所有buffer用于最终合成完整音频\r\n        this.allRecordedBuffers = buffers;\r\n\r\n        // 获取最新的音频片段用于实时ASR\r\n        let frameBuffer = buffers.length && buffers[buffers.length - 1];\r\n        if (!frameBuffer) return;\r\n\r\n        // 重采样到16kHz用于ASR\r\n        frameBuffer = resampleBuffer(frameBuffer, 44100, 16000);\r\n\r\n        // 发送到WebSocket进行实时ASR\r\n        this.sendAudioToWebSocket(frameBuffer);\r\n\r\n        // 更新波形显示\r\n        this.waveClient?.input(buffers[buffers.length - 1], powerLevel, bufferSampleRate);\r\n      }\r\n    });\r\n  }\r\n\r\n  private cleanupIndependentRecorder(): void {\r\n    if (this.currentRecorder) {\r\n      try {\r\n        this.currentRecorder.close();\r\n      } catch (error) {\r\n        console.error('Error cleaning up independent recorder:', error);\r\n      }\r\n      this.currentRecorder = null;\r\n    }\r\n  }\r\n\r\n  private sendAudioToWebSocket(frameBuffer: any): void {\r\n    if (this.iatWS && this.iatWS.readyState === WebSocket.OPEN && !this.disableASR) {\r\n      try {\r\n        this.iatWS.send(JSON.stringify({\r\n          data: {\r\n            status: 1,\r\n            format: \"audio/L16;rate=16000\",\r\n            encoding: this.encodingType,\r\n            audio: convertFrameBufferToBase64(frameBuffer)\r\n          },\r\n        }));\r\n      } catch (error) {\r\n        console.error(\"Error sending audio data to WebSocket:\", error);\r\n      }\r\n    }\r\n  }\r\n\r\n  // WebSocket相关方法 - 仅用于ASR，不控制录制\r\n  connectWebSocket() {\r\n    console.log(\"connectWebSocket called, shouldReconnect:\", this.shouldReconnect);\r\n    \r\n    if (!this.shouldReconnect || !this.isRecording) {\r\n      console.log(\"Not connecting WebSocket - shouldReconnect:\", this.shouldReconnect, \"isRecording:\", this.isRecording);\r\n      return;\r\n    }\r\n\r\n    // 关闭现有连接\r\n    this.closeWebSocket();\r\n\r\n    const websocketUrl = this.getWebSocketUrl();\r\n    console.log(\"Connecting to:\", websocketUrl);\r\n\r\n    try {\r\n      if (\"WebSocket\" in window) {\r\n        this.iatWS = new WebSocket(websocketUrl);\r\n      } else {\r\n        console.error(\"浏览器不支持WebSocket\");\r\n        alert(\"浏览器不支持WebSocket\");\r\n        return;\r\n      }\r\n\r\n      this.changeBtnStatus(\"CONNECTING\");\r\n\r\n      this.iatWS.onopen = (e) => {\r\n        console.log(\"WebSocket connected for ASR\");\r\n        this.reconnectAttempts = 0;\r\n        this.changeBtnStatus(\"OPEN\");\r\n\r\n        // 发送初始参数\r\n        const params = {\r\n          common: {\r\n            app_id: this.APPID,\r\n          },\r\n          business: {\r\n            language: \"zh_cn\",\r\n            domain: \"iat\",\r\n            accent: \"mandarin\",\r\n            vad_eos: 10000, // 增加到10秒\r\n            dwa: \"wpgs\",\r\n          },\r\n          data: {\r\n            status: 0,\r\n            format: \"audio/L16;rate=16000\",\r\n            encoding: this.encodingType,\r\n          },\r\n        };\r\n        this.iatWS.send(JSON.stringify(params));\r\n      };\r\n\r\n      this.iatWS.onmessage = (e) => {\r\n        this.renderResult(e.data);\r\n      };\r\n\r\n      this.iatWS.onerror = (e) => {\r\n        console.error(\"WebSocket error:\", e);\r\n        this.handleWebSocketError();\r\n      };\r\n\r\n      this.iatWS.onclose = (e) => {\r\n        console.log(\"WebSocket closed:\", e.code, e.reason);\r\n        this.handleWebSocketClose(e);\r\n      };\r\n\r\n    } catch (error) {\r\n      console.error(\"Failed to create WebSocket:\", error);\r\n      this.handleWebSocketError();\r\n    }\r\n  }\r\n\r\n  private handleWebSocketClose(event: CloseEvent) {\r\n    console.log(\"handleWebSocketClose\", {\r\n      code: event.code,\r\n      reason: event.reason,\r\n      shouldReconnect: this.shouldReconnect,\r\n      isRecording: this.isRecording,\r\n      isUserFinish: this.isUserFinish\r\n    });\r\n\r\n    // 如果用户已经结束或不应该重连，则不处理\r\n    if (this.isUserFinish || !this.shouldReconnect || !this.isRecording) {\r\n      return;\r\n    }\r\n\r\n    // 发送结束帧（如果需要）\r\n    try {\r\n      if (this.iatWS && this.iatWS.readyState === WebSocket.OPEN) {\r\n        this.iatWS.send(JSON.stringify({\r\n          \"data\": {\r\n            \"status\": 2,\r\n            \"format\": \"audio/L16;rate=16000\",\r\n            \"encoding\": this.encodingType,\r\n            \"audio\": \"\"\r\n          }\r\n        }));\r\n      }\r\n    } catch (err) {\r\n      console.error('Error sending end frame on close:', err);\r\n    }\r\n\r\n    // 正常关闭码或服务端主动关闭，尝试重连\r\n    if (event.code === 1000 || event.code === 1006 || event.code === 1011) {\r\n      this.attemptReconnect();\r\n    } else {\r\n      console.error(\"WebSocket closed with error code:\", event.code);\r\n      this.handleWebSocketError();\r\n    }\r\n  }\r\n\r\n  private handleWebSocketError() {\r\n    if (!this.shouldReconnect || !this.isRecording) {\r\n      return;\r\n    }\r\n    this.attemptReconnect();\r\n  }\r\n\r\n  private attemptReconnect() {\r\n    if (this.reconnectAttempts >= this.maxReconnectAttempts) {\r\n      console.error(\"Max reconnect attempts reached\");\r\n      // 不改变录制状态，只是ASR暂时不可用\r\n      return;\r\n    }\r\n\r\n    this.reconnectAttempts++;\r\n    console.log(`Attempting WebSocket reconnect ${this.reconnectAttempts}/${this.maxReconnectAttempts}`);\r\n\r\n    setTimeout(() => {\r\n      if (this.shouldReconnect && this.isRecording && !this.isUserFinish) {\r\n        this.connectWebSocket();\r\n      }\r\n    }, this.reconnectDelay * this.reconnectAttempts);\r\n  }\r\n\r\n  private closeWebSocket() {\r\n    if (this.iatWS) {\r\n      try {\r\n        // 发送结束帧\r\n        if (this.iatWS.readyState === WebSocket.OPEN) {\r\n          this.iatWS.send(JSON.stringify({\r\n            \"data\": {\r\n              \"status\": 2,\r\n              \"format\": \"audio/L16;rate=16000\",\r\n              \"encoding\": this.encodingType,\r\n              \"audio\": \"\"\r\n            }\r\n          }));\r\n        }\r\n        \r\n        if (this.iatWS.readyState === WebSocket.OPEN || this.iatWS.readyState === WebSocket.CONNECTING) {\r\n          this.iatWS.close(1000, \"User initiated close\");\r\n        }\r\n      } catch (error) {\r\n        console.error(\"Error closing WebSocket:\", error);\r\n      }\r\n      this.iatWS = null;\r\n    }\r\n  }\r\n\r\n  // 状态管理\r\n  private resetState() {\r\n    this.resultText = \"\";\r\n    this.resultTextTemp = \"\";\r\n    this.durationStr = \"00:00\";\r\n    this.duration = 0;\r\n    this.recordDuration = 0;\r\n    this.allRecordedBuffers = [];\r\n    this.clearTimers();\r\n  }\r\n\r\n  private clearTimers() {\r\n    if (this.countdownInterval) {\r\n      clearInterval(this.countdownInterval);\r\n      this.countdownInterval = null;\r\n    }\r\n    if (this.durationInterval) {\r\n      clearInterval(this.durationInterval);\r\n      this.durationInterval = null;\r\n    }\r\n  }\r\n\r\n  changeBtnStatus(status: string) {\r\n    this.btnStatus = status;\r\n    switch (status) {\r\n      case \"CONNECTING\":\r\n        this.connStatus = \"建立连接中\";\r\n        break;\r\n      case \"OPEN\":\r\n        this.connStatus = `录音中（${this.durationStr}）`;\r\n        break;\r\n      case \"CLOSING\":\r\n        this.connStatus = \"关闭连接中\";\r\n        break;\r\n      case \"CLOSED\":\r\n        this.connStatus = \"开始录音\";\r\n        break;\r\n    }\r\n  }\r\n\r\n  // 计时器\r\n  startCountdown() {\r\n    this.clearTimers();\r\n    this.recordDuration = 0;\r\n    this.duration = 0;\r\n    this.now = new Date();\r\n\r\n    // 录音时长计时器（100ms精度）\r\n    this.durationInterval = setInterval(() => {\r\n      if (this.isRecording) {\r\n        this.recordDuration += 100;\r\n      }\r\n    }, 100);\r\n\r\n    // 显示时长计时器（1秒精度）\r\n    this.countdownInterval = setInterval(() => {\r\n      if (this.isRecording) {\r\n        this.countTimer();\r\n      }\r\n    }, 1000);\r\n  }\r\n\r\n  countTimer() {\r\n    this.duration++;\r\n    let minuteStr = String(Math.floor(this.duration / 60)).padStart(2, \"0\");\r\n    let secondStr = String(this.duration % 60).padStart(2, \"0\");\r\n    let durationStr = minuteStr + \":\" + secondStr;\r\n    this.durationStr = durationStr;\r\n    this.connStatus = `录音中（${this.durationStr}）`;\r\n    this.onDurationStrChange && this.onDurationStrChange(durationStr);\r\n  }\r\n\r\n  // 结果处理\r\n  renderResult(resultData: string) {\r\n    try {\r\n      let jsonData = JSON.parse(resultData);\r\n      \r\n      if (jsonData.data && jsonData.data.result) {\r\n        let data = jsonData.data.result;\r\n        let str = \"\";\r\n        let ws = data.ws;\r\n        \r\n        for (let i = 0; i < ws.length; i++) {\r\n          str = str + ws[i].cw[0].w;\r\n        }\r\n\r\n        // 处理动态修正\r\n        if (data.pgs) {\r\n          if (data.pgs === \"apd\") {\r\n            this.resultText = this.resultTextTemp;\r\n          }\r\n          this.resultTextTemp = this.resultText + str;\r\n        } else {\r\n          this.resultText = this.resultText + str;\r\n        }\r\n\r\n        this.onInputChange && this.onInputChange(this.getUserInput());\r\n      }\r\n\r\n      // 处理结束状态\r\n      if (jsonData.code === 0 && jsonData.data.status === 2) {\r\n        console.log(\"ASR session completed, will reconnect if still recording\");\r\n        if (this.iatWS && this.iatWS.readyState === WebSocket.OPEN) {\r\n          this.iatWS.close();\r\n        }\r\n      }\r\n\r\n      // 处理错误\r\n      if (jsonData.code !== 0) {\r\n        console.error(\"ASR error:\", jsonData);\r\n        if (this.iatWS && this.iatWS.readyState === WebSocket.OPEN) {\r\n          this.iatWS.close();\r\n        }\r\n      }\r\n    } catch (error) {\r\n      console.error(\"Error parsing result:\", error);\r\n    }\r\n  }\r\n\r\n  getUserInput(): string {\r\n    return \"\" + (this.resultTextTemp || this.resultText);\r\n  }\r\n\r\n  // 权限和初始化\r\n  async openWithPriviledge(): Promise<boolean> {\r\n    await this.requestPermission();\r\n    \r\n    if (Recorder.IsOpen()) return true;\r\n\r\n    // 这里不创建录制器，因为我们使用独立的录制器\r\n    return true;\r\n  }\r\n\r\n  // 音频处理方法\r\n  async pcmBlobToWavBlob(pcmBlob: Blob, sampleRate: number): Promise<Blob> {\r\n    return new Promise(resolve => {\r\n      let fileReader = new FileReader();\r\n      \r\n      fileReader.onload = function(event) {\r\n        let pcmData: any = event.target.result;\r\n        let wavBlob = pcmtoWav(pcmData, sampleRate, 1, 16);\r\n        resolve(wavBlob);\r\n      };\r\n      \r\n      fileReader.readAsArrayBuffer(pcmBlob);\r\n    });\r\n  }\r\n\r\n  async playPCM(pcmBlob: Blob, sampleRate: number) {\r\n    let wavBlob = await this.pcmBlobToWavBlob(pcmBlob, sampleRate);\r\n    let wavUrl = window.URL.createObjectURL(wavBlob);\r\n    let audio = new Audio();\r\n    audio.src = wavUrl;\r\n    audio.play();\r\n  }\r\n\r\n  playRecord() {\r\n    if (this.recordPcmBlob) {\r\n      this.playPCM(this.recordPcmBlob, 44100);\r\n    }\r\n  }\r\n\r\n  // WebSocket URL生成\r\n  getWebSocketUrl(): string {\r\n    let url = \"wss://iat-api.xfyun.cn/v2/iat\";\r\n    let host = \"iat-api.xfyun.cn\";\r\n    \r\n    let apiKey = this.API_KEY;\r\n    let apiSecret = this.API_SECRET;\r\n    let date = new Date().toUTCString();\r\n    let algorithm = \"hmac-sha256\";\r\n    let headers = \"host date request-line\";\r\n    let signatureOrigin = `host: ${host}\\ndate: ${date}\\nGET /v2/iat HTTP/1.1`;\r\n    let signatureSha = CryptoJS.HmacSHA256(signatureOrigin, apiSecret);\r\n    let signature = CryptoJS.enc.Base64.stringify(signatureSha);\r\n    let authorizationOrigin = `api_key=\"${apiKey}\", algorithm=\"${algorithm}\", headers=\"${headers}\", signature=\"${signature}\"`;\r\n    let authorization = btoa(authorizationOrigin);\r\n    url = `${url}?authorization=${authorization}&date=${date}&host=${host}`;\r\n    return url;\r\n  }\r\n\r\n  toBase64(buffer: ArrayBuffer): string {\r\n    var binary = \"\";\r\n    var bytes = new Uint8Array(buffer);\r\n    var len = bytes.byteLength;\r\n    for (var i = 0; i < len; i++) {\r\n      binary += String.fromCharCode(bytes[i]);\r\n    }\r\n    return window.btoa(binary);\r\n  }\r\n\r\n  // 移动端权限方法\r\n  isCapacitor(): boolean {\r\n    return this.platform.is(\"capacitor\") || this.platform.is(\"cordova\");\r\n  }\r\n\r\n  async requestPermission() {\r\n    if (!this.isCapacitor()) return;\r\n    \r\n    try {\r\n      await this.requestStoagePermission();\r\n      await this.requestCameraPermission();\r\n      await this.requestMicPermission();\r\n      await this.requestRecordAudioPermission();\r\n    } catch (err) {\r\n      console.error(err);\r\n    }\r\n  }\r\n\r\n  async requestRecordAudioPermission() {let data = await this.diagnostic.requestRuntimePermissions([this.diagnostic.permission.RECORD_AUDIO]);\r\n    console.log(\"record permission request:\", data);\r\n    return;\r\n  }\r\n\r\n  async requestMicPermission() {\r\n    let isAvailable = await this.diagnostic.isMicrophoneAuthorized();\r\n    console.log(\"permisson_MIC:\", isAvailable);\r\n    if (!isAvailable) {\r\n      let data = await this.diagnostic.requestMicrophoneAuthorization();\r\n    }\r\n    return;\r\n  }\r\n\r\n  async requestStoagePermission() {\r\n    let isAvailable = await this.diagnostic.isExternalStorageAuthorized();\r\n    console.log(\"permisson_STORAGE:\", isAvailable);\r\n    if (!isAvailable) {\r\n      let data = await this.diagnostic.requestExternalStorageAuthorization();\r\n    }\r\n    return;\r\n  }\r\n\r\n  async requestCameraPermission() {\r\n    let isAvailable = await this.diagnostic.isCameraAuthorized();\r\n    console.log(\"permisson_Camera:\", isAvailable);\r\n    if (!isAvailable) {\r\n      let data = await this.diagnostic.requestCameraAuthorization();\r\n    }\r\n    return;\r\n  }\r\n\r\n  // 其他辅助方法\r\n  splitAudioData(audioData: any) {\r\n    const segmentSize = 1280;\r\n    const segmentCount = Math.ceil(audioData.length / segmentSize);\r\n    const segments = [];\r\n\r\n    for (let i = 0; i < segmentCount; i++) {\r\n      const start = i * segmentSize;\r\n      const end = start + segmentSize;\r\n      const segment = audioData.slice(start, end);\r\n      segments.push(segment);\r\n    }\r\n\r\n    return segments;\r\n  }\r\n\r\n  BufferToBlob(buffer: any) {\r\n    return new Blob([buffer], { type: 'audio/pcm' });\r\n  }\r\n\r\n  async playBuffers() {\r\n    let audioBlob = await this.BuffersToBlob(this.allRecordedBuffers);\r\n    this.playPCM(audioBlob, 44100);\r\n  }\r\n\r\n  BuffersToBlob(buffers: Array<any>) {\r\n    let audioBuffer: any = [];\r\n    buffers.forEach(buffer => {\r\n      buffer.forEach(int16 => {\r\n        audioBuffer.push(int16);\r\n      });\r\n    });\r\n    return new Blob([audioBuffer], { type: 'audio/pcm' });\r\n  }\r\n}"]}
636
+ //# sourceMappingURL=data:application/json;base64,{"version":3,"file":"fmode-voice.service.js","sourceRoot":"","sources":["../../../../../../projects/fmode-ng/src/lib/aigc/voice/fmode-voice.service.ts"],"names":[],"mappings":"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","sourcesContent":["// import RecorderManager from \"./lib/xunfei-recorder\"\r\n// import {RecorderManager} from \"./lib/recorder/recorder-manager\"\r\nimport Recorder from 'recorder-core'\r\n\r\n// 检测浏览器支持情况\r\nfunction checkWorkletSupport() {\r\n  try {\r\n    return (\r\n      window.AudioWorkletNode &&\r\n      window.AudioContext && \r\n      window.AudioContext.audioWorklet &&\r\n      !/Android|webOS|iPhone|iPad|iPod|BlackBerry|IEMobile|Opera Mini/i.test(navigator.userAgent)\r\n    );\r\n  } catch (e) {\r\n    return false;\r\n  }\r\n}\r\nRecorder.ConnectEnableWorklet = checkWorkletSupport(); // 明确禁用AudioWorklet 兼容不支持的场景\r\n\r\nimport 'recorder-core/src/engine/pcm'\r\nimport 'recorder-core/src/engine/wav'\r\nimport 'recorder-core/src/extensions/waveview'\r\n// import './lib/recorder/engine-pcm'\r\n// import './lib/recorder/engine-wav'\r\n// import './lib/recorder/extension-waveview'\r\nimport CryptoJS from \"crypto-js\"\r\nimport { pcmtoWav } from './lib/pcm2wav';\r\nimport { convertFrameBufferToBase64, resampleAudio, resampleBuffer } from './lib/resample';\r\n// import { WebSpeech } from './class-asr';\r\nimport { Platform } from '@ionic/angular';\r\n// import { Diagnostic } from '@awesome-cordova-plugins/diagnostic/ngx';\r\nimport { Diagnostic } from '@awesome-cordova-plugins/diagnostic/ngx';\r\n// import { FmodeTTSXunfei } fro./class-tts-xunfei.ts.bakfei'\r\n\r\nexport class FmodeVoiceService {\r\n  disableASR: boolean = false;\r\n  // webSpeech = WebSpeech;\r\n\r\n  // 状态管理\r\n  private isRecording: boolean = false;\r\n  private isUserFinish: boolean = false;\r\n  private shouldReconnect: boolean = true;\r\n  private reconnectAttempts: number = 0;\r\n  private maxReconnectAttempts: number = 3;\r\n  private reconnectDelay: number = 1000;\r\n\r\n  // 录制相关 - 独立于WebSocket\r\n  recordWavBlob: Blob = null;\r\n  recordPcmBlob: Blob = null;\r\n  recordDuration: number = 0;\r\n  private allRecordedBuffers: Array<any> = []; // 存储所有录制的buffer\r\n  private currentRecorder: any = null; // 当前的录制器实例\r\n  \r\n  // WebSocket相关\r\n  iatWS: WebSocket;\r\n  btnStatus = \"UNDEFINED\";\r\n  connStatus = \"\";\r\n  \r\n  // 结果相关\r\n  resultText = \"\";\r\n  resultTextTemp = \"\";\r\n  \r\n  // 计时相关\r\n  countdownInterval: any;\r\n  durationInterval: any;\r\n  durationStr: string = \"00:00\";\r\n  duration: number = 0;\r\n  now: Date;\r\n  \r\n  // 波形相关\r\n  waveClient: any;\r\n  recordType = \"pcm\";\r\n  encodingType = \"raw\";\r\n\r\n  // 回调函数\r\n  onResultTextChanged: Function;\r\n  onBeforeFinishTalk: Function;\r\n  onAfterFinishTalk: Function;\r\n  onInputChange: Function;\r\n  onBeforeStartTalk: Function;\r\n  onAfterStartTalk: Function;\r\n  onBeforeCancelTalk: Function;\r\n  onAfterCancelTalk: Function;\r\n  onAfterRecordStart: Function;\r\n  onDurationStrChange: Function;\r\n\r\n  // API配置\r\n  APPID = \"50f4a46c\";\r\n  API_SECRET = \"NzFlNmFhZDJjMDNkZGM3NzI0Mzg2OGNm\";\r\n  API_KEY = \"106ddc40dfd4b9ca6d7b47c70fada749\";\r\n\r\n  constructor(\r\n    private platform: Platform,\r\n    private diagnostic: Diagnostic,\r\n  ) {\r\n    this.requestPermission();\r\n  }\r\n\r\n  // 用户操作方法\r\n  toggleRecord() {\r\n    console.log('toggleRecord', this.btnStatus);\r\n    if (this.btnStatus === \"UNDEFINED\" || this.btnStatus === \"CLOSED\") {\r\n      this.startTalk();\r\n    } else if (this.btnStatus === \"CONNECTING\" || this.btnStatus === \"OPEN\") {\r\n      this.finishTalk();\r\n    }\r\n  }\r\n\r\n  async startTalk(event?) {\r\n    console.log('startTalk called');\r\n    this.resetState();\r\n    this.onBeforeStartTalk && this.onBeforeStartTalk();\r\n    \r\n    event?.preventDefault();\r\n    \r\n    try {\r\n      await this.openWithPriviledge();\r\n      \r\n      // 1. 首先启动独立录制\r\n      await this.startIndependentRecording();\r\n      \r\n      // 2. 然后启动WebSocket连接\r\n      this.isRecording = true;\r\n      this.shouldReconnect = true;\r\n      this.reconnectAttempts = 0;\r\n      \r\n      setTimeout(() => {\r\n        this.connectWebSocket();\r\n      }, 100);\r\n\r\n      this.startCountdown();\r\n      this.onAfterStartTalk && this.onAfterStartTalk();\r\n    } catch (error) {\r\n      console.error('Failed to start talk:', error);\r\n      this.resetState();\r\n    }\r\n  }\r\n\r\n  async finishTalk() {\r\n    console.log('finishTalk called');\r\n    this.isUserFinish = true;\r\n    this.shouldReconnect = false;\r\n    this.onBeforeFinishTalk && this.onBeforeFinishTalk();\r\n    \r\n    try {\r\n      // 1. 先关闭WebSocket\r\n      this.closeWebSocket();\r\n      \r\n      // 2. 停止独立录制并获取完整音频\r\n      await this.stopIndependentRecording();\r\n      \r\n      // 3. 清理状态\r\n      this.isRecording = false;\r\n      this.clearTimers();\r\n      this.changeBtnStatus(\"CLOSED\");\r\n      \r\n      // 4. 延迟回调确保所有处理完成\r\n      setTimeout(() => {\r\n        if (this.isUserFinish) {\r\n          this.onAfterFinishTalk && this.onAfterFinishTalk();\r\n          this.isUserFinish = false;\r\n        }\r\n      }, 500);\r\n      \r\n    } catch (error) {\r\n      console.error('Error finishing talk:', error);\r\n    }\r\n  }\r\n\r\n  cancelTalk() {\r\n    console.log('cancelTalk called');\r\n    this.onBeforeCancelTalk && this.onBeforeCancelTalk();\r\n    \r\n    this.isUserFinish = false;\r\n    this.shouldReconnect = false;\r\n    this.isRecording = false;\r\n    \r\n    // 关闭WebSocket\r\n    this.closeWebSocket();\r\n    \r\n    // 停止独立录制但不保存结果\r\n    this.cancelIndependentRecording();\r\n    \r\n    this.resetState();\r\n    this.onAfterCancelTalk && this.onAfterCancelTalk();\r\n  }\r\n\r\n  // 独立录制方法 - 与WebSocket完全解耦\r\n  async startIndependentRecording(): Promise<void> {\r\n    console.log('Starting independent recording');\r\n    \r\n    this.allRecordedBuffers = [];\r\n    this.createIndependentRecorder();\r\n    \r\n    return new Promise((resolve, reject) => {\r\n      this.currentRecorder.open(() => {\r\n        console.log('Independent recorder opened');\r\n        \r\n        // 创建可视化\r\n        let waveDiv = document.querySelector(\".record-wave\");\r\n        if (waveDiv && Recorder.WaveView) {\r\n          this.waveClient = Recorder.WaveView({ elem: \".record-wave\" });\r\n        }\r\n        \r\n        this.currentRecorder.start();\r\n        this.onAfterRecordStart && this.onAfterRecordStart();\r\n        resolve();\r\n      }, (msg, isUserNotAllow) => {\r\n        console.error('Failed to open independent recorder:', msg);\r\n        reject(new Error(msg));\r\n      });\r\n    });\r\n  }\r\n\r\n  async stopIndependentRecording(): Promise<void> {\r\n    console.log('Stopping independent recording');\r\n    \r\n    if (!this.currentRecorder) {\r\n      console.log('No independent recorder to stop');\r\n      return;\r\n    }\r\n\r\n    return new Promise((resolve) => {\r\n      this.currentRecorder.stop(\r\n        async (blob: Blob, duration: number) => {\r\n          console.log('Independent recording stopped successfully', blob, duration);\r\n          \r\n          // 保存完整的录制结果\r\n          this.recordPcmBlob = blob;\r\n          this.recordWavBlob = await this.pcmBlobToWavBlob(blob, 44100);\r\n          \r\n          // 清理录制器\r\n          this.cleanupIndependentRecorder();\r\n          resolve();\r\n        },\r\n        (msg) => {\r\n          console.error(\"独立录音停止失败:\" + msg);\r\n          this.cleanupIndependentRecorder();\r\n          resolve();\r\n        }\r\n      );\r\n    });\r\n  }\r\n\r\n  cancelIndependentRecording(): void {\r\n    console.log('Cancelling independent recording');\r\n    \r\n    if (this.currentRecorder) {\r\n      try {\r\n        // 直接关闭，不保存结果\r\n        this.currentRecorder.close();\r\n      } catch (error) {\r\n        console.error('Error closing independent recorder:', error);\r\n      }\r\n    }\r\n    \r\n    this.cleanupIndependentRecorder();\r\n    this.recordPcmBlob = null;\r\n    this.recordWavBlob = null;\r\n  }\r\n\r\n  private createIndependentRecorder(): void {\r\n    if (this.currentRecorder) {\r\n      this.cleanupIndependentRecorder();\r\n    }\r\n\r\n    this.currentRecorder = Recorder({\r\n      type: this.recordType,\r\n      sampleRate: 44100,\r\n      bitRate: 16,\r\n      onProcess: (buffers, powerLevel, bufferDuration, bufferSampleRate, newBufferIdx, asyncEnd) => {\r\n        if (!this.isRecording) return;\r\n\r\n        // 保存所有buffer用于最终合成完整音频\r\n        this.allRecordedBuffers = buffers;\r\n\r\n        // 获取最新的音频片段用于实时ASR\r\n        let frameBuffer = buffers.length && buffers[buffers.length - 1];\r\n        if (!frameBuffer) return;\r\n\r\n        // 重采样到16kHz用于ASR\r\n        frameBuffer = resampleBuffer(frameBuffer, 44100, 16000);\r\n\r\n        // 发送到WebSocket进行实时ASR\r\n        this.sendAudioToWebSocket(frameBuffer);\r\n\r\n        // 更新波形显示\r\n        this.waveClient?.input(buffers[buffers.length - 1], powerLevel, bufferSampleRate);\r\n      }\r\n    });\r\n  }\r\n\r\n  private cleanupIndependentRecorder(): void {\r\n    if (this.currentRecorder) {\r\n      try {\r\n        this.currentRecorder.close();\r\n      } catch (error) {\r\n        console.error('Error cleaning up independent recorder:', error);\r\n      }\r\n      this.currentRecorder = null;\r\n    }\r\n  }\r\n\r\n  private sendAudioToWebSocket(frameBuffer: any): void {\r\n    if (this.iatWS && this.iatWS.readyState === WebSocket.OPEN && !this.disableASR) {\r\n      try {\r\n        this.iatWS.send(JSON.stringify({\r\n          data: {\r\n            status: 1,\r\n            format: \"audio/L16;rate=16000\",\r\n            encoding: this.encodingType,\r\n            audio: convertFrameBufferToBase64(frameBuffer)\r\n          },\r\n        }));\r\n      } catch (error) {\r\n        console.error(\"Error sending audio data to WebSocket:\", error);\r\n      }\r\n    }\r\n  }\r\n\r\n  // WebSocket相关方法 - 仅用于ASR，不控制录制\r\n  connectWebSocket() {\r\n    console.log(\"connectWebSocket called, shouldReconnect:\", this.shouldReconnect);\r\n    \r\n    if (!this.shouldReconnect || !this.isRecording) {\r\n      console.log(\"Not connecting WebSocket - shouldReconnect:\", this.shouldReconnect, \"isRecording:\", this.isRecording);\r\n      return;\r\n    }\r\n\r\n    // 关闭现有连接\r\n    this.closeWebSocket();\r\n\r\n    const websocketUrl = this.getWebSocketUrl();\r\n    console.log(\"Connecting to:\", websocketUrl);\r\n\r\n    try {\r\n      if (\"WebSocket\" in window) {\r\n        this.iatWS = new WebSocket(websocketUrl);\r\n      } else {\r\n        console.error(\"浏览器不支持WebSocket\");\r\n        alert(\"浏览器不支持WebSocket\");\r\n        return;\r\n      }\r\n\r\n      this.changeBtnStatus(\"CONNECTING\");\r\n\r\n      this.iatWS.onopen = (e) => {\r\n        console.log(\"WebSocket connected for ASR\");\r\n        this.reconnectAttempts = 0;\r\n        this.changeBtnStatus(\"OPEN\");\r\n\r\n        // 发送初始参数\r\n        const params = {\r\n          common: {\r\n            app_id: this.APPID,\r\n          },\r\n          business: {\r\n            language: \"zh_cn\",\r\n            domain: \"iat\",\r\n            accent: \"mandarin\",\r\n            vad_eos: 10000, // 增加到10秒\r\n            dwa: \"wpgs\",\r\n          },\r\n          data: {\r\n            status: 0,\r\n            format: \"audio/L16;rate=16000\",\r\n            encoding: this.encodingType,\r\n          },\r\n        };\r\n        this.iatWS.send(JSON.stringify(params));\r\n      };\r\n\r\n      this.iatWS.onmessage = (e) => {\r\n        this.renderResult(e.data);\r\n      };\r\n\r\n      this.iatWS.onerror = (e) => {\r\n        console.error(\"WebSocket error:\", e);\r\n        this.handleWebSocketError();\r\n      };\r\n\r\n      this.iatWS.onclose = (e) => {\r\n        console.log(\"WebSocket closed:\", e.code, e.reason);\r\n        this.handleWebSocketClose(e);\r\n      };\r\n\r\n    } catch (error) {\r\n      console.error(\"Failed to create WebSocket:\", error);\r\n      this.handleWebSocketError();\r\n    }\r\n  }\r\n\r\n  private handleWebSocketClose(event: CloseEvent) {\r\n    console.log(\"handleWebSocketClose\", {\r\n      code: event.code,\r\n      reason: event.reason,\r\n      shouldReconnect: this.shouldReconnect,\r\n      isRecording: this.isRecording,\r\n      isUserFinish: this.isUserFinish\r\n    });\r\n\r\n    // 如果用户已经结束或不应该重连，则不处理\r\n    if (this.isUserFinish || !this.shouldReconnect || !this.isRecording) {\r\n      return;\r\n    }\r\n\r\n    // 发送结束帧（如果需要）\r\n    try {\r\n      if (this.iatWS && this.iatWS.readyState === WebSocket.OPEN) {\r\n        this.iatWS.send(JSON.stringify({\r\n          \"data\": {\r\n            \"status\": 2,\r\n            \"format\": \"audio/L16;rate=16000\",\r\n            \"encoding\": this.encodingType,\r\n            \"audio\": \"\"\r\n          }\r\n        }));\r\n      }\r\n    } catch (err) {\r\n      console.error('Error sending end frame on close:', err);\r\n    }\r\n\r\n    // 正常关闭码或服务端主动关闭，尝试重连\r\n    if (event.code === 1000 || event.code === 1006 || event.code === 1011) {\r\n      this.attemptReconnect();\r\n    } else {\r\n      console.error(\"WebSocket closed with error code:\", event.code);\r\n      this.handleWebSocketError();\r\n    }\r\n  }\r\n\r\n  private handleWebSocketError() {\r\n    if (!this.shouldReconnect || !this.isRecording) {\r\n      return;\r\n    }\r\n    this.attemptReconnect();\r\n  }\r\n\r\n  private attemptReconnect() {\r\n    if (this.reconnectAttempts >= this.maxReconnectAttempts) {\r\n      console.error(\"Max reconnect attempts reached\");\r\n      // 不改变录制状态，只是ASR暂时不可用\r\n      return;\r\n    }\r\n\r\n    this.reconnectAttempts++;\r\n    console.log(`Attempting WebSocket reconnect ${this.reconnectAttempts}/${this.maxReconnectAttempts}`);\r\n\r\n    setTimeout(() => {\r\n      if (this.shouldReconnect && this.isRecording && !this.isUserFinish) {\r\n        this.connectWebSocket();\r\n      }\r\n    }, this.reconnectDelay * this.reconnectAttempts);\r\n  }\r\n\r\n  private closeWebSocket() {\r\n    if (this.iatWS) {\r\n      try {\r\n        // 发送结束帧\r\n        if (this.iatWS.readyState === WebSocket.OPEN) {\r\n          this.iatWS.send(JSON.stringify({\r\n            \"data\": {\r\n              \"status\": 2,\r\n              \"format\": \"audio/L16;rate=16000\",\r\n              \"encoding\": this.encodingType,\r\n              \"audio\": \"\"\r\n            }\r\n          }));\r\n        }\r\n        \r\n        if (this.iatWS.readyState === WebSocket.OPEN || this.iatWS.readyState === WebSocket.CONNECTING) {\r\n          this.iatWS.close(1000, \"User initiated close\");\r\n        }\r\n      } catch (error) {\r\n        console.error(\"Error closing WebSocket:\", error);\r\n      }\r\n      this.iatWS = null;\r\n    }\r\n  }\r\n\r\n  // 状态管理\r\n  private resetState() {\r\n    this.resultText = \"\";\r\n    this.resultTextTemp = \"\";\r\n    this.durationStr = \"00:00\";\r\n    this.duration = 0;\r\n    this.recordDuration = 0;\r\n    this.allRecordedBuffers = [];\r\n    this.clearTimers();\r\n  }\r\n\r\n  private clearTimers() {\r\n    if (this.countdownInterval) {\r\n      clearInterval(this.countdownInterval);\r\n      this.countdownInterval = null;\r\n    }\r\n    if (this.durationInterval) {\r\n      clearInterval(this.durationInterval);\r\n      this.durationInterval = null;\r\n    }\r\n  }\r\n\r\n  changeBtnStatus(status: string) {\r\n    this.btnStatus = status;\r\n    switch (status) {\r\n      case \"CONNECTING\":\r\n        this.connStatus = \"建立连接中\";\r\n        break;\r\n      case \"OPEN\":\r\n        this.connStatus = `录音中（${this.durationStr}）`;\r\n        break;\r\n      case \"CLOSING\":\r\n        this.connStatus = \"关闭连接中\";\r\n        break;\r\n      case \"CLOSED\":\r\n        this.connStatus = \"开始录音\";\r\n        break;\r\n    }\r\n  }\r\n\r\n  // 计时器\r\n  startCountdown() {\r\n    this.clearTimers();\r\n    this.recordDuration = 0;\r\n    this.duration = 0;\r\n    this.now = new Date();\r\n\r\n    // 录音时长计时器（100ms精度）\r\n    this.durationInterval = setInterval(() => {\r\n      if (this.isRecording) {\r\n        this.recordDuration += 100;\r\n      }\r\n    }, 100);\r\n\r\n    // 显示时长计时器（1秒精度）\r\n    this.countdownInterval = setInterval(() => {\r\n      if (this.isRecording) {\r\n        this.countTimer();\r\n      }\r\n    }, 1000);\r\n  }\r\n\r\n  countTimer() {\r\n    this.duration++;\r\n    let minuteStr = String(Math.floor(this.duration / 60)).padStart(2, \"0\");\r\n    let secondStr = String(this.duration % 60).padStart(2, \"0\");\r\n    let durationStr = minuteStr + \":\" + secondStr;\r\n    this.durationStr = durationStr;\r\n    this.connStatus = `录音中（${this.durationStr}）`;\r\n    this.onDurationStrChange && this.onDurationStrChange(durationStr);\r\n  }\r\n\r\n  // 结果处理\r\n  renderResult(resultData: string) {\r\n    try {\r\n      let jsonData = JSON.parse(resultData);\r\n      \r\n      if (jsonData.data && jsonData.data.result) {\r\n        let data = jsonData.data.result;\r\n        let str = \"\";\r\n        let ws = data.ws;\r\n        \r\n        for (let i = 0; i < ws.length; i++) {\r\n          str = str + ws[i].cw[0].w;\r\n        }\r\n\r\n        // 处理动态修正\r\n        if (data.pgs) {\r\n          if (data.pgs === \"apd\") {\r\n            this.resultText = this.resultTextTemp;\r\n          }\r\n          this.resultTextTemp = this.resultText + str;\r\n        } else {\r\n          this.resultText = this.resultText + str;\r\n        }\r\n\r\n        this.onInputChange && this.onInputChange(this.getUserInput());\r\n      }\r\n\r\n      // 处理结束状态\r\n      if (jsonData.code === 0 && jsonData.data.status === 2) {\r\n        console.log(\"ASR session completed, will reconnect if still recording\");\r\n        if (this.iatWS && this.iatWS.readyState === WebSocket.OPEN) {\r\n          this.iatWS.close();\r\n        }\r\n      }\r\n\r\n      // 处理错误\r\n      if (jsonData.code !== 0) {\r\n        console.error(\"ASR error:\", jsonData);\r\n        if (this.iatWS && this.iatWS.readyState === WebSocket.OPEN) {\r\n          this.iatWS.close();\r\n        }\r\n      }\r\n    } catch (error) {\r\n      console.error(\"Error parsing result:\", error);\r\n    }\r\n  }\r\n\r\n  getUserInput(): string {\r\n    return \"\" + (this.resultTextTemp || this.resultText);\r\n  }\r\n\r\n  // 权限和初始化\r\n  async openWithPriviledge(): Promise<boolean> {\r\n    await this.requestPermission();\r\n    \r\n    if (Recorder.IsOpen()) return true;\r\n\r\n    // 这里不创建录制器，因为我们使用独立的录制器\r\n    return true;\r\n  }\r\n\r\n  // 音频处理方法\r\n  async pcmBlobToWavBlob(pcmBlob: Blob, sampleRate: number): Promise<Blob> {\r\n    return new Promise(resolve => {\r\n      let fileReader = new FileReader();\r\n      \r\n      fileReader.onload = function(event) {\r\n        let pcmData: any = event.target.result;\r\n        let wavBlob = pcmtoWav(pcmData, sampleRate, 1, 16);\r\n        resolve(wavBlob);\r\n      };\r\n      \r\n      fileReader.readAsArrayBuffer(pcmBlob);\r\n    });\r\n  }\r\n\r\n  async playPCM(pcmBlob: Blob, sampleRate: number) {\r\n    let wavBlob = await this.pcmBlobToWavBlob(pcmBlob, sampleRate);\r\n    let wavUrl = window.URL.createObjectURL(wavBlob);\r\n    let audio = new Audio();\r\n    audio.src = wavUrl;\r\n    audio.play();\r\n  }\r\n\r\n  playRecord() {\r\n    if (this.recordPcmBlob) {\r\n      this.playPCM(this.recordPcmBlob, 44100);\r\n    }\r\n  }\r\n\r\n  // WebSocket URL生成\r\n  getWebSocketUrl(): string {\r\n    let url = \"wss://iat-api.xfyun.cn/v2/iat\";\r\n    let host = \"iat-api.xfyun.cn\";\r\n    \r\n    let apiKey = this.API_KEY;\r\n    let apiSecret = this.API_SECRET;\r\n    let date = new Date().toUTCString();\r\n    let algorithm = \"hmac-sha256\";\r\n    let headers = \"host date request-line\";\r\n    let signatureOrigin = `host: ${host}\\ndate: ${date}\\nGET /v2/iat HTTP/1.1`;\r\n    let signatureSha = CryptoJS.HmacSHA256(signatureOrigin, apiSecret);\r\n    let signature = CryptoJS.enc.Base64.stringify(signatureSha);\r\n    let authorizationOrigin = `api_key=\"${apiKey}\", algorithm=\"${algorithm}\", headers=\"${headers}\", signature=\"${signature}\"`;\r\n    let authorization = btoa(authorizationOrigin);\r\n    url = `${url}?authorization=${authorization}&date=${date}&host=${host}`;\r\n    return url;\r\n  }\r\n\r\n  toBase64(buffer: ArrayBuffer): string {\r\n    var binary = \"\";\r\n    var bytes = new Uint8Array(buffer);\r\n    var len = bytes.byteLength;\r\n    for (var i = 0; i < len; i++) {\r\n      binary += String.fromCharCode(bytes[i]);\r\n    }\r\n    return window.btoa(binary);\r\n  }\r\n\r\n  // 移动端权限方法\r\n  isCapacitor(): boolean {\r\n    if(!this.platform?.is) return false\r\n    return this.platform.is(\"capacitor\") || this.platform.is(\"cordova\");\r\n  }\r\n\r\n  async requestPermission() {\r\n    if (!this.isCapacitor()) return;\r\n    \r\n    try {\r\n      await this.requestStoagePermission();\r\n      await this.requestCameraPermission();\r\n      await this.requestMicPermission();\r\n      await this.requestRecordAudioPermission();\r\n    } catch (err) {\r\n      console.error(err);\r\n    }\r\n  }\r\n\r\n  async requestRecordAudioPermission() {let data = await this.diagnostic.requestRuntimePermissions([this.diagnostic.permission.RECORD_AUDIO]);\r\n    console.log(\"record permission request:\", data);\r\n    return;\r\n  }\r\n\r\n  async requestMicPermission() {\r\n    let isAvailable = await this.diagnostic.isMicrophoneAuthorized();\r\n    console.log(\"permisson_MIC:\", isAvailable);\r\n    if (!isAvailable) {\r\n      let data = await this.diagnostic.requestMicrophoneAuthorization();\r\n    }\r\n    return;\r\n  }\r\n\r\n  async requestStoagePermission() {\r\n    let isAvailable = await this.diagnostic.isExternalStorageAuthorized();\r\n    console.log(\"permisson_STORAGE:\", isAvailable);\r\n    if (!isAvailable) {\r\n      let data = await this.diagnostic.requestExternalStorageAuthorization();\r\n    }\r\n    return;\r\n  }\r\n\r\n  async requestCameraPermission() {\r\n    let isAvailable = await this.diagnostic.isCameraAuthorized();\r\n    console.log(\"permisson_Camera:\", isAvailable);\r\n    if (!isAvailable) {\r\n      let data = await this.diagnostic.requestCameraAuthorization();\r\n    }\r\n    return;\r\n  }\r\n\r\n  // 其他辅助方法\r\n  splitAudioData(audioData: any) {\r\n    const segmentSize = 1280;\r\n    const segmentCount = Math.ceil(audioData.length / segmentSize);\r\n    const segments = [];\r\n\r\n    for (let i = 0; i < segmentCount; i++) {\r\n      const start = i * segmentSize;\r\n      const end = start + segmentSize;\r\n      const segment = audioData.slice(start, end);\r\n      segments.push(segment);\r\n    }\r\n\r\n    return segments;\r\n  }\r\n\r\n  BufferToBlob(buffer: any) {\r\n    return new Blob([buffer], { type: 'audio/pcm' });\r\n  }\r\n\r\n  async playBuffers() {\r\n    let audioBlob = await this.BuffersToBlob(this.allRecordedBuffers);\r\n    this.playPCM(audioBlob, 44100);\r\n  }\r\n\r\n  BuffersToBlob(buffers: Array<any>) {\r\n    let audioBuffer: any = [];\r\n    buffers.forEach(buffer => {\r\n      buffer.forEach(int16 => {\r\n        audioBuffer.push(int16);\r\n      });\r\n    });\r\n    return new Blob([audioBuffer], { type: 'audio/pcm' });\r\n  }\r\n}"]}
@@ -1,7 +1,7 @@
1
1
  export * from "./fmode-voice.service";
2
2
  export * from "./lib/audio/audio.player";
3
3
  export * from "./lib/audio/streamer.microsoft";
4
- export * from "./class-asr";
4
+ // export * from "./class-asr"
5
5
  export * from "./tts";
6
6
  export * from "./lib/audio/streamer.microsoft";
7
- //# sourceMappingURL=data:application/json;base64,eyJ2ZXJzaW9uIjozLCJmaWxlIjoiaW5kZXguanMiLCJzb3VyY2VSb290IjoiIiwic291cmNlcyI6WyIuLi8uLi8uLi8uLi8uLi8uLi9wcm9qZWN0cy9mbW9kZS1uZy9zcmMvbGliL2FpZ2Mvdm9pY2UvaW5kZXgudHMiXSwibmFtZXMiOltdLCJtYXBwaW5ncyI6IkFBQUEsY0FBYyx1QkFBdUIsQ0FBQTtBQUVyQyxjQUFjLDBCQUEwQixDQUFBO0FBQ3hDLGNBQWMsZ0NBQWdDLENBQUE7QUFFOUMsY0FBYyxhQUFhLENBQUE7QUFDM0IsY0FBYyxPQUFPLENBQUE7QUFDckIsY0FBYyxnQ0FBZ0MsQ0FBQSIsInNvdXJjZXNDb250ZW50IjpbImV4cG9ydCAqIGZyb20gXCIuL2Ztb2RlLXZvaWNlLnNlcnZpY2VcIlxyXG5cclxuZXhwb3J0ICogZnJvbSBcIi4vbGliL2F1ZGlvL2F1ZGlvLnBsYXllclwiXHJcbmV4cG9ydCAqIGZyb20gXCIuL2xpYi9hdWRpby9zdHJlYW1lci5taWNyb3NvZnRcIlxyXG5cclxuZXhwb3J0ICogZnJvbSBcIi4vY2xhc3MtYXNyXCJcclxuZXhwb3J0ICogZnJvbSBcIi4vdHRzXCJcclxuZXhwb3J0ICogZnJvbSBcIi4vbGliL2F1ZGlvL3N0cmVhbWVyLm1pY3Jvc29mdFwiIl19
7
+ //# sourceMappingURL=data:application/json;base64,eyJ2ZXJzaW9uIjozLCJmaWxlIjoiaW5kZXguanMiLCJzb3VyY2VSb290IjoiIiwic291cmNlcyI6WyIuLi8uLi8uLi8uLi8uLi8uLi9wcm9qZWN0cy9mbW9kZS1uZy9zcmMvbGliL2FpZ2Mvdm9pY2UvaW5kZXgudHMiXSwibmFtZXMiOltdLCJtYXBwaW5ncyI6IkFBQUEsY0FBYyx1QkFBdUIsQ0FBQTtBQUVyQyxjQUFjLDBCQUEwQixDQUFBO0FBQ3hDLGNBQWMsZ0NBQWdDLENBQUE7QUFFOUMsOEJBQThCO0FBQzlCLGNBQWMsT0FBTyxDQUFBO0FBQ3JCLGNBQWMsZ0NBQWdDLENBQUEiLCJzb3VyY2VzQ29udGVudCI6WyJleHBvcnQgKiBmcm9tIFwiLi9mbW9kZS12b2ljZS5zZXJ2aWNlXCJcclxuXHJcbmV4cG9ydCAqIGZyb20gXCIuL2xpYi9hdWRpby9hdWRpby5wbGF5ZXJcIlxyXG5leHBvcnQgKiBmcm9tIFwiLi9saWIvYXVkaW8vc3RyZWFtZXIubWljcm9zb2Z0XCJcclxuXHJcbi8vIGV4cG9ydCAqIGZyb20gXCIuL2NsYXNzLWFzclwiXHJcbmV4cG9ydCAqIGZyb20gXCIuL3R0c1wiXHJcbmV4cG9ydCAqIGZyb20gXCIuL2xpYi9hdWRpby9zdHJlYW1lci5taWNyb3NvZnRcIiJdfQ==
@@ -1,7 +1,8 @@
1
1
  import Parse from "parse";
2
2
  import { AudioPlayer } from "../lib/audio/audio.player";
3
3
  import { FmPushAudioOutputStreamCallback } from "../lib/audio/streamer.microsoft";
4
- import { FmodeTTSProviderMicrosoft } from "./provider-microsoft";
4
+ // import { FmodeTTSProviderMicrosoft } from "./provider-microsoft";
5
+ // class FmodeTTSProviderMicrosoft{}
5
6
  import { PCMStreamer } from "../lib/audio/streamer.pcm";
6
7
  /**
7
8
  * FmodeTTS
@@ -27,7 +28,7 @@ export class FmodeTTS {
27
28
  this.audioPlayer = AudioPlayer.getInstance();
28
29
  this.audioStream = new FmPushAudioOutputStreamCallback();
29
30
  // 初始化provider
30
- this.provider = config.provider || new FmodeTTSProviderMicrosoft();
31
+ this.provider = config.provider; // || new FmodeTTSProviderMicrosoft();
31
32
  console.log("初始化provider", config, this.provider, this.provider.constructor);
32
33
  if (this.provider.constructor.toString().indexOf("Doubao") > -1) {
33
34
  // 创建 PCM 流播放器
@@ -185,4 +186,4 @@ export class FmodeTTS {
185
186
  return fileResult?.url || null;
186
187
  }
187
188
  }
188
- //# sourceMappingURL=data:application/json;base64,{"version":3,"file":"fmode-tts-class.js","sourceRoot":"","sources":["../../../../../../../projects/fmode-ng/src/lib/aigc/voice/tts/fmode-tts-class.ts"],"names":[],"mappings":"AAAA,OAAO,KAAK,MAAM,OAAO,CAAC;AAE1B,OAAO,EAAE,WAAW,EAAE,MAAM,2BAA2B,CAAC;AACxD,OAAO,EAAE,+BAA+B,EAAE,MAAM,iCAAiC,CAAC;AAElF,OAAO,EAAE,yBAAyB,EAAE,MAAM,sBAAsB,CAAC;AACjE,OAAO,EAAE,WAAW,EAAE,MAAM,2BAA2B,CAAC;AAKxD;;;GAGG;AACH,MAAM,OAAO,QAAQ;IACjB,mBAAmB;IACnB,gBAAgB;aACD,mBAAc,GAAoB,IAAI,AAAxB,CAAyB;IAYtD,IAAI;QACA,IAAI,QAAQ,CAAC,cAAc,KAAK,IAAI,EAAE,CAAC;YACnC,QAAQ,CAAC,cAAc,GAAG,IAAI,CAAC;QACnC,CAAC;QACD,IAAI,CAAC,WAAW,EAAE,IAAI,EAAE,CAAC;QACzB,IAAI,CAAC,WAAW,EAAE,IAAI,EAAE,CAAC;QACzB,IAAI,CAAC,QAAQ,EAAE,IAAI,EAAE,CAAC;QACtB,IAAI,CAAC,SAAS,GAAG,KAAK,CAAC;IAC3B,CAAC;IAED,YAAY,MAOX,EAAE,UAA8B;QAvB1B,cAAS,GAAY,KAAK,CAAC;QAwB9B,IAAI,CAAC,UAAU,GAAG,UAAU,CAAC;QAC7B,IAAI,CAAC,MAAM,GAAG,MAAM,CAAA;QAEpB,IAAI,CAAC,WAAW,GAAG,WAAW,CAAC,WAAW,EAAE,CAAC;QAC7C,IAAI,CAAC,WAAW,GAAG,IAAI,+BAA+B,EAAE,CAAC;QAEzD,cAAc;QACd,IAAI,CAAC,QAAQ,GAAG,MAAM,CAAC,QAAQ,IAAI,IAAI,yBAAyB,EAAE,CAAC;QACnE,OAAO,CAAC,GAAG,CAAC,aAAa,EAAC,MAAM,EAAC,IAAI,CAAC,QAAQ,EAAC,IAAI,CAAC,QAAQ,CAAC,WAAW,CAAC,CAAA;QACzE,IAAG,IAAI,CAAC,QAAQ,CAAC,WAAW,CAAC,QAAQ,EAAE,CAAC,OAAO,CAAC,QAAQ,CAAC,GAAC,CAAC,CAAC,EAAE,CAAC;YAC3D,cAAc;YACd,MAAM,WAAW,GAAG,IAAI,WAAW,CAAC,KAAK,EAAE,CAAC,EAAE,EAAE,CAAC,CAAC;YAClD,MAAM,CAAC,WAAW,GAAG,WAAW,CAAA;QACpC,CAAC;aAAI,CAAC;YACF,MAAM,CAAC,WAAW,GAAG,IAAI,CAAC,WAAW,CAAA;QACzC,CAAC;QAED,IAAI,CAAC,WAAW,GAAG,MAAM,EAAE,WAAW,IAAI,IAAI,CAAC,WAAW,CAAC;QAG3D,IAAI,CAAC,QAAQ,CAAC,UAAU,CAAC,MAAM,CAAC,CAAC;QACjC,IAAI,CAAC,QAAQ,CAAC,WAAW,GAAG,MAAM,EAAE,WAAW,CAAC;QAChD,IAAI,CAAC,QAAQ,CAAC,QAAQ,GAAG,IAAI,CAAC,QAAQ,CAAC;IAE3C,CAAC;IAED,kBAAkB,CAAC,WAAmB;QAClC,IAAI,CAAC,WAAW,IAAI,OAAO,WAAW,KAAK,QAAQ,EAAE,CAAC;YAClD,OAAO,EAAE,CAAC;QACd,CAAC;QAED,MAAM,KAAK,GAAG,YAAY,CAAC;QAC3B,MAAM,MAAM,GAAa,EAAE,CAAC;QAC5B,IAAI,OAA+B,CAAC;QAEpC,OAAO,CAAC,OAAO,GAAG,KAAK,CAAC,IAAI,CAAC,WAAW,CAAC,CAAC,KAAK,IAAI,EAAE,CAAC;YAClD,MAAM,IAAI,GAAG,OAAO,CAAC,CAAC,CAAC,EAAE,IAAI,EAAE,CAAC;YAChC,IAAI,IAAI,EAAE,CAAC;gBACP,MAAM,CAAC,IAAI,CAAC,IAAI,CAAC,CAAC;YACtB,CAAC;QACL,CAAC;QAED,OAAO,MAAM,CAAC,IAAI,CAAC,GAAG,CAAC,CAAC;IAC5B,CAAC;IAED,kBAAkB,CAAC,WAAmB;QAClC,IAAI,CAAC,WAAW,IAAI,OAAO,WAAW,KAAK,QAAQ,EAAE,CAAC;YAClD,OAAO,WAAW,IAAI,EAAE,CAAC;QAC7B,CAAC;QAED,MAAM,KAAK,GAAG,qBAAqB,CAAC;QACpC,MAAM,KAAK,GAAG,WAAW,CAAC,KAAK,CAAC,KAAK,CAAC,CAAC;QACvC,OAAO,KAAK,EAAE,CAAC,CAAC,CAAC,IAAI,WAAW,CAAC;IACrC,CAAC;IAEM,KAAK,CAAC,UAAU,CAAC,UAAkB,EAAE,SAAwB,EAAE,QAAc;QAC/E,cAAc;QACd,IAAI,QAAQ,CAAC,cAAc,IAAI,QAAQ,CAAC,cAAc,KAAK,IAAI,EAAE,CAAC;YAC/D,QAAQ,CAAC,cAAc,CAAC,IAAI,EAAE,CAAC;QACnC,CAAC;QACD,QAAQ,CAAC,cAAc,GAAG,IAAI,CAAC;QAE/B,IAAI,CAAC,SAAS,EAAE,CAAC;YACb,MAAM,SAAS,GAAG,KAAK,CAAC,MAAM,CAAC,MAAM,CAAC,WAAW,CAAC,CAAC;YACnD,SAAS,GAAG,IAAI,SAAS,EAAE,CAAC;YAC5B,SAAS,CAAC,GAAG,CAAC,MAAM,EAAE,UAAU,CAAC,CAAC;YAClC,SAAS,CAAC,GAAG,CAAC,SAAS,EAAE,IAAI,CAAC,kBAAkB,CAAC,UAAU,CAAC,CAAC,CAAC;YAC9D,MAAM,OAAO,GAAG,YAAY,CAAC,OAAO,CAAC,SAAS,CAAC,CAAC;YAChD,OAAO,IAAI,SAAS,CAAC,GAAG,CAAC,SAAS,EAAE,EAAE,MAAM,EAAE,SAAS,EAAE,SAAS,EAAE,SAAS,EAAE,QAAQ,EAAE,OAAO,EAAE,CAAC,CAAC;YACpG,KAAK,CAAC,IAAI,CAAC,OAAO,EAAE,EAAE,EAAE,IAAI,SAAS,CAAC,GAAG,CAAC,MAAM,EAAE,KAAK,CAAC,IAAI,CAAC,OAAO,EAAE,CAAC,SAAS,EAAE,CAAC,CAAC;QACxF,CAAC;QAED,OAAO;QACP,IAAI,CAAC,SAAS,EAAE,GAAG,CAAC,WAAW,CAAC,EAAE,CAAC;YAC/B,MAAM,MAAM,GAAG,EAAE,CAAC;YAClB,mCAAmC;YACnC,2DAA2D;YAC3D,IAAI;YACJ,IAAI,UAAU,EAAE,CAAC;gBACb,MAAM,CAAC,IAAI,CAAC,EAAE,IAAI,EAAE,UAAU,EAAE,CAAC,CAAC;YACtC,CAAC;YAED,MAAM,KAAK,GAAG,KAAK,CAAC,KAAK,CAAC,QAAQ,CAAC,WAAW,EAAE;gBAC5C,OAAO,EAAE,WAAW;gBACpB,KAAK,EAAE,EAAE,GAAG,EAAE,MAAM,EAAE;aACzB,CAAC,CAAC;YACH,KAAK,CAAC,MAAM,CAAC,WAAW,CAAC,CAAC;YAC1B,KAAK,CAAC,aAAa,CAAC,WAAW,CAAC,CAAC;YACjC,MAAM,MAAM,GAAG,MAAM,KAAK,CAAC,KAAK,EAAE,CAAC;YACnC,SAAS,CAAC,GAAG,CAAC,WAAW,EAAE,MAAM,EAAE,GAAG,CAAC,WAAW,CAAC,CAAC,CAAC;QACzD,CAAC;QAED,OAAO,CAAC,GAAG,CAAC,sBAAsB,EAAE,SAAS,EAAE,GAAG,CAAC,WAAW,CAAC,CAAC,CAAA;QAChE,IAAI,SAAS,EAAE,GAAG,CAAC,WAAW,CAAC,EAAE,CAAC;YAC9B,IAAI,CAAC,aAAa,CAAC,SAAS,EAAE,GAAG,CAAC,WAAW,CAAC,EAAE,GAAG,CAAC,KAAK,CAAC,EAAE,SAAS,EAAE,QAAQ,CAAC,CAAC;YACjF,SAAS,CAAC,IAAI,EAAE,CAAC;YACjB,OAAO,SAAS,CAAC;QACrB,CAAC;QAED,UAAU,GAAG,IAAI,CAAC,kBAAkB,CAAC,UAAU,CAAC,CAAC;QACjD,IAAI,CAAC,SAAS,GAAG,IAAI,CAAC;QAEtB,OAAO,IAAI,OAAO,CAAC,CAAC,OAAO,EAAE,MAAM,EAAE,EAAE;YACnC,MAAM,SAAS,GAAG,IAAI,CAAC,GAAG,EAAE,CAAC;YAE7B,IAAI,CAAC,QAAQ,CAAC,UAAU,CACpB,UAAU,EACV,KAAK,EAAE,MAAM,EAAE,EAAE;gBACb,MAAM,OAAO,GAAG,IAAI,CAAC,GAAG,EAAE,CAAC;gBAC3B,MAAM,SAAS,GAAG,MAAM,EAAE,SAAS,CAAC;gBAEpC,IAAI,QAAQ,GAAG,MAAM,CAAC,MAAM,EAAE,aAAa,CAAC,CAAA;gBAC5C,IAAG,IAAI,CAAC,MAAM,CAAC,QAAQ,IAAE,WAAW,EAAC,CAAC;oBAClC,QAAQ,GAAG,QAAQ,GAAG,KAAK,CAAA;gBAC/B,CAAC;gBAED,SAAS,CAAC,GAAG,CAAC,UAAU,EAAE,QAAQ,CAAC,CAAC;gBAEpC,OAAO,CAAC,GAAG,CAAC,oBAAoB,EAAC,QAAQ,EAAC,QAAQ,CAAC,CAAA;gBACnD,QAAQ,EAAE,QAAQ,EAAE,CAAC,EAAC,QAAQ,EAAC,QAAQ,EAAC,CAAC,CAAC;gBAE1C,gFAAgF;gBAChF,MAAM,IAAI,GAAG,IAAI,IAAI,CAAC,CAAC,SAAS,CAAC,EAAE,EAAE,IAAI,EAAE,WAAW,EAAE,CAAC,CAAC;gBAC1D,MAAM,IAAI,CAAC,kBAAkB,CAAC,IAAI,EAAE,SAAS,CAAC,CAAC;gBAC/C,OAAO,CAAC,GAAG,CAAC,uCAAuC,OAAO,GAAG,SAAS,KAAK,CAAC,CAAC;gBAE7E,UAAU,CAAC,GAAG,EAAE;oBACZ,IAAI,CAAC,SAAS,GAAG,KAAK,CAAC;oBACvB,QAAQ,EAAE,MAAM,EAAE,EAAE,CAAC;gBACzB,CAAC,EAAE,IAAI,CAAC,CAAC;gBAET,OAAO,CAAC,SAAS,CAAC,CAAC;YACvB,CAAC,EACD,CAAC,KAAK,EAAE,EAAE;gBACN,IAAI,CAAC,SAAS,GAAG,KAAK,CAAC;gBACvB,MAAM,CAAC,oCAAoC,KAAK,EAAE,CAAC,CAAC;YACxD,CAAC,EACD,GAAE,EAAE;gBACA,QAAQ,EAAE,OAAO,EAAE,EAAE,CAAC;YAC1B,CAAC,CACJ,CAAC;QACN,CAAC,CAAC,CAAC;IACP,CAAC;IAEO,KAAK,CAAC,aAAa,CAAC,SAAiB,EAAE,SAAuB,EAAE,QAAa;QACjF,IAAI,GAAW,CAAC;QAEhB,IAAI,SAAS,EAAE,OAAO,EAAE,CAAC,MAAM,CAAC,IAAI,CAAC,CAAC,EAAE,CAAC;YACrC,GAAG,GAAG,SAAS,CAAC;QACpB,CAAC;aAAM,CAAC;YACJ,MAAM,IAAI,GAAG,IAAI,IAAI,CAAC,CAAC,SAAS,CAAC,EAAE,EAAE,IAAI,EAAE,WAAW,EAAE,CAAC,CAAC;YAC1D,GAAG,GAAG,GAAG,CAAC,eAAe,CAAC,IAAI,CAAC,CAAC;YAChC,MAAM,IAAI,CAAC,kBAAkB,CAAC,IAAI,EAAE,SAAS,CAAC,CAAC;QACnD,CAAC;QAED,QAAQ,EAAE,OAAO,EAAE,CAAC,SAAS,CAAC,CAAC;QAC/B,IAAI,CAAC,SAAS,GAAG,IAAI,CAAC;QAEtB,MAAM,WAAW,GAAG,IAAI,CAAC,WAAW,CAAC;QACrC,WAAW,CAAC,aAAa,CAAC,cAAc,EAAE,GAAG,EAAE;YAC3C,MAAM,QAAQ,GAAG,WAAW,CAAC,QAAQ,GAAG,IAAI,CAAC;YAC7C,IAAI,CAAC,SAAS,EAAE,GAAG,CAAC,UAAU,CAAC,EAAE,CAAC;gBAC9B,SAAS,CAAC,GAAG,CAAC,UAAU,EAAE,QAAQ,CAAC,CAAC;gBACpC,SAAS,CAAC,IAAI,EAAE,CAAC;YACrB,CAAC;YACD,QAAQ,EAAE,QAAQ,EAAE,CAAC,WAAW,CAAC,CAAC;QACtC,CAAC,CAAC,CAAC;QAEH,SAAS;QACT,CAAC,SAAS,EAAE,SAAS,EAAE,SAAS,EAAE,SAAS,EAAE,SAAS,CAAC,CAAC,OAAO,CAAC,KAAK,CAAC,EAAE;YACpE,WAAW,CAAC,aAAa,CAAC,KAAK,EAAE,GAAG,EAAE;gBAClC,IAAI,CAAC,SAAS,GAAG,KAAK,CAAC;gBACvB,QAAQ,EAAE,MAAM,EAAE,EAAE,CAAC;YACzB,CAAC,CAAC,CAAC;QACP,CAAC,CAAC,CAAC;QAEH,MAAM,SAAS,GAAG,GAAG,EAAE;YACnB,WAAW,CAAC,IAAI,CAAC,GAAG,CAAC,CAAC,KAAK,CAAC,GAAG,EAAE;gBAC7B,UAAU,CAAC,SAAS,EAAE,GAAG,CAAC,CAAC;YAC/B,CAAC,CAAC,CAAC;QACP,CAAC,CAAC;QAEF,SAAS,EAAE,CAAC;IAChB,CAAC;IAEO,KAAK,CAAC,kBAAkB,CAAC,IAAU,EAAE,SAAuB;QAChE,IAAI,CAAC,IAAI,CAAC,UAAU;YAAE,OAAO,IAAI,CAAC;QAElC,MAAM,EAAE,GAAG,SAAS,EAAE,EAAE,IAAI,IAAI,CAAC,UAAU,CAAC,MAAM,CAAC,SAAS,EAAE,GAAG,CAAC,SAAS,CAAC,IAAI,SAAS,EAAE,GAAG,CAAC,MAAM,CAAC,CAAC,CAAC;QACxG,MAAM,GAAG,GAAG,IAAI,IAAI,EAAE,CAAC;QACvB,MAAM,QAAQ,GAAG,GAAG,EAAE,GAAG,GAAG,CAAC,WAAW,EAAE,GAAG,GAAG,CAAC,QAAQ,EAAE,GAAG,CAAC,GAAG,GAAG,CAAC,OAAO,EAAE,GAAG,GAAG,CAAC,QAAQ,EAAE,GAAG,GAAG,CAAC,UAAU,EAAE,GAAG,GAAG,CAAC,UAAU,EAAE,MAAM,CAAC;QAE7I,MAAM,IAAI,GAAG,IAAI,IAAI,CAAC,CAAC,IAAI,CAAC,EAAE,QAAQ,EAAE,EAAE,IAAI,EAAE,WAAW,EAAE,CAAC,CAAC;QAC/D,MAAM,UAAU,GAAG,MAAM,IAAI,CAAC,UAAU,CAAC,MAAM,CAAC,IAAI,CAAC,CAAC;QAEtD,IAAI,UAAU,EAAE,EAAE,EAAE,CAAC;YACjB,SAAS,CAAC,GAAG,CAAC,WAAW,EAAE,EAAE,MAAM,EAAE,SAAS,EAAE,SAAS,EAAE,YAAY,EAAE,QAAQ,EAAE,UAAU,CAAC,EAAE,EAAE,CAAC,CAAC;YACpG,MAAM,SAAS,CAAC,IAAI,EAAE,CAAC;QAC3B,CAAC;QAED,OAAO,UAAU,EAAE,GAAG,IAAI,IAAI,CAAC;IACnC,CAAC","sourcesContent":["import Parse from \"parse\";\r\nimport { NovaUploadService } from \"../../../storage/service-upload/nova-upload.service\";\r\nimport { AudioPlayer } from \"../lib/audio/audio.player\";\r\nimport { FmPushAudioOutputStreamCallback } from \"../lib/audio/streamer.microsoft\";\r\nimport { FmodeTTSEvent, FmodeTTSProvider } from \"./int-tts-provider\";\r\nimport { FmodeTTSProviderMicrosoft } from \"./provider-microsoft\";\r\nimport { PCMStreamer } from \"../lib/audio/streamer.pcm\";\r\nimport { AudioStreamer } from \"../lib/audio/audio.streamer\";\r\nimport { FmodeChatVoiceConfig } from \"../../../core/agent/chat/interface\";\r\n\r\n\r\n/**\r\n * FmodeTTS\r\n * 每个实例表示一次独立的语音合成服务\r\n */\r\nexport class FmodeTTS {\r\n    // 私有属性：全局配置 当前激活实例\r\n    // 实现全局仅一个TTS在播放\r\n    private static activeInstance: FmodeTTS | null = null;\r\n\r\n\r\n    private provider: FmodeTTSProvider;\r\n    eventMap: FmodeTTSEvent;\r\n    uploadServ?: NovaUploadService;\r\n    public isPlaying: boolean = false;\r\n    audioPlayer: AudioPlayer;\r\n    audioStream: AudioStreamer;\r\n    // 音色\r\n    voiceConfig:FmodeChatVoiceConfig|undefined\r\n\r\n    stop() {\r\n        if (FmodeTTS.activeInstance === this) {\r\n            FmodeTTS.activeInstance = null;\r\n        }\r\n        this.audioPlayer?.stop();\r\n        this.audioStream?.stop();\r\n        this.provider?.stop();\r\n        this.isPlaying = false;\r\n    }\r\n    config:any\r\n    constructor(config: {\r\n        region: string,\r\n        audioStream?:AudioStreamer,\r\n        subscriptionKey?: string,\r\n        token?: string,\r\n        provider?: FmodeTTSProvider // 允许传入自定义provider\r\n        voiceConfig?: FmodeChatVoiceConfig\r\n    }, uploadServ?: NovaUploadService) {\r\n        this.uploadServ = uploadServ;\r\n        this.config = config\r\n\r\n        this.audioPlayer = AudioPlayer.getInstance();\r\n        this.audioStream = new FmPushAudioOutputStreamCallback();\r\n\r\n        // 初始化provider\r\n        this.provider = config.provider || new FmodeTTSProviderMicrosoft();\r\n        console.log(\"初始化provider\",config,this.provider,this.provider.constructor)\r\n        if(this.provider.constructor.toString().indexOf(\"Doubao\")>-1 ){\r\n            // 创建 PCM 流播放器\r\n            const pcmStreamer = new PCMStreamer(24000, 1, 16);\r\n            config.audioStream = pcmStreamer\r\n        }else{\r\n            config.audioStream = this.audioStream\r\n        }\r\n\r\n        this.audioStream = config?.audioStream || this.audioStream;\r\n\r\n        \r\n        this.provider.initialize(config);\r\n        this.provider.voiceConfig = config?.voiceConfig;\r\n        this.provider.eventMap = this.eventMap;\r\n\r\n    }\r\n\r\n    extractTextFromXML(inputString: string): string {\r\n        if (!inputString || typeof inputString !== 'string') {\r\n            return '';\r\n        }\r\n    \r\n        const regex = />([^<]+)</g;\r\n        const result: string[] = [];\r\n        let matches: RegExpExecArray | null;\r\n    \r\n        while ((matches = regex.exec(inputString)) !== null) {\r\n            const text = matches[1]?.trim();\r\n            if (text) {\r\n                result.push(text);\r\n            }\r\n        }\r\n    \r\n        return result.join(' ');\r\n    }\r\n    \r\n    extractSSMLContent(inputString: string): string {\r\n        if (!inputString || typeof inputString !== 'string') {\r\n            return inputString || '';\r\n        }\r\n    \r\n        const regex = /<speak.*?<\\/speak>/s;\r\n        const match = inputString.match(regex);\r\n        return match?.[0] || inputString;\r\n    }\r\n\r\n    public async speakAsync(textOrSSML: string, chatVoice?: Parse.Object, eventMap?: any): Promise<Parse.Object> {\r\n         // 停止当前活动的其他实例\r\n         if (FmodeTTS.activeInstance && FmodeTTS.activeInstance !== this) {\r\n            FmodeTTS.activeInstance.stop();\r\n        }\r\n        FmodeTTS.activeInstance = this;\r\n\r\n        if (!chatVoice) {\r\n            const ChatVoice = Parse.Object.extend(\"ChatVoice\");\r\n            chatVoice = new ChatVoice();\r\n            chatVoice.set(\"ssml\", textOrSSML);\r\n            chatVoice.set(\"content\", this.extractTextFromXML(textOrSSML));\r\n            const company = localStorage.getItem(\"company\");\r\n            company && chatVoice.set(\"company\", { __type: \"Pointer\", className: \"Company\", objectId: company });\r\n            Parse.User.current()?.id && chatVoice.set(\"user\", Parse.User.current().toPointer());\r\n        }\r\n\r\n        // 检查缓存\r\n        if (!chatVoice?.get(\"voiceFile\")) {\r\n            const orList = [];\r\n            // if (chatVoice?.get(\"content\")) {\r\n            //     orList.push({ content: chatVoice?.get(\"content\") });\r\n            // }\r\n            if (textOrSSML) {\r\n                orList.push({ ssml: textOrSSML });\r\n            }\r\n\r\n            const query = Parse.Query.fromJSON(\"ChatVoice\", {\r\n                include: \"voiceFile\",\r\n                where: { $or: orList }\r\n            });\r\n            query.exists(\"voiceFile\");\r\n            query.addDescending('updatedAt');\r\n            const exists = await query.first();\r\n            chatVoice.set(\"voiceFile\", exists?.get(\"voiceFile\"));\r\n        }\r\n\r\n        console.log(\"Exists playAudioData\", chatVoice?.get(\"voiceFile\"))\r\n        if (chatVoice?.get(\"voiceFile\")) {\r\n            this.playAudioData(chatVoice?.get(\"voiceFile\")?.get(\"url\"), chatVoice, eventMap);\r\n            chatVoice.save();\r\n            return chatVoice;\r\n        }\r\n\r\n        textOrSSML = this.extractSSMLContent(textOrSSML);\r\n        this.isPlaying = true;\r\n\r\n        return new Promise((resolve, reject) => {\r\n            const startTime = Date.now();\r\n\r\n            this.provider.synthesize(\r\n                textOrSSML,\r\n                async (result) => {\r\n                    const endTime = Date.now();\r\n                    const audioData = result?.audioData;\r\n\r\n                    let duration = Number(result?.audioDuration)\r\n                    if(this.config.provider==\"microsoft\"){\r\n                        duration = duration / 10000\r\n                    }\r\n\r\n                    chatVoice.set(\"duration\", duration);\r\n                    \r\n                    console.log(\"eventMap?.onResult\",eventMap,duration)\r\n                    eventMap?.onResult?.({duration:duration});\r\n                    \r\n                    // await this.playAudioData(audioData, chatVoice, eventMap); // 生成时已经在播放了，不需要再播放\r\n                    const blob = new Blob([audioData], { type: 'audio/wav' });\r\n                    await this.uploadAndSaveVoice(blob, chatVoice);\r\n                    console.log(`Audio synthesis finished. Duration: ${endTime - startTime} ms`);\r\n                    \r\n                    setTimeout(() => {\r\n                        this.isPlaying = false;\r\n                        eventMap?.onStop?.();\r\n                    }, 2000);\r\n                    \r\n                    resolve(chatVoice);\r\n                },\r\n                (error) => {\r\n                    this.isPlaying = false;\r\n                    reject(`Error occurred during synthesis: ${error}`);\r\n                },\r\n                ()=>{\r\n                    eventMap?.onStart?.();\r\n                }\r\n            );\r\n        });\r\n    }\r\n\r\n    private async playAudioData(audioData: string, chatVoice: Parse.Object, eventMap: any): Promise<void> {\r\n        let url: string;\r\n        \r\n        if (audioData?.indexOf?.(\"http\") >= -1) {\r\n            url = audioData;\r\n        } else {\r\n            const blob = new Blob([audioData], { type: 'audio/wav' });\r\n            url = URL.createObjectURL(blob);\r\n            await this.uploadAndSaveVoice(blob, chatVoice);\r\n        }\r\n\r\n        eventMap?.onStart?.(chatVoice);\r\n        this.isPlaying = true;\r\n        \r\n        const audioPlayer = this.audioPlayer;\r\n        audioPlayer.setAudioEvent(\"onloadeddata\", () => {\r\n            const duration = audioPlayer.duration * 1000;\r\n            if (!chatVoice?.get(\"duration\")) {\r\n                chatVoice.set(\"duration\", duration);\r\n                chatVoice.save();\r\n            }\r\n            eventMap?.onLoaded?.(audioPlayer);\r\n        });\r\n\r\n        // 设置事件监听\r\n        [\"onabort\", \"onerror\", \"onpause\", \"onended\", \"onclose\"].forEach(event => {\r\n            audioPlayer.setAudioEvent(event, () => {\r\n                this.isPlaying = false;\r\n                eventMap?.onStop?.();\r\n            });\r\n        });\r\n\r\n        const playAudio = () => {\r\n            audioPlayer.play(url).catch(() => {\r\n                setTimeout(playAudio, 200);\r\n            });\r\n        };\r\n        \r\n        playAudio();\r\n    }\r\n\r\n    private async uploadAndSaveVoice(blob: Blob, chatVoice: Parse.Object): Promise<string | null> {\r\n        if (!this.uploadServ) return null;\r\n\r\n        const id = chatVoice?.id || this.uploadServ.genMd5(chatVoice?.get(\"content\") || chatVoice?.get(\"ssml\"));\r\n        const now = new Date();\r\n        const filename = `${id}${now.getFullYear()}${now.getMonth() + 1}${now.getDate()}${now.getHours()}${now.getMinutes()}${now.getSeconds()}.wav`;\r\n        \r\n        const file = new File([blob], filename, { type: 'audio/wav' });\r\n        const fileResult = await this.uploadServ.upload(file);\r\n        \r\n        if (fileResult?.id) {\r\n            chatVoice.set(\"voiceFile\", { __type: \"Pointer\", className: \"Attachment\", objectId: fileResult.id });\r\n            await chatVoice.save();\r\n        }\r\n        \r\n        return fileResult?.url || null;\r\n    }\r\n\r\n    \r\n}"]}
189
+ //# sourceMappingURL=data:application/json;base64,{"version":3,"file":"fmode-tts-class.js","sourceRoot":"","sources":["../../../../../../../projects/fmode-ng/src/lib/aigc/voice/tts/fmode-tts-class.ts"],"names":[],"mappings":"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","sourcesContent":["import Parse from \"parse\";\r\nimport { NovaUploadService } from \"../../../storage/service-upload/nova-upload.service\";\r\nimport { AudioPlayer } from \"../lib/audio/audio.player\";\r\nimport { FmPushAudioOutputStreamCallback } from \"../lib/audio/streamer.microsoft\";\r\nimport { FmodeTTSEvent, FmodeTTSProvider } from \"./int-tts-provider\";\r\n// import { FmodeTTSProviderMicrosoft } from \"./provider-microsoft\";\r\n// class FmodeTTSProviderMicrosoft{}\r\nimport { PCMStreamer } from \"../lib/audio/streamer.pcm\";\r\nimport { AudioStreamer } from \"../lib/audio/audio.streamer\";\r\nimport { FmodeChatVoiceConfig } from \"../../../core/agent/chat/interface\";\r\n\r\n\r\n/**\r\n * FmodeTTS\r\n * 每个实例表示一次独立的语音合成服务\r\n */\r\nexport class FmodeTTS {\r\n    // 私有属性：全局配置 当前激活实例\r\n    // 实现全局仅一个TTS在播放\r\n    private static activeInstance: FmodeTTS | null = null;\r\n\r\n\r\n    private provider: FmodeTTSProvider;\r\n    eventMap: FmodeTTSEvent;\r\n    uploadServ?: NovaUploadService;\r\n    public isPlaying: boolean = false;\r\n    audioPlayer: AudioPlayer;\r\n    audioStream: AudioStreamer;\r\n    // 音色\r\n    voiceConfig:FmodeChatVoiceConfig|undefined\r\n\r\n    stop() {\r\n        if (FmodeTTS.activeInstance === this) {\r\n            FmodeTTS.activeInstance = null;\r\n        }\r\n        this.audioPlayer?.stop();\r\n        this.audioStream?.stop();\r\n        this.provider?.stop();\r\n        this.isPlaying = false;\r\n    }\r\n    config:any\r\n    constructor(config: {\r\n        region: string,\r\n        audioStream?:AudioStreamer,\r\n        subscriptionKey?: string,\r\n        token?: string,\r\n        provider?: FmodeTTSProvider // 允许传入自定义provider\r\n        voiceConfig?: FmodeChatVoiceConfig\r\n    }, uploadServ?: NovaUploadService) {\r\n        this.uploadServ = uploadServ;\r\n        this.config = config\r\n\r\n        this.audioPlayer = AudioPlayer.getInstance();\r\n        this.audioStream = new FmPushAudioOutputStreamCallback();\r\n\r\n        // 初始化provider\r\n        this.provider = config.provider // || new FmodeTTSProviderMicrosoft();\r\n        console.log(\"初始化provider\",config,this.provider,this.provider.constructor)\r\n        if(this.provider.constructor.toString().indexOf(\"Doubao\")>-1 ){\r\n            // 创建 PCM 流播放器\r\n            const pcmStreamer = new PCMStreamer(24000, 1, 16);\r\n            config.audioStream = pcmStreamer\r\n        }else{\r\n            config.audioStream = this.audioStream\r\n        }\r\n\r\n        this.audioStream = config?.audioStream || this.audioStream;\r\n\r\n        \r\n        this.provider.initialize(config);\r\n        this.provider.voiceConfig = config?.voiceConfig;\r\n        this.provider.eventMap = this.eventMap;\r\n\r\n    }\r\n\r\n    extractTextFromXML(inputString: string): string {\r\n        if (!inputString || typeof inputString !== 'string') {\r\n            return '';\r\n        }\r\n    \r\n        const regex = />([^<]+)</g;\r\n        const result: string[] = [];\r\n        let matches: RegExpExecArray | null;\r\n    \r\n        while ((matches = regex.exec(inputString)) !== null) {\r\n            const text = matches[1]?.trim();\r\n            if (text) {\r\n                result.push(text);\r\n            }\r\n        }\r\n    \r\n        return result.join(' ');\r\n    }\r\n    \r\n    extractSSMLContent(inputString: string): string {\r\n        if (!inputString || typeof inputString !== 'string') {\r\n            return inputString || '';\r\n        }\r\n    \r\n        const regex = /<speak.*?<\\/speak>/s;\r\n        const match = inputString.match(regex);\r\n        return match?.[0] || inputString;\r\n    }\r\n\r\n    public async speakAsync(textOrSSML: string, chatVoice?: Parse.Object, eventMap?: any): Promise<Parse.Object> {\r\n         // 停止当前活动的其他实例\r\n         if (FmodeTTS.activeInstance && FmodeTTS.activeInstance !== this) {\r\n            FmodeTTS.activeInstance.stop();\r\n        }\r\n        FmodeTTS.activeInstance = this;\r\n\r\n        if (!chatVoice) {\r\n            const ChatVoice = Parse.Object.extend(\"ChatVoice\");\r\n            chatVoice = new ChatVoice();\r\n            chatVoice.set(\"ssml\", textOrSSML);\r\n            chatVoice.set(\"content\", this.extractTextFromXML(textOrSSML));\r\n            const company = localStorage.getItem(\"company\");\r\n            company && chatVoice.set(\"company\", { __type: \"Pointer\", className: \"Company\", objectId: company });\r\n            Parse.User.current()?.id && chatVoice.set(\"user\", Parse.User.current().toPointer());\r\n        }\r\n\r\n        // 检查缓存\r\n        if (!chatVoice?.get(\"voiceFile\")) {\r\n            const orList = [];\r\n            // if (chatVoice?.get(\"content\")) {\r\n            //     orList.push({ content: chatVoice?.get(\"content\") });\r\n            // }\r\n            if (textOrSSML) {\r\n                orList.push({ ssml: textOrSSML });\r\n            }\r\n\r\n            const query = Parse.Query.fromJSON(\"ChatVoice\", {\r\n                include: \"voiceFile\",\r\n                where: { $or: orList }\r\n            });\r\n            query.exists(\"voiceFile\");\r\n            query.addDescending('updatedAt');\r\n            const exists = await query.first();\r\n            chatVoice.set(\"voiceFile\", exists?.get(\"voiceFile\"));\r\n        }\r\n\r\n        console.log(\"Exists playAudioData\", chatVoice?.get(\"voiceFile\"))\r\n        if (chatVoice?.get(\"voiceFile\")) {\r\n            this.playAudioData(chatVoice?.get(\"voiceFile\")?.get(\"url\"), chatVoice, eventMap);\r\n            chatVoice.save();\r\n            return chatVoice;\r\n        }\r\n\r\n        textOrSSML = this.extractSSMLContent(textOrSSML);\r\n        this.isPlaying = true;\r\n\r\n        return new Promise((resolve, reject) => {\r\n            const startTime = Date.now();\r\n\r\n            this.provider.synthesize(\r\n                textOrSSML,\r\n                async (result) => {\r\n                    const endTime = Date.now();\r\n                    const audioData = result?.audioData;\r\n\r\n                    let duration = Number(result?.audioDuration)\r\n                    if(this.config.provider==\"microsoft\"){\r\n                        duration = duration / 10000\r\n                    }\r\n\r\n                    chatVoice.set(\"duration\", duration);\r\n                    \r\n                    console.log(\"eventMap?.onResult\",eventMap,duration)\r\n                    eventMap?.onResult?.({duration:duration});\r\n                    \r\n                    // await this.playAudioData(audioData, chatVoice, eventMap); // 生成时已经在播放了，不需要再播放\r\n                    const blob = new Blob([audioData], { type: 'audio/wav' });\r\n                    await this.uploadAndSaveVoice(blob, chatVoice);\r\n                    console.log(`Audio synthesis finished. Duration: ${endTime - startTime} ms`);\r\n                    \r\n                    setTimeout(() => {\r\n                        this.isPlaying = false;\r\n                        eventMap?.onStop?.();\r\n                    }, 2000);\r\n                    \r\n                    resolve(chatVoice);\r\n                },\r\n                (error) => {\r\n                    this.isPlaying = false;\r\n                    reject(`Error occurred during synthesis: ${error}`);\r\n                },\r\n                ()=>{\r\n                    eventMap?.onStart?.();\r\n                }\r\n            );\r\n        });\r\n    }\r\n\r\n    private async playAudioData(audioData: string, chatVoice: Parse.Object, eventMap: any): Promise<void> {\r\n        let url: string;\r\n        \r\n        if (audioData?.indexOf?.(\"http\") >= -1) {\r\n            url = audioData;\r\n        } else {\r\n            const blob = new Blob([audioData], { type: 'audio/wav' });\r\n            url = URL.createObjectURL(blob);\r\n            await this.uploadAndSaveVoice(blob, chatVoice);\r\n        }\r\n\r\n        eventMap?.onStart?.(chatVoice);\r\n        this.isPlaying = true;\r\n        \r\n        const audioPlayer = this.audioPlayer;\r\n        audioPlayer.setAudioEvent(\"onloadeddata\", () => {\r\n            const duration = audioPlayer.duration * 1000;\r\n            if (!chatVoice?.get(\"duration\")) {\r\n                chatVoice.set(\"duration\", duration);\r\n                chatVoice.save();\r\n            }\r\n            eventMap?.onLoaded?.(audioPlayer);\r\n        });\r\n\r\n        // 设置事件监听\r\n        [\"onabort\", \"onerror\", \"onpause\", \"onended\", \"onclose\"].forEach(event => {\r\n            audioPlayer.setAudioEvent(event, () => {\r\n                this.isPlaying = false;\r\n                eventMap?.onStop?.();\r\n            });\r\n        });\r\n\r\n        const playAudio = () => {\r\n            audioPlayer.play(url).catch(() => {\r\n                setTimeout(playAudio, 200);\r\n            });\r\n        };\r\n        \r\n        playAudio();\r\n    }\r\n\r\n    private async uploadAndSaveVoice(blob: Blob, chatVoice: Parse.Object): Promise<string | null> {\r\n        if (!this.uploadServ) return null;\r\n\r\n        const id = chatVoice?.id || this.uploadServ.genMd5(chatVoice?.get(\"content\") || chatVoice?.get(\"ssml\"));\r\n        const now = new Date();\r\n        const filename = `${id}${now.getFullYear()}${now.getMonth() + 1}${now.getDate()}${now.getHours()}${now.getMinutes()}${now.getSeconds()}.wav`;\r\n        \r\n        const file = new File([blob], filename, { type: 'audio/wav' });\r\n        const fileResult = await this.uploadServ.upload(file);\r\n        \r\n        if (fileResult?.id) {\r\n            chatVoice.set(\"voiceFile\", { __type: \"Pointer\", className: \"Attachment\", objectId: fileResult.id });\r\n            await chatVoice.save();\r\n        }\r\n        \r\n        return fileResult?.url || null;\r\n    }\r\n\r\n    \r\n}"]}
@@ -1,5 +1,5 @@
1
1
  export * from "./fmode-tts-class";
2
2
  export * from "./provider-doubao";
3
- export * from "./provider-microsoft";
4
3
  export * from "../lib/audio/streamer.pcm";
5
- //# sourceMappingURL=data:application/json;base64,eyJ2ZXJzaW9uIjozLCJmaWxlIjoiaW5kZXguanMiLCJzb3VyY2VSb290IjoiIiwic291cmNlcyI6WyIuLi8uLi8uLi8uLi8uLi8uLi8uLi9wcm9qZWN0cy9mbW9kZS1uZy9zcmMvbGliL2FpZ2Mvdm9pY2UvdHRzL2luZGV4LnRzIl0sIm5hbWVzIjpbXSwibWFwcGluZ3MiOiJBQUFBLGNBQWMsbUJBQW1CLENBQUE7QUFDakMsY0FBYyxtQkFBbUIsQ0FBQTtBQUNqQyxjQUFjLHNCQUFzQixDQUFBO0FBQ3BDLGNBQWMsMkJBQTJCLENBQUEiLCJzb3VyY2VzQ29udGVudCI6WyJleHBvcnQgKiBmcm9tIFwiLi9mbW9kZS10dHMtY2xhc3NcIlxyXG5leHBvcnQgKiBmcm9tIFwiLi9wcm92aWRlci1kb3ViYW9cIlxyXG5leHBvcnQgKiBmcm9tIFwiLi9wcm92aWRlci1taWNyb3NvZnRcIlxyXG5leHBvcnQgKiBmcm9tIFwiLi4vbGliL2F1ZGlvL3N0cmVhbWVyLnBjbVwiIl19
4
+ // export * from "./provider-microsoft"
5
+ //# sourceMappingURL=data:application/json;base64,eyJ2ZXJzaW9uIjozLCJmaWxlIjoiaW5kZXguanMiLCJzb3VyY2VSb290IjoiIiwic291cmNlcyI6WyIuLi8uLi8uLi8uLi8uLi8uLi8uLi9wcm9qZWN0cy9mbW9kZS1uZy9zcmMvbGliL2FpZ2Mvdm9pY2UvdHRzL2luZGV4LnRzIl0sIm5hbWVzIjpbXSwibWFwcGluZ3MiOiJBQUFBLGNBQWMsbUJBQW1CLENBQUE7QUFDakMsY0FBYyxtQkFBbUIsQ0FBQTtBQUNqQyxjQUFjLDJCQUEyQixDQUFBO0FBRXpDLHVDQUF1QyIsInNvdXJjZXNDb250ZW50IjpbImV4cG9ydCAqIGZyb20gXCIuL2Ztb2RlLXR0cy1jbGFzc1wiXHJcbmV4cG9ydCAqIGZyb20gXCIuL3Byb3ZpZGVyLWRvdWJhb1wiXHJcbmV4cG9ydCAqIGZyb20gXCIuLi9saWIvYXVkaW8vc3RyZWFtZXIucGNtXCJcclxuXHJcbi8vIGV4cG9ydCAqIGZyb20gXCIuL3Byb3ZpZGVyLW1pY3Jvc29mdFwiIl19