@modochats/widget 0.1.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.
Files changed (195) hide show
  1. package/.github/workflows/build-and-publish.yml +174 -0
  2. package/.vscode/settings.json +3 -0
  3. package/.yarn/install-state.gz +0 -0
  4. package/README.md +163 -0
  5. package/cdn-dist/README.md +42 -0
  6. package/cdn-dist/modo-web-component.js +1344 -0
  7. package/cdn-dist/modo-web-component.min.js +1 -0
  8. package/cdn-dist/package.json +27 -0
  9. package/dist/src/app.d.ts +29 -0
  10. package/dist/src/app.js +1 -0
  11. package/dist/src/app.js.map +1 -0
  12. package/dist/src/constants/index.d.ts +9 -0
  13. package/dist/src/constants/index.js +1 -0
  14. package/dist/src/constants/index.js.map +1 -0
  15. package/dist/src/constants/regex.d.ts +2 -0
  16. package/dist/src/constants/regex.js +1 -0
  17. package/dist/src/constants/regex.js.map +1 -0
  18. package/dist/src/index.d.ts +9 -0
  19. package/dist/src/index.js +1 -0
  20. package/dist/src/index.js.map +1 -0
  21. package/dist/src/models/chatbot.d.ts +23 -0
  22. package/dist/src/models/chatbot.js +1 -0
  23. package/dist/src/models/chatbot.js.map +1 -0
  24. package/dist/src/models/conversation.d.ts +22 -0
  25. package/dist/src/models/conversation.js +1 -0
  26. package/dist/src/models/conversation.js.map +1 -0
  27. package/dist/src/models/customer-data.d.ts +31 -0
  28. package/dist/src/models/customer-data.js +1 -0
  29. package/dist/src/models/customer-data.js.map +1 -0
  30. package/dist/src/models/message-utils.d.ts +12 -0
  31. package/dist/src/models/message-utils.js +1 -0
  32. package/dist/src/models/message-utils.js.map +1 -0
  33. package/dist/src/services/chat/conversation.d.ts +22 -0
  34. package/dist/src/services/chat/conversation.js +1 -0
  35. package/dist/src/services/chat/conversation.js.map +1 -0
  36. package/dist/src/services/chat/message-utils.d.ts +12 -0
  37. package/dist/src/services/chat/message-utils.js +1 -0
  38. package/dist/src/services/chat/message-utils.js.map +1 -0
  39. package/dist/src/services/chat/model.d.ts +27 -0
  40. package/dist/src/services/chat/model.js +1 -0
  41. package/dist/src/services/chat/model.js.map +1 -0
  42. package/dist/src/services/chatbot/chatbot.d.ts +23 -0
  43. package/dist/src/services/chatbot/chatbot.js +1 -0
  44. package/dist/src/services/chatbot/chatbot.js.map +1 -0
  45. package/dist/src/services/checker.d.ts +3 -0
  46. package/dist/src/services/checker.js +1 -0
  47. package/dist/src/services/checker.js.map +1 -0
  48. package/dist/src/services/listeners/adders.d.ts +3 -0
  49. package/dist/src/services/listeners/adders.js +1 -0
  50. package/dist/src/services/listeners/adders.js.map +1 -0
  51. package/dist/src/services/listeners/fn.d.ts +3 -0
  52. package/dist/src/services/listeners/fn.js +1 -0
  53. package/dist/src/services/listeners/fn.js.map +1 -0
  54. package/dist/src/services/socket/utils.d.ts +2 -0
  55. package/dist/src/services/socket/utils.js +1 -0
  56. package/dist/src/services/socket/utils.js.map +1 -0
  57. package/dist/src/services/ui/fn.d.ts +13 -0
  58. package/dist/src/services/ui/fn.js +1 -0
  59. package/dist/src/services/ui/fn.js.map +1 -0
  60. package/dist/src/services/ui/html.d.ts +3 -0
  61. package/dist/src/services/ui/html.js +1 -0
  62. package/dist/src/services/ui/html.js.map +1 -0
  63. package/dist/src/services/user/customer-data.d.ts +31 -0
  64. package/dist/src/services/user/customer-data.js +1 -0
  65. package/dist/src/services/user/customer-data.js.map +1 -0
  66. package/dist/src/services/voice-chat/model.d.ts +12 -0
  67. package/dist/src/services/voice-chat/model.js +1 -0
  68. package/dist/src/services/voice-chat/model.js.map +1 -0
  69. package/dist/src/services/voice-chat/utils.d.ts +9 -0
  70. package/dist/src/services/voice-chat/utils.js +1 -0
  71. package/dist/src/services/voice-chat/utils.js.map +1 -0
  72. package/dist/src/tools/fetch.d.ts +2 -0
  73. package/dist/src/tools/fetch.js +1 -0
  74. package/dist/src/tools/fetch.js.map +1 -0
  75. package/dist/src/types/app.d.ts +17 -0
  76. package/dist/src/types/app.js.map +1 -0
  77. package/dist/src/types/conversation.d.ts +14 -0
  78. package/dist/src/types/conversation.js +1 -0
  79. package/dist/src/types/conversation.js.map +1 -0
  80. package/dist/src/types/socket.d.ts +6 -0
  81. package/dist/src/types/socket.js +1 -0
  82. package/dist/src/types/socket.js.map +1 -0
  83. package/dist/src/types/window.d.ts +9 -0
  84. package/dist/src/types/window.js +1 -0
  85. package/dist/src/types/window.js.map +1 -0
  86. package/dist/src/utils/audio.d.ts +3 -0
  87. package/dist/src/utils/audio.js +1 -0
  88. package/dist/src/utils/audio.js.map +1 -0
  89. package/dist/src/utils/browser.d.ts +2 -0
  90. package/dist/src/utils/browser.js +1 -0
  91. package/dist/src/utils/browser.js.map +1 -0
  92. package/dist/src/utils/fetch.d.ts +18 -0
  93. package/dist/src/utils/fetch.js +1 -0
  94. package/dist/src/utils/fetch.js.map +1 -0
  95. package/dist/src/utils/uuid.d.ts +6 -0
  96. package/dist/src/utils/uuid.js +1 -0
  97. package/dist/src/utils/uuid.js.map +1 -0
  98. package/dist/types/src/app.d.ts +30 -0
  99. package/dist/types/src/app.d.ts.map +1 -0
  100. package/dist/types/src/constants/index.d.ts +10 -0
  101. package/dist/types/src/constants/index.d.ts.map +1 -0
  102. package/dist/types/src/constants/regex.d.ts +3 -0
  103. package/dist/types/src/constants/regex.d.ts.map +1 -0
  104. package/dist/types/src/index.d.ts +10 -0
  105. package/dist/types/src/index.d.ts.map +1 -0
  106. package/dist/types/src/models/chatbot.d.ts +24 -0
  107. package/dist/types/src/models/chatbot.d.ts.map +1 -0
  108. package/dist/types/src/models/conversation.d.ts +23 -0
  109. package/dist/types/src/models/conversation.d.ts.map +1 -0
  110. package/dist/types/src/models/customer-data.d.ts +32 -0
  111. package/dist/types/src/models/customer-data.d.ts.map +1 -0
  112. package/dist/types/src/models/message-utils.d.ts +13 -0
  113. package/dist/types/src/models/message-utils.d.ts.map +1 -0
  114. package/dist/types/src/services/chat/conversation.d.ts +23 -0
  115. package/dist/types/src/services/chat/conversation.d.ts.map +1 -0
  116. package/dist/types/src/services/chat/message-utils.d.ts +13 -0
  117. package/dist/types/src/services/chat/message-utils.d.ts.map +1 -0
  118. package/dist/types/src/services/chat/model.d.ts +28 -0
  119. package/dist/types/src/services/chat/model.d.ts.map +1 -0
  120. package/dist/types/src/services/chatbot/chatbot.d.ts +24 -0
  121. package/dist/types/src/services/chatbot/chatbot.d.ts.map +1 -0
  122. package/dist/types/src/services/checker.d.ts +4 -0
  123. package/dist/types/src/services/checker.d.ts.map +1 -0
  124. package/dist/types/src/services/listeners/adders.d.ts +4 -0
  125. package/dist/types/src/services/listeners/adders.d.ts.map +1 -0
  126. package/dist/types/src/services/listeners/fn.d.ts +4 -0
  127. package/dist/types/src/services/listeners/fn.d.ts.map +1 -0
  128. package/dist/types/src/services/socket/utils.d.ts +3 -0
  129. package/dist/types/src/services/socket/utils.d.ts.map +1 -0
  130. package/dist/types/src/services/ui/fn.d.ts +14 -0
  131. package/dist/types/src/services/ui/fn.d.ts.map +1 -0
  132. package/dist/types/src/services/ui/html.d.ts +4 -0
  133. package/dist/types/src/services/ui/html.d.ts.map +1 -0
  134. package/dist/types/src/services/user/customer-data.d.ts +32 -0
  135. package/dist/types/src/services/user/customer-data.d.ts.map +1 -0
  136. package/dist/types/src/services/voice-chat/model.d.ts +13 -0
  137. package/dist/types/src/services/voice-chat/model.d.ts.map +1 -0
  138. package/dist/types/src/services/voice-chat/utils.d.ts +10 -0
  139. package/dist/types/src/services/voice-chat/utils.d.ts.map +1 -0
  140. package/dist/types/src/tools/fetch.d.ts +3 -0
  141. package/dist/types/src/tools/fetch.d.ts.map +1 -0
  142. package/dist/types/src/types/app.d.ts +18 -0
  143. package/dist/types/src/types/app.d.ts.map +1 -0
  144. package/dist/types/src/types/conversation.d.ts +15 -0
  145. package/dist/types/src/types/conversation.d.ts.map +1 -0
  146. package/dist/types/src/types/socket.d.ts +7 -0
  147. package/dist/types/src/types/socket.d.ts.map +1 -0
  148. package/dist/types/src/types/window.d.ts +10 -0
  149. package/dist/types/src/types/window.d.ts.map +1 -0
  150. package/dist/types/src/utils/audio.d.ts +4 -0
  151. package/dist/types/src/utils/audio.d.ts.map +1 -0
  152. package/dist/types/src/utils/browser.d.ts +3 -0
  153. package/dist/types/src/utils/browser.d.ts.map +1 -0
  154. package/dist/types/src/utils/fetch.d.ts +19 -0
  155. package/dist/types/src/utils/fetch.d.ts.map +1 -0
  156. package/dist/types/src/utils/uuid.d.ts +7 -0
  157. package/dist/types/src/utils/uuid.d.ts.map +1 -0
  158. package/package.json +76 -0
  159. package/rollup.config.js +18 -0
  160. package/rollup.dev.config.js +22 -0
  161. package/scripts/create-umd-bundle.js +213 -0
  162. package/scripts/terser-minify.js +112 -0
  163. package/src/app.ts +117 -0
  164. package/src/constants/index.ts +21 -0
  165. package/src/constants/regex.ts +2 -0
  166. package/src/index.ts +16 -0
  167. package/src/services/chat/conversation.ts +135 -0
  168. package/src/services/chat/message-utils.ts +221 -0
  169. package/src/services/chat/model.ts +139 -0
  170. package/src/services/chatbot/chatbot.ts +66 -0
  171. package/src/services/checker.ts +10 -0
  172. package/src/services/listeners/adders.ts +178 -0
  173. package/src/services/listeners/fn.ts +77 -0
  174. package/src/services/socket/utils.ts +9 -0
  175. package/src/services/ui/fn.ts +254 -0
  176. package/src/services/ui/html.ts +192 -0
  177. package/src/services/user/customer-data.ts +78 -0
  178. package/src/services/voice-chat/model.ts +79 -0
  179. package/src/services/voice-chat/utils.ts +137 -0
  180. package/src/tools/fetch.ts +7 -0
  181. package/src/types/app.ts +17 -0
  182. package/src/types/conversation.ts +14 -0
  183. package/src/types/socket.ts +7 -0
  184. package/src/types/window.ts +12 -0
  185. package/src/utils/audio.ts +67 -0
  186. package/src/utils/browser.ts +4 -0
  187. package/src/utils/fetch.ts +98 -0
  188. package/src/utils/uuid.ts +13 -0
  189. package/temp/audio/new-message.mp3 +0 -0
  190. package/temp/audio/on-hold.mp3 +0 -0
  191. package/temp/audio-processor.js +261 -0
  192. package/temp/css/index.css +2283 -0
  193. package/temp/dev.html +87 -0
  194. package/temp/index.html +16 -0
  195. package/tsconfig.json +119 -0
@@ -0,0 +1,98 @@
1
+ import {$fetch} from "#src/tools/fetch.js";
2
+ import {FetchPaginationRes} from "#src/types/app.js";
3
+
4
+ const fetchChatbot = async (publicKey: string) => {
5
+ return await $fetch<Record<string, any>>(`/v1/chatbot/public/${publicKey}`);
6
+ };
7
+
8
+ const fetchSendMessage = async (
9
+ chatbotId: number,
10
+ content: string,
11
+ uniqueId: string,
12
+ conversationUuid?: string,
13
+ phoneNumber?: string,
14
+ options?: {
15
+ file?: File;
16
+ replyTo?: number;
17
+ }
18
+ ) => {
19
+ const formData = new FormData();
20
+ formData.append("chatbot_id", chatbotId.toString());
21
+ formData.append("content", content);
22
+ formData.append("message_type", "0");
23
+ formData.append("unique_id", uniqueId);
24
+ if (conversationUuid) formData.append("conversation_id", conversationUuid);
25
+ formData.append("url", window?.location?.href || "");
26
+ formData.append("title", document?.title || "");
27
+ if (phoneNumber && phoneNumber !== "no phone number") formData.append("phone_number", phoneNumber);
28
+ if (options?.file) formData.append("file", options.file);
29
+ if (options?.replyTo) formData.append("response_to", options.replyTo.toString());
30
+
31
+ return await $fetch("/v2/conversations/website/send-message/", {
32
+ method: "POST",
33
+ body: formData
34
+ });
35
+ };
36
+ const fetchGetAccessTokenForSocket = async (chatbotId: string, conversationUuid: string, uniqueId: string) => {
37
+ return await $fetch<{access_token: string; conversation_uuid: string; expires_in: number}>("/v2/conversations/websocket/auth/", {
38
+ method: "POST",
39
+ body: {
40
+ chatbot_id: chatbotId,
41
+ conversation_uuid: conversationUuid,
42
+ unique_id: uniqueId
43
+ }
44
+ });
45
+ };
46
+ const fetchConversationMessages = async (conversationUuid: string, chatbotUuid: string) => {
47
+ return await $fetch<Record<string, any>>(`/v2/conversations/website/conversations/${conversationUuid}/chatbot/${chatbotUuid}/messages/`);
48
+ };
49
+ const fetchUpdateUserData = async (chatbotUuid: string, uniqueId: string, userData: Record<string, any>) => {
50
+ return await $fetch("/v1/chatbot/customners/set-user-data", {
51
+ method: "POST",
52
+ body: {
53
+ chatbot_uuid: chatbotUuid,
54
+ unique_id: uniqueId,
55
+ user_data: userData
56
+ }
57
+ });
58
+ };
59
+ const fetchReadMessage = async (id: number) => {
60
+ return await $fetch(`/v2/conversations/messages/${id}/`, {
61
+ method: "POST"
62
+ });
63
+ };
64
+ const fetchMarkConversationAsRead = async (conversationUuid: string, uniqueId: string) => {
65
+ return await $fetch(`/v2/conversations/website/conversations/${conversationUuid}/messages/seen`, {
66
+ method: "POST",
67
+ body: {
68
+ unique_id: uniqueId
69
+ }
70
+ });
71
+ };
72
+
73
+ const fetchMessageFeedback = async (id: number, uniqueId: string, conversationUuid: string, liked: boolean) => {
74
+ return await $fetch(`/v2/conversations/website/conversations/messages/feedback`, {
75
+ method: "POST",
76
+ body: {
77
+ unique_id: uniqueId,
78
+ feedback: liked ? 1 : 0,
79
+ message_id: id,
80
+ conversation_uuid: conversationUuid
81
+ }
82
+ });
83
+ };
84
+
85
+ const fetchConversations = async (conversationUuid: string, uniqueId: string) => {
86
+ return await $fetch<FetchPaginationRes>(`/v2/conversations/website/conversations/${conversationUuid}/customer/${uniqueId}`);
87
+ };
88
+ export {
89
+ fetchChatbot,
90
+ fetchSendMessage,
91
+ fetchGetAccessTokenForSocket,
92
+ fetchConversationMessages,
93
+ fetchUpdateUserData,
94
+ fetchReadMessage,
95
+ fetchMarkConversationAsRead,
96
+ fetchMessageFeedback,
97
+ fetchConversations
98
+ };
@@ -0,0 +1,13 @@
1
+ /**
2
+ * Generates a random UUID v4
3
+ * @returns A random UUID string
4
+ */
5
+ const generateUUID = (): string => {
6
+ return "xxxxxxxx-xxxx-4xxx-yxxx-xxxxxxxxxxxx".replace(/[xy]/g, function (c) {
7
+ const r = (Math.random() * 16) | 0;
8
+ const v = c === "x" ? r : (r & 0x3) | 0x8;
9
+ return v.toString(16);
10
+ });
11
+ };
12
+
13
+ export {generateUUID};
Binary file
Binary file
@@ -0,0 +1,261 @@
1
+ class AudioProcessor extends AudioWorkletProcessor {
2
+ constructor() {
3
+ super();
4
+ this.bufferSize = 4096;
5
+ this.buffer = new Float32Array(this.bufferSize);
6
+ this.bufferIndex = 0;
7
+
8
+ this.noiseThreshold = 0.01;
9
+ this.voiceThreshold = 0.08;
10
+ this.silenceFrames = 0;
11
+ this.maxSilenceFrames = 8;
12
+ this.isVoiceActive = false;
13
+ this.isPaused = false;
14
+
15
+ this.noiseProfile = new Float32Array(128).fill(0);
16
+ this.noiseProfileIndex = 0;
17
+ this.noiseProfileReady = false;
18
+
19
+ this.previousSamples = new Float32Array(this.bufferSize).fill(0);
20
+
21
+ this.preRollBuffer = [];
22
+ this.maxPreRollBuffers = 5;
23
+
24
+ this.tailBuffer = [];
25
+ this.maxTailBuffers = 3;
26
+ this.sendingTail = false;
27
+
28
+ this.justResumed = false;
29
+ this.resumedFrameCount = 0;
30
+ this.resumedFrameThreshold = 150;
31
+
32
+ this.port.onmessage = (event) => {
33
+ if (event.data.type === 'set-voice-threshold') {
34
+ this.voiceThreshold = event.data.value;
35
+ } else if (event.data.type === 'set-noise-threshold') {
36
+ this.noiseThreshold = event.data.value;
37
+ } else if (event.data.type === 'pause') {
38
+ this.isPaused = true;
39
+ console.log('AudioProcessor: Input paused');
40
+ } else if (event.data.type === 'resume') {
41
+ this.isPaused = false;
42
+ // Reset VAD state when resuming but KEEP pre-roll buffer
43
+ this.isVoiceActive = false;
44
+ this.silenceFrames = 0;
45
+ // Reset noise profile to quickly adapt to new environment
46
+ this.noiseProfile.fill(0);
47
+ this.noiseProfileIndex = 0;
48
+ this.noiseProfileReady = false;
49
+ // Be extra sensitive for the first ~1 second after resume
50
+ this.justResumed = true;
51
+ this.resumedFrameCount = 0;
52
+ // Don't clear preRollBuffer - it may contain the start of speech
53
+ console.log('AudioProcessor: Input resumed - VAD and noise profile reset, sensitivity boosted');
54
+ }
55
+ };
56
+ }
57
+
58
+ calculateRMS(samples) {
59
+ let sum = 0;
60
+ for (let i = 0; i < samples.length; i++) {
61
+ sum += samples[i] * samples[i];
62
+ }
63
+ return Math.sqrt(sum / samples.length);
64
+ }
65
+
66
+ calculateDB(rms) {
67
+ if (rms <= 0 || isNaN(rms)) return -Infinity;
68
+ const db = 20 * Math.log10(rms);
69
+ return isNaN(db) ? -Infinity : db;
70
+ }
71
+
72
+ updateNoiseProfile(samples) {
73
+ const rms = this.calculateRMS(samples);
74
+ this.noiseProfile[this.noiseProfileIndex] = rms;
75
+ this.noiseProfileIndex = (this.noiseProfileIndex + 1) % this.noiseProfile.length;
76
+
77
+ if (this.noiseProfileIndex === 0) {
78
+ this.noiseProfileReady = true;
79
+ }
80
+ }
81
+
82
+ getNoiseFloor() {
83
+ if (!this.noiseProfileReady) {
84
+ return this.noiseThreshold;
85
+ }
86
+
87
+ const sorted = Array.from(this.noiseProfile).sort((a, b) => a - b);
88
+ return sorted[Math.floor(sorted.length * 0.25)];
89
+ }
90
+
91
+ applySpectralSubtraction(samples) {
92
+ const output = new Float32Array(samples.length);
93
+ const noiseFloor = this.getNoiseFloor();
94
+
95
+ for (let i = 0; i < samples.length; i++) {
96
+ const signal = Math.abs(samples[i]);
97
+
98
+ if (signal < noiseFloor * 2) {
99
+ output[i] = 0;
100
+ } else {
101
+ const cleaned = signal - noiseFloor;
102
+ output[i] = samples[i] >= 0 ? cleaned : -cleaned;
103
+ }
104
+ }
105
+
106
+ return output;
107
+ }
108
+
109
+ applyHighPassFilter(samples) {
110
+ const output = new Float32Array(samples.length);
111
+ const alpha = 0.95;
112
+
113
+ output[0] = samples[0];
114
+ for (let i = 1; i < samples.length; i++) {
115
+ output[i] = alpha * (output[i-1] + samples[i] - this.previousSamples[i]);
116
+ }
117
+
118
+ this.previousSamples.set(samples);
119
+ return output;
120
+ }
121
+
122
+ applyNoiseGate(samples, rms) {
123
+ const noiseFloor = this.getNoiseFloor();
124
+ let adaptiveThreshold = Math.max(this.noiseThreshold, noiseFloor * 1.8);
125
+
126
+ if (this.justResumed) {
127
+ adaptiveThreshold = adaptiveThreshold * 0.3;
128
+ }
129
+
130
+ if (rms < adaptiveThreshold) {
131
+ return new Float32Array(samples.length);
132
+ }
133
+
134
+ return samples;
135
+ }
136
+
137
+ detectVoice(rms) {
138
+ const noiseFloor = this.getNoiseFloor();
139
+
140
+ let adaptiveVoiceThreshold;
141
+ if (this.justResumed) {
142
+ this.resumedFrameCount++;
143
+
144
+ const boostProgress = Math.min(this.resumedFrameCount / this.resumedFrameThreshold, 1.0);
145
+ const boostMultiplier = 0.2 + (0.8 * boostProgress);
146
+
147
+ adaptiveVoiceThreshold = this.voiceThreshold * boostMultiplier;
148
+
149
+ if (this.resumedFrameCount % 20 === 0) {
150
+ console.log(`AudioProcessor: Boosted threshold: ${adaptiveVoiceThreshold.toFixed(4)} (${(boostMultiplier * 100).toFixed(0)}% of normal, RMS: ${rms.toFixed(4)})`);
151
+ }
152
+
153
+ if (this.resumedFrameCount > this.resumedFrameThreshold) {
154
+ this.justResumed = false;
155
+ console.log('AudioProcessor: Sensitivity boost ended, returning to adaptive threshold');
156
+ }
157
+ } else {
158
+ const multiplier = 1.8;
159
+ adaptiveVoiceThreshold = Math.max(this.voiceThreshold, noiseFloor * multiplier);
160
+ }
161
+
162
+ const wasActive = this.isVoiceActive;
163
+
164
+ if (rms > adaptiveVoiceThreshold) {
165
+ if (!this.isVoiceActive) {
166
+ this.isVoiceActive = true;
167
+ if (this.justResumed) {
168
+ console.log('AudioProcessor: Voice detected with boost!');
169
+ }
170
+ return { justActivated: true, isActive: true };
171
+ }
172
+ this.isVoiceActive = true;
173
+ this.silenceFrames = 0;
174
+ } else if (this.isVoiceActive) {
175
+ this.silenceFrames++;
176
+ if (this.silenceFrames > this.maxSilenceFrames) {
177
+ this.isVoiceActive = false;
178
+ this.sendingTail = true;
179
+ this.preRollBuffer = [];
180
+ return { justActivated: false, isActive: false, justDeactivated: true };
181
+ }
182
+ } else {
183
+ this.updateNoiseProfile(this.buffer);
184
+ }
185
+
186
+ return { justActivated: false, isActive: this.isVoiceActive, justDeactivated: false };
187
+ }
188
+
189
+ process(inputs, outputs, parameters) {
190
+ const input = inputs[0];
191
+ if (input.length > 0) {
192
+ const inputChannel = input[0];
193
+
194
+ for (let i = 0; i < inputChannel.length; i++) {
195
+ this.buffer[this.bufferIndex] = inputChannel[i];
196
+ this.bufferIndex++;
197
+
198
+ if (this.bufferIndex >= this.bufferSize) {
199
+ const rms = this.calculateRMS(this.buffer);
200
+ const db = this.calculateDB(rms);
201
+ const voiceStatus = this.detectVoice(rms);
202
+
203
+ this.port.postMessage({
204
+ type: 'voice-level',
205
+ rms: rms,
206
+ db: db,
207
+ isActive: voiceStatus.isActive,
208
+ noiseFloor: this.getNoiseFloor(),
209
+ isPaused: this.isPaused
210
+ });
211
+
212
+ if (!this.isPaused) {
213
+ let processedBuffer = this.buffer;
214
+
215
+ const int16Buffer = new Int16Array(this.bufferSize);
216
+ for (let j = 0; j < this.bufferSize; j++) {
217
+ const s = Math.max(-1, Math.min(1, processedBuffer[j]));
218
+ int16Buffer[j] = s < 0 ? s * 0x8000 : s * 0x7FFF;
219
+ }
220
+
221
+ if (voiceStatus.justActivated && this.preRollBuffer.length > 0) {
222
+ for (const preRollBuffer of this.preRollBuffer) {
223
+ this.port.postMessage(preRollBuffer);
224
+ }
225
+ this.preRollBuffer = [];
226
+ this.tailBuffer = [];
227
+ this.sendingTail = false;
228
+ }
229
+
230
+ if (voiceStatus.isActive) {
231
+ this.port.postMessage(int16Buffer.buffer);
232
+ this.tailBuffer = [];
233
+ this.sendingTail = false;
234
+ } else if (this.sendingTail) {
235
+ this.tailBuffer.push(int16Buffer.buffer);
236
+ if (this.tailBuffer.length >= this.maxTailBuffers) {
237
+ for (const tailBuffer of this.tailBuffer) {
238
+ this.port.postMessage(tailBuffer);
239
+ }
240
+ this.port.postMessage({ type: 'voice-ended' });
241
+ this.tailBuffer = [];
242
+ this.sendingTail = false;
243
+ }
244
+ } else {
245
+ this.preRollBuffer.push(int16Buffer.buffer);
246
+ if (this.preRollBuffer.length > this.maxPreRollBuffers) {
247
+ this.preRollBuffer.shift();
248
+ }
249
+ }
250
+ }
251
+
252
+ this.bufferIndex = 0;
253
+ }
254
+ }
255
+ }
256
+
257
+ return true;
258
+ }
259
+ }
260
+
261
+ registerProcessor('audio-processor', AudioProcessor);