@easyv/biz-components 0.0.49 → 0.0.51-beta.0

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.
@@ -1,5 +1,5 @@
1
- var propTypes = { exports: {} };
1
+ var coWebWorker = { exports: {} };
2
2
  export {
3
- propTypes as __module
3
+ coWebWorker as __module
4
4
  };
5
5
  //# sourceMappingURL=index.es2.js.map
@@ -1,5 +1,5 @@
1
- var classnames = { exports: {} };
1
+ var propTypes = { exports: {} };
2
2
  export {
3
- classnames as __module
3
+ propTypes as __module
4
4
  };
5
5
  //# sourceMappingURL=index.es3.js.map
@@ -1,5 +1,5 @@
1
- var coWebWorker = { exports: {} };
1
+ var classnames = { exports: {} };
2
2
  export {
3
- coWebWorker as __module
3
+ classnames as __module
4
4
  };
5
5
  //# sourceMappingURL=index.es4.js.map
@@ -1,5 +1,5 @@
1
1
  import { getDefaultExportFromCjs } from "../../../../../_virtual/_commonjsHelpers.es.js";
2
- import { __module as classnames$1 } from "../../../../../_virtual/index.es3.js";
2
+ import { __module as classnames$1 } from "../../../../../_virtual/index.es4.js";
3
3
  /*!
4
4
  Copyright (c) 2018 Jed Watson.
5
5
  Licensed under the MIT License (MIT), see
@@ -1,5 +1,5 @@
1
1
  import { getDefaultExportFromCjs } from "../../../../../_virtual/_commonjsHelpers.es.js";
2
- import { __module as coWebWorker } from "../../../../../_virtual/index.es4.js";
2
+ import { __module as coWebWorker } from "../../../../../_virtual/index.es2.js";
3
3
  class CrossOriginWorker extends Worker {
4
4
  constructor(scriptUrl) {
5
5
  const b = new Blob([`importScripts('${new URL(scriptUrl).toString()}')`], { type: "application/javascript" });
@@ -1,5 +1,5 @@
1
1
  import { getDefaultExportFromCjs } from "../../../../../_virtual/_commonjsHelpers.es.js";
2
- import { __module as propTypes } from "../../../../../_virtual/index.es2.js";
2
+ import { __module as propTypes } from "../../../../../_virtual/index.es3.js";
3
3
  import { __require as requireReactIs } from "../../../react-is@16.13.1/node_modules/react-is/index.es.js";
4
4
  import { __require as requireFactoryWithTypeCheckers } from "./factoryWithTypeCheckers.es.js";
5
5
  import { __require as requireFactoryWithThrowingShims } from "./factoryWithThrowingShims.es.js";
@@ -18,14 +18,14 @@ export interface AudiManagerConfig {
18
18
  export declare class AudioManager {
19
19
  workletUrl?: string;
20
20
  sampleRate?: SampleRate;
21
- status: 'init' | 'starting' | 'running' | 'closed';
22
21
  audioContext?: AudioContext;
23
22
  mediaSource?: MediaStreamAudioSourceNode;
24
23
  mediaStream?: MediaStream;
25
24
  config: AudiManagerConfig;
26
25
  constructor(config: AudiManagerConfig);
26
+ init(): Promise<void>;
27
27
  start(): Promise<void>;
28
28
  stop(): Promise<void>;
29
29
  close(): Promise<void> | undefined;
30
- destroy(): void;
30
+ destroy(): Promise<void>;
31
31
  }
@@ -13,7 +13,6 @@ class AudioManager {
13
13
  constructor(config) {
14
14
  __publicField(this, "workletUrl");
15
15
  __publicField(this, "sampleRate", defaultSampleRate);
16
- __publicField(this, "status", "init");
17
16
  __publicField(this, "audioContext");
18
17
  __publicField(this, "mediaSource");
19
18
  __publicField(this, "mediaStream");
@@ -24,37 +23,48 @@ class AudioManager {
24
23
  this.sampleRate = config.sampleRate;
25
24
  this.config = config;
26
25
  }
26
+ async init() {
27
+ if (this.audioContext) {
28
+ await this.destroy();
29
+ }
30
+ const audioContext = new window.AudioContext({
31
+ sampleRate: this.sampleRate
32
+ });
33
+ if (!this.workletUrl) {
34
+ throw Error("加载 workletProcessor 失败!");
35
+ }
36
+ await audioContext.audioWorklet.addModule(this.workletUrl);
37
+ const audioWorkletNode = new AudioWorkletNode(audioContext, "audio-worklet-processor");
38
+ const mediaStream = await requestMicrophonePermission();
39
+ const mediaSource = audioContext.createMediaStreamSource(mediaStream);
40
+ audioWorkletNode.port.postMessage({
41
+ type: "init",
42
+ frameSize: this.config.frameSize
43
+ });
44
+ audioWorkletNode.port.onmessage = (e) => {
45
+ var _a, _b;
46
+ (_b = (_a = this.config).onFrameRecorded) == null ? void 0 : _b.call(_a, e.data);
47
+ };
48
+ mediaSource.connect(audioWorkletNode);
49
+ this.audioContext = audioContext;
50
+ this.mediaSource = mediaSource;
51
+ this.mediaStream = mediaStream;
52
+ }
27
53
  async start() {
28
- var _a, _b;
54
+ var _a, _b, _c, _d;
29
55
  try {
30
- if (this.status === "starting") {
31
- return;
56
+ console.log(
57
+ "%c 🐶 dog ==== audio manager start:",
58
+ "color: green; font-size: 16px;",
59
+ this.audioContext
60
+ );
61
+ if (((_a = this.audioContext) == null ? void 0 : _a.state) === "suspended") {
62
+ this.audioContext.resume();
32
63
  }
33
- this.status = "starting";
34
- const audioContext = new window.AudioContext({
35
- sampleRate: this.sampleRate
36
- });
37
- if (!this.workletUrl) {
38
- throw Error("加载 workletProcessor 失败!");
64
+ if (((_b = this.audioContext) == null ? void 0 : _b.state) === "closed") {
65
+ await this.init();
39
66
  }
40
- await audioContext.audioWorklet.addModule(this.workletUrl);
41
- const audioWorkletNode = new AudioWorkletNode(audioContext, "audio-worklet-processor");
42
- const mediaStream = await requestMicrophonePermission();
43
- const mediaSource = audioContext.createMediaStreamSource(mediaStream);
44
- audioWorkletNode.port.postMessage({
45
- type: "init",
46
- frameSize: this.config.frameSize
47
- });
48
- audioWorkletNode.port.onmessage = (e) => {
49
- var _a2, _b2;
50
- (_b2 = (_a2 = this.config).onFrameRecorded) == null ? void 0 : _b2.call(_a2, e.data);
51
- };
52
- mediaSource.connect(audioWorkletNode);
53
- this.audioContext = audioContext;
54
- this.mediaSource = mediaSource;
55
- this.mediaStream = mediaStream;
56
- this.status = "running";
57
- (_b = (_a = this.config).onStart) == null ? void 0 : _b.call(_a);
67
+ (_d = (_c = this.config).onStart) == null ? void 0 : _d.call(_c);
58
68
  } catch (error) {
59
69
  console.error("Failed to start recording:", error);
60
70
  throw error;
@@ -62,23 +72,21 @@ class AudioManager {
62
72
  }
63
73
  async stop() {
64
74
  var _a, _b, _c;
65
- this.status = "closed";
66
75
  if (((_a = this.audioContext) == null ? void 0 : _a.state) === "running") {
67
76
  await this.audioContext.suspend();
68
77
  }
69
78
  (_c = (_b = this.config).onStop) == null ? void 0 : _c.call(_b);
70
79
  }
71
80
  close() {
72
- var _a, _b, _c;
81
+ var _a;
73
82
  (_a = this.mediaSource) == null ? void 0 : _a.disconnect();
74
- this.status = "closed";
75
- if (((_b = this.audioContext) == null ? void 0 : _b.state) !== "closed") {
76
- return (_c = this.audioContext) == null ? void 0 : _c.close();
83
+ if (this.audioContext && this.audioContext.state !== "closed") {
84
+ return this.audioContext.close();
77
85
  }
78
86
  }
79
- destroy() {
87
+ async destroy() {
80
88
  var _a, _b;
81
- this.close();
89
+ await this.close();
82
90
  (_b = (_a = this.mediaStream) == null ? void 0 : _a.getTracks()) == null ? void 0 : _b.forEach((track) => track.stop());
83
91
  }
84
92
  }
@@ -1 +1 @@
1
- {"version":3,"file":"AudioManager.es.js","sources":["../../../../src/utils/voskBrowserManager/AudioManager/AudioManager.ts"],"sourcesContent":["import { requestMicrophonePermission } from '../../xunFeiVoice/utils';\nimport { SampleRate } from './types';\nimport defaultWorkletProcessorStr from './workletProcessor.js?raw';\n\nexport const defaultSampleRate = 48000;\n\nexport interface AudiManagerConfig {\n /** 一个转换成 string js 文件。其内容要符合 AudioWorkletProcessor 的要求。\n * - 需使用 registerProcessor 注册,第一个参数为 “audio-worklet-processor”,如: registerProcessor('audio-worklet-processor', audioToFloat32Processor);\n * - 参考:https://developer.mozilla.org/en-US/docs/Web/API/AudioWorkletProcessor\n */\n workletProcessor?: string;\n /** 采样率(每秒采样数)的浮点数,常用值:16000, 32000, 44100, 48000 */\n sampleRate?: SampleRate;\n onFrameRecorded?: (data: Float32Array) => void;\n onStart?: () => void;\n onStop?: () => void;\n frameSize?: number;\n}\n\nconst generateWorkletUrl = (jsStr: string) => {\n const blob = new Blob([jsStr], { type: 'application/javascript' });\n const scriptUrl = URL.createObjectURL(blob);\n return scriptUrl;\n};\n\n/** 对实时音频录制进行管理。 */\nexport class AudioManager {\n workletUrl?: string;\n sampleRate?: SampleRate = defaultSampleRate;\n status: 'init' | 'starting' | 'running' | 'closed' = 'init';\n audioContext?: AudioContext;\n mediaSource?: MediaStreamAudioSourceNode;\n mediaStream?: MediaStream;\n config: AudiManagerConfig = {};\n\n constructor(config: AudiManagerConfig) {\n console.log('audio manager constructor: ====', config);\n const processorStr = config.workletProcessor || defaultWorkletProcessorStr;\n this.workletUrl = generateWorkletUrl(processorStr);\n this.sampleRate = config.sampleRate;\n this.config = config;\n }\n\n async start() {\n try {\n if (this.status === 'starting') {\n return;\n }\n this.status = 'starting';\n // 1. 创建 AudioContext\n const audioContext = new window.AudioContext({\n sampleRate: this.sampleRate,\n });\n if (!this.workletUrl) {\n throw Error('加载 workletProcessor 失败!');\n }\n // 加载给定 JavaScript 文件中的模块,并将其添加到当前 Worklet 中\n await audioContext.audioWorklet.addModule(this.workletUrl);\n const audioWorkletNode = new AudioWorkletNode(audioContext, 'audio-worklet-processor');\n // 获取音频流并创建处理节点\n const mediaStream = await requestMicrophonePermission();\n const mediaSource = audioContext.createMediaStreamSource(mediaStream);\n // 发送配置给 workletProcessor\n audioWorkletNode.port.postMessage({\n type: 'init',\n frameSize: this.config.frameSize,\n });\n // 监听消息\n audioWorkletNode.port.onmessage = (e) => {\n this.config.onFrameRecorded?.(e.data);\n };\n // 连接音频节点\n mediaSource.connect(audioWorkletNode);\n // 赋值到 this 上\n this.audioContext = audioContext;\n this.mediaSource = mediaSource;\n this.mediaStream = mediaStream;\n this.status = 'running';\n this.config.onStart?.();\n } catch (error) {\n console.error('Failed to start recording:', error);\n throw error;\n }\n }\n\n async stop() {\n this.status = 'closed';\n /** 暂停音频上下文对象中的进度,并暂时剥离进程对音频设备硬件的访问权限,减少 CPU 和电池的使用 */\n if (this.audioContext?.state === 'running') {\n await this.audioContext.suspend();\n }\n this.config.onStop?.();\n }\n\n close() {\n this.mediaSource?.disconnect();\n this.status = 'closed';\n if (this.audioContext?.state !== 'closed') {\n return this.audioContext?.close();\n }\n }\n\n destroy() {\n this.close();\n this.mediaStream?.getTracks()?.forEach((track) => track.stop());\n }\n}\n"],"names":["_b","_a"],"mappings":";;;;;AAIO,MAAM,oBAAoB;AAgBjC,MAAM,qBAAqB,CAAC,UAAkB;AACtC,QAAA,OAAO,IAAI,KAAK,CAAC,KAAK,GAAG,EAAE,MAAM,0BAA0B;AAC3D,QAAA,YAAY,IAAI,gBAAgB,IAAI;AACnC,SAAA;AACT;AAGO,MAAM,aAAa;AAAA,EASxB,YAAY,QAA2B;AARvC;AACA,sCAA0B;AAC1B,kCAAqD;AACrD;AACA;AACA;AACA,kCAA4B,CAAC;AAGnB,YAAA,IAAI,mCAAmC,MAAM;AAC/C,UAAA,eAAe,OAAO,oBAAoB;AAC3C,SAAA,aAAa,mBAAmB,YAAY;AACjD,SAAK,aAAa,OAAO;AACzB,SAAK,SAAS;AAAA,EAAA;AAAA,EAGhB,MAAM,QAAQ;;AACR,QAAA;AACE,UAAA,KAAK,WAAW,YAAY;AAC9B;AAAA,MAAA;AAEF,WAAK,SAAS;AAER,YAAA,eAAe,IAAI,OAAO,aAAa;AAAA,QAC3C,YAAY,KAAK;AAAA,MAAA,CAClB;AACG,UAAA,CAAC,KAAK,YAAY;AACpB,cAAM,MAAM,yBAAyB;AAAA,MAAA;AAGvC,YAAM,aAAa,aAAa,UAAU,KAAK,UAAU;AACzD,YAAM,mBAAmB,IAAI,iBAAiB,cAAc,yBAAyB;AAE/E,YAAA,cAAc,MAAM,4BAA4B;AAChD,YAAA,cAAc,aAAa,wBAAwB,WAAW;AAEpE,uBAAiB,KAAK,YAAY;AAAA,QAChC,MAAM;AAAA,QACN,WAAW,KAAK,OAAO;AAAA,MAAA,CACxB;AAEgB,uBAAA,KAAK,YAAY,CAAC,MAAM;;AAClC,SAAAA,OAAAC,MAAA,KAAA,QAAO,oBAAP,gBAAAD,IAAA,KAAAC,KAAyB,EAAE;AAAA,MAClC;AAEA,kBAAY,QAAQ,gBAAgB;AAEpC,WAAK,eAAe;AACpB,WAAK,cAAc;AACnB,WAAK,cAAc;AACnB,WAAK,SAAS;AACd,uBAAK,QAAO,YAAZ;AAAA,aACO,OAAO;AACN,cAAA,MAAM,8BAA8B,KAAK;AAC3C,YAAA;AAAA,IAAA;AAAA,EACR;AAAA,EAGF,MAAM,OAAO;;AACX,SAAK,SAAS;AAEV,UAAA,UAAK,iBAAL,mBAAmB,WAAU,WAAW;AACpC,YAAA,KAAK,aAAa,QAAQ;AAAA,IAAA;AAElC,qBAAK,QAAO,WAAZ;AAAA,EAAqB;AAAA,EAGvB,QAAQ;;AACN,eAAK,gBAAL,mBAAkB;AAClB,SAAK,SAAS;AACV,UAAA,UAAK,iBAAL,mBAAmB,WAAU,UAAU;AAClC,cAAA,UAAK,iBAAL,mBAAmB;AAAA,IAAM;AAAA,EAClC;AAAA,EAGF,UAAU;;AACR,SAAK,MAAM;AACN,qBAAA,gBAAA,mBAAa,gBAAb,mBAA0B,QAAQ,CAAC,UAAU,MAAM;EAAM;AAElE;"}
1
+ {"version":3,"file":"AudioManager.es.js","sources":["../../../../src/utils/voskBrowserManager/AudioManager/AudioManager.ts"],"sourcesContent":["import { requestMicrophonePermission } from '../../xunFeiVoice/utils';\nimport { SampleRate } from './types';\nimport defaultWorkletProcessorStr from './workletProcessor.js?raw';\n\nexport const defaultSampleRate = 48000;\n\nexport interface AudiManagerConfig {\n /** 一个转换成 string js 文件。其内容要符合 AudioWorkletProcessor 的要求。\n * - 需使用 registerProcessor 注册,第一个参数为 “audio-worklet-processor”,如: registerProcessor('audio-worklet-processor', audioToFloat32Processor);\n * - 参考:https://developer.mozilla.org/en-US/docs/Web/API/AudioWorkletProcessor\n */\n workletProcessor?: string;\n /** 采样率(每秒采样数)的浮点数,常用值:16000, 32000, 44100, 48000 */\n sampleRate?: SampleRate;\n onFrameRecorded?: (data: Float32Array) => void;\n onStart?: () => void;\n onStop?: () => void;\n frameSize?: number;\n}\n\nconst generateWorkletUrl = (jsStr: string) => {\n const blob = new Blob([jsStr], { type: 'application/javascript' });\n const scriptUrl = URL.createObjectURL(blob);\n return scriptUrl;\n};\n\n/** 对实时音频录制进行管理。 */\nexport class AudioManager {\n workletUrl?: string;\n sampleRate?: SampleRate = defaultSampleRate;\n audioContext?: AudioContext;\n mediaSource?: MediaStreamAudioSourceNode;\n mediaStream?: MediaStream;\n config: AudiManagerConfig = {};\n\n constructor(config: AudiManagerConfig) {\n console.log('audio manager constructor: ====', config);\n const processorStr = config.workletProcessor || defaultWorkletProcessorStr;\n this.workletUrl = generateWorkletUrl(processorStr);\n this.sampleRate = config.sampleRate;\n this.config = config;\n }\n\n async init() {\n if (this.audioContext) {\n await this.destroy();\n }\n // 1. 创建 AudioContext\n const audioContext = new window.AudioContext({\n sampleRate: this.sampleRate,\n });\n if (!this.workletUrl) {\n throw Error('加载 workletProcessor 失败!');\n }\n // 加载给定 JavaScript 文件中的模块,并将其添加到当前 Worklet 中\n await audioContext.audioWorklet.addModule(this.workletUrl);\n const audioWorkletNode = new AudioWorkletNode(audioContext, 'audio-worklet-processor');\n // 获取音频流并创建处理节点\n const mediaStream = await requestMicrophonePermission();\n const mediaSource = audioContext.createMediaStreamSource(mediaStream);\n // 发送配置给 workletProcessor\n audioWorkletNode.port.postMessage({\n type: 'init',\n frameSize: this.config.frameSize,\n });\n // 监听消息\n audioWorkletNode.port.onmessage = (e) => {\n this.config.onFrameRecorded?.(e.data);\n };\n // 连接音频节点\n mediaSource.connect(audioWorkletNode);\n // 赋值到 this 上\n this.audioContext = audioContext;\n this.mediaSource = mediaSource;\n this.mediaStream = mediaStream;\n }\n\n async start() {\n try {\n console.log(\n '%c 🐶 dog ==== audio manager start:',\n 'color: green; font-size: 16px;',\n this.audioContext,\n );\n if (this.audioContext?.state === 'suspended') {\n this.audioContext.resume();\n }\n if (this.audioContext?.state === 'closed') {\n await this.init();\n }\n this.config.onStart?.();\n } catch (error) {\n console.error('Failed to start recording:', error);\n throw error;\n }\n }\n\n async stop() {\n /** 暂停音频上下文对象中的进度,并暂时剥离进程对音频设备硬件的访问权限,减少 CPU 和电池的使用 */\n if (this.audioContext?.state === 'running') {\n await this.audioContext.suspend();\n }\n this.config.onStop?.();\n }\n\n close() {\n this.mediaSource?.disconnect();\n if (this.audioContext && this.audioContext.state !== 'closed') {\n return this.audioContext.close();\n }\n }\n\n async destroy() {\n await this.close();\n this.mediaStream?.getTracks()?.forEach((track) => track.stop());\n }\n}\n"],"names":[],"mappings":";;;;;AAIO,MAAM,oBAAoB;AAgBjC,MAAM,qBAAqB,CAAC,UAAkB;AACtC,QAAA,OAAO,IAAI,KAAK,CAAC,KAAK,GAAG,EAAE,MAAM,0BAA0B;AAC3D,QAAA,YAAY,IAAI,gBAAgB,IAAI;AACnC,SAAA;AACT;AAGO,MAAM,aAAa;AAAA,EAQxB,YAAY,QAA2B;AAPvC;AACA,sCAA0B;AAC1B;AACA;AACA;AACA,kCAA4B,CAAC;AAGnB,YAAA,IAAI,mCAAmC,MAAM;AAC/C,UAAA,eAAe,OAAO,oBAAoB;AAC3C,SAAA,aAAa,mBAAmB,YAAY;AACjD,SAAK,aAAa,OAAO;AACzB,SAAK,SAAS;AAAA,EAAA;AAAA,EAGhB,MAAM,OAAO;AACX,QAAI,KAAK,cAAc;AACrB,YAAM,KAAK,QAAQ;AAAA,IAAA;AAGf,UAAA,eAAe,IAAI,OAAO,aAAa;AAAA,MAC3C,YAAY,KAAK;AAAA,IAAA,CAClB;AACG,QAAA,CAAC,KAAK,YAAY;AACpB,YAAM,MAAM,yBAAyB;AAAA,IAAA;AAGvC,UAAM,aAAa,aAAa,UAAU,KAAK,UAAU;AACzD,UAAM,mBAAmB,IAAI,iBAAiB,cAAc,yBAAyB;AAE/E,UAAA,cAAc,MAAM,4BAA4B;AAChD,UAAA,cAAc,aAAa,wBAAwB,WAAW;AAEpE,qBAAiB,KAAK,YAAY;AAAA,MAChC,MAAM;AAAA,MACN,WAAW,KAAK,OAAO;AAAA,IAAA,CACxB;AAEgB,qBAAA,KAAK,YAAY,CAAC,MAAM;;AAClC,uBAAA,QAAO,oBAAP,4BAAyB,EAAE;AAAA,IAClC;AAEA,gBAAY,QAAQ,gBAAgB;AAEpC,SAAK,eAAe;AACpB,SAAK,cAAc;AACnB,SAAK,cAAc;AAAA,EAAA;AAAA,EAGrB,MAAM,QAAQ;;AACR,QAAA;AACM,cAAA;AAAA,QACN;AAAA,QACA;AAAA,QACA,KAAK;AAAA,MACP;AACI,YAAA,UAAK,iBAAL,mBAAmB,WAAU,aAAa;AAC5C,aAAK,aAAa,OAAO;AAAA,MAAA;AAEvB,YAAA,UAAK,iBAAL,mBAAmB,WAAU,UAAU;AACzC,cAAM,KAAK,KAAK;AAAA,MAAA;AAElB,uBAAK,QAAO,YAAZ;AAAA,aACO,OAAO;AACN,cAAA,MAAM,8BAA8B,KAAK;AAC3C,YAAA;AAAA,IAAA;AAAA,EACR;AAAA,EAGF,MAAM,OAAO;;AAEP,UAAA,UAAK,iBAAL,mBAAmB,WAAU,WAAW;AACpC,YAAA,KAAK,aAAa,QAAQ;AAAA,IAAA;AAElC,qBAAK,QAAO,WAAZ;AAAA,EAAqB;AAAA,EAGvB,QAAQ;;AACN,eAAK,gBAAL,mBAAkB;AAClB,QAAI,KAAK,gBAAgB,KAAK,aAAa,UAAU,UAAU;AACtD,aAAA,KAAK,aAAa,MAAM;AAAA,IAAA;AAAA,EACjC;AAAA,EAGF,MAAM,UAAU;;AACd,UAAM,KAAK,MAAM;AACZ,qBAAA,gBAAA,mBAAa,gBAAb,mBAA0B,QAAQ,CAAC,UAAU,MAAM;EAAM;AAElE;"}
@@ -15,11 +15,15 @@ export declare class VoskBrowserManager {
15
15
  /** 采样率(每秒采样数)的浮点数,常用值:16000, 32000, 44100, 48000 */
16
16
  sampleRate: SampleRate;
17
17
  audioManager?: AudioManager;
18
+ model?: Vosk.Model;
18
19
  autoStart: boolean;
20
+ startResolve?: (value?: unknown) => void;
21
+ status?: 'initializing' | 'initialized';
19
22
  constructor(config: VoskBrowserManagerConfig);
20
23
  setConfig(config: Partial<Pick<VoskBrowserManagerConfig, 'onResult' | 'onPartialResult'>>): void;
21
24
  init(vosk: typeof Vosk): Promise<void>;
22
- start(): void;
25
+ start(): Promise<void>;
23
26
  stop(): void;
27
+ destroy(): void;
24
28
  }
25
29
  export {};
@@ -19,7 +19,10 @@ class VoskBrowserManager {
19
19
  /** 采样率(每秒采样数)的浮点数,常用值:16000, 32000, 44100, 48000 */
20
20
  __publicField(this, "sampleRate", defaultSampleRate);
21
21
  __publicField(this, "audioManager");
22
+ __publicField(this, "model");
22
23
  __publicField(this, "autoStart", true);
24
+ __publicField(this, "startResolve");
25
+ __publicField(this, "status");
23
26
  this.config = config;
24
27
  this.sampleRate = config.sampleRate || defaultSampleRate;
25
28
  this.autoStart = config.autoStart || true;
@@ -32,11 +35,15 @@ class VoskBrowserManager {
32
35
  };
33
36
  }
34
37
  async init(vosk) {
35
- var _a, _b;
38
+ var _a, _b, _c;
39
+ if (this.model) {
40
+ this.model.terminate();
41
+ }
42
+ this.status = "initializing";
36
43
  console.time("load model:");
37
- const model = await vosk.createModel(this.config.modelUrl);
44
+ this.model = await vosk.createModel(this.config.modelUrl);
38
45
  console.timeEnd("load model:");
39
- const recognizer = new model.KaldiRecognizer(this.sampleRate);
46
+ const recognizer = new this.model.KaldiRecognizer(this.sampleRate);
40
47
  recognizer.on("result", (message) => {
41
48
  var _a2, _b2;
42
49
  const resultText = message.result.text;
@@ -47,6 +54,9 @@ class VoskBrowserManager {
47
54
  const partialText = message.result.partial;
48
55
  (_b2 = (_a2 = this.config).onPartialResult) == null ? void 0 : _b2.call(_a2, partialText);
49
56
  });
57
+ if (this.audioManager) {
58
+ this.audioManager.destroy();
59
+ }
50
60
  this.audioManager = new AudioManager({
51
61
  ...pickAudioManagerConfig(this.config),
52
62
  sampleRate: this.sampleRate,
@@ -56,19 +66,41 @@ class VoskBrowserManager {
56
66
  (_b2 = (_a2 = this.config).onFrameRecorded) == null ? void 0 : _b2.call(_a2, data);
57
67
  }
58
68
  });
69
+ await this.audioManager.init();
59
70
  (_b = (_a = this.config).onInitialized) == null ? void 0 : _b.call(_a);
71
+ (_c = this.startResolve) == null ? void 0 : _c.call(this);
72
+ this.status = "initialized";
73
+ console.log("%c 🐱 cat ==== init:", "color: orange; font-size: 16px;", this);
60
74
  if (this.autoStart) {
61
75
  this.start();
62
76
  }
63
77
  }
64
- start() {
65
- var _a;
66
- (_a = this.audioManager) == null ? void 0 : _a.start();
78
+ async start() {
79
+ var _a, _b, _c, _d;
80
+ console.log(
81
+ "vosk manager start ====",
82
+ this.audioManager,
83
+ (_b = (_a = this.audioManager) == null ? void 0 : _a.audioContext) == null ? void 0 : _b.state
84
+ );
85
+ if (this.status === "initialized") {
86
+ await ((_c = this.audioManager) == null ? void 0 : _c.start());
87
+ } else {
88
+ await new Promise((resolve) => {
89
+ this.startResolve = resolve;
90
+ });
91
+ console.log("%c 🐱 cat ==== start resolved", "color: orange; font-size: 16px;");
92
+ await ((_d = this.audioManager) == null ? void 0 : _d.start());
93
+ }
67
94
  }
68
95
  stop() {
69
96
  var _a;
70
97
  (_a = this.audioManager) == null ? void 0 : _a.stop();
71
98
  }
99
+ destroy() {
100
+ var _a, _b;
101
+ (_a = this.audioManager) == null ? void 0 : _a.destroy();
102
+ (_b = this.model) == null ? void 0 : _b.terminate();
103
+ }
72
104
  }
73
105
  export {
74
106
  VoskBrowserManager
@@ -1 +1 @@
1
- {"version":3,"file":"voskBrowserManager.es.js","sources":["../../../src/utils/voskBrowserManager/voskBrowserManager.ts"],"sourcesContent":["import { omit } from 'lodash-es';\nimport type * as Vosk from 'vosk-browser';\nimport { AudiManagerConfig, AudioManager, defaultSampleRate } from './AudioManager/AudioManager';\nimport { SampleRate } from './AudioManager/types';\n\ninterface VoskBrowserManagerConfig extends AudiManagerConfig {\n vosk: typeof Vosk;\n modelUrl: string;\n /** 加载模型等初始化配置完成后,自动开始监听 */\n autoStart?: boolean;\n onInitialized?: () => void;\n onResult?: (text: string) => void;\n onPartialResult?: (text: string) => void;\n}\n\nconst pickAudioManagerConfig = (config: VoskBrowserManagerConfig) => {\n return omit(config, [\n 'vosk',\n 'modelUrl',\n 'autoStart',\n 'onResult',\n 'onPartialResult',\n 'onInitialized',\n ]);\n};\n\nexport class VoskBrowserManager {\n config: VoskBrowserManagerConfig;\n /** 采样率(每秒采样数)的浮点数,常用值:16000, 32000, 44100, 48000 */\n sampleRate: SampleRate = defaultSampleRate;\n audioManager?: AudioManager;\n autoStart = true;\n\n constructor(config: VoskBrowserManagerConfig) {\n this.config = config;\n this.sampleRate = config.sampleRate || defaultSampleRate;\n this.autoStart = config.autoStart || true;\n this.init(config.vosk);\n }\n\n setConfig(config: Partial<Pick<VoskBrowserManagerConfig, 'onResult' | 'onPartialResult'>>) {\n this.config = {\n ...this.config,\n ...config,\n };\n }\n\n async init(vosk: typeof Vosk) {\n console.time('load model:');\n const model = await vosk.createModel(this.config.modelUrl);\n console.timeEnd('load model:');\n\n const recognizer = new model.KaldiRecognizer(this.sampleRate);\n recognizer.on('result', (message) => {\n const resultText = (message as any).result.text;\n this.config.onResult?.(resultText);\n });\n recognizer.on('partialresult', (message) => {\n const partialText = (message as any).result.partial;\n this.config.onPartialResult?.(partialText);\n });\n\n this.audioManager = new AudioManager({\n ...pickAudioManagerConfig(this.config),\n sampleRate: this.sampleRate,\n onFrameRecorded: (data) => {\n recognizer.acceptWaveformFloat(data, this.sampleRate);\n this.config.onFrameRecorded?.(data);\n },\n });\n\n this.config.onInitialized?.();\n if (this.autoStart) {\n this.start();\n }\n }\n\n start() {\n this.audioManager?.start();\n }\n\n stop() {\n this.audioManager?.stop();\n }\n}\n"],"names":["_b","_a"],"mappings":";;;;;AAeA,MAAM,yBAAyB,CAAC,WAAqC;AACnE,SAAO,KAAK,QAAQ;AAAA,IAClB;AAAA,IACA;AAAA,IACA;AAAA,IACA;AAAA,IACA;AAAA,IACA;AAAA,EAAA,CACD;AACH;AAEO,MAAM,mBAAmB;AAAA,EAO9B,YAAY,QAAkC;AAN9C;AAEA;AAAA,sCAAyB;AACzB;AACA,qCAAY;AAGV,SAAK,SAAS;AACT,SAAA,aAAa,OAAO,cAAc;AAClC,SAAA,YAAY,OAAO,aAAa;AAChC,SAAA,KAAK,OAAO,IAAI;AAAA,EAAA;AAAA,EAGvB,UAAU,QAAiF;AACzF,SAAK,SAAS;AAAA,MACZ,GAAG,KAAK;AAAA,MACR,GAAG;AAAA,IACL;AAAA,EAAA;AAAA,EAGF,MAAM,KAAK,MAAmB;;AAC5B,YAAQ,KAAK,aAAa;AAC1B,UAAM,QAAQ,MAAM,KAAK,YAAY,KAAK,OAAO,QAAQ;AACzD,YAAQ,QAAQ,aAAa;AAE7B,UAAM,aAAa,IAAI,MAAM,gBAAgB,KAAK,UAAU;AACjD,eAAA,GAAG,UAAU,CAAC,YAAY;;AAC7B,YAAA,aAAc,QAAgB,OAAO;AACtC,OAAAA,OAAAC,MAAA,KAAA,QAAO,aAAP,gBAAAD,IAAA,KAAAC,KAAkB;AAAA,IAAU,CAClC;AACU,eAAA,GAAG,iBAAiB,CAAC,YAAY;;AACpC,YAAA,cAAe,QAAgB,OAAO;AACvC,OAAAD,OAAAC,MAAA,KAAA,QAAO,oBAAP,gBAAAD,IAAA,KAAAC,KAAyB;AAAA,IAAW,CAC1C;AAEI,SAAA,eAAe,IAAI,aAAa;AAAA,MACnC,GAAG,uBAAuB,KAAK,MAAM;AAAA,MACrC,YAAY,KAAK;AAAA,MACjB,iBAAiB,CAAC,SAAS;;AACd,mBAAA,oBAAoB,MAAM,KAAK,UAAU;AAC/C,SAAAD,OAAAC,MAAA,KAAA,QAAO,oBAAP,gBAAAD,IAAA,KAAAC,KAAyB;AAAA,MAAI;AAAA,IACpC,CACD;AAED,qBAAK,QAAO,kBAAZ;AACA,QAAI,KAAK,WAAW;AAClB,WAAK,MAAM;AAAA,IAAA;AAAA,EACb;AAAA,EAGF,QAAQ;;AACN,eAAK,iBAAL,mBAAmB;AAAA,EAAM;AAAA,EAG3B,OAAO;;AACL,eAAK,iBAAL,mBAAmB;AAAA,EAAK;AAE5B;"}
1
+ {"version":3,"file":"voskBrowserManager.es.js","sources":["../../../src/utils/voskBrowserManager/voskBrowserManager.ts"],"sourcesContent":["import { omit } from 'lodash-es';\nimport type * as Vosk from 'vosk-browser';\nimport { AudiManagerConfig, AudioManager, defaultSampleRate } from './AudioManager/AudioManager';\nimport { SampleRate } from './AudioManager/types';\n\ninterface VoskBrowserManagerConfig extends AudiManagerConfig {\n vosk: typeof Vosk;\n modelUrl: string;\n /** 加载模型等初始化配置完成后,自动开始监听 */\n autoStart?: boolean;\n onInitialized?: () => void;\n onResult?: (text: string) => void;\n onPartialResult?: (text: string) => void;\n}\n\nconst pickAudioManagerConfig = (config: VoskBrowserManagerConfig) => {\n return omit(config, [\n 'vosk',\n 'modelUrl',\n 'autoStart',\n 'onResult',\n 'onPartialResult',\n 'onInitialized',\n ]);\n};\n\nexport class VoskBrowserManager {\n config: VoskBrowserManagerConfig;\n /** 采样率(每秒采样数)的浮点数,常用值:16000, 32000, 44100, 48000 */\n sampleRate: SampleRate = defaultSampleRate;\n audioManager?: AudioManager;\n model?: Vosk.Model;\n autoStart = true;\n startResolve?: (value?: unknown) => void;\n status?: 'initializing' | 'initialized';\n\n constructor(config: VoskBrowserManagerConfig) {\n this.config = config;\n this.sampleRate = config.sampleRate || defaultSampleRate;\n this.autoStart = config.autoStart || true;\n this.init(config.vosk);\n }\n\n setConfig(config: Partial<Pick<VoskBrowserManagerConfig, 'onResult' | 'onPartialResult'>>) {\n this.config = {\n ...this.config,\n ...config,\n };\n }\n\n async init(vosk: typeof Vosk) {\n if (this.model) {\n this.model.terminate();\n }\n this.status = 'initializing';\n console.time('load model:');\n this.model = await vosk.createModel(this.config.modelUrl);\n console.timeEnd('load model:');\n\n const recognizer = new this.model.KaldiRecognizer(this.sampleRate);\n recognizer.on('result', (message) => {\n const resultText = (message as any).result.text;\n this.config.onResult?.(resultText);\n });\n recognizer.on('partialresult', (message) => {\n const partialText = (message as any).result.partial;\n this.config.onPartialResult?.(partialText);\n });\n\n if (this.audioManager) {\n this.audioManager.destroy();\n }\n this.audioManager = new AudioManager({\n ...pickAudioManagerConfig(this.config),\n sampleRate: this.sampleRate,\n onFrameRecorded: (data) => {\n recognizer.acceptWaveformFloat(data, this.sampleRate);\n this.config.onFrameRecorded?.(data);\n },\n });\n await this.audioManager.init();\n\n this.config.onInitialized?.();\n this.startResolve?.();\n this.status = 'initialized';\n console.log('%c 🐱 cat ==== init:', 'color: orange; font-size: 16px;', this);\n if (this.autoStart) {\n this.start();\n }\n }\n\n async start() {\n console.log(\n 'vosk manager start ====',\n this.audioManager,\n this.audioManager?.audioContext?.state,\n );\n if (this.status === 'initialized') {\n await this.audioManager?.start();\n } else {\n await new Promise((resolve) => {\n this.startResolve = resolve;\n });\n console.log('%c 🐱 cat ==== start resolved', 'color: orange; font-size: 16px;');\n await this.audioManager?.start();\n }\n }\n\n stop() {\n this.audioManager?.stop();\n }\n\n destroy() {\n this.audioManager?.destroy();\n this.model?.terminate();\n }\n}\n"],"names":["_b","_a"],"mappings":";;;;;AAeA,MAAM,yBAAyB,CAAC,WAAqC;AACnE,SAAO,KAAK,QAAQ;AAAA,IAClB;AAAA,IACA;AAAA,IACA;AAAA,IACA;AAAA,IACA;AAAA,IACA;AAAA,EAAA,CACD;AACH;AAEO,MAAM,mBAAmB;AAAA,EAU9B,YAAY,QAAkC;AAT9C;AAEA;AAAA,sCAAyB;AACzB;AACA;AACA,qCAAY;AACZ;AACA;AAGE,SAAK,SAAS;AACT,SAAA,aAAa,OAAO,cAAc;AAClC,SAAA,YAAY,OAAO,aAAa;AAChC,SAAA,KAAK,OAAO,IAAI;AAAA,EAAA;AAAA,EAGvB,UAAU,QAAiF;AACzF,SAAK,SAAS;AAAA,MACZ,GAAG,KAAK;AAAA,MACR,GAAG;AAAA,IACL;AAAA,EAAA;AAAA,EAGF,MAAM,KAAK,MAAmB;;AAC5B,QAAI,KAAK,OAAO;AACd,WAAK,MAAM,UAAU;AAAA,IAAA;AAEvB,SAAK,SAAS;AACd,YAAQ,KAAK,aAAa;AAC1B,SAAK,QAAQ,MAAM,KAAK,YAAY,KAAK,OAAO,QAAQ;AACxD,YAAQ,QAAQ,aAAa;AAE7B,UAAM,aAAa,IAAI,KAAK,MAAM,gBAAgB,KAAK,UAAU;AACtD,eAAA,GAAG,UAAU,CAAC,YAAY;;AAC7B,YAAA,aAAc,QAAgB,OAAO;AACtC,OAAAA,OAAAC,MAAA,KAAA,QAAO,aAAP,gBAAAD,IAAA,KAAAC,KAAkB;AAAA,IAAU,CAClC;AACU,eAAA,GAAG,iBAAiB,CAAC,YAAY;;AACpC,YAAA,cAAe,QAAgB,OAAO;AACvC,OAAAD,OAAAC,MAAA,KAAA,QAAO,oBAAP,gBAAAD,IAAA,KAAAC,KAAyB;AAAA,IAAW,CAC1C;AAED,QAAI,KAAK,cAAc;AACrB,WAAK,aAAa,QAAQ;AAAA,IAAA;AAEvB,SAAA,eAAe,IAAI,aAAa;AAAA,MACnC,GAAG,uBAAuB,KAAK,MAAM;AAAA,MACrC,YAAY,KAAK;AAAA,MACjB,iBAAiB,CAAC,SAAS;;AACd,mBAAA,oBAAoB,MAAM,KAAK,UAAU;AAC/C,SAAAD,OAAAC,MAAA,KAAA,QAAO,oBAAP,gBAAAD,IAAA,KAAAC,KAAyB;AAAA,MAAI;AAAA,IACpC,CACD;AACK,UAAA,KAAK,aAAa,KAAK;AAE7B,qBAAK,QAAO,kBAAZ;AACA,eAAK,iBAAL;AACA,SAAK,SAAS;AACN,YAAA,IAAI,0BAA0B,mCAAmC,IAAI;AAC7E,QAAI,KAAK,WAAW;AAClB,WAAK,MAAM;AAAA,IAAA;AAAA,EACb;AAAA,EAGF,MAAM,QAAQ;;AACJ,YAAA;AAAA,MACN;AAAA,MACA,KAAK;AAAA,OACL,gBAAK,iBAAL,mBAAmB,iBAAnB,mBAAiC;AAAA,IACnC;AACI,QAAA,KAAK,WAAW,eAAe;AAC3B,cAAA,UAAK,iBAAL,mBAAmB;AAAA,IAAM,OAC1B;AACC,YAAA,IAAI,QAAQ,CAAC,YAAY;AAC7B,aAAK,eAAe;AAAA,MAAA,CACrB;AACO,cAAA,IAAI,mCAAmC,iCAAiC;AAC1E,cAAA,UAAK,iBAAL,mBAAmB;AAAA,IAAM;AAAA,EACjC;AAAA,EAGF,OAAO;;AACL,eAAK,iBAAL,mBAAmB;AAAA,EAAK;AAAA,EAG1B,UAAU;;AACR,eAAK,iBAAL,mBAAmB;AACnB,eAAK,UAAL,mBAAY;AAAA,EAAU;AAE1B;"}
package/package.json CHANGED
@@ -1,6 +1,6 @@
1
1
  {
2
2
  "name": "@easyv/biz-components",
3
- "version": "0.0.49",
3
+ "version": "0.0.51-beta.0",
4
4
  "type": "module",
5
5
  "main": "dist/index.es.js",
6
6
  "module": "dist/index.es.js",