fmode-ng 0.0.39 → 0.0.41
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.
- package/esm2022/fmode-ng.mjs +5 -10
- package/esm2022/lib/aigc/agent/agent.prompt.mjs +122 -10
- package/esm2022/lib/aigc/agent/index.mjs +2 -10
- package/esm2022/lib/aigc/avatar/avatar.module.mjs +45 -10
- package/esm2022/lib/aigc/avatar/comp-avatar-particle/avatar.role.mjs +2 -10
- package/esm2022/lib/aigc/avatar/comp-avatar-particle/comp-avatar-particle.component.mjs +315 -10
- package/esm2022/lib/aigc/avatar/comp-avatar-particle/index.mjs +3 -10
- package/esm2022/lib/aigc/avatar/comp-avatar-particle/role-points.class.mjs +57 -10
- package/esm2022/lib/aigc/avatar/comp-avatar-role-image/comp-avatar-role-image.component.mjs +97 -10
- package/esm2022/lib/aigc/avatar/comp-avatar-role-video/comp-avatar-role-video.component.mjs +104 -10
- package/esm2022/lib/aigc/avatar/comp-avatar-talk/comp-avatar-talk.component.mjs +111 -10
- package/esm2022/lib/aigc/avatar/index.mjs +8 -10
- package/esm2022/lib/aigc/avatar/interface-avatar-role.mjs +2 -10
- package/esm2022/lib/aigc/avatar/modal-chat-voice-input/modal-chat-voice-input.component.mjs +166 -8
- package/esm2022/lib/aigc/chat/chat-header-area/comp-header-area.component.mjs +36 -10
- package/esm2022/lib/aigc/chat/chat-header-area/index.mjs +2 -10
- package/esm2022/lib/aigc/chat/chat-list/chat-list.component.mjs +141 -8
- package/esm2022/lib/aigc/chat/chat-list/index.mjs +2 -10
- package/esm2022/lib/aigc/chat/chat-message-area/comp-message-area.component.mjs +40 -10
- package/esm2022/lib/aigc/chat/chat-message-area/index.mjs +2 -10
- package/esm2022/lib/aigc/chat/chat-message-card/comp-message-card.component.mjs +92 -10
- package/esm2022/lib/aigc/chat/chat-message-card/index.mjs +2 -10
- package/esm2022/lib/aigc/chat/chat-modal-input/index.mjs +2 -10
- package/esm2022/lib/aigc/chat/chat-modal-input/modal-audio-message/modal-audio-message.component.mjs +207 -8
- package/esm2022/lib/aigc/chat/chat-modal-input/modal-input.component.mjs +236 -10
- package/esm2022/lib/aigc/chat/chat-panel/chat-panel.component.mjs +137 -10
- package/esm2022/lib/aigc/chat/comp-role-prompt/comp-role-prompt.component.mjs +69 -10
- package/esm2022/lib/aigc/chat/comp-role-prompt/index.mjs +2 -10
- package/esm2022/lib/aigc/chat/index.mjs +8 -10
- package/esm2022/lib/aigc/comp-markdown-preview/clipboard.service.mjs +82 -10
- package/esm2022/lib/aigc/comp-markdown-preview/markdown-parse.mjs +269 -8
- package/esm2022/lib/aigc/comp-markdown-preview/markdown-preview.component.mjs +51 -10
- package/esm2022/lib/aigc/comp-markdown-preview/markdown-preview.module.mjs +24 -10
- package/esm2022/lib/aigc/comp-markdown-preview/plugins/md-mathjax/index.mjs +94 -10
- package/esm2022/lib/aigc/index.mjs +13 -10
- package/esm2022/lib/aigc/service-fmai/fmai.service.mjs +21 -10
- package/esm2022/lib/aigc/service-fmai/service-chat/chat-class.mjs +736 -8
- package/esm2022/lib/aigc/service-fmai/service-chat/chat.service.mjs +181 -8
- package/esm2022/lib/aigc/service-fmai/service-chat/index.mjs +7 -10
- package/esm2022/lib/aigc/service-fmai/service-chat/mask-list.mjs +194 -9
- package/esm2022/lib/aigc/service-fmai/service-chat/pipes/chat-content.pipe.mjs +27 -10
- package/esm2022/lib/aigc/service-fmai/service-chat/pipes/hidexml.pipe.mjs +27 -10
- package/esm2022/lib/aigc/service-fmai/service-chat/utilnow.pipe.mjs +68 -10
- package/esm2022/lib/aigc/service-fmai/service-imagine/imagine.service.mjs +229 -8
- package/esm2022/lib/aigc/service-fmai/service-imagine/index.mjs +2 -10
- package/esm2022/lib/aigc/voice/audio.player.mjs +52 -10
- package/esm2022/lib/aigc/voice/class-asr.mjs +79 -8
- package/esm2022/lib/aigc/voice/fmode-voice.service.mjs +501 -8
- package/esm2022/lib/aigc/voice/index.mjs +3 -10
- package/esm2022/lib/aigc/voice/lib/pcm2wav.mjs +38 -10
- package/esm2022/lib/aigc/voice/lib/resample.mjs +34 -10
- package/esm2022/lib/aigc/voice/tts/fmode-tts-class.mjs +233 -8
- package/esm2022/lib/aigc/voice/tts/index.mjs +2 -10
- package/esm2022/lib/map/comp-poi-picker/comp-poi-picker.component.mjs +190 -10
- package/esm2022/lib/map/comp-poi-picker/comp-poi-picker.module.mjs +33 -10
- package/esm2022/lib/map/index.mjs +4 -10
- package/esm2022/lib/map/map.module.mjs +61 -10
- package/esm2022/lib/map/page-loca-scatter/page-loca-scatter.component.mjs +110 -10
- package/esm2022/lib/map/page-map.start/page-map.start.component.mjs +98 -8
- package/esm2022/lib/map/page-plan-route/page-plan-route.component.mjs +100 -8
- package/esm2022/lib/nova-cloud/index.mjs +2 -10
- package/esm2022/lib/nova-cloud/nova-cloud.service.mjs +148 -10
- package/esm2022/lib/platform/cross.service.mjs +63 -10
- package/esm2022/lib/platform/index.mjs +2 -10
- package/esm2022/lib/social/index.mjs +2 -10
- package/esm2022/lib/social/wechat/wechat-jssdk.service.mjs +236 -8
- package/esm2022/lib/storage/comp-hwobs-manager/hwobs-manager.component.mjs +59 -10
- package/esm2022/lib/storage/index.mjs +5 -10
- package/esm2022/lib/storage/service-hwobs/hwobs.service.mjs +130 -8
- package/esm2022/lib/storage/service-upload/index.mjs +2 -10
- package/esm2022/lib/storage/service-upload/nova-upload.service.mjs +462 -8
- package/esm2022/lib/storage/service-upload/util-file-md5.mjs +28 -10
- package/esm2022/lib/storage/storage.module.mjs +41 -10
- package/esm2022/lib/user/account/account.service.mjs +221 -10
- package/esm2022/lib/user/captcha/captcha.component.mjs +135 -10
- package/esm2022/lib/user/comp-user-avatar/comp-user-avatar.component.mjs +62 -10
- package/esm2022/lib/user/index.mjs +17 -10
- package/esm2022/lib/user/login/auth.guard.mjs +28 -10
- package/esm2022/lib/user/login/auth.service.mjs +373 -8
- package/esm2022/lib/user/login/login.component.mjs +913 -10
- package/esm2022/lib/user/modal-user-login/modal-user-login.component.mjs +273 -10
- package/esm2022/lib/user/profile/auth-profile.guard.mjs +27 -10
- package/esm2022/lib/user/profile/auth-profile.service.mjs +122 -10
- package/esm2022/lib/user/profile/profile-bind/profile-bind.component.mjs +115 -10
- package/esm2022/lib/user/profile/profile.module.mjs +57 -10
- package/esm2022/lib/user/staff/index.mjs +4 -10
- package/esm2022/lib/user/staff/staff.guard.mjs +26 -10
- package/esm2022/lib/user/staff/staff.module.mjs +18 -10
- package/esm2022/lib/user/staff/staff.service.mjs +85 -10
- package/esm2022/lib/user/user-name.pipe.mjs +29 -10
- package/esm2022/lib/user/user.module.mjs +106 -10
- package/esm2022/lib/video/fm-video/fm-video.component.mjs +67 -10
- package/esm2022/lib/video/index.mjs +2 -10
- package/esm2022/public-api.mjs +13 -10
- package/fesm2022/fmode-ng.mjs +8895 -7
- package/fesm2022/fmode-ng.mjs.map +1 -1
- package/lib/user/login/auth.service.d.ts +18 -3
- package/lib/user/modal-user-login/modal-user-login.component.d.ts +4 -2
- package/package.json +1 -1
- package/LICENSE.md +0 -8
|
@@ -1,10 +1,503 @@
|
|
|
1
|
-
|
|
1
|
+
import { Injectable } from '@angular/core';
|
|
2
|
+
// import RecorderManager from "./lib/xunfei-recorder"
|
|
3
|
+
// import {RecorderManager} from "./lib/recorder/recorder-manager"
|
|
4
|
+
import Recorder from 'recorder-core';
|
|
5
|
+
import 'recorder-core/src/engine/pcm';
|
|
6
|
+
import 'recorder-core/src/engine/wav';
|
|
7
|
+
import 'recorder-core/src/extensions/waveview';
|
|
8
|
+
import CryptoJS from "crypto-js";
|
|
9
|
+
import { pcmtoWav } from './lib/pcm2wav';
|
|
10
|
+
import { convertFrameBufferToBase64, resampleBuffer } from './lib/resample';
|
|
11
|
+
import { WebSpeech } from './class-asr';
|
|
12
|
+
import { Platform } from '@ionic/angular';
|
|
13
|
+
import { Diagnostic } from '@awesome-cordova-plugins/diagnostic/ngx';
|
|
14
|
+
import * as i0 from "@angular/core";
|
|
15
|
+
import * as i1 from "@ionic/angular";
|
|
16
|
+
import * as i2 from "@awesome-cordova-plugins/diagnostic/ngx";
|
|
17
|
+
// import { FmodeTTSXunfei } fro./class-tts-xunfei.ts.bakfei'
|
|
18
|
+
export class FmodeVoiceService {
|
|
19
|
+
constructor(platform, diagnostic) {
|
|
20
|
+
this.platform = platform;
|
|
21
|
+
this.diagnostic = diagnostic;
|
|
22
|
+
/**
|
|
23
|
+
* 讯飞TTS语音合成
|
|
24
|
+
*/
|
|
25
|
+
// ttsXunfei = new FmodeTTSXunfei()
|
|
26
|
+
/**
|
|
27
|
+
* WebSpeech 语音库
|
|
28
|
+
*/
|
|
29
|
+
this.webSpeech = WebSpeech;
|
|
30
|
+
/**
|
|
31
|
+
* 用户操作:完成录音,并处理转录结果
|
|
32
|
+
*/
|
|
33
|
+
this.isUserFinish = false;
|
|
34
|
+
this.recordWavBlob = null;
|
|
35
|
+
this.recordPcmBlob = null;
|
|
36
|
+
this.recordDuration = 0;
|
|
37
|
+
this.recordType = "pcm";
|
|
38
|
+
this.encodingType = "raw";
|
|
39
|
+
this.connStatus = ""; // 录音中 建立连接中 关闭连接中
|
|
40
|
+
this.btnStatus = "UNDEFINED"; // "UNDEFINED" "CONNECTING" "OPEN" "CLOSING" "CLOSED"
|
|
41
|
+
this.resultText = "";
|
|
42
|
+
this.resultTextTemp = "";
|
|
43
|
+
/**
|
|
44
|
+
* 获取websocket url
|
|
45
|
+
* 该接口需要后端提供,这里为了方便前端处理
|
|
46
|
+
*/
|
|
47
|
+
this.APPID = "50f4a46c";
|
|
48
|
+
this.API_SECRET = "NzFlNmFhZDJjMDNkZGM3NzI0Mzg2OGNm";
|
|
49
|
+
this.API_KEY = "106ddc40dfd4b9ca6d7b47c70fada749";
|
|
50
|
+
this.requestPermission();
|
|
51
|
+
// this.recorder.on("start",() => {
|
|
52
|
+
// })
|
|
53
|
+
// this.recorder.on("frameRecorded", ({ isLastFrame, frameBuffer }) => {
|
|
54
|
+
// if (this.iatWS.readyState === this.iatWS.OPEN) {
|
|
55
|
+
// this.iatWS.send(
|
|
56
|
+
// JSON.stringify({
|
|
57
|
+
// data: {
|
|
58
|
+
// status: isLastFrame ? 2 : 1,
|
|
59
|
+
// format: "audio/L16;rate=16000",
|
|
60
|
+
// encoding: this.encodingType,
|
|
61
|
+
// audio: this.toBase64(frameBuffer),
|
|
62
|
+
// },
|
|
63
|
+
// })
|
|
64
|
+
// );
|
|
65
|
+
// if (isLastFrame) {
|
|
66
|
+
// this.changeBtnStatus("CLOSING");
|
|
67
|
+
// }
|
|
68
|
+
// }
|
|
69
|
+
// })
|
|
70
|
+
// this.recorder.on("stop",() => {
|
|
71
|
+
// })
|
|
72
|
+
}
|
|
73
|
+
/**()
|
|
74
|
+
* 用户操作:录音按钮快捷触发操作
|
|
75
|
+
*/
|
|
76
|
+
toggleRecord() {
|
|
77
|
+
console.log(this.btnStatus);
|
|
78
|
+
if (this.btnStatus === "UNDEFINED" || this.btnStatus === "CLOSED") {
|
|
79
|
+
// 开始讲话`
|
|
80
|
+
this.startTalk();
|
|
81
|
+
}
|
|
82
|
+
else if (this.btnStatus === "CONNECTING" || this.btnStatus === "OPEN") {
|
|
83
|
+
// 结束录音 并发送消息等待回复
|
|
84
|
+
this.finishTalk();
|
|
85
|
+
}
|
|
86
|
+
}
|
|
87
|
+
finishTalk() {
|
|
88
|
+
this.isUserFinish = true;
|
|
89
|
+
this.onBeforeFinishTalk && this.onBeforeFinishTalk();
|
|
90
|
+
this.recordStop();
|
|
91
|
+
}
|
|
92
|
+
/**
|
|
93
|
+
* 用户操作:开始讲话,实时转录语言
|
|
94
|
+
*/
|
|
95
|
+
async startTalk(evnet) {
|
|
96
|
+
this.resultText = "";
|
|
97
|
+
this.resultTextTemp = "";
|
|
98
|
+
this.onBeforeStartTalk && this.onBeforeStartTalk();
|
|
99
|
+
event?.preventDefault();
|
|
100
|
+
await this.openWithPriviledge();
|
|
101
|
+
setTimeout(() => {
|
|
102
|
+
this.connectWebSocket();
|
|
103
|
+
}, 100);
|
|
104
|
+
this.onAfterStartTalk && this.onAfterStartTalk();
|
|
105
|
+
}
|
|
106
|
+
/**
|
|
107
|
+
* 用户操作:取消讲话,且不发送结果
|
|
108
|
+
*/
|
|
109
|
+
cancelTalk() {
|
|
110
|
+
this.onBeforeCancelTalk && this.onBeforeCancelTalk();
|
|
111
|
+
this.recordStop(); // 停止录制
|
|
112
|
+
this.iatWS?.close(); // 断开连接
|
|
113
|
+
this.resultText = null; // 设置转录结果为空
|
|
114
|
+
this.onAfterCancelTalk && this.onAfterCancelTalk();
|
|
115
|
+
}
|
|
2
116
|
/**
|
|
3
|
-
*
|
|
4
|
-
* 版权所有 © 未来飞马 © 江西脑控科技有限公司 Copyright © Fmode Technology Co., Ltd.
|
|
5
|
-
* 保留所有权利 All Rights Reserved.
|
|
6
|
-
* /home/ryan/workspace/nova/nova-admin/dist/fmode-ng/esm2022/lib/aigc/voice/fmode-voice.service.mjs
|
|
117
|
+
* 程序逻辑
|
|
7
118
|
*/
|
|
8
|
-
|
|
9
|
-
|
|
10
|
-
|
|
119
|
+
async recordStart() {
|
|
120
|
+
this.createRecorder();
|
|
121
|
+
await this.openWithPriviledge();
|
|
122
|
+
this.recorder.start();
|
|
123
|
+
this.changeBtnStatus("OPEN");
|
|
124
|
+
this.onAfterRecordStart && this.onAfterRecordStart();
|
|
125
|
+
}
|
|
126
|
+
recordStop() {
|
|
127
|
+
return new Promise((resolve) => {
|
|
128
|
+
clearInterval(this.countdownInterval);
|
|
129
|
+
this.changeBtnStatus("CLOSED");
|
|
130
|
+
this.recorder?.stop(async (blob, duration) => {
|
|
131
|
+
//录音结束后,发送录音结束帧
|
|
132
|
+
try {
|
|
133
|
+
this.iatWS.send(JSON.stringify({ "data": { "status": 2, "format": "audio/L16;rate=16000", "encoding": this.encodingType, "audio": "" } }));
|
|
134
|
+
}
|
|
135
|
+
catch (err) { }
|
|
136
|
+
//简单利用URL生成本地文件地址,注意不用了时需要revokeObjectURL,否则霸占内存
|
|
137
|
+
//此地址只能本地使用,比如赋值给audio.src进行播放,赋值给a.href然后a.click()进行下载(a需提供download="xxx.mp3"属性)
|
|
138
|
+
let localUrl = (window.URL || webkitURL).createObjectURL(blob);
|
|
139
|
+
console.log(blob, localUrl, "时长:" + duration + "ms");
|
|
140
|
+
this.recordPcmBlob = blob;
|
|
141
|
+
this.recordWavBlob = await this.pcmBlobToWavBlob(blob, 44100);
|
|
142
|
+
this.recordDuration = duration;
|
|
143
|
+
console.log("this.recordWavBlob", this.recordWavBlob);
|
|
144
|
+
setTimeout(() => {
|
|
145
|
+
if (this.isUserFinish) {
|
|
146
|
+
this.onAfterFinishTalk && this.onAfterFinishTalk();
|
|
147
|
+
this.isUserFinish = false;
|
|
148
|
+
}
|
|
149
|
+
}, 2000);
|
|
150
|
+
this.recorder?.close(); //释放录音资源,当然可以不释放,后面可以连续调用start;但不释放时系统或浏览器会一直提示在录音,最佳操作是录完就close掉
|
|
151
|
+
this.recorder = null;
|
|
152
|
+
//已经拿到blob文件对象想干嘛就干嘛:立即播放、上传、下载保存
|
|
153
|
+
/*** 【立即播放例子】 ***/
|
|
154
|
+
// this.audioPlayer= new Audio();
|
|
155
|
+
// this.audioPlayer.controls=true;
|
|
156
|
+
// this.audioPlayer.src=localUrl;
|
|
157
|
+
console.log("localUrl", localUrl);
|
|
158
|
+
// 等待语音转义完成,处理转录结果
|
|
159
|
+
// 需要准确判断websocket已经close了
|
|
160
|
+
resolve(true);
|
|
161
|
+
}, (msg) => {
|
|
162
|
+
console.log("录音失败:" + msg);
|
|
163
|
+
this.recorder.close(); //可以通过stop方法的第3个参数来自动调用close
|
|
164
|
+
this.recorder = null;
|
|
165
|
+
resolve(null);
|
|
166
|
+
});
|
|
167
|
+
});
|
|
168
|
+
}
|
|
169
|
+
playRecord() {
|
|
170
|
+
this.playPCM(this.recordPcmBlob, 44100);
|
|
171
|
+
}
|
|
172
|
+
async pcmBlobToWavBlob(pcmBlob, sampleRate) {
|
|
173
|
+
return new Promise(resolve => {
|
|
174
|
+
let fileReader = new FileReader();
|
|
175
|
+
// 读取Blob数据
|
|
176
|
+
fileReader.onload = function (event) {
|
|
177
|
+
let pcmData = event.target.result;
|
|
178
|
+
// Convert PCM data to WAV format
|
|
179
|
+
let wavBlob = pcmtoWav(pcmData, sampleRate, 1, 16);
|
|
180
|
+
resolve(wavBlob);
|
|
181
|
+
};
|
|
182
|
+
// 将Blob转换为ArrayBuffer
|
|
183
|
+
fileReader.readAsArrayBuffer(pcmBlob);
|
|
184
|
+
});
|
|
185
|
+
}
|
|
186
|
+
async playPCM(pcmBlob, sampleRate) {
|
|
187
|
+
let wavBlob = await this.pcmBlobToWavBlob(pcmBlob, sampleRate);
|
|
188
|
+
let wavUrl = window.URL.createObjectURL(wavBlob);
|
|
189
|
+
let audio = new Audio();
|
|
190
|
+
audio.src = wavUrl;
|
|
191
|
+
audio.play();
|
|
192
|
+
}
|
|
193
|
+
async playBuffers() {
|
|
194
|
+
// let audioBuffer = []
|
|
195
|
+
// this.buffers.forEach(buffer=>{
|
|
196
|
+
// audioBuffer.push(buffer)
|
|
197
|
+
// })
|
|
198
|
+
let audioBlob = await this.BuffersToBlob(this.buffers);
|
|
199
|
+
this.playPCM(audioBlob, 44100);
|
|
200
|
+
}
|
|
201
|
+
BuffersToBlob(buffers) {
|
|
202
|
+
let audioBuffer = [];
|
|
203
|
+
buffers.forEach(buffer => {
|
|
204
|
+
buffer.forEach(int16 => {
|
|
205
|
+
audioBuffer.push(int16);
|
|
206
|
+
});
|
|
207
|
+
});
|
|
208
|
+
return new Blob([audioBuffer], { type: 'audio/pcm' });
|
|
209
|
+
}
|
|
210
|
+
// 将录音数据分割为每次发送的音频片段 1280字节
|
|
211
|
+
splitAudioData(audioData) {
|
|
212
|
+
const segmentSize = 1280; // 目标切片大小,单位为字节
|
|
213
|
+
const segmentCount = Math.ceil(audioData.length / segmentSize); // 切片数量
|
|
214
|
+
const segments = [];
|
|
215
|
+
for (let i = 0; i < segmentCount; i++) {
|
|
216
|
+
const start = i * segmentSize;
|
|
217
|
+
const end = start + segmentSize;
|
|
218
|
+
const segment = audioData.slice(start, end);
|
|
219
|
+
segments.push(segment);
|
|
220
|
+
}
|
|
221
|
+
return segments;
|
|
222
|
+
}
|
|
223
|
+
BufferToBlob(buffer) {
|
|
224
|
+
return new Blob([buffer], { type: 'audio/pcm' });
|
|
225
|
+
}
|
|
226
|
+
createRecorder() {
|
|
227
|
+
if (this.recorder)
|
|
228
|
+
return;
|
|
229
|
+
this.recorder = Recorder({
|
|
230
|
+
type: this.recordType, sampleRate: 44100, bitRate: 16 //pcm格式,指定采样率hz、比特率kbps,其他参数使用默认配置;注意:是数字的参数必须提供数字,不要用字符串;需要使用的type类型,需提前把格式支持文件加载进来,比如使用wav格式需要提前加载wav.js编码引擎
|
|
231
|
+
,
|
|
232
|
+
onProcess: (buffers, powerLevel, bufferDuration, bufferSampleRate, newBufferIdx, asyncEnd) => {
|
|
233
|
+
//录音实时回调,大约1秒调用12次本回调,buffers为开始到现在的所有录音pcm数据块(16位小端LE)
|
|
234
|
+
//可利用extensions/sonic.js插件实时变速变调,此插件计算量巨大,onProcess需要返回true开启异步模式
|
|
235
|
+
//可实时上传(发送)数据,配合Recorder.SampleData方法,将buffers中的新数据连续的转换成pcm上传,或使用mock方法将新数据连续的转码成其他格式上传,可以参考文档里面的:Demo片段列表 -> 实时转码并上传-通用版;基于本功能可以做到:实时转发数据、实时保存数据、实时语音识别(ASR)等
|
|
236
|
+
// 实时发送录音至接口
|
|
237
|
+
// console.log("asyncEnd:",asyncEnd)
|
|
238
|
+
// console.log(buffers.length,bufferDuration,bufferSampleRate,newBufferIdx)
|
|
239
|
+
let isLastFrame = false && this.btnStatus == "CLOSED"; // 主动关闭作为最后帧
|
|
240
|
+
// 录音默认为 44100 采样率 pcm不支持再process中实时转换
|
|
241
|
+
// 需要压缩为 16000 上传
|
|
242
|
+
let frameBuffer = buffers.length && buffers[buffers.length - 1];
|
|
243
|
+
this.buffers = buffers;
|
|
244
|
+
frameBuffer = resampleBuffer(frameBuffer, 44100, 16000);
|
|
245
|
+
// this.playPCM(this.BufferToBlob(frameBuffer),16000) // 切片数据播放,采样率测试
|
|
246
|
+
if (this.iatWS.readyState === this.iatWS.OPEN) {
|
|
247
|
+
if (this.disableASR)
|
|
248
|
+
return;
|
|
249
|
+
this.iatWS.send(JSON.stringify({
|
|
250
|
+
data: {
|
|
251
|
+
status: isLastFrame ? 2 : 1,
|
|
252
|
+
format: "audio/L16;rate=16000",
|
|
253
|
+
encoding: this.encodingType,
|
|
254
|
+
audio: convertFrameBufferToBase64(frameBuffer) // this.toBase64(audioData),
|
|
255
|
+
},
|
|
256
|
+
}));
|
|
257
|
+
if (isLastFrame) {
|
|
258
|
+
this.changeBtnStatus("CLOSING");
|
|
259
|
+
}
|
|
260
|
+
}
|
|
261
|
+
//可实时绘制波形(extensions目录内的waveview.js、wavesurfer.view.js、frequency.histogram.view.js插件功能)
|
|
262
|
+
this.waveClient?.input(buffers[buffers.length - 1], powerLevel, bufferSampleRate);
|
|
263
|
+
}
|
|
264
|
+
});
|
|
265
|
+
}
|
|
266
|
+
async openWithPriviledge() {
|
|
267
|
+
console.log(this.btnStatus);
|
|
268
|
+
await this.requestPermission();
|
|
269
|
+
this.createRecorder();
|
|
270
|
+
if (Recorder.IsOpen())
|
|
271
|
+
return true;
|
|
272
|
+
return new Promise(resolve => {
|
|
273
|
+
//var dialog=createDelayDialog(); 我们可以选择性的弹一个对话框:为了防止移动端浏览器存在第三种情况:用户忽略,并且(或者国产系统UC系)浏览器没有任何回调,此处demo省略了弹窗的代码
|
|
274
|
+
this.recorder.open(() => {
|
|
275
|
+
//dialog&&dialog.Cancel(); 如果开启了弹框,此处需要取消
|
|
276
|
+
//rec.start() 此处可以立即开始录音,但不建议这样编写,因为open是一个延迟漫长的操作,通过两次用户操作来分别调用open和start是推荐的最佳流程
|
|
277
|
+
//创建可视化,指定一个要显示的div
|
|
278
|
+
let waveDiv = document.querySelector(".record-wave");
|
|
279
|
+
if (waveDiv) { // 存在波形区域时,加载渲染组件
|
|
280
|
+
console.log(waveDiv);
|
|
281
|
+
if (Recorder.WaveView)
|
|
282
|
+
this.waveClient = Recorder.WaveView({ elem: ".record-wave" });
|
|
283
|
+
}
|
|
284
|
+
resolve(true);
|
|
285
|
+
}, (msg, isUserNotAllow) => {
|
|
286
|
+
//dialog&&dialog.Cancel(); 如果开启了弹框,此处需要取消
|
|
287
|
+
console.log((isUserNotAllow ? "UserNotAllow," : "") + "无法录音:" + msg);
|
|
288
|
+
});
|
|
289
|
+
});
|
|
290
|
+
}
|
|
291
|
+
getWebSocketUrl() {
|
|
292
|
+
// 请求地址根据语种不同变化
|
|
293
|
+
let url = "wss://iat-api.xfyun.cn/v2/iat";
|
|
294
|
+
let host = "iat-api.xfyun.cn";
|
|
295
|
+
let apiKey = this.API_KEY;
|
|
296
|
+
let apiSecret = this.API_SECRET;
|
|
297
|
+
let date = new Date().toUTCString();
|
|
298
|
+
let algorithm = "hmac-sha256";
|
|
299
|
+
let headers = "host date request-line";
|
|
300
|
+
let signatureOrigin = `host: ${host}\ndate: ${date}\nGET /v2/iat HTTP/1.1`;
|
|
301
|
+
let signatureSha = CryptoJS.HmacSHA256(signatureOrigin, apiSecret);
|
|
302
|
+
let signature = CryptoJS.enc.Base64.stringify(signatureSha);
|
|
303
|
+
let authorizationOrigin = `api_key="${apiKey}", algorithm="${algorithm}", headers="${headers}", signature="${signature}"`;
|
|
304
|
+
let authorization = btoa(authorizationOrigin);
|
|
305
|
+
url = `${url}?authorization=${authorization}&date=${date}&host=${host}`;
|
|
306
|
+
return url;
|
|
307
|
+
}
|
|
308
|
+
toBase64(buffer) {
|
|
309
|
+
var binary = "";
|
|
310
|
+
var bytes = new Uint8Array(buffer);
|
|
311
|
+
var len = bytes.byteLength;
|
|
312
|
+
for (var i = 0; i < len; i++) {
|
|
313
|
+
binary += String.fromCharCode(bytes[i]);
|
|
314
|
+
}
|
|
315
|
+
return window.btoa(binary);
|
|
316
|
+
}
|
|
317
|
+
/**
|
|
318
|
+
* 倒计时:最长支持60秒实时语音转录,如果持续7-10秒无声音,服务端将自动CLOSE链接
|
|
319
|
+
*/
|
|
320
|
+
countdown() {
|
|
321
|
+
let seconds = 60;
|
|
322
|
+
this.connStatus = `录音中(${seconds}s)`;
|
|
323
|
+
this.countdownInterval = setInterval(() => {
|
|
324
|
+
seconds = seconds - 1;
|
|
325
|
+
console.log(seconds);
|
|
326
|
+
if (seconds <= 0) {
|
|
327
|
+
clearInterval(this.countdownInterval);
|
|
328
|
+
this.recordStop();
|
|
329
|
+
}
|
|
330
|
+
else {
|
|
331
|
+
this.connStatus = `录音中(${seconds}s)`;
|
|
332
|
+
}
|
|
333
|
+
}, 1000);
|
|
334
|
+
}
|
|
335
|
+
changeBtnStatus(status) {
|
|
336
|
+
this.btnStatus = status;
|
|
337
|
+
if (status === "CONNECTING") {
|
|
338
|
+
this.connStatus = "建立连接中";
|
|
339
|
+
}
|
|
340
|
+
else if (status === "OPEN") {
|
|
341
|
+
this.countdown();
|
|
342
|
+
}
|
|
343
|
+
else if (status === "CLOSING") {
|
|
344
|
+
this.connStatus = "关闭连接中";
|
|
345
|
+
}
|
|
346
|
+
else if (status === "CLOSED") {
|
|
347
|
+
this.connStatus = "开始录音";
|
|
348
|
+
}
|
|
349
|
+
}
|
|
350
|
+
renderResult(resultData) {
|
|
351
|
+
// 识别结束
|
|
352
|
+
let jsonData = JSON.parse(resultData);
|
|
353
|
+
// console.log(jsonData.data.result)
|
|
354
|
+
if (jsonData.data && jsonData.data.result) {
|
|
355
|
+
let data = jsonData.data.result;
|
|
356
|
+
let str = "";
|
|
357
|
+
let ws = data.ws;
|
|
358
|
+
for (let i = 0; i < ws.length; i++) {
|
|
359
|
+
str = str + ws[i].cw[0].w;
|
|
360
|
+
console.log(str);
|
|
361
|
+
}
|
|
362
|
+
// 开启wpgs会有此字段(前提:在控制台开通动态修正功能)
|
|
363
|
+
// 取值为 "apd"时表示该片结果是追加到前面的最终结果;取值为"rpl" 时表示替换前面的部分结果,替换范围为rg字段
|
|
364
|
+
if (data.pgs) {
|
|
365
|
+
if (data.pgs === "apd") {
|
|
366
|
+
// 将this.resultTextTemp同步给this.resultText
|
|
367
|
+
this.resultText = this.resultTextTemp;
|
|
368
|
+
}
|
|
369
|
+
// 将结果存储在this.resultTextTemp中
|
|
370
|
+
this.resultTextTemp = this.resultText + str;
|
|
371
|
+
}
|
|
372
|
+
else {
|
|
373
|
+
this.resultText = this.resultText + str;
|
|
374
|
+
}
|
|
375
|
+
this.resultTextTemp || this.resultText || "";
|
|
376
|
+
console.log("diff temp", this.resultTextTemp);
|
|
377
|
+
console.log("diff result", this.resultText);
|
|
378
|
+
}
|
|
379
|
+
if (jsonData.code === 0 && jsonData.data.status === 2) {
|
|
380
|
+
this.iatWS.close();
|
|
381
|
+
}
|
|
382
|
+
if (jsonData.code !== 0) {
|
|
383
|
+
this.iatWS.close();
|
|
384
|
+
console.error(jsonData);
|
|
385
|
+
}
|
|
386
|
+
}
|
|
387
|
+
connectWebSocket() {
|
|
388
|
+
console.log("connectWebSocket");
|
|
389
|
+
const websocketUrl = this.getWebSocketUrl();
|
|
390
|
+
if ("WebSocket" in window) {
|
|
391
|
+
this.iatWS = new WebSocket(websocketUrl);
|
|
392
|
+
}
|
|
393
|
+
else if ("MozWebSocket" in window) {
|
|
394
|
+
// this.iatWS = new MozWebSocket(websocketUrl); // Moz兼容
|
|
395
|
+
}
|
|
396
|
+
else {
|
|
397
|
+
alert("浏览器不支持WebSocket");
|
|
398
|
+
return;
|
|
399
|
+
}
|
|
400
|
+
console.log("connectWebSocket", this.btnStatus);
|
|
401
|
+
this.changeBtnStatus("CONNECTING");
|
|
402
|
+
this.iatWS.onopen = (e) => {
|
|
403
|
+
// 开始录音
|
|
404
|
+
this.recordStart();
|
|
405
|
+
var params = {
|
|
406
|
+
common: {
|
|
407
|
+
app_id: this.APPID,
|
|
408
|
+
},
|
|
409
|
+
business: {
|
|
410
|
+
language: "zh_cn",
|
|
411
|
+
domain: "iat",
|
|
412
|
+
accent: "mandarin",
|
|
413
|
+
vad_eos: 5000,
|
|
414
|
+
dwa: "wpgs",
|
|
415
|
+
},
|
|
416
|
+
data: {
|
|
417
|
+
status: 0,
|
|
418
|
+
format: "audio/L16;rate=16000",
|
|
419
|
+
encoding: this.encodingType,
|
|
420
|
+
// encoding: "speex-wb",
|
|
421
|
+
},
|
|
422
|
+
};
|
|
423
|
+
this.iatWS.send(JSON.stringify(params));
|
|
424
|
+
};
|
|
425
|
+
this.iatWS.onmessage = (e) => {
|
|
426
|
+
console.log("onmessage" + this.resultText);
|
|
427
|
+
this.renderResult(e.data);
|
|
428
|
+
};
|
|
429
|
+
this.iatWS.onerror = (e) => {
|
|
430
|
+
console.error("error", e);
|
|
431
|
+
this.recordStop();
|
|
432
|
+
this.changeBtnStatus("CLOSED");
|
|
433
|
+
};
|
|
434
|
+
this.iatWS.onclose = async (e) => {
|
|
435
|
+
// 5秒停顿会导致结束录制
|
|
436
|
+
console.log("onclose" + this.resultText);
|
|
437
|
+
console.error("close", e);
|
|
438
|
+
await this.recordStop();
|
|
439
|
+
this.changeBtnStatus("CLOSED");
|
|
440
|
+
};
|
|
441
|
+
}
|
|
442
|
+
/**
|
|
443
|
+
* 移动端兼容方法
|
|
444
|
+
*/
|
|
445
|
+
/**
|
|
446
|
+
* 移动端上传专用方法
|
|
447
|
+
* @returns
|
|
448
|
+
*/
|
|
449
|
+
isCapacitor() {
|
|
450
|
+
return this.platform.is("capacitor") || this.platform.is("cordova");
|
|
451
|
+
}
|
|
452
|
+
async requestPermission() {
|
|
453
|
+
if (!this.isCapacitor())
|
|
454
|
+
return;
|
|
455
|
+
try {
|
|
456
|
+
await this.requestStoagePermission();
|
|
457
|
+
await this.requestCameraPermission();
|
|
458
|
+
await this.requestMicPermission();
|
|
459
|
+
await this.requestRecordAudioPermission();
|
|
460
|
+
}
|
|
461
|
+
catch (err) {
|
|
462
|
+
console.error(err);
|
|
463
|
+
}
|
|
464
|
+
}
|
|
465
|
+
async requestRecordAudioPermission() {
|
|
466
|
+
let data = await this.diagnostic.requestRuntimePermissions([this.diagnostic.permission.RECORD_AUDIO]);
|
|
467
|
+
console.log("record permission request:", data);
|
|
468
|
+
return;
|
|
469
|
+
}
|
|
470
|
+
async requestMicPermission() {
|
|
471
|
+
let isAvailable = await this.diagnostic.isMicrophoneAuthorized();
|
|
472
|
+
console.log("permisson_MIC:", isAvailable);
|
|
473
|
+
if (!isAvailable) {
|
|
474
|
+
let data = await this.diagnostic.requestMicrophoneAuthorization();
|
|
475
|
+
}
|
|
476
|
+
return;
|
|
477
|
+
}
|
|
478
|
+
async requestStoagePermission() {
|
|
479
|
+
let isAvailable = await this.diagnostic.isExternalStorageAuthorized();
|
|
480
|
+
console.log("permisson_STORAGE:", isAvailable);
|
|
481
|
+
if (!isAvailable) {
|
|
482
|
+
let data = await this.diagnostic.requestExternalStorageAuthorization();
|
|
483
|
+
}
|
|
484
|
+
return;
|
|
485
|
+
}
|
|
486
|
+
async requestCameraPermission() {
|
|
487
|
+
let isAvailable = await this.diagnostic.isCameraAuthorized();
|
|
488
|
+
console.log("permisson_Camera:", isAvailable);
|
|
489
|
+
if (!isAvailable) {
|
|
490
|
+
let data = await this.diagnostic.requestCameraAuthorization();
|
|
491
|
+
}
|
|
492
|
+
return;
|
|
493
|
+
}
|
|
494
|
+
static { this.ɵfac = i0.ɵɵngDeclareFactory({ minVersion: "12.0.0", version: "17.3.12", ngImport: i0, type: FmodeVoiceService, deps: [{ token: i1.Platform }, { token: i2.Diagnostic }], target: i0.ɵɵFactoryTarget.Injectable }); }
|
|
495
|
+
static { this.ɵprov = i0.ɵɵngDeclareInjectable({ minVersion: "12.0.0", version: "17.3.12", ngImport: i0, type: FmodeVoiceService, providedIn: 'root' }); }
|
|
496
|
+
}
|
|
497
|
+
i0.ɵɵngDeclareClassMetadata({ minVersion: "12.0.0", version: "17.3.12", ngImport: i0, type: FmodeVoiceService, decorators: [{
|
|
498
|
+
type: Injectable,
|
|
499
|
+
args: [{
|
|
500
|
+
providedIn: 'root'
|
|
501
|
+
}]
|
|
502
|
+
}], ctorParameters: () => [{ type: i1.Platform }, { type: i2.Diagnostic }] });
|
|
503
|
+
//# 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 { EventEmitter, Injectable, Output } from '@angular/core';\n// import RecorderManager from \"./lib/xunfei-recorder\"\n// import {RecorderManager} from \"./lib/recorder/recorder-manager\"\nimport Recorder from 'recorder-core'\nimport 'recorder-core/src/engine/pcm'\nimport 'recorder-core/src/engine/wav'\nimport 'recorder-core/src/extensions/waveview'\nimport CryptoJS from \"crypto-js\"\nimport { pcmtoWav } from './lib/pcm2wav';\nimport { convertFrameBufferToBase64, resampleAudio, resampleBuffer } from './lib/resample';\nimport { WebSpeech } from './class-asr';\nimport { Platform } from '@ionic/angular';\nimport { Diagnostic } from '@awesome-cordova-plugins/diagnostic/ngx';\n// import { FmodeTTSXunfei } fro./class-tts-xunfei.ts.bakfei'\n\n@Injectable({\n  providedIn: 'root'\n})\nexport class FmodeVoiceService {\n\n  /**\n   * 关闭ASR：用于测试其他功能时，可关闭ASR节省语音接口消耗\n   */\n  disableASR:false;\n  /**\n   * 讯飞TTS语音合成\n   */\n  // ttsXunfei = new FmodeTTSXunfei()\n  /**\n   * WebSpeech 语音库\n   */\n  webSpeech = WebSpeech\n  constructor(\n    private platform:Platform,\n    private diagnostic: Diagnostic, \n  ) { \n    this.requestPermission();\n    // this.recorder.on(\"start\",() => {\n    // })\n\n    // this.recorder.on(\"frameRecorded\", ({ isLastFrame, frameBuffer }) => {\n    //   if (this.iatWS.readyState === this.iatWS.OPEN) {\n    //     this.iatWS.send(\n    //       JSON.stringify({\n    //         data: {\n    //           status: isLastFrame ? 2 : 1,\n    //           format: \"audio/L16;rate=16000\",\n    //           encoding: this.encodingType,\n    //           audio: this.toBase64(frameBuffer),\n    //         },\n    //       })\n    //     );\n    //     if (isLastFrame) {\n    //       this.changeBtnStatus(\"CLOSING\");\n    //     }\n    //   }\n    // })\n\n    // this.recorder.on(\"stop\",() => {\n    // })\n\n\n  }\n\n\n\n  onResultTextChanged:Function\n  /**()\n   * 用户操作：录音按钮快捷触发操作\n   */\n   toggleRecord(){\n    console.log(this.btnStatus)\n    if (this.btnStatus === \"UNDEFINED\" || this.btnStatus === \"CLOSED\") {\n      // 开始讲话`\n      this.startTalk()\n    } else if (this.btnStatus === \"CONNECTING\" || this.btnStatus === \"OPEN\") {\n      // 结束录音 并发送消息等待回复\n      this.finishTalk()\n    }\n  }\n /**\n   * 用户操作：完成录音，并处理转录结果\n  */\n isUserFinish:boolean = false;\n finishTalk(){\n    this.isUserFinish = true;\n    this.onBeforeFinishTalk&&this.onBeforeFinishTalk()\n    this.recordStop()\n }\n onBeforeFinishTalk: Function\n onAfterFinishTalk: Function\n  /**\n   * 用户操作：开始讲话，实时转录语言\n   */\n  async startTalk(evnet?){\n    this.resultText = \"\"\n    this.resultTextTemp = \"\"\n    this.onBeforeStartTalk&&this.onBeforeStartTalk()\n    event?.preventDefault()\n    await this.openWithPriviledge()\n    setTimeout(() => {\n      this.connectWebSocket()\n    }, 100);\n\n    this.onAfterStartTalk&&this.onAfterStartTalk()\n  }\n  onBeforeStartTalk: Function\n  onAfterStartTalk: Function\n  \n  /**\n   * 用户操作：取消讲话，且不发送结果\n   */\n  cancelTalk(){\n    this.onBeforeCancelTalk&&this.onBeforeCancelTalk()\n    this.recordStop(); // 停止录制\n    this.iatWS?.close(); // 断开连接\n    this.resultText = null; // 设置转录结果为空\n    this.onAfterCancelTalk&&this.onAfterCancelTalk()\n  }\n  /**\n   * 操作回调\n   */\n  onBeforeCancelTalk: Function\n  onAfterCancelTalk: Function\n  onAfterRecordStart: Function\n\n  /**\n   * 程序逻辑\n   */\n  async recordStart(){\n    this.createRecorder();\n    await this.openWithPriviledge()\n    this.recorder.start();\n    this.changeBtnStatus(\"OPEN\");\n    this.onAfterRecordStart&&this.onAfterRecordStart()\n  }\n  recordStop(){\n    return new Promise((resolve)=>{\n      clearInterval(this.countdownInterval);\n      this.changeBtnStatus(\"CLOSED\");\n      this.recorder?.stop(async (blob:Blob,duration:number)=>{\n          //录音结束后，发送录音结束帧\n          try{\n            this.iatWS.send(\n              JSON.stringify({\"data\":{\"status\":2,\"format\":\"audio/L16;rate=16000\",\"encoding\":this.encodingType,\"audio\":\"\"}})\n              );\n          }catch(err){}\n\n          //简单利用URL生成本地文件地址，注意不用了时需要revokeObjectURL，否则霸占内存\n          //此地址只能本地使用，比如赋值给audio.src进行播放，赋值给a.href然后a.click()进行下载（a需提供download=\"xxx.mp3\"属性）\n          let localUrl=(window.URL||webkitURL).createObjectURL(blob);\n          console.log(blob,localUrl,\"时长:\"+duration+\"ms\");\n          this.recordPcmBlob = blob;\n          this.recordWavBlob = await this.pcmBlobToWavBlob(blob,44100);\n          this.recordDuration = duration;\n          console.log(\"this.recordWavBlob\",this.recordWavBlob)\n          setTimeout(() => {\n            if(this.isUserFinish){\n              this.onAfterFinishTalk&&this.onAfterFinishTalk()\n              this.isUserFinish = false;\n            }\n          }, 2000);\n          this.recorder?.close();//释放录音资源，当然可以不释放，后面可以连续调用start；但不释放时系统或浏览器会一直提示在录音，最佳操作是录完就close掉\n          this.recorder=null;\n          \n          //已经拿到blob文件对象想干嘛就干嘛：立即播放、上传、下载保存\n          \n          /*** 【立即播放例子】 ***/\n          // this.audioPlayer= new Audio();\n          // this.audioPlayer.controls=true;\n          // this.audioPlayer.src=localUrl;\n          console.log(\"localUrl\",localUrl)\n          // 等待语音转义完成，处理转录结果\n          // 需要准确判断websocket已经close了\n          resolve(true);\n      },(msg)=>{\n          console.log(\"录音失败:\"+msg);\n          this.recorder.close();//可以通过stop方法的第3个参数来自动调用close\n          this.recorder=null;\n          resolve(null)\n      })\n    })\n  }\n  recordWavBlob:Blob = null\n  recordPcmBlob:Blob = null\n  recordDuration:number = 0;\n  playRecord(){\n    this.playPCM(this.recordPcmBlob,44100)\n  }\n  async pcmBlobToWavBlob(pcmBlob, sampleRate):Promise<Blob>{\n    return new Promise(resolve=>{\n      let fileReader = new FileReader();\n      \n      // 读取Blob数据\n      fileReader.onload = function(event) {\n        let pcmData:any = event.target.result;\n        // Convert PCM data to WAV format\n        let wavBlob = pcmtoWav(pcmData, sampleRate,1,16);\n        resolve(wavBlob)\n      };\n      \n      // 将Blob转换为ArrayBuffer\n      fileReader.readAsArrayBuffer(pcmBlob);\n    })\n  }\n  async playPCM(pcmBlob, sampleRate) {\n    let wavBlob = await this.pcmBlobToWavBlob(pcmBlob,sampleRate);\n    let wavUrl = window.URL.createObjectURL(wavBlob)\n    let audio = new Audio();\n    audio.src = wavUrl;\n    audio.play();\n  }\n  buffers:Array<any>\n  async playBuffers(){\n    // let audioBuffer = []\n    // this.buffers.forEach(buffer=>{\n    //   audioBuffer.push(buffer)\n    // })\n    let audioBlob = await this.BuffersToBlob(this.buffers)\n    this.playPCM(audioBlob,44100)\n  }\n  BuffersToBlob(buffers) {\n    let audioBuffer:any = []\n    buffers.forEach(buffer=>{\n      buffer.forEach(int16=>{\n        audioBuffer.push(int16)\n      })\n    })\n    return new Blob([audioBuffer], { type: 'audio/pcm' });\n  }\n  // 将录音数据分割为每次发送的音频片段 1280字节\n   splitAudioData(audioData) {\n    const segmentSize = 1280; // 目标切片大小，单位为字节\n    const segmentCount = Math.ceil(audioData.length / segmentSize); // 切片数量\n    const segments = [];\n  \n    for (let i = 0; i < segmentCount; i++) {\n      const start = i * segmentSize;\n      const end = start + segmentSize;\n      const segment = audioData.slice(start, end);\n  \n      segments.push(segment);\n    }\n  \n    return segments;\n  }\n  BufferToBlob(buffer) {\n    return new Blob([buffer], { type: 'audio/pcm' });\n  }\n  recordType = \"pcm\"\n  encodingType = \"raw\"\n  createRecorder(){\n    if(this.recorder) return\n    this.recorder = Recorder({ //本配置参数请参考下面的文档，有详细介绍\n      type:this.recordType,sampleRate:44100,bitRate:16 //pcm格式，指定采样率hz、比特率kbps，其他参数使用默认配置；注意：是数字的参数必须提供数字，不要用字符串；需要使用的type类型，需提前把格式支持文件加载进来，比如使用wav格式需要提前加载wav.js编码引擎\n      ,onProcess:(buffers,powerLevel,bufferDuration,bufferSampleRate,newBufferIdx,asyncEnd)=>{\n          //录音实时回调，大约1秒调用12次本回调，buffers为开始到现在的所有录音pcm数据块(16位小端LE)\n          //可利用extensions/sonic.js插件实时变速变调，此插件计算量巨大，onProcess需要返回true开启异步模式\n          //可实时上传（发送）数据，配合Recorder.SampleData方法，将buffers中的新数据连续的转换成pcm上传，或使用mock方法将新数据连续的转码成其他格式上传，可以参考文档里面的：Demo片段列表 -> 实时转码并上传-通用版；基于本功能可以做到：实时转发数据、实时保存数据、实时语音识别（ASR）等\n          // 实时发送录音至接口\n          // console.log(\"asyncEnd:\",asyncEnd)\n          // console.log(buffers.length,bufferDuration,bufferSampleRate,newBufferIdx)\n          let isLastFrame = false && this.btnStatus==\"CLOSED\"; // 主动关闭作为最后帧\n          // 录音默认为 44100 采样率 pcm不支持再process中实时转换\n          // 需要压缩为 16000 上传\n          let frameBuffer = buffers.length&&buffers[buffers.length-1];\n          this.buffers = buffers\n          frameBuffer = resampleBuffer(frameBuffer,44100,16000);\n          // this.playPCM(this.BufferToBlob(frameBuffer),16000) // 切片数据播放，采样率测试\n\n          if (this.iatWS.readyState === this.iatWS.OPEN) {\n                if(this.disableASR) return\n                this.iatWS.send(\n                  JSON.stringify({\n                    data: {\n                      status: isLastFrame ? 2 : 1,\n                      format: \"audio/L16;rate=16000\",\n                      encoding: this.encodingType,\n                      audio: convertFrameBufferToBase64(frameBuffer)// this.toBase64(audioData),\n                    },\n                  })\n                );\n           \n            if (isLastFrame) {\n              this.changeBtnStatus(\"CLOSING\");\n            }\n          }\n          //可实时绘制波形（extensions目录内的waveview.js、wavesurfer.view.js、frequency.histogram.view.js插件功能）\n          this.waveClient?.input(buffers[buffers.length-1],powerLevel,bufferSampleRate);\n      }\n  });\n  }\n  \n  async openWithPriviledge(){\n    console.log(this.btnStatus)\n    await this.requestPermission();\n    this.createRecorder();\n    if(Recorder.IsOpen()) return true\n    return new Promise(resolve=>{\n      //var dialog=createDelayDialog(); 我们可以选择性的弹一个对话框：为了防止移动端浏览器存在第三种情况：用户忽略，并且（或者国产系统UC系）浏览器没有任何回调，此处demo省略了弹窗的代码\n      this.recorder.open(()=>{//打开麦克风授权获得相关资源\n        //dialog&&dialog.Cancel(); 如果开启了弹框，此处需要取消\n        //rec.start() 此处可以立即开始录音，但不建议这样编写，因为open是一个延迟漫长的操作，通过两次用户操作来分别调用open和start是推荐的最佳流程\n        //创建可视化，指定一个要显示的div\n        let waveDiv = document.querySelector(\".record-wave\");\n        if(waveDiv){ // 存在波形区域时，加载渲染组件\n          console.log(waveDiv)\n          if(Recorder.WaveView)this.waveClient=Recorder.WaveView({elem:\".record-wave\"});\n        }\n        resolve(true);\n    },(msg,isUserNotAllow)=>{//用户拒绝未授权或不支持\n        //dialog&&dialog.Cancel(); 如果开启了弹框，此处需要取消\n        console.log((isUserNotAllow?\"UserNotAllow，\":\"\")+\"无法录音:\"+msg);\n    });\n  })\n\n  }\n\n  connStatus = \"\" // 录音中 建立连接中 关闭连接中\n\n  btnStatus = \"UNDEFINED\"; // \"UNDEFINED\" \"CONNECTING\" \"OPEN\" \"CLOSING\" \"CLOSED\"\n\n  waveClient:any // 用户波形\n  recorder:any\n  // recorder = new RecorderManager();\n  // recorder = new RecorderManager(\"./lib/xunfei-recorder\");\n  \n  iatWS;\n  resultText = \"\";\n  resultTextTemp = \"\";\n  countdownInterval;\n\n  /**\n   * 获取websocket url\n   * 该接口需要后端提供，这里为了方便前端处理\n   */\n  APPID = \"50f4a46c\";\n  API_SECRET = \"NzFlNmFhZDJjMDNkZGM3NzI0Mzg2OGNm\";\n  API_KEY = \"106ddc40dfd4b9ca6d7b47c70fada749\";\n  getWebSocketUrl() {\n    // 请求地址根据语种不同变化\n    let url = \"wss://iat-api.xfyun.cn/v2/iat\";\n    let host = \"iat-api.xfyun.cn\";\n    \n    let apiKey = this.API_KEY;\n    let apiSecret = this.API_SECRET;\n    let date = new Date().toUTCString();\n    let algorithm = \"hmac-sha256\";\n    let headers = \"host date request-line\";\n    let signatureOrigin = `host: ${host}\\ndate: ${date}\\nGET /v2/iat HTTP/1.1`;\n    let signatureSha = CryptoJS.HmacSHA256(signatureOrigin, apiSecret);\n    let signature = CryptoJS.enc.Base64.stringify(signatureSha);\n    let authorizationOrigin = `api_key=\"${apiKey}\", algorithm=\"${algorithm}\", headers=\"${headers}\", signature=\"${signature}\"`;\n    let authorization = btoa(authorizationOrigin);\n    url = `${url}?authorization=${authorization}&date=${date}&host=${host}`;\n    return url;\n  }\n\n  toBase64(buffer) {\n    var binary = \"\";\n    var bytes = new Uint8Array(buffer);\n    var len = bytes.byteLength;\n    for (var i = 0; i < len; i++) {\n      binary += String.fromCharCode(bytes[i]);\n    }\n    return window.btoa(binary);\n  }\n\n  /**\n   * 倒计时：最长支持60秒实时语音转录，如果持续7-10秒无声音，服务端将自动CLOSE链接\n   */\n  countdown() {\n    let seconds = 60;\n    this.connStatus = `录音中（${seconds}s）`;\n    this.countdownInterval = setInterval(() => {\n      seconds = seconds - 1;\n      console.log(seconds);\n      if (seconds <= 0) {\n        clearInterval(this.countdownInterval);\n        this.recordStop();\n      } else {\n        this.connStatus = `录音中（${seconds}s）`;\n      }\n    }, 1000);\n  }\n\n  changeBtnStatus(status) {\n    this.btnStatus = status;\n    if (status === \"CONNECTING\") {\n      this.connStatus = \"建立连接中\";\n    } else if (status === \"OPEN\") {\n      this.countdown();\n    } else if (status === \"CLOSING\") {\n      this.connStatus = \"关闭连接中\";\n    } else if (status === \"CLOSED\") {\n      this.connStatus = \"开始录音\";\n    }\n  }\n\n  renderResult(resultData) {\n    // 识别结束\n    let jsonData = JSON.parse(resultData);\n    // console.log(jsonData.data.result)\n    if (jsonData.data && jsonData.data.result) {\n      let data = jsonData.data.result;\n      let str = \"\";\n      let ws = data.ws;\n      for (let i = 0; i < ws.length; i++) {\n        str = str + ws[i].cw[0].w;\n        console.log(str)\n      }\n      // 开启wpgs会有此字段(前提：在控制台开通动态修正功能)\n      // 取值为 \"apd\"时表示该片结果是追加到前面的最终结果；取值为\"rpl\" 时表示替换前面的部分结果，替换范围为rg字段\n      if (data.pgs) {\n        if (data.pgs === \"apd\") {\n          // 将this.resultTextTemp同步给this.resultText\n          this.resultText = this.resultTextTemp;\n        }\n        // 将结果存储在this.resultTextTemp中\n        this.resultTextTemp = this.resultText + str;\n      } else {\n        this.resultText = this.resultText + str;\n      }\n        this.resultTextTemp || this.resultText || \"\";\n      console.log(\"diff temp\",this.resultTextTemp)\n      console.log(\"diff result\",this.resultText)\n    }\n    if (jsonData.code === 0 && jsonData.data.status === 2) {\n      this.iatWS.close();\n    }\n    if (jsonData.code !== 0) {\n      this.iatWS.close();\n      console.error(jsonData);\n    }\n  }\n\n  connectWebSocket() {\n    console.log(\"connectWebSocket\");\n    const websocketUrl = this.getWebSocketUrl();\n    if (\"WebSocket\" in window) {\n      this.iatWS = new WebSocket(websocketUrl);\n    } else if (\"MozWebSocket\" in window) {\n      // this.iatWS = new MozWebSocket(websocketUrl); // Moz兼容\n    } else {\n      alert(\"浏览器不支持WebSocket\");\n      return;\n    }\n    console.log(\"connectWebSocket\",this.btnStatus);\n    this.changeBtnStatus(\"CONNECTING\");\n    this.iatWS.onopen = (e) => {\n      // 开始录音\n      this.recordStart();\n      var params = {\n        common: {\n          app_id: this.APPID,\n        },\n        business: {\n          language: \"zh_cn\",\n          domain: \"iat\",\n          accent: \"mandarin\",\n          vad_eos: 5000,\n          dwa: \"wpgs\",\n        },\n        data: {\n          status: 0,\n          format: \"audio/L16;rate=16000\",\n          encoding: this.encodingType,\n          // encoding: \"speex-wb\",\n        },\n      };\n      this.iatWS.send(JSON.stringify(params));\n    };\n    this.iatWS.onmessage = (e) => {\n      console.log(\"onmessage\"+this.resultText)\n      this.renderResult(e.data);\n    };\n    this.iatWS.onerror = (e) => {\n      console.error(\"error\",e);\n      this.recordStop();\n      this.changeBtnStatus(\"CLOSED\");\n    };\n    this.iatWS.onclose = async (e) => {\n      // 5秒停顿会导致结束录制\n      console.log(\"onclose\"+this.resultText)\n      console.error(\"close\",e)\n      await this.recordStop();\n      this.changeBtnStatus(\"CLOSED\");\n\n    };\n  }\n\n  /**\n   * 移动端兼容方法\n   */\n\n   /**\n   * 移动端上传专用方法\n   * @returns \n   */\n    isCapacitor(){\n      return this.platform.is(\"capacitor\") || this.platform.is(\"cordova\")\n    }\n    async requestPermission(){\n      if(!this.isCapacitor()) return\n      try{\n        await this.requestStoagePermission();\n        await this.requestCameraPermission();\n        await this.requestMicPermission();\n        await this.requestRecordAudioPermission();\n      }catch(err){\n        console.error(err)\n      }\n    }\n    async requestRecordAudioPermission(){\n      let data = await this.diagnostic.requestRuntimePermissions([this.diagnostic.permission.RECORD_AUDIO])\n      console.log(\"record permission request:\",data)\n      return\n  }\n    async requestMicPermission(){\n        let isAvailable = await this.diagnostic.isMicrophoneAuthorized()\n        console.log(\"permisson_MIC:\", isAvailable)\n        if (!isAvailable) {\n          let data = await this.diagnostic.requestMicrophoneAuthorization()\n        }\n        return\n    }\n    async requestStoagePermission(){\n        let isAvailable = await this.diagnostic.isExternalStorageAuthorized()\n        console.log(\"permisson_STORAGE:\", isAvailable)\n        if (!isAvailable) {\n          let data = await this.diagnostic.requestExternalStorageAuthorization()\n        }\n        return\n    }\n    async requestCameraPermission(){\n      let isAvailable = await this.diagnostic.isCameraAuthorized()\n      console.log(\"permisson_Camera:\", isAvailable)\n      if (!isAvailable) {\n        let data = await this.diagnostic.requestCameraAuthorization()\n      }\n      return\n  }\n}\n"]}
|
|
@@ -1,10 +1,3 @@
|
|
|
1
|
-
|
|
2
|
-
|
|
3
|
-
|
|
4
|
-
* 版权所有 © 未来飞马 © 江西脑控科技有限公司 Copyright © Fmode Technology Co., Ltd.
|
|
5
|
-
* 保留所有权利 All Rights Reserved.
|
|
6
|
-
* /home/ryan/workspace/nova/nova-admin/dist/fmode-ng/esm2022/lib/aigc/voice/index.mjs
|
|
7
|
-
*/
|
|
8
|
-
export*from"./fmode-voice.service";export*from"./audio.player";
|
|
9
|
-
var MODULE_PATH_NEED = `6K+l5paH5Lu25piv5pys6aG555uu55qE5LiA6YOo5YiGIFRoaXMgZmlsZSBpcyBwYXJ0IG9mIHRoZSBDb21wb25lbnRzIGluIEZtb2RlIEluYy4KICAgIOeJiOadg+aJgOaciSDCqSDmnKrmnaXpo57pqawgwqkg5rGf6KW/6ISR5o6n56eR5oqA5pyJ6ZmQ5YWs5Y+4IENvcHlyaWdodCDCqSBGbW9kZSBUZWNobm9sb2d5IENvLiwgTHRkLgogICAg5L+d55WZ5omA5pyJ5p2D5YipIEFsbCBSaWdodHMgUmVzZXJ2ZWQuCiAgICDkuKXnpoHlnKjmnKrnu4/mjojmnYPnmoTmg4XlhrXkuIvvvIzpgJrov4fku7vkvZXlqpLku4vlpI3liLbmraTmlofku7YgVW5hdXRob3JpemVkIGNvcHlpbmcgb2YgdGhpcyBmaWxlLCB2aWEgYW55IG1lZGl1bSBpcyBzdHJpY3RseSBwcm9oaWJpdGVkCiAgICDor6Xmlofku7bmmK/kuJPmnInnmoTmnLrlr4bmlofku7YgUHJvcHJpZXRhcnkgYW5kIGNvbmZpZGVudGlhbAogICAKICAgIENvcHlyaWdodCAyMDIxLW5vdyBGbW9kZSBJbmMuIHN1cHBvcnRAZm1vZGUuY24uIDE4NjA3MDA3MDczLgogICAg5L+d55WZ5omA5pyJ5p2D5YipIEFsbCByaWdodHMgcmVzZXJ2ZWQuCgogICAgUEFUSDovaG9tZS9yeWFuL3dvcmtzcGFjZS9ub3ZhL25vdmEtYWRtaW4vZGlzdC9mbW9kZS1uZy9lc20yMDIyL2xpYi9haWdjL3ZvaWNlL2luZGV4Lm1qcw==`
|
|
10
|
-
|
|
1
|
+
export * from "./fmode-voice.service";
|
|
2
|
+
export * from "./audio.player";
|
|
3
|
+
//# sourceMappingURL=data:application/json;base64,eyJ2ZXJzaW9uIjozLCJmaWxlIjoiaW5kZXguanMiLCJzb3VyY2VSb290IjoiIiwic291cmNlcyI6WyIuLi8uLi8uLi8uLi8uLi8uLi9wcm9qZWN0cy9mbW9kZS1uZy9zcmMvbGliL2FpZ2Mvdm9pY2UvaW5kZXgudHMiXSwibmFtZXMiOltdLCJtYXBwaW5ncyI6IkFBQUEsY0FBYyx1QkFBdUIsQ0FBQTtBQUNyQyxjQUFjLGdCQUFnQixDQUFBIiwic291cmNlc0NvbnRlbnQiOlsiZXhwb3J0ICogZnJvbSBcIi4vZm1vZGUtdm9pY2Uuc2VydmljZVwiXG5leHBvcnQgKiBmcm9tIFwiLi9hdWRpby5wbGF5ZXJcIiJdfQ==
|