@tiny-codes/react-easy 1.4.7 → 1.4.8
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/CHANGELOG.md +14 -0
- package/es/assets/request-camera-en.js +1 -1
- package/es/assets/request-camera-zh.js +1 -1
- package/es/assets/request-microphone-en.js +1 -1
- package/es/assets/request-microphone-zh.js +1 -1
- package/es/assets/reset-camera-en.js +1 -1
- package/es/assets/reset-camera-zh.js +1 -1
- package/es/assets/reset-microphone-en.js +1 -1
- package/es/assets/reset-microphone-zh.js +1 -1
- package/es/assets/save-default-audio1-en.js +1 -1
- package/es/assets/save-default-audio1-zh.js +1 -1
- package/es/assets/save-default-audio2-en.js +1 -1
- package/es/assets/save-default-audio2-zh.js +1 -1
- package/es/hooks/index.d.ts +3 -0
- package/es/hooks/index.js +3 -0
- package/es/hooks/index.js.map +1 -1
- package/es/hooks/useStompSocket.d.ts +63 -0
- package/es/hooks/useStompSocket.js +182 -0
- package/es/hooks/useStompSocket.js.map +1 -0
- package/es/hooks/useUserMedia.js +12 -24
- package/es/hooks/useUserMedia.js.map +1 -1
- package/es/locales/index.d.ts +6 -0
- package/es/locales/langs/en-US.d.ts +2 -0
- package/es/locales/langs/en-US.js +3 -1
- package/es/locales/langs/en-US.js.map +1 -1
- package/es/locales/langs/zh-CN.d.ts +2 -0
- package/es/locales/langs/zh-CN.js +3 -1
- package/es/locales/langs/zh-CN.js.map +1 -1
- package/lib/hooks/index.d.ts +3 -0
- package/lib/hooks/index.js +9 -1
- package/lib/hooks/index.js.map +2 -2
- package/lib/hooks/useStompSocket.d.ts +63 -0
- package/lib/hooks/useStompSocket.js +154 -0
- package/lib/hooks/useStompSocket.js.map +7 -0
- package/lib/hooks/useUserMedia.js +12 -24
- package/lib/hooks/useUserMedia.js.map +2 -2
- package/lib/locales/index.d.ts +6 -0
- package/lib/locales/langs/en-US.d.ts +2 -0
- package/lib/locales/langs/en-US.js +3 -1
- package/lib/locales/langs/en-US.js.map +2 -2
- package/lib/locales/langs/zh-CN.d.ts +2 -0
- package/lib/locales/langs/zh-CN.js +3 -1
- package/lib/locales/langs/zh-CN.js.map +2 -2
- package/package.json +9 -3
|
@@ -0,0 +1,63 @@
|
|
|
1
|
+
import type { StompConfig } from '@stomp/stompjs';
|
|
2
|
+
import { Client } from '@stomp/stompjs';
|
|
3
|
+
export interface UseSocketOptions<M = string> {
|
|
4
|
+
/**
|
|
5
|
+
* - **EN:** Socket connection address
|
|
6
|
+
* - **CN:** Socket连接地址
|
|
7
|
+
*/
|
|
8
|
+
url: string;
|
|
9
|
+
/**
|
|
10
|
+
* - **EN:** STOMP connection configuration
|
|
11
|
+
* - **CN:** - **CN:** STOMP连接配置
|
|
12
|
+
*/
|
|
13
|
+
connectConfig?: StompConfig;
|
|
14
|
+
/**
|
|
15
|
+
* - **EN:** Channel path for publishing messages
|
|
16
|
+
* - **CN:** 发布消息的频道路径
|
|
17
|
+
*/
|
|
18
|
+
sendEndpoint?: string;
|
|
19
|
+
/**
|
|
20
|
+
* - **EN:** Endpoint path for receiving messages
|
|
21
|
+
* - **CN:** 接收消息的端点路径
|
|
22
|
+
*/
|
|
23
|
+
subscribeEndpoint?: string;
|
|
24
|
+
/**
|
|
25
|
+
* - **EN:** Connection success callback
|
|
26
|
+
* - **CN:** 连接成功回调
|
|
27
|
+
*/
|
|
28
|
+
onConnected?: () => void;
|
|
29
|
+
/**
|
|
30
|
+
* - **EN:** Message received callback
|
|
31
|
+
* - **CN:** 接收到消息回调
|
|
32
|
+
*/
|
|
33
|
+
onMessage?: (message: M) => void;
|
|
34
|
+
/**
|
|
35
|
+
* - **EN:** Parse message body, return value as input parameter for `onMessage`
|
|
36
|
+
* - **CN:** 解析消息体,返回值作为`onMessage`的输入参数。
|
|
37
|
+
*
|
|
38
|
+
* @param body - The raw message body received from the server, needs to be deserialized based on
|
|
39
|
+
* actual conditions. | 从服务端接收到的原始消息体,需要根据实际情况进行反序列化。
|
|
40
|
+
*
|
|
41
|
+
* @returns Parsed message body as the input parameter of `onMessage`| 解析后的消息体,作为`onMessage`的输入参数
|
|
42
|
+
*/
|
|
43
|
+
parseMessageBody?: (body: string) => M;
|
|
44
|
+
/**
|
|
45
|
+
* - **EN:** Connection close callback
|
|
46
|
+
* - **CN:** 连接关闭回调
|
|
47
|
+
*/
|
|
48
|
+
onClose?: () => void;
|
|
49
|
+
}
|
|
50
|
+
/**
|
|
51
|
+
* - **EN:** Establish a WebSocket based on the STOMP protocol, mainly used for bidirectional
|
|
52
|
+
* transmission of serializable character messages.
|
|
53
|
+
* - **CN:** 建立基于STOMP协议的WebSocket,主要用于双向传递可序列化的字符型消息
|
|
54
|
+
*/
|
|
55
|
+
declare function useStompSocket<M = string>(options: UseSocketOptions<M>): {
|
|
56
|
+
connect: () => Promise<void>;
|
|
57
|
+
close: () => void;
|
|
58
|
+
send: (body: string) => void;
|
|
59
|
+
connecting: boolean;
|
|
60
|
+
socket: WebSocket | undefined;
|
|
61
|
+
stompClient: Client | undefined;
|
|
62
|
+
};
|
|
63
|
+
export default useStompSocket;
|
|
@@ -0,0 +1,154 @@
|
|
|
1
|
+
var __create = Object.create;
|
|
2
|
+
var __defProp = Object.defineProperty;
|
|
3
|
+
var __getOwnPropDesc = Object.getOwnPropertyDescriptor;
|
|
4
|
+
var __getOwnPropNames = Object.getOwnPropertyNames;
|
|
5
|
+
var __getProtoOf = Object.getPrototypeOf;
|
|
6
|
+
var __hasOwnProp = Object.prototype.hasOwnProperty;
|
|
7
|
+
var __export = (target, all) => {
|
|
8
|
+
for (var name in all)
|
|
9
|
+
__defProp(target, name, { get: all[name], enumerable: true });
|
|
10
|
+
};
|
|
11
|
+
var __copyProps = (to, from, except, desc) => {
|
|
12
|
+
if (from && typeof from === "object" || typeof from === "function") {
|
|
13
|
+
for (let key of __getOwnPropNames(from))
|
|
14
|
+
if (!__hasOwnProp.call(to, key) && key !== except)
|
|
15
|
+
__defProp(to, key, { get: () => from[key], enumerable: !(desc = __getOwnPropDesc(from, key)) || desc.enumerable });
|
|
16
|
+
}
|
|
17
|
+
return to;
|
|
18
|
+
};
|
|
19
|
+
var __toESM = (mod, isNodeMode, target) => (target = mod != null ? __create(__getProtoOf(mod)) : {}, __copyProps(
|
|
20
|
+
// If the importer is in node compatibility mode or this is not an ESM
|
|
21
|
+
// file that has been converted to a CommonJS file using a Babel-
|
|
22
|
+
// compatible transform (i.e. "__esModule" has not been set), then set
|
|
23
|
+
// "default" to the CommonJS "module.exports" for node compatibility.
|
|
24
|
+
isNodeMode || !mod || !mod.__esModule ? __defProp(target, "default", { value: mod, enumerable: true }) : target,
|
|
25
|
+
mod
|
|
26
|
+
));
|
|
27
|
+
var __toCommonJS = (mod) => __copyProps(__defProp({}, "__esModule", { value: true }), mod);
|
|
28
|
+
|
|
29
|
+
// src/hooks/useStompSocket.ts
|
|
30
|
+
var useStompSocket_exports = {};
|
|
31
|
+
__export(useStompSocket_exports, {
|
|
32
|
+
default: () => useStompSocket_default
|
|
33
|
+
});
|
|
34
|
+
module.exports = __toCommonJS(useStompSocket_exports);
|
|
35
|
+
var import_react = require("react");
|
|
36
|
+
var import_stompjs = require("@stomp/stompjs");
|
|
37
|
+
var import_antd = require("antd");
|
|
38
|
+
var import_sockjs_client = __toESM(require("sockjs-client"));
|
|
39
|
+
var import_useRefFunction = __toESM(require("./useRefFunction"));
|
|
40
|
+
var import_useT = __toESM(require("./useT"));
|
|
41
|
+
function useStompSocket(options) {
|
|
42
|
+
const { url, sendEndpoint, subscribeEndpoint, connectConfig, onMessage, parseMessageBody, onConnected, onClose } = options;
|
|
43
|
+
const t = (0, import_useT.default)();
|
|
44
|
+
const socketRef = (0, import_react.useRef)(void 0);
|
|
45
|
+
const stompClientRef = (0, import_react.useRef)(void 0);
|
|
46
|
+
const [connecting, setConnecting] = (0, import_react.useState)(false);
|
|
47
|
+
const isConnectedRef = (0, import_react.useRef)(false);
|
|
48
|
+
const [, refresh] = (0, import_react.useState)();
|
|
49
|
+
const connect = (0, import_useRefFunction.default)(async () => {
|
|
50
|
+
const promise = new Promise((resolve, reject) => {
|
|
51
|
+
try {
|
|
52
|
+
setConnecting(true);
|
|
53
|
+
socketRef.current = new import_sockjs_client.default(url);
|
|
54
|
+
stompClientRef.current = new import_stompjs.Client({
|
|
55
|
+
heartbeatIncoming: 5e3,
|
|
56
|
+
heartbeatOutgoing: 5e3,
|
|
57
|
+
...connectConfig,
|
|
58
|
+
webSocketFactory: () => socketRef.current
|
|
59
|
+
});
|
|
60
|
+
stompClientRef.current.activate();
|
|
61
|
+
stompClientRef.current.onConnect = () => {
|
|
62
|
+
var _a;
|
|
63
|
+
setConnecting(false);
|
|
64
|
+
isConnectedRef.current = true;
|
|
65
|
+
onConnected == null ? void 0 : onConnected();
|
|
66
|
+
if (subscribeEndpoint) {
|
|
67
|
+
(_a = stompClientRef.current) == null ? void 0 : _a.subscribe(subscribeEndpoint, (response) => {
|
|
68
|
+
if (parseMessageBody) {
|
|
69
|
+
onMessage == null ? void 0 : onMessage(parseMessageBody(response.body));
|
|
70
|
+
} else {
|
|
71
|
+
onMessage == null ? void 0 : onMessage(response.body);
|
|
72
|
+
}
|
|
73
|
+
});
|
|
74
|
+
}
|
|
75
|
+
resolve();
|
|
76
|
+
};
|
|
77
|
+
stompClientRef.current.onStompError = (error) => {
|
|
78
|
+
console.error("STOMP Error:", error);
|
|
79
|
+
};
|
|
80
|
+
stompClientRef.current.onWebSocketError = (error) => {
|
|
81
|
+
console.error("WebSocket Error:", error);
|
|
82
|
+
};
|
|
83
|
+
if (socketRef.current) {
|
|
84
|
+
socketRef.current.onerror = (error) => {
|
|
85
|
+
console.error(error);
|
|
86
|
+
};
|
|
87
|
+
}
|
|
88
|
+
stompClientRef.current.onWebSocketClose = (event) => {
|
|
89
|
+
var _a;
|
|
90
|
+
setConnecting(false);
|
|
91
|
+
if (event.type === "close" && event.code === 1e3) {
|
|
92
|
+
return;
|
|
93
|
+
}
|
|
94
|
+
(_a = stompClientRef.current) == null ? void 0 : _a.debug("StompClient closed");
|
|
95
|
+
if (isConnectedRef.current) {
|
|
96
|
+
isConnectedRef.current = false;
|
|
97
|
+
onClose == null ? void 0 : onClose();
|
|
98
|
+
import_antd.notification.error({ message: void 0, description: t("hooks.useStompSocket.serverDisconnected") });
|
|
99
|
+
} else {
|
|
100
|
+
import_antd.notification.error({ message: t("hooks.useStompSocket.connectError") });
|
|
101
|
+
}
|
|
102
|
+
};
|
|
103
|
+
if (socketRef.current) {
|
|
104
|
+
socketRef.current.onclose = (event) => {
|
|
105
|
+
var _a;
|
|
106
|
+
setConnecting(false);
|
|
107
|
+
isConnectedRef.current = false;
|
|
108
|
+
(_a = stompClientRef.current) == null ? void 0 : _a.debug("Socket closed");
|
|
109
|
+
console.log("event", event);
|
|
110
|
+
onClose == null ? void 0 : onClose();
|
|
111
|
+
};
|
|
112
|
+
}
|
|
113
|
+
} catch (error) {
|
|
114
|
+
console.error(error);
|
|
115
|
+
reject(error);
|
|
116
|
+
}
|
|
117
|
+
});
|
|
118
|
+
refresh();
|
|
119
|
+
await new Promise((resolve) => setTimeout(resolve));
|
|
120
|
+
return promise;
|
|
121
|
+
});
|
|
122
|
+
const close = (0, import_useRefFunction.default)(() => {
|
|
123
|
+
var _a, _b;
|
|
124
|
+
try {
|
|
125
|
+
(_a = stompClientRef.current) == null ? void 0 : _a.deactivate();
|
|
126
|
+
(_b = socketRef.current) == null ? void 0 : _b.close();
|
|
127
|
+
isConnectedRef.current = false;
|
|
128
|
+
setConnecting(false);
|
|
129
|
+
} catch (error) {
|
|
130
|
+
console.error(error);
|
|
131
|
+
}
|
|
132
|
+
});
|
|
133
|
+
const send = (0, import_useRefFunction.default)((body) => {
|
|
134
|
+
var _a;
|
|
135
|
+
if (!sendEndpoint) {
|
|
136
|
+
console.error("No publish endpoint defined, unable to send message");
|
|
137
|
+
return;
|
|
138
|
+
}
|
|
139
|
+
(_a = stompClientRef.current) == null ? void 0 : _a.publish({
|
|
140
|
+
destination: sendEndpoint,
|
|
141
|
+
body
|
|
142
|
+
});
|
|
143
|
+
});
|
|
144
|
+
return {
|
|
145
|
+
connect,
|
|
146
|
+
close,
|
|
147
|
+
send,
|
|
148
|
+
connecting,
|
|
149
|
+
socket: socketRef.current,
|
|
150
|
+
stompClient: stompClientRef.current
|
|
151
|
+
};
|
|
152
|
+
}
|
|
153
|
+
var useStompSocket_default = useStompSocket;
|
|
154
|
+
//# sourceMappingURL=useStompSocket.js.map
|
|
@@ -0,0 +1,7 @@
|
|
|
1
|
+
{
|
|
2
|
+
"version": 3,
|
|
3
|
+
"sources": ["../../src/hooks/useStompSocket.ts"],
|
|
4
|
+
"sourcesContent": ["import { useRef, useState } from 'react';\nimport type { StompConfig } from '@stomp/stompjs';\nimport { Client } from '@stomp/stompjs';\nimport { notification } from 'antd';\nimport SockJS from 'sockjs-client';\nimport useRefFunction from './useRefFunction';\nimport useT from './useT';\n\nexport interface UseSocketOptions<M = string> {\n /**\n * - **EN:** Socket connection address\n * - **CN:** Socket连接地址\n */\n url: string;\n /**\n * - **EN:** STOMP connection configuration\n * - **CN:** - **CN:** STOMP连接配置\n */\n connectConfig?: StompConfig;\n /**\n * - **EN:** Channel path for publishing messages\n * - **CN:** 发布消息的频道路径\n */\n sendEndpoint?: string;\n /**\n * - **EN:** Endpoint path for receiving messages\n * - **CN:** 接收消息的端点路径\n */\n subscribeEndpoint?: string;\n /**\n * - **EN:** Connection success callback\n * - **CN:** 连接成功回调\n */\n onConnected?: () => void;\n /**\n * - **EN:** Message received callback\n * - **CN:** 接收到消息回调\n */\n onMessage?: (message: M) => void;\n /**\n * - **EN:** Parse message body, return value as input parameter for `onMessage`\n * - **CN:** 解析消息体,返回值作为`onMessage`的输入参数。\n *\n * @param body - The raw message body received from the server, needs to be deserialized based on\n * actual conditions. | 从服务端接收到的原始消息体,需要根据实际情况进行反序列化。\n *\n * @returns Parsed message body as the input parameter of `onMessage`| 解析后的消息体,作为`onMessage`的输入参数\n */\n parseMessageBody?: (body: string) => M;\n /**\n * - **EN:** Connection close callback\n * - **CN:** 连接关闭回调\n */\n onClose?: () => void;\n}\n\n/**\n * - **EN:** Establish a WebSocket based on the STOMP protocol, mainly used for bidirectional\n * transmission of serializable character messages.\n * - **CN:** 建立基于STOMP协议的WebSocket,主要用于双向传递可序列化的字符型消息\n */\nfunction useStompSocket<M = string>(options: UseSocketOptions<M>) {\n const { url, sendEndpoint, subscribeEndpoint, connectConfig, onMessage, parseMessageBody, onConnected, onClose } =\n options;\n const t = useT();\n const socketRef = useRef<WebSocket | undefined>(undefined);\n const stompClientRef = useRef<Client | undefined>(undefined);\n const [connecting, setConnecting] = useState(false);\n const isConnectedRef = useRef(false);\n // eslint-disable-next-line @typescript-eslint/no-invalid-void-type\n const [, refresh] = useState<void>();\n\n const connect = useRefFunction(async () => {\n const promise = new Promise<void>((resolve, reject) => {\n try {\n setConnecting(true);\n // Create SockJS instance\n socketRef.current = new SockJS(url);\n\n // Create STOMP client\n stompClientRef.current = new Client({\n heartbeatIncoming: 5000,\n heartbeatOutgoing: 5000,\n ...connectConfig,\n webSocketFactory: () => socketRef.current,\n });\n // Connect to STOMP server\n stompClientRef.current.activate();\n\n // STOMP server connection established\n stompClientRef.current.onConnect = () => {\n setConnecting(false);\n isConnectedRef.current = true;\n onConnected?.();\n if (subscribeEndpoint) {\n stompClientRef.current?.subscribe(subscribeEndpoint, (response) => {\n if (parseMessageBody) {\n onMessage?.(parseMessageBody(response.body));\n } else {\n onMessage?.(response.body as M);\n }\n });\n }\n resolve();\n };\n stompClientRef.current.onStompError = (error) => {\n console.error('STOMP Error:', error);\n };\n stompClientRef.current.onWebSocketError = (error) => {\n console.error('WebSocket Error:', error);\n };\n if (socketRef.current) {\n socketRef.current.onerror = (error: unknown) => {\n console.error(error);\n };\n }\n\n stompClientRef.current.onWebSocketClose = (event) => {\n setConnecting(false);\n // Normal close\n if (event.type === 'close' && event.code === 1000) {\n return;\n }\n stompClientRef.current?.debug('StompClient closed');\n if (isConnectedRef.current) {\n isConnectedRef.current = false;\n onClose?.();\n notification.error({ message: undefined, description: t('hooks.useStompSocket.serverDisconnected') });\n } else {\n // Client connection failed\n notification.error({ message: t('hooks.useStompSocket.connectError') });\n }\n };\n if (socketRef.current) {\n socketRef.current.onclose = (event) => {\n setConnecting(false);\n isConnectedRef.current = false;\n stompClientRef.current?.debug('Socket closed');\n console.log('event', event);\n onClose?.();\n };\n }\n } catch (error: unknown) {\n console.error(error);\n // notification.error({ message: error?.message ?? JSON.stringify(error) });\n reject(error);\n }\n });\n refresh();\n await new Promise((resolve) => setTimeout(resolve));\n return promise;\n });\n const close = useRefFunction(() => {\n try {\n stompClientRef.current?.deactivate();\n socketRef.current?.close();\n isConnectedRef.current = false;\n setConnecting(false);\n } catch (error) {\n console.error(error);\n }\n });\n const send = useRefFunction((body: string) => {\n if (!sendEndpoint) {\n console.error('No publish endpoint defined, unable to send message');\n return;\n }\n stompClientRef.current?.publish({\n destination: sendEndpoint,\n body,\n });\n });\n return {\n connect,\n close,\n send,\n connecting,\n socket: socketRef.current,\n stompClient: stompClientRef.current,\n };\n}\n\nexport default useStompSocket;\n"],
|
|
5
|
+
"mappings": "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",
|
|
6
|
+
"names": ["useT", "useRefFunction", "SockJS"]
|
|
7
|
+
}
|
|
@@ -40,18 +40,6 @@ var import_stream = require("../utils/stream");
|
|
|
40
40
|
var import_useRefFunction = __toESM(require("./useRefFunction"));
|
|
41
41
|
var import_useRefValue = __toESM(require("./useRefValue"));
|
|
42
42
|
var import_useT = __toESM(require("./useT"));
|
|
43
|
-
var requestMicrophoneEnUrlPromise = import("../assets/request-microphone-en.js");
|
|
44
|
-
var requestMicrophoneZhUrlPromise = import("../assets/request-microphone-zh.js");
|
|
45
|
-
var resetMicrophoneEnUrlPromise = import("../assets/reset-microphone-en.js");
|
|
46
|
-
var resetMicrophoneZhUrlPromise = import("../assets/reset-microphone-zh.js");
|
|
47
|
-
var saveAudioDeviceEnUrlPromise1 = import("../assets/save-default-audio1-en.js");
|
|
48
|
-
var saveAudioDeviceZhUrlPromise1 = import("../assets/save-default-audio1-zh.js");
|
|
49
|
-
var saveAudioDeviceEnUrlPromise2 = import("../assets/save-default-audio2-en.js");
|
|
50
|
-
var saveAudioDeviceZhUrlPromise2 = import("../assets/save-default-audio2-zh.js");
|
|
51
|
-
var requestCameraEnUrlPromise = import("../assets/request-camera-en.js");
|
|
52
|
-
var requestCameraZhUrlPromise = import("../assets/request-camera-zh.js");
|
|
53
|
-
var resetCameraEnUrlPromise = import("../assets/reset-camera-en.js");
|
|
54
|
-
var resetCameraZhUrlPromise = import("../assets/reset-camera-zh.js");
|
|
55
43
|
var useUserMedia = (props) => {
|
|
56
44
|
var _a;
|
|
57
45
|
const {
|
|
@@ -382,28 +370,28 @@ var useUserMedia = (props) => {
|
|
|
382
370
|
};
|
|
383
371
|
}, [includeAudio, mediaStream, t]);
|
|
384
372
|
(0, import_react.useEffect)(() => {
|
|
385
|
-
|
|
373
|
+
import("../assets/request-microphone-en.js").then((module2) => {
|
|
386
374
|
setRequestMicrophoneEnUrl(module2.default);
|
|
387
375
|
});
|
|
388
|
-
|
|
376
|
+
import("../assets/request-microphone-zh.js").then((module2) => {
|
|
389
377
|
setRequestMicrophoneZhUrl(module2.default);
|
|
390
378
|
});
|
|
391
|
-
|
|
379
|
+
import("../assets/reset-microphone-en.js").then((module2) => {
|
|
392
380
|
setResetMicrophoneEnUrl(module2.default);
|
|
393
381
|
});
|
|
394
|
-
|
|
382
|
+
import("../assets/reset-microphone-zh.js").then((module2) => {
|
|
395
383
|
setResetMicrophoneZhUrl(module2.default);
|
|
396
384
|
});
|
|
397
|
-
|
|
385
|
+
import("../assets/request-camera-en.js").then((module2) => {
|
|
398
386
|
setRequestCameraEnUrl(module2.default);
|
|
399
387
|
});
|
|
400
|
-
|
|
388
|
+
import("../assets/request-camera-zh.js").then((module2) => {
|
|
401
389
|
setRequestCameraZhUrl(module2.default);
|
|
402
390
|
});
|
|
403
|
-
|
|
391
|
+
import("../assets/reset-camera-en.js").then((module2) => {
|
|
404
392
|
setResetCameraEnUrl(module2.default);
|
|
405
393
|
});
|
|
406
|
-
|
|
394
|
+
import("../assets/reset-camera-zh.js").then((module2) => {
|
|
407
395
|
setResetCameraZhUrl(module2.default);
|
|
408
396
|
});
|
|
409
397
|
}, []);
|
|
@@ -446,16 +434,16 @@ function SaveAudioDeviceForm(props) {
|
|
|
446
434
|
window.open(url);
|
|
447
435
|
});
|
|
448
436
|
(0, import_react.useEffect)(() => {
|
|
449
|
-
|
|
437
|
+
import("../assets/save-default-audio1-en.js").then((module2) => {
|
|
450
438
|
setSaveAudioDeviceEnUrl1(module2.default);
|
|
451
439
|
});
|
|
452
|
-
|
|
440
|
+
import("../assets/save-default-audio2-en.js").then((module2) => {
|
|
453
441
|
setSaveAudioDeviceEnUrl2(module2.default);
|
|
454
442
|
});
|
|
455
|
-
|
|
443
|
+
import("../assets/save-default-audio1-zh.js").then((module2) => {
|
|
456
444
|
setSaveAudioDeviceZhUrl1(module2.default);
|
|
457
445
|
});
|
|
458
|
-
|
|
446
|
+
import("../assets/save-default-audio2-zh.js").then((module2) => {
|
|
459
447
|
setSaveAudioDeviceZhUrl2(module2.default);
|
|
460
448
|
});
|
|
461
449
|
}, []);
|
|
@@ -1,7 +1,7 @@
|
|
|
1
1
|
{
|
|
2
2
|
"version": 3,
|
|
3
3
|
"sources": ["../../src/hooks/useUserMedia.tsx"],
|
|
4
|
-
"sourcesContent": ["import { useContext, useEffect, useMemo, useRef, useState } from 'react';\nimport { App, Checkbox, Flex, Form, Modal, notification, Select, Typography } from 'antd';\nimport EasyConfigProvider from '../components/ConfigProvider';\nimport ReactEasyContext from '../components/ConfigProvider/context';\nimport { StreamTimeSlicerClass } from '../utils/stream';\nimport useRefFunction from './useRefFunction';\nimport useRefValue from './useRefValue';\nimport useT from './useT';\n\nconst requestMicrophoneEnUrlPromise = import('../assets/request-microphone-en.js');\nconst requestMicrophoneZhUrlPromise = import('../assets/request-microphone-zh.js');\nconst resetMicrophoneEnUrlPromise = import('../assets/reset-microphone-en.js');\nconst resetMicrophoneZhUrlPromise = import('../assets/reset-microphone-zh.js');\nconst saveAudioDeviceEnUrlPromise1 = import('../assets/save-default-audio1-en.js');\nconst saveAudioDeviceZhUrlPromise1 = import('../assets/save-default-audio1-zh.js');\nconst saveAudioDeviceEnUrlPromise2 = import('../assets/save-default-audio2-en.js');\nconst saveAudioDeviceZhUrlPromise2 = import('../assets/save-default-audio2-zh.js');\nconst requestCameraEnUrlPromise = import('../assets/request-camera-en.js');\nconst requestCameraZhUrlPromise = import('../assets/request-camera-zh.js');\nconst resetCameraEnUrlPromise = import('../assets/reset-camera-en.js');\nconst resetCameraZhUrlPromise = import('../assets/reset-camera-zh.js');\n\nexport interface UseUserMediaProps {\n /**\n * - **EN:** The media constraints for the audio and video stream.\n * - **CN:** 媒体流的媒体约束。\n */\n media: Pick<MediaStreamConstraints, 'audio' | 'video'>;\n /**\n * - **EN:** Whether to enable PCM output, only effective when recording audio. Please use\n * `onPcmStreamChunk` callback to get PCM data. Default is `false`.\n * - **CN:** 是否启用 PCM 输出,仅在录制音频时有效。请使用 `onPcmStreamChunk` 回调获取 PCM 数据。默认`false`\n */\n pcmAudioOptions?: {\n /**\n * - **EN:** The audio context options for the PCM output.\n * - **CN:** PCM 输出的音频上下文选项。\n */\n audioContext?: AudioContextOptions;\n /**\n * - **EN:** The worklet options for the PCM output.\n * - **CN:** PCM 输出的工作线程选项。\n */\n workletOptions?: AudioWorkletNodeOptions;\n };\n /**\n * - **EN:** Callback function that is triggered when the recording starts, providing the media\n * stream.\n * - **CN:** 开始录制时触发的回调函数,提供媒体流。\n *\n * @param {MediaStream} stream - The media stream.\n */\n onStartRecording?: (stream: MediaStream) => void;\n /**\n * - **EN:** Callback function that is triggered when the recording stops.\n * - **CN:** 停止录制时触发的回调函数。\n */\n onStopRecording?: () => void;\n /**\n * - **EN:** Callback function that is triggered when a new chunk of media data is available.\n * - **CN:** 当录制媒体流时,每个时间分片会触发一次 `onStreamChunk` 回调,提供媒体数据块。\n *\n * > 注意音频流编码格式为:audio/webm;codecs=opus。如果希望获取 PCM 数据,请使用 `onPcmData` 回调。\n *\n * @param {Blob} chunk - The media data chunk (MIME: audio/webm;codecs=opus) | 媒体数据块 (MIME:\n * audio/webm;codecs=opus)\n */\n onStreamChunk?: (chunk: Blob) => void;\n /**\n * - **EN:** Callback for raw PCM float data (per render quantum)\n * - **CN:** 获取原始 PCM 浮点数据的回调(每个渲染量子)\n *\n * @param data Monophonic or polyphonic spliced data | 单声道或多声道拼接数据\n * @param sampleRate Sample rate | 采样率\n */\n onPcmStreamChunk?: (channels: Float32Array[], sampleRate: number) => void;\n /**\n * - **EN:** Whether to disable this hook.\n * - **CN:** 是否禁用此工具\n */\n disabled?: boolean;\n /**\n * - **EN:** The slicing time period (milliseconds) for each fragment of the audio and video stream,\n * each time slice will trigger the `onStreamChunk` callback. Default is `500`.\n * - **CN:** 媒体流每个分片的切片时间段(毫秒),每个时间分片会触发一次 `onStreamChunk` 回调,默认值为 `500`。\n */\n streamSliceMs?: number;\n /**\n * - **EN:** The silence detection threshold (0-1) for the audio stream, below which the audio is\n * considered silent. Default is `0`.\n * - **CN:** 音频流的静音检测阈值(0-1),低于该值音频被视为静音。默认值为 `0`。\n */\n soundDetectionThreshold?: number;\n /**\n * - **EN:** The timeout duration (milliseconds) for detecting sound input. If no sound is detected\n * within this period, the user will be prompted to re-select the audio device. Default is\n * `3000`.\n * - **CN:** 检测是否有声音输入的超时时间(毫秒),如果在该时间段内没有检测到声音,则会提示用户重新选择音频设备。默认值为 `3000`。\n */\n soundDetectionTimeout?: number;\n}\n\nconst useUserMedia = (props: UseUserMediaProps): UseUserMediaResult => {\n const {\n media,\n pcmAudioOptions,\n disabled,\n streamSliceMs = 500,\n soundDetectionThreshold = 0,\n soundDetectionTimeout = 3000,\n onStartRecording,\n onStopRecording,\n onStreamChunk,\n onPcmStreamChunk,\n } = props;\n const context = useContext(ReactEasyContext);\n const { lang } = context;\n const contextRef = useRefValue(context);\n const t = useT();\n const app = App.useApp();\n // @ts-expect-error: because app may return a stub object when App is not used\n const modal = app.modal?.confirm ? app.modal : Modal;\n const modalRef = useRefValue(modal);\n const [requestMicrophoneEnUrl, setRequestMicrophoneEnUrl] = useState<string>();\n const [requestMicrophoneZhUrl, setRequestMicrophoneZhUrl] = useState<string>();\n const [resetMicrophoneEnUrl, setResetMicrophoneEnUrl] = useState<string>();\n const [resetMicrophoneZhUrl, setResetMicrophoneZhUrl] = useState<string>();\n const [requestCameraEnUrl, setRequestCameraEnUrl] = useState<string>();\n const [requestCameraZhUrl, setRequestCameraZhUrl] = useState<string>();\n const [resetCameraEnUrl, setResetCameraEnUrl] = useState<string>();\n const [resetCameraZhUrl, setResetCameraZhUrl] = useState<string>();\n const [isRecording, setIsRecording] = useState(false);\n const [mediaStream, setMediaStream] = useState<MediaStream>();\n const [mediaRecorder, setMediaRecorder] = useState<MediaRecorder | null>(null);\n const stopSoundListeningRef = useRef<(() => void) | undefined>(undefined);\n const closePcmRef = useRef<(() => void) | undefined>(undefined);\n const includeAudio = !!media.audio;\n const exactAudioDeviceIdRef = useRef<string | undefined>(undefined);\n const rafRef = useRef<number>(0);\n const isSpeakingRef = useRef(false);\n const silenceVolumeThresholdRef = useRefValue(soundDetectionThreshold);\n const silenceDetectDurationRef = useRefValue(soundDetectionTimeout);\n const soundDetectStart = useRef<number>(0);\n const pcmSampleRateRef = useRef<number>(0);\n const onPcmStreamChunkRef = useRefValue(onPcmStreamChunk);\n const pcmStreamSlicerRef = useRef(\n new StreamTimeSlicerClass({\n timeSlice: streamSliceMs,\n onSlice: (channels) => {\n onPcmStreamChunkRef.current?.(channels, pcmSampleRateRef.current);\n },\n })\n );\n const deviceType = useMemo(\n () => (media.video ? t('hooks.useUserMedia.camera') : t('hooks.useUserMedia.microphone')),\n [media, t]\n );\n const featureName = useMemo(\n () => (media.video ? t('hooks.featureName.camera') : t('hooks.featureName.microphone')),\n [media, t]\n );\n\n const showDeniedPopup = () => {\n const resetMicrophoneUrl = lang === 'zh-CN' ? resetMicrophoneZhUrl : resetMicrophoneEnUrl;\n const resetCameraUrl = lang === 'zh-CN' ? resetCameraZhUrl : resetCameraEnUrl;\n modal.error({\n title: t('hooks.useUserMedia.devicePermission', { deviceType }),\n width: 500,\n content: (\n <div>\n <Typography.Paragraph></Typography.Paragraph>\n <Typography.Paragraph>\n <Typography.Text strong type=\"danger\">\n {t('hooks.useUserMedia.deniedPermission', { deviceType, featureName })}\n </Typography.Text>\n </Typography.Paragraph>\n <Typography.Paragraph>{t('hooks.useUserMedia.reopenPermissionGuide', { deviceType })}</Typography.Paragraph>\n <img\n src={media.video ? resetCameraUrl : resetMicrophoneUrl}\n alt=\"microphone-permission\"\n style={{ width: 380 }}\n />\n </div>\n ),\n });\n };\n const recordStream = async () => {\n let stream: MediaStream;\n try {\n const options = media;\n if (media.audio) {\n if (exactAudioDeviceIdRef.current) {\n if (media.audio === true) {\n options.audio = { deviceId: { exact: exactAudioDeviceIdRef.current } };\n } else {\n options.audio = { deviceId: { exact: exactAudioDeviceIdRef.current }, ...media.audio };\n }\n }\n }\n stream = await navigator.mediaDevices.getUserMedia(options);\n setMediaStream(stream);\n onStartRecording?.(stream);\n\n const recorder = new MediaRecorder(stream);\n recorder.ondataavailable = (event) => {\n if (event.data.size > 0) {\n onStreamChunk?.(event.data);\n }\n };\n if (streamSliceMs) {\n recorder.start(streamSliceMs);\n } else {\n recorder.start();\n }\n setMediaRecorder(recorder);\n setIsRecording(true);\n\n // Capture PCM data if enabled\n if (options.audio && onPcmStreamChunkRef.current) {\n try {\n const ctx = new AudioContext(pcmAudioOptions?.audioContext);\n pcmSampleRateRef.current = ctx.sampleRate;\n // Safari might need resume\n if (ctx.state === 'suspended') {\n await ctx.resume();\n }\n const sourceNode = ctx.createMediaStreamSource(stream);\n let node: AudioWorkletNode | undefined;\n\n closePcmRef.current = () => {\n node?.port.close();\n node?.disconnect();\n sourceNode.disconnect();\n ctx.close().catch(() => {\n // Ignore errors\n });\n };\n\n const setupWorklet = async () => {\n try {\n // Load the worklet module\n await ctx.audioWorklet.addModule(generatePcmCaptureProcessorModule());\n node = new AudioWorkletNode(ctx, 'pcm-capture', pcmAudioOptions?.workletOptions);\n node.port.onmessage = (e: MessageEvent) => {\n if (e.data?.type === 'pcm') {\n const channels = e.data.channels as Float32Array[];\n pcmStreamSlicerRef.current.push(channels);\n }\n };\n sourceNode.connect(node);\n } catch (err) {\n fallbackScriptProcessor({ ctx, sourceNode, streamSlicer: pcmStreamSlicerRef.current });\n }\n };\n\n if ('audioWorklet' in ctx) {\n setupWorklet();\n } else {\n fallbackScriptProcessor({ ctx, sourceNode, streamSlicer: pcmStreamSlicerRef.current });\n }\n } catch (e) {\n console.error('setup pcm worklet failed', e);\n }\n }\n return recorder;\n } catch (error) {\n console.error(error);\n if (error instanceof Error && error.name === 'NotAllowedError') {\n showDeniedPopup();\n notification.error({ message: t('hooks.useUserMedia.deniedPermission', { deviceType, featureName }) });\n throw new Error(t('hooks.useUserMedia.deniedPermission', { deviceType, featureName }));\n }\n notification.error({ message: t('hooks.useUserMedia.notSupport') });\n throw new Error(t('hooks.useUserMedia.notSupport'));\n }\n };\n\n const startRecording = useRefFunction(async () => {\n if (disabled) {\n throw new Error(t('hooks.useUserMedia.disabledWarning'));\n }\n if (isRecording) {\n throw new Error(t('hooks.useUserMedia.isRecordingNow'));\n }\n if (!navigator.mediaDevices?.getUserMedia || !navigator.permissions?.query) {\n notification.error({\n message: t('hooks.useUserMedia.notSupport'),\n });\n throw new Error(t('hooks.useUserMedia.notSupport'));\n }\n try {\n const result = await window.navigator.permissions.query({\n // eslint-disable-next-line @typescript-eslint/no-explicit-any\n name: media.video ? ('camera' as any) : ('microphone' as any),\n });\n if (result.state === 'denied') {\n showDeniedPopup();\n notification.error({ message: t('hooks.useUserMedia.deniedPermission', { deviceType, featureName }) });\n throw new Error(t('hooks.useUserMedia.deniedPermission', { deviceType, featureName }));\n } else {\n if (result.state === 'prompt') {\n const requestMicrophoneUrl = lang === 'zh-CN' ? requestMicrophoneZhUrl : requestMicrophoneEnUrl;\n const requestCameraUrl = lang === 'zh-CN' ? requestCameraZhUrl : requestCameraEnUrl;\n return new Promise<MediaRecorder>((resolve, reject) => {\n modal.warning({\n title: t('hooks.useUserMedia.devicePermission', { deviceType }),\n content: (\n <div>\n <Typography.Paragraph></Typography.Paragraph>\n <Typography.Paragraph>\n <Typography.Text strong>{t('hooks.useUserMedia.requestTip1', { deviceType })}</Typography.Text>\n </Typography.Paragraph>\n <Typography.Paragraph>{t('hooks.useUserMedia.requestTip2', { featureName })}</Typography.Paragraph>\n <div>\n <img\n src={media.video ? requestCameraUrl : requestMicrophoneUrl}\n alt=\"microphone-permission\"\n style={{ width: 380 }}\n />\n </div>\n </div>\n ),\n onOk: () => {\n try {\n recordStream()\n .then((recorder) => {\n resolve(recorder);\n })\n .catch((error) => {\n reject(error);\n });\n } catch (error) {\n console.error(error);\n reject(error);\n }\n },\n width: 500,\n });\n });\n } else {\n return await recordStream();\n }\n }\n } catch (error) {\n console.error(error);\n throw error;\n }\n });\n\n const stopRecording = useRefFunction(() => {\n mediaRecorder?.stop();\n mediaStream?.getTracks().forEach((t) => t.stop());\n setMediaStream(undefined);\n setIsRecording(false);\n isSpeakingRef.current = false;\n cancelAnimationFrame(rafRef.current);\n stopSoundListeningRef.current?.();\n closePcmRef.current?.();\n onStopRecording?.();\n });\n\n // Wait for sound. If keep silent for a while, show a modal to let user reselect audio device\n const waitForSound = useRefFunction((mediaStream: MediaStream) => {\n const ctx = new AudioContext();\n const source = ctx.createMediaStreamSource(mediaStream);\n const analyser = ctx.createAnalyser();\n analyser.fftSize = 2048;\n source.connect(analyser);\n soundDetectStart.current = Date.now();\n const data = new Uint8Array(analyser.fftSize);\n const cancelDetect = () => {\n cancelAnimationFrame(rafRef.current);\n rafRef.current = 0;\n stopSoundListeningRef.current?.();\n };\n const loop = () => {\n analyser.getByteTimeDomainData(data);\n // Calculate RMS\n let sum = 0;\n for (const value of data) {\n const v = (value - 128) / 128;\n sum += v * v;\n }\n const rms = Math.sqrt(sum / data.length); // 0~1\n if (rms > silenceVolumeThresholdRef.current) {\n if (!isSpeakingRef.current) {\n isSpeakingRef.current = true;\n cancelDetect();\n return;\n }\n } else {\n if (Date.now() > soundDetectStart.current + silenceDetectDurationRef.current) {\n navigator.mediaDevices.enumerateDevices().then((devices) => {\n const audioInputs = devices.filter((d) => d.kind === 'audioinput');\n modalRef.current.confirm({\n title: t('hooks.useUserMedia.soundDetectTitle'),\n content: (\n <EasyConfigProvider {...contextRef.current}>\n <SaveAudioDeviceForm\n devices={audioInputs}\n mediaStream={mediaStream}\n onDeviceChange={(deviceId) => (exactAudioDeviceIdRef.current = deviceId)}\n />\n </EasyConfigProvider>\n ),\n width: 500,\n onOk: () => {\n if (exactAudioDeviceIdRef.current) {\n stopRecording();\n setTimeout(() => {\n startRecording();\n });\n }\n },\n onCancel: () => {\n cancelDetect();\n },\n });\n });\n\n cancelDetect();\n return;\n }\n }\n rafRef.current = requestAnimationFrame(loop);\n };\n loop();\n\n stopSoundListeningRef.current = () => {\n analyser.disconnect();\n source.disconnect();\n ctx.close().catch(() => {\n // Ignore errors\n });\n stopSoundListeningRef.current = undefined;\n };\n });\n\n // Component destroy\n useEffect(() => {\n return stopRecording;\n }, []);\n\n // Stop recording when disabled has been changed\n useEffect(() => {\n if (disabled && isRecording) {\n stopRecording();\n }\n }, [disabled, isRecording]);\n\n // Update PCM stream slicer time slice when input sample rate changes\n useEffect(() => {\n if (streamSliceMs && pcmStreamSlicerRef.current.timeSlice !== streamSliceMs) {\n pcmStreamSlicerRef.current.timeSlice = streamSliceMs;\n }\n }, [streamSliceMs]);\n\n // Detect sound activity (only for audio or media with audio)\n useEffect(() => {\n if (includeAudio && mediaStream && !isSpeakingRef.current) {\n try {\n waitForSound(mediaStream);\n } catch (e) {\n console.warn('Audio volume detecting failed:', e);\n }\n }\n return () => {\n stopSoundListeningRef.current?.();\n };\n }, [includeAudio, mediaStream, t]);\n\n useEffect(() => {\n requestMicrophoneEnUrlPromise.then((module) => {\n setRequestMicrophoneEnUrl(module.default);\n });\n requestMicrophoneZhUrlPromise.then((module) => {\n setRequestMicrophoneZhUrl(module.default);\n });\n resetMicrophoneEnUrlPromise.then((module) => {\n setResetMicrophoneEnUrl(module.default);\n });\n resetMicrophoneZhUrlPromise.then((module) => {\n setResetMicrophoneZhUrl(module.default);\n });\n requestCameraEnUrlPromise.then((module) => {\n setRequestCameraEnUrl(module.default);\n });\n requestCameraZhUrlPromise.then((module) => {\n setRequestCameraZhUrl(module.default);\n });\n resetCameraEnUrlPromise.then((module) => {\n setResetCameraEnUrl(module.default);\n });\n resetCameraZhUrlPromise.then((module) => {\n setResetCameraZhUrl(module.default);\n });\n }, []);\n\n return {\n isRecording,\n startRecording,\n stopRecording,\n };\n};\n\nexport interface UseUserMediaResult {\n /**\n * - **EN** Whether the media stream is currently being recorded\n * - **CN** 是否正在录制媒体流\n */\n isRecording: boolean;\n /**\n * - **EN** Start recording the media stream\n * - **CN** 开始录制媒体流\n */\n startRecording: () => Promise<MediaRecorder>;\n /**\n * - **EN** Stop recording the media stream\n * - **CN** 停止录制媒体流\n */\n stopRecording: () => void;\n /**\n * - **EN** Get the media stream being recorded, returns the stream if recording, otherwise returns\n * `undefined`\n * - **CN** 获取正在录制的媒体流,如果正在录制则返回该流,否则返回 `undefined`\n */\n mediaStream?: MediaStream;\n}\n\nfunction SaveAudioDeviceForm(props: {\n devices: MediaDeviceInfo[];\n mediaStream: MediaStream;\n onDeviceChange: (deviceId: string) => void;\n}) {\n const { devices, mediaStream, onDeviceChange } = props;\n const [form] = Form.useForm();\n const t = useT();\n const { lang } = useContext(ReactEasyContext);\n const [saveAudioDeviceEnUrl1, setSaveAudioDeviceEnUrl1] = useState<string>();\n const [saveAudioDeviceEnUrl2, setSaveAudioDeviceEnUrl2] = useState<string>();\n const [saveAudioDeviceZhUrl1, setSaveAudioDeviceZhUrl1] = useState<string>();\n const [saveAudioDeviceZhUrl2, setSaveAudioDeviceZhUrl2] = useState<string>();\n const [saveDefaultAudioDevicePermanently, setSaveDefaultAudioDevicePermanently] = useState(false);\n const audioInputs = useMemo(() => devices.filter((d) => d.kind === 'audioinput'), [devices]);\n const [selectedDeviceId, setSelectedDeviceId] = useState<string | undefined>(\n () => mediaStream.getAudioTracks()[0]?.getSettings()?.deviceId ?? audioInputs[0]?.deviceId\n );\n\n const openDataImageInNewTab = useRefFunction((dataUrl: string | undefined) => {\n if (!dataUrl) return;\n const [meta, b64] = dataUrl.split(',');\n const mime = meta.match(/data:(.*);base64/)?.[1] || 'image/png';\n const binary = atob(b64);\n const len = binary.length;\n const bytes = new Uint8Array(len);\n for (let i = 0; i < len; i++) bytes[i] = binary.charCodeAt(i);\n const blob = new Blob([bytes], { type: mime });\n const url = URL.createObjectURL(blob);\n window.open(url);\n });\n\n useEffect(() => {\n saveAudioDeviceEnUrlPromise1.then((module) => {\n setSaveAudioDeviceEnUrl1(module.default);\n });\n saveAudioDeviceEnUrlPromise2.then((module) => {\n setSaveAudioDeviceEnUrl2(module.default);\n });\n saveAudioDeviceZhUrlPromise1.then((module) => {\n setSaveAudioDeviceZhUrl1(module.default);\n });\n saveAudioDeviceZhUrlPromise2.then((module) => {\n setSaveAudioDeviceZhUrl2(module.default);\n });\n }, []);\n\n return (\n <Form layout=\"vertical\" form={form}>\n <Typography.Paragraph></Typography.Paragraph>\n <Typography.Paragraph>\n <Typography.Text>{t('hooks.useUserMedia.soundDetectDescription')}</Typography.Text>\n </Typography.Paragraph>\n <Form.Item label={t('hooks.useUserMedia.chooseMicrophoneDevice')}>\n <Select\n options={audioInputs.map((input) => ({\n label: input.label,\n value: input.deviceId,\n }))}\n defaultValue={selectedDeviceId}\n onChange={(id) => {\n setSelectedDeviceId(id);\n onDeviceChange(id);\n }}\n />\n </Form.Item>\n <Form.Item style={{ marginBottom: 0 }}>\n <Checkbox onChange={(e) => setSaveDefaultAudioDevicePermanently(e.target.checked)}>\n {t('hooks.useUserMedia.rememberDefaultAudioDevice')}\n </Checkbox>\n </Form.Item>\n {saveDefaultAudioDevicePermanently && (\n <div>\n <Typography.Paragraph>\n <Typography.Text>{t('hooks.useUserMedia.rememberDefaultAudioDeviceTip')}</Typography.Text>\n </Typography.Paragraph>\n <Flex gap={8} align=\"flex-start\">\n <div style={{ flex: 1, minWidth: 0 }}>\n <a\n href=\"#\"\n onClick={(e) => {\n e.preventDefault();\n openDataImageInNewTab(lang === 'zh-CN' ? saveAudioDeviceZhUrl1 : saveAudioDeviceEnUrl1);\n }}\n >\n <img\n src={lang === 'zh-CN' ? saveAudioDeviceZhUrl1 : saveAudioDeviceEnUrl1}\n alt=\"the first step to save default audio device\"\n style={{ width: '100%', height: 'auto' }}\n />\n </a>\n </div>\n <div style={{ flex: 1, minWidth: 0 }}>\n <a\n href=\"#\"\n onClick={(e) => {\n e.preventDefault();\n openDataImageInNewTab(lang === 'zh-CN' ? saveAudioDeviceZhUrl2 : saveAudioDeviceEnUrl2);\n }}\n >\n <img\n src={lang === 'zh-CN' ? saveAudioDeviceZhUrl2 : saveAudioDeviceEnUrl2}\n alt=\"the second step to save default audio device\"\n style={{ width: '100%', height: 'auto' }}\n />\n </a>\n </div>\n </Flex>\n </div>\n )}\n </Form>\n );\n}\n\nfunction generatePcmCaptureProcessorModule() {\n const workletCode = `\nclass PcmCaptureProcessor extends AudioWorkletProcessor {\n process(inputs, outputs, parameters) {\n const channelsIn = inputs[0];\n if (channelsIn && channelsIn[0]) {\n const channels = channelsIn.map((ch) => {\n const copy = new Float32Array(ch.length);\n copy.set(ch);\n return copy;\n });\n this.port.postMessage({ type: 'pcm', channels }, channels.map(ch => ch.buffer));\n }\n return true;\n }\n}\nregisterProcessor('pcm-capture', PcmCaptureProcessor);\n`;\n\n // Create a Blob from the worklet code and return its URL\n const blob = new Blob([workletCode], { type: 'application/javascript' });\n const blobUrl = URL.createObjectURL(blob);\n return blobUrl;\n}\n\nfunction fallbackScriptProcessor(options: {\n ctx: AudioContext;\n sourceNode: MediaStreamAudioSourceNode;\n streamSlicer: StreamTimeSlicerClass;\n}) {\n const { ctx, sourceNode, streamSlicer } = options;\n const bufferSize = 128;\n const processor = ctx.createScriptProcessor(bufferSize, 1, 1);\n processor.onaudioprocess = (ev) => {\n const channels = [];\n for (let i = 0; i < ev.inputBuffer.numberOfChannels; i++) {\n const input = ev.inputBuffer.getChannelData(i);\n const copy = new Float32Array(input.length);\n copy.set(input);\n channels.push(copy);\n }\n streamSlicer.push(channels);\n };\n sourceNode.connect(processor);\n // 可不输出:processor.connect(ctx!.destination);不连接 destination 在部分浏览器会被自动 GC,可连到 destination 或 gain(0)\n const gain = ctx.createGain();\n gain.gain.value = 0;\n processor.connect(gain).connect(ctx.destination);\n}\n\nexport default useUserMedia;\n"],
|
|
5
|
-
"mappings": "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",
|
|
4
|
+
"sourcesContent": ["import { useContext, useEffect, useMemo, useRef, useState } from 'react';\nimport { App, Checkbox, Flex, Form, Modal, notification, Select, Typography } from 'antd';\nimport EasyConfigProvider from '../components/ConfigProvider';\nimport ReactEasyContext from '../components/ConfigProvider/context';\nimport { StreamTimeSlicerClass } from '../utils/stream';\nimport useRefFunction from './useRefFunction';\nimport useRefValue from './useRefValue';\nimport useT from './useT';\n\nexport interface UseUserMediaProps {\n /**\n * - **EN:** The media constraints for the audio and video stream.\n * - **CN:** 媒体流的媒体约束。\n */\n media: Pick<MediaStreamConstraints, 'audio' | 'video'>;\n /**\n * - **EN:** Whether to enable PCM output, only effective when recording audio. Please use\n * `onPcmStreamChunk` callback to get PCM data. Default is `false`.\n * - **CN:** 是否启用 PCM 输出,仅在录制音频时有效。请使用 `onPcmStreamChunk` 回调获取 PCM 数据。默认`false`\n */\n pcmAudioOptions?: {\n /**\n * - **EN:** The audio context options for the PCM output.\n * - **CN:** PCM 输出的音频上下文选项。\n */\n audioContext?: AudioContextOptions;\n /**\n * - **EN:** The worklet options for the PCM output.\n * - **CN:** PCM 输出的工作线程选项。\n */\n workletOptions?: AudioWorkletNodeOptions;\n };\n /**\n * - **EN:** Callback function that is triggered when the recording starts, providing the media\n * stream.\n * - **CN:** 开始录制时触发的回调函数,提供媒体流。\n *\n * @param {MediaStream} stream - The media stream.\n */\n onStartRecording?: (stream: MediaStream) => void;\n /**\n * - **EN:** Callback function that is triggered when the recording stops.\n * - **CN:** 停止录制时触发的回调函数。\n */\n onStopRecording?: () => void;\n /**\n * - **EN:** Callback function that is triggered when a new chunk of media data is available.\n * - **CN:** 当录制媒体流时,每个时间分片会触发一次 `onStreamChunk` 回调,提供媒体数据块。\n *\n * > 注意音频流编码格式为:audio/webm;codecs=opus。如果希望获取 PCM 数据,请使用 `onPcmData` 回调。\n *\n * @param {Blob} chunk - The media data chunk (MIME: audio/webm;codecs=opus) | 媒体数据块 (MIME:\n * audio/webm;codecs=opus)\n */\n onStreamChunk?: (chunk: Blob) => void;\n /**\n * - **EN:** Callback for raw PCM float data (per render quantum)\n * - **CN:** 获取原始 PCM 浮点数据的回调(每个渲染量子)\n *\n * @param data Monophonic or polyphonic spliced data | 单声道或多声道拼接数据\n * @param sampleRate Sample rate | 采样率\n */\n onPcmStreamChunk?: (channels: Float32Array[], sampleRate: number) => void;\n /**\n * - **EN:** Whether to disable this hook.\n * - **CN:** 是否禁用此工具\n */\n disabled?: boolean;\n /**\n * - **EN:** The slicing time period (milliseconds) for each fragment of the audio and video stream,\n * each time slice will trigger the `onStreamChunk` callback. Default is `500`.\n * - **CN:** 媒体流每个分片的切片时间段(毫秒),每个时间分片会触发一次 `onStreamChunk` 回调,默认值为 `500`。\n */\n streamSliceMs?: number;\n /**\n * - **EN:** The silence detection threshold (0-1) for the audio stream, below which the audio is\n * considered silent. Default is `0`.\n * - **CN:** 音频流的静音检测阈值(0-1),低于该值音频被视为静音。默认值为 `0`。\n */\n soundDetectionThreshold?: number;\n /**\n * - **EN:** The timeout duration (milliseconds) for detecting sound input. If no sound is detected\n * within this period, the user will be prompted to re-select the audio device. Default is\n * `3000`.\n * - **CN:** 检测是否有声音输入的超时时间(毫秒),如果在该时间段内没有检测到声音,则会提示用户重新选择音频设备。默认值为 `3000`。\n */\n soundDetectionTimeout?: number;\n}\n\nconst useUserMedia = (props: UseUserMediaProps): UseUserMediaResult => {\n const {\n media,\n pcmAudioOptions,\n disabled,\n streamSliceMs = 500,\n soundDetectionThreshold = 0,\n soundDetectionTimeout = 3000,\n onStartRecording,\n onStopRecording,\n onStreamChunk,\n onPcmStreamChunk,\n } = props;\n const context = useContext(ReactEasyContext);\n const { lang } = context;\n const contextRef = useRefValue(context);\n const t = useT();\n const app = App.useApp();\n // @ts-expect-error: because app may return a stub object when App is not used\n const modal = app.modal?.confirm ? app.modal : Modal;\n const modalRef = useRefValue(modal);\n const [requestMicrophoneEnUrl, setRequestMicrophoneEnUrl] = useState<string>();\n const [requestMicrophoneZhUrl, setRequestMicrophoneZhUrl] = useState<string>();\n const [resetMicrophoneEnUrl, setResetMicrophoneEnUrl] = useState<string>();\n const [resetMicrophoneZhUrl, setResetMicrophoneZhUrl] = useState<string>();\n const [requestCameraEnUrl, setRequestCameraEnUrl] = useState<string>();\n const [requestCameraZhUrl, setRequestCameraZhUrl] = useState<string>();\n const [resetCameraEnUrl, setResetCameraEnUrl] = useState<string>();\n const [resetCameraZhUrl, setResetCameraZhUrl] = useState<string>();\n const [isRecording, setIsRecording] = useState(false);\n const [mediaStream, setMediaStream] = useState<MediaStream>();\n const [mediaRecorder, setMediaRecorder] = useState<MediaRecorder | null>(null);\n const stopSoundListeningRef = useRef<(() => void) | undefined>(undefined);\n const closePcmRef = useRef<(() => void) | undefined>(undefined);\n const includeAudio = !!media.audio;\n const exactAudioDeviceIdRef = useRef<string | undefined>(undefined);\n const rafRef = useRef<number>(0);\n const isSpeakingRef = useRef(false);\n const silenceVolumeThresholdRef = useRefValue(soundDetectionThreshold);\n const silenceDetectDurationRef = useRefValue(soundDetectionTimeout);\n const soundDetectStart = useRef<number>(0);\n const pcmSampleRateRef = useRef<number>(0);\n const onPcmStreamChunkRef = useRefValue(onPcmStreamChunk);\n const pcmStreamSlicerRef = useRef(\n new StreamTimeSlicerClass({\n timeSlice: streamSliceMs,\n onSlice: (channels) => {\n onPcmStreamChunkRef.current?.(channels, pcmSampleRateRef.current);\n },\n })\n );\n const deviceType = useMemo(\n () => (media.video ? t('hooks.useUserMedia.camera') : t('hooks.useUserMedia.microphone')),\n [media, t]\n );\n const featureName = useMemo(\n () => (media.video ? t('hooks.featureName.camera') : t('hooks.featureName.microphone')),\n [media, t]\n );\n\n const showDeniedPopup = () => {\n const resetMicrophoneUrl = lang === 'zh-CN' ? resetMicrophoneZhUrl : resetMicrophoneEnUrl;\n const resetCameraUrl = lang === 'zh-CN' ? resetCameraZhUrl : resetCameraEnUrl;\n modal.error({\n title: t('hooks.useUserMedia.devicePermission', { deviceType }),\n width: 500,\n content: (\n <div>\n <Typography.Paragraph></Typography.Paragraph>\n <Typography.Paragraph>\n <Typography.Text strong type=\"danger\">\n {t('hooks.useUserMedia.deniedPermission', { deviceType, featureName })}\n </Typography.Text>\n </Typography.Paragraph>\n <Typography.Paragraph>{t('hooks.useUserMedia.reopenPermissionGuide', { deviceType })}</Typography.Paragraph>\n <img\n src={media.video ? resetCameraUrl : resetMicrophoneUrl}\n alt=\"microphone-permission\"\n style={{ width: 380 }}\n />\n </div>\n ),\n });\n };\n const recordStream = async () => {\n let stream: MediaStream;\n try {\n const options = media;\n if (media.audio) {\n if (exactAudioDeviceIdRef.current) {\n if (media.audio === true) {\n options.audio = { deviceId: { exact: exactAudioDeviceIdRef.current } };\n } else {\n options.audio = { deviceId: { exact: exactAudioDeviceIdRef.current }, ...media.audio };\n }\n }\n }\n stream = await navigator.mediaDevices.getUserMedia(options);\n setMediaStream(stream);\n onStartRecording?.(stream);\n\n const recorder = new MediaRecorder(stream);\n recorder.ondataavailable = (event) => {\n if (event.data.size > 0) {\n onStreamChunk?.(event.data);\n }\n };\n if (streamSliceMs) {\n recorder.start(streamSliceMs);\n } else {\n recorder.start();\n }\n setMediaRecorder(recorder);\n setIsRecording(true);\n\n // Capture PCM data if enabled\n if (options.audio && onPcmStreamChunkRef.current) {\n try {\n const ctx = new AudioContext(pcmAudioOptions?.audioContext);\n pcmSampleRateRef.current = ctx.sampleRate;\n // Safari might need resume\n if (ctx.state === 'suspended') {\n await ctx.resume();\n }\n const sourceNode = ctx.createMediaStreamSource(stream);\n let node: AudioWorkletNode | undefined;\n\n closePcmRef.current = () => {\n node?.port.close();\n node?.disconnect();\n sourceNode.disconnect();\n ctx.close().catch(() => {\n // Ignore errors\n });\n };\n\n const setupWorklet = async () => {\n try {\n // Load the worklet module\n await ctx.audioWorklet.addModule(generatePcmCaptureProcessorModule());\n node = new AudioWorkletNode(ctx, 'pcm-capture', pcmAudioOptions?.workletOptions);\n node.port.onmessage = (e: MessageEvent) => {\n if (e.data?.type === 'pcm') {\n const channels = e.data.channels as Float32Array[];\n pcmStreamSlicerRef.current.push(channels);\n }\n };\n sourceNode.connect(node);\n } catch (err) {\n fallbackScriptProcessor({ ctx, sourceNode, streamSlicer: pcmStreamSlicerRef.current });\n }\n };\n\n if ('audioWorklet' in ctx) {\n setupWorklet();\n } else {\n fallbackScriptProcessor({ ctx, sourceNode, streamSlicer: pcmStreamSlicerRef.current });\n }\n } catch (e) {\n console.error('setup pcm worklet failed', e);\n }\n }\n return recorder;\n } catch (error) {\n console.error(error);\n if (error instanceof Error && error.name === 'NotAllowedError') {\n showDeniedPopup();\n notification.error({ message: t('hooks.useUserMedia.deniedPermission', { deviceType, featureName }) });\n throw new Error(t('hooks.useUserMedia.deniedPermission', { deviceType, featureName }));\n }\n notification.error({ message: t('hooks.useUserMedia.notSupport') });\n throw new Error(t('hooks.useUserMedia.notSupport'));\n }\n };\n\n const startRecording = useRefFunction(async () => {\n if (disabled) {\n throw new Error(t('hooks.useUserMedia.disabledWarning'));\n }\n if (isRecording) {\n throw new Error(t('hooks.useUserMedia.isRecordingNow'));\n }\n if (!navigator.mediaDevices?.getUserMedia || !navigator.permissions?.query) {\n notification.error({\n message: t('hooks.useUserMedia.notSupport'),\n });\n throw new Error(t('hooks.useUserMedia.notSupport'));\n }\n try {\n const result = await window.navigator.permissions.query({\n // eslint-disable-next-line @typescript-eslint/no-explicit-any\n name: media.video ? ('camera' as any) : ('microphone' as any),\n });\n if (result.state === 'denied') {\n showDeniedPopup();\n notification.error({ message: t('hooks.useUserMedia.deniedPermission', { deviceType, featureName }) });\n throw new Error(t('hooks.useUserMedia.deniedPermission', { deviceType, featureName }));\n } else {\n if (result.state === 'prompt') {\n const requestMicrophoneUrl = lang === 'zh-CN' ? requestMicrophoneZhUrl : requestMicrophoneEnUrl;\n const requestCameraUrl = lang === 'zh-CN' ? requestCameraZhUrl : requestCameraEnUrl;\n return new Promise<MediaRecorder>((resolve, reject) => {\n modal.warning({\n title: t('hooks.useUserMedia.devicePermission', { deviceType }),\n content: (\n <div>\n <Typography.Paragraph></Typography.Paragraph>\n <Typography.Paragraph>\n <Typography.Text strong>{t('hooks.useUserMedia.requestTip1', { deviceType })}</Typography.Text>\n </Typography.Paragraph>\n <Typography.Paragraph>{t('hooks.useUserMedia.requestTip2', { featureName })}</Typography.Paragraph>\n <div>\n <img\n src={media.video ? requestCameraUrl : requestMicrophoneUrl}\n alt=\"microphone-permission\"\n style={{ width: 380 }}\n />\n </div>\n </div>\n ),\n onOk: () => {\n try {\n recordStream()\n .then((recorder) => {\n resolve(recorder);\n })\n .catch((error) => {\n reject(error);\n });\n } catch (error) {\n console.error(error);\n reject(error);\n }\n },\n width: 500,\n });\n });\n } else {\n return await recordStream();\n }\n }\n } catch (error) {\n console.error(error);\n throw error;\n }\n });\n\n const stopRecording = useRefFunction(() => {\n mediaRecorder?.stop();\n mediaStream?.getTracks().forEach((t) => t.stop());\n setMediaStream(undefined);\n setIsRecording(false);\n isSpeakingRef.current = false;\n cancelAnimationFrame(rafRef.current);\n stopSoundListeningRef.current?.();\n closePcmRef.current?.();\n onStopRecording?.();\n });\n\n // Wait for sound. If keep silent for a while, show a modal to let user reselect audio device\n const waitForSound = useRefFunction((mediaStream: MediaStream) => {\n const ctx = new AudioContext();\n const source = ctx.createMediaStreamSource(mediaStream);\n const analyser = ctx.createAnalyser();\n analyser.fftSize = 2048;\n source.connect(analyser);\n soundDetectStart.current = Date.now();\n const data = new Uint8Array(analyser.fftSize);\n const cancelDetect = () => {\n cancelAnimationFrame(rafRef.current);\n rafRef.current = 0;\n stopSoundListeningRef.current?.();\n };\n const loop = () => {\n analyser.getByteTimeDomainData(data);\n // Calculate RMS\n let sum = 0;\n for (const value of data) {\n const v = (value - 128) / 128;\n sum += v * v;\n }\n const rms = Math.sqrt(sum / data.length); // 0~1\n if (rms > silenceVolumeThresholdRef.current) {\n if (!isSpeakingRef.current) {\n isSpeakingRef.current = true;\n cancelDetect();\n return;\n }\n } else {\n if (Date.now() > soundDetectStart.current + silenceDetectDurationRef.current) {\n navigator.mediaDevices.enumerateDevices().then((devices) => {\n const audioInputs = devices.filter((d) => d.kind === 'audioinput');\n modalRef.current.confirm({\n title: t('hooks.useUserMedia.soundDetectTitle'),\n content: (\n <EasyConfigProvider {...contextRef.current}>\n <SaveAudioDeviceForm\n devices={audioInputs}\n mediaStream={mediaStream}\n onDeviceChange={(deviceId) => (exactAudioDeviceIdRef.current = deviceId)}\n />\n </EasyConfigProvider>\n ),\n width: 500,\n onOk: () => {\n if (exactAudioDeviceIdRef.current) {\n stopRecording();\n setTimeout(() => {\n startRecording();\n });\n }\n },\n onCancel: () => {\n cancelDetect();\n },\n });\n });\n\n cancelDetect();\n return;\n }\n }\n rafRef.current = requestAnimationFrame(loop);\n };\n loop();\n\n stopSoundListeningRef.current = () => {\n analyser.disconnect();\n source.disconnect();\n ctx.close().catch(() => {\n // Ignore errors\n });\n stopSoundListeningRef.current = undefined;\n };\n });\n\n // Component destroy\n useEffect(() => {\n return stopRecording;\n }, []);\n\n // Stop recording when disabled has been changed\n useEffect(() => {\n if (disabled && isRecording) {\n stopRecording();\n }\n }, [disabled, isRecording]);\n\n // Update PCM stream slicer time slice when input sample rate changes\n useEffect(() => {\n if (streamSliceMs && pcmStreamSlicerRef.current.timeSlice !== streamSliceMs) {\n pcmStreamSlicerRef.current.timeSlice = streamSliceMs;\n }\n }, [streamSliceMs]);\n\n // Detect sound activity (only for audio or media with audio)\n useEffect(() => {\n if (includeAudio && mediaStream && !isSpeakingRef.current) {\n try {\n waitForSound(mediaStream);\n } catch (e) {\n console.warn('Audio volume detecting failed:', e);\n }\n }\n return () => {\n stopSoundListeningRef.current?.();\n };\n }, [includeAudio, mediaStream, t]);\n\n useEffect(() => {\n import('../assets/request-microphone-en.js').then((module) => {\n setRequestMicrophoneEnUrl(module.default);\n });\n import('../assets/request-microphone-zh.js').then((module) => {\n setRequestMicrophoneZhUrl(module.default);\n });\n import('../assets/reset-microphone-en.js').then((module) => {\n setResetMicrophoneEnUrl(module.default);\n });\n import('../assets/reset-microphone-zh.js').then((module) => {\n setResetMicrophoneZhUrl(module.default);\n });\n import('../assets/request-camera-en.js').then((module) => {\n setRequestCameraEnUrl(module.default);\n });\n import('../assets/request-camera-zh.js').then((module) => {\n setRequestCameraZhUrl(module.default);\n });\n import('../assets/reset-camera-en.js').then((module) => {\n setResetCameraEnUrl(module.default);\n });\n import('../assets/reset-camera-zh.js').then((module) => {\n setResetCameraZhUrl(module.default);\n });\n }, []);\n\n return {\n isRecording,\n startRecording,\n stopRecording,\n };\n};\n\nexport interface UseUserMediaResult {\n /**\n * - **EN** Whether the media stream is currently being recorded\n * - **CN** 是否正在录制媒体流\n */\n isRecording: boolean;\n /**\n * - **EN** Start recording the media stream\n * - **CN** 开始录制媒体流\n */\n startRecording: () => Promise<MediaRecorder>;\n /**\n * - **EN** Stop recording the media stream\n * - **CN** 停止录制媒体流\n */\n stopRecording: () => void;\n /**\n * - **EN** Get the media stream being recorded, returns the stream if recording, otherwise returns\n * `undefined`\n * - **CN** 获取正在录制的媒体流,如果正在录制则返回该流,否则返回 `undefined`\n */\n mediaStream?: MediaStream;\n}\n\nfunction SaveAudioDeviceForm(props: {\n devices: MediaDeviceInfo[];\n mediaStream: MediaStream;\n onDeviceChange: (deviceId: string) => void;\n}) {\n const { devices, mediaStream, onDeviceChange } = props;\n const [form] = Form.useForm();\n const t = useT();\n const { lang } = useContext(ReactEasyContext);\n const [saveAudioDeviceEnUrl1, setSaveAudioDeviceEnUrl1] = useState<string>();\n const [saveAudioDeviceEnUrl2, setSaveAudioDeviceEnUrl2] = useState<string>();\n const [saveAudioDeviceZhUrl1, setSaveAudioDeviceZhUrl1] = useState<string>();\n const [saveAudioDeviceZhUrl2, setSaveAudioDeviceZhUrl2] = useState<string>();\n const [saveDefaultAudioDevicePermanently, setSaveDefaultAudioDevicePermanently] = useState(false);\n const audioInputs = useMemo(() => devices.filter((d) => d.kind === 'audioinput'), [devices]);\n const [selectedDeviceId, setSelectedDeviceId] = useState<string | undefined>(\n () => mediaStream.getAudioTracks()[0]?.getSettings()?.deviceId ?? audioInputs[0]?.deviceId\n );\n\n const openDataImageInNewTab = useRefFunction((dataUrl: string | undefined) => {\n if (!dataUrl) return;\n const [meta, b64] = dataUrl.split(',');\n const mime = meta.match(/data:(.*);base64/)?.[1] || 'image/png';\n const binary = atob(b64);\n const len = binary.length;\n const bytes = new Uint8Array(len);\n for (let i = 0; i < len; i++) bytes[i] = binary.charCodeAt(i);\n const blob = new Blob([bytes], { type: mime });\n const url = URL.createObjectURL(blob);\n window.open(url);\n });\n\n useEffect(() => {\n import('../assets/save-default-audio1-en.js').then((module) => {\n setSaveAudioDeviceEnUrl1(module.default);\n });\n import('../assets/save-default-audio2-en.js').then((module) => {\n setSaveAudioDeviceEnUrl2(module.default);\n });\n import('../assets/save-default-audio1-zh.js').then((module) => {\n setSaveAudioDeviceZhUrl1(module.default);\n });\n import('../assets/save-default-audio2-zh.js').then((module) => {\n setSaveAudioDeviceZhUrl2(module.default);\n });\n }, []);\n\n return (\n <Form layout=\"vertical\" form={form}>\n <Typography.Paragraph></Typography.Paragraph>\n <Typography.Paragraph>\n <Typography.Text>{t('hooks.useUserMedia.soundDetectDescription')}</Typography.Text>\n </Typography.Paragraph>\n <Form.Item label={t('hooks.useUserMedia.chooseMicrophoneDevice')}>\n <Select\n options={audioInputs.map((input) => ({\n label: input.label,\n value: input.deviceId,\n }))}\n defaultValue={selectedDeviceId}\n onChange={(id) => {\n setSelectedDeviceId(id);\n onDeviceChange(id);\n }}\n />\n </Form.Item>\n <Form.Item style={{ marginBottom: 0 }}>\n <Checkbox onChange={(e) => setSaveDefaultAudioDevicePermanently(e.target.checked)}>\n {t('hooks.useUserMedia.rememberDefaultAudioDevice')}\n </Checkbox>\n </Form.Item>\n {saveDefaultAudioDevicePermanently && (\n <div>\n <Typography.Paragraph>\n <Typography.Text>{t('hooks.useUserMedia.rememberDefaultAudioDeviceTip')}</Typography.Text>\n </Typography.Paragraph>\n <Flex gap={8} align=\"flex-start\">\n <div style={{ flex: 1, minWidth: 0 }}>\n <a\n href=\"#\"\n onClick={(e) => {\n e.preventDefault();\n openDataImageInNewTab(lang === 'zh-CN' ? saveAudioDeviceZhUrl1 : saveAudioDeviceEnUrl1);\n }}\n >\n <img\n src={lang === 'zh-CN' ? saveAudioDeviceZhUrl1 : saveAudioDeviceEnUrl1}\n alt=\"the first step to save default audio device\"\n style={{ width: '100%', height: 'auto' }}\n />\n </a>\n </div>\n <div style={{ flex: 1, minWidth: 0 }}>\n <a\n href=\"#\"\n onClick={(e) => {\n e.preventDefault();\n openDataImageInNewTab(lang === 'zh-CN' ? saveAudioDeviceZhUrl2 : saveAudioDeviceEnUrl2);\n }}\n >\n <img\n src={lang === 'zh-CN' ? saveAudioDeviceZhUrl2 : saveAudioDeviceEnUrl2}\n alt=\"the second step to save default audio device\"\n style={{ width: '100%', height: 'auto' }}\n />\n </a>\n </div>\n </Flex>\n </div>\n )}\n </Form>\n );\n}\n\nfunction generatePcmCaptureProcessorModule() {\n const workletCode = `\nclass PcmCaptureProcessor extends AudioWorkletProcessor {\n process(inputs, outputs, parameters) {\n const channelsIn = inputs[0];\n if (channelsIn && channelsIn[0]) {\n const channels = channelsIn.map((ch) => {\n const copy = new Float32Array(ch.length);\n copy.set(ch);\n return copy;\n });\n this.port.postMessage({ type: 'pcm', channels }, channels.map(ch => ch.buffer));\n }\n return true;\n }\n}\nregisterProcessor('pcm-capture', PcmCaptureProcessor);\n`;\n\n // Create a Blob from the worklet code and return its URL\n const blob = new Blob([workletCode], { type: 'application/javascript' });\n const blobUrl = URL.createObjectURL(blob);\n return blobUrl;\n}\n\nfunction fallbackScriptProcessor(options: {\n ctx: AudioContext;\n sourceNode: MediaStreamAudioSourceNode;\n streamSlicer: StreamTimeSlicerClass;\n}) {\n const { ctx, sourceNode, streamSlicer } = options;\n const bufferSize = 128;\n const processor = ctx.createScriptProcessor(bufferSize, 1, 1);\n processor.onaudioprocess = (ev) => {\n const channels = [];\n for (let i = 0; i < ev.inputBuffer.numberOfChannels; i++) {\n const input = ev.inputBuffer.getChannelData(i);\n const copy = new Float32Array(input.length);\n copy.set(input);\n channels.push(copy);\n }\n streamSlicer.push(channels);\n };\n sourceNode.connect(processor);\n // 可不输出:processor.connect(ctx!.destination);不连接 destination 在部分浏览器会被自动 GC,可连到 destination 或 gain(0)\n const gain = ctx.createGain();\n gain.gain.value = 0;\n processor.connect(gain).connect(ctx.destination);\n}\n\nexport default useUserMedia;\n"],
|
|
5
|
+
"mappings": "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",
|
|
6
6
|
"names": ["ReactEasyContext", "useRefValue", "useT", "_a", "useRefFunction", "t", "mediaStream", "EasyConfigProvider", "module"]
|
|
7
7
|
}
|
package/lib/locales/index.d.ts
CHANGED
|
@@ -45,6 +45,8 @@ export declare const resources: {
|
|
|
45
45
|
readonly 'hooks.useUserMedia.chooseMicrophoneDevice': "Please reselect the audio device";
|
|
46
46
|
readonly 'hooks.useUserMedia.rememberDefaultAudioDevice': "Remember default device?";
|
|
47
47
|
readonly 'hooks.useUserMedia.rememberDefaultAudioDeviceTip': "Please follow these steps to save the default audio device for automatic selection next time.";
|
|
48
|
+
readonly 'hooks.useStompSocket.connectError': "Connection failed, please try again later";
|
|
49
|
+
readonly 'hooks.useStompSocket.serverDisconnected': "Server disconnected";
|
|
48
50
|
};
|
|
49
51
|
};
|
|
50
52
|
readonly 'en-US': {
|
|
@@ -90,6 +92,8 @@ export declare const resources: {
|
|
|
90
92
|
readonly 'hooks.useUserMedia.chooseMicrophoneDevice': "Please reselect the audio device";
|
|
91
93
|
readonly 'hooks.useUserMedia.rememberDefaultAudioDevice': "Remember default device?";
|
|
92
94
|
readonly 'hooks.useUserMedia.rememberDefaultAudioDeviceTip': "Please follow these steps to save the default audio device for automatic selection next time.";
|
|
95
|
+
readonly 'hooks.useStompSocket.connectError': "Connection failed, please try again later";
|
|
96
|
+
readonly 'hooks.useStompSocket.serverDisconnected': "Server disconnected";
|
|
93
97
|
};
|
|
94
98
|
};
|
|
95
99
|
readonly 'zh-CN': {
|
|
@@ -135,6 +139,8 @@ export declare const resources: {
|
|
|
135
139
|
readonly 'hooks.useUserMedia.chooseMicrophoneDevice': "请重新选择音频设备";
|
|
136
140
|
readonly 'hooks.useUserMedia.rememberDefaultAudioDevice': "记住默认设备?";
|
|
137
141
|
readonly 'hooks.useUserMedia.rememberDefaultAudioDeviceTip': "请按照以下步骤保存默认音频设备,以便下次使用时自动选择。";
|
|
142
|
+
readonly 'hooks.useStompSocket.connectError': "连接失败,请稍后重试";
|
|
143
|
+
readonly 'hooks.useStompSocket.serverDisconnected': "服务端已断开连接";
|
|
138
144
|
};
|
|
139
145
|
};
|
|
140
146
|
};
|
|
@@ -40,5 +40,7 @@ declare const enUS: {
|
|
|
40
40
|
readonly 'hooks.useUserMedia.chooseMicrophoneDevice': "Please reselect the audio device";
|
|
41
41
|
readonly 'hooks.useUserMedia.rememberDefaultAudioDevice': "Remember default device?";
|
|
42
42
|
readonly 'hooks.useUserMedia.rememberDefaultAudioDeviceTip': "Please follow these steps to save the default audio device for automatic selection next time.";
|
|
43
|
+
readonly 'hooks.useStompSocket.connectError': "Connection failed, please try again later";
|
|
44
|
+
readonly 'hooks.useStompSocket.serverDisconnected': "Server disconnected";
|
|
43
45
|
};
|
|
44
46
|
export default enUS;
|
|
@@ -63,7 +63,9 @@ var enUS = {
|
|
|
63
63
|
"hooks.useUserMedia.soundDetectDescription": "No sound seems to be detected. Please check if the microphone is working properly or if the correct audio input device is selected.",
|
|
64
64
|
"hooks.useUserMedia.chooseMicrophoneDevice": "Please reselect the audio device",
|
|
65
65
|
"hooks.useUserMedia.rememberDefaultAudioDevice": "Remember default device?",
|
|
66
|
-
"hooks.useUserMedia.rememberDefaultAudioDeviceTip": "Please follow these steps to save the default audio device for automatic selection next time."
|
|
66
|
+
"hooks.useUserMedia.rememberDefaultAudioDeviceTip": "Please follow these steps to save the default audio device for automatic selection next time.",
|
|
67
|
+
"hooks.useStompSocket.connectError": "Connection failed, please try again later",
|
|
68
|
+
"hooks.useStompSocket.serverDisconnected": "Server disconnected"
|
|
67
69
|
};
|
|
68
70
|
var en_US_default = enUS;
|
|
69
71
|
//# sourceMappingURL=en-US.js.map
|
|
@@ -1,7 +1,7 @@
|
|
|
1
1
|
{
|
|
2
2
|
"version": 3,
|
|
3
3
|
"sources": ["../../../src/locales/langs/en-US.ts"],
|
|
4
|
-
"sourcesContent": ["const enUS = {\n 'components.EditableText.requiredMsg': 'Please enter text content',\n 'components.EditableText.placeholder': 'Please input text content, press Enter or click the OK button',\n 'components.EditableText.edit': 'Edit',\n 'components.EditableText.save': 'Save',\n 'components.EditableText.cancel': 'Cancel',\n 'validation.rule.number.message': 'Please enter a number',\n 'validation.rule.floatNumber.message': 'Please enter a number',\n 'validation.rule.email.message': 'Please enter the correct email address',\n 'validation.rule.ip.message': 'Please enter the correct ip address',\n 'validation.rule.chineseMobile.message': 'Please enter the correct mobile number',\n 'validation.rule.password.message': '8-16 digit password, at least two of numbers, letters, and symbols',\n 'validation.rule.buildRule.includeMandatory': 'At least one field in the include option is true',\n 'validation.rule.buildRule.messageFormat': 'Allow {{content}}',\n 'validation.rule.buildRule.startsWithFormat': `, start with {{prefix}}`,\n 'validation.rule.buildRule.minLength': `, at least {{min}} characters`,\n 'validation.rule.buildRule.maxLength': `, up to {{max}} characters`,\n 'validation.rule.buildRule.rangeLength': `, {{min}}~{{max}} characters`,\n 'validation.rule.buildRule.token.separator': ', ',\n 'validation.rule.buildRule.token.chinese': 'Chinese characters',\n 'validation.rule.buildRule.token.chinesePunctuation': 'Chinese punctuation',\n 'validation.rule.buildRule.token.letter': 'letter',\n 'validation.rule.buildRule.token.lowerLetter': 'lowercase letter',\n 'validation.rule.buildRule.token.upperLetter': 'uppercase letter',\n 'validation.rule.buildRule.token.number': 'number',\n 'hooks.useUserMedia.microphone': 'Microphone',\n 'hooks.useUserMedia.camera': 'Camera',\n 'hooks.featureName.microphone': 'Speech',\n 'hooks.featureName.camera': 'Video',\n 'hooks.useUserMedia.disabledWarning': 'User media is disabled',\n 'hooks.useUserMedia.isRecordingNow': 'User media is currently recording and cannot be operated on',\n 'hooks.useUserMedia.devicePermission': '{{deviceType}} Permission',\n 'hooks.useUserMedia.requestTip1':\n 'The browser is requesting {{deviceType}} permission. Please click the \"Allow\" button in the pop-up window that appears later.',\n 'hooks.useUserMedia.requestTip2':\n 'Please do not click the \"Deny\" button, otherwise you will not be able to use the {{featureName}} related feature.',\n 'hooks.useUserMedia.deniedPermission':\n 'You have denied the {{deviceType}} permission, which will prevent you from using the {{featureName}} related feature!',\n 'hooks.useUserMedia.reopenPermissionGuide':\n 'Please click the site settings icon in the address bar, and then follow these steps to reopen {{deviceType}} permission.',\n 'hooks.useUserMedia.notSupport': 'Current browser does not support audio recording',\n 'hooks.useUserMedia.soundDetectTitle': 'Sound Issue Detected',\n 'hooks.useUserMedia.soundDetectDescription':\n 'No sound seems to be detected. Please check if the microphone is working properly or if the correct audio input device is selected.',\n 'hooks.useUserMedia.chooseMicrophoneDevice': 'Please reselect the audio device',\n 'hooks.useUserMedia.rememberDefaultAudioDevice': 'Remember default device?',\n 'hooks.useUserMedia.rememberDefaultAudioDeviceTip':\n 'Please follow these steps to save the default audio device for automatic selection next time.',\n} as const;\nexport default enUS;\n"],
|
|
5
|
-
"mappings": ";;;;;;;;;;;;;;;;;;;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA,IAAM,OAAO;AAAA,EACX,uCAAuC;AAAA,EACvC,uCAAuC;AAAA,EACvC,gCAAgC;AAAA,EAChC,gCAAgC;AAAA,EAChC,kCAAkC;AAAA,EAClC,kCAAkC;AAAA,EAClC,uCAAuC;AAAA,EACvC,iCAAiC;AAAA,EACjC,8BAA8B;AAAA,EAC9B,yCAAyC;AAAA,EACzC,oCAAoC;AAAA,EACpC,8CAA8C;AAAA,EAC9C,2CAA2C;AAAA,EAC3C,8CAA8C;AAAA,EAC9C,uCAAuC;AAAA,EACvC,uCAAuC;AAAA,EACvC,yCAAyC;AAAA,EACzC,6CAA6C;AAAA,EAC7C,2CAA2C;AAAA,EAC3C,sDAAsD;AAAA,EACtD,0CAA0C;AAAA,EAC1C,+CAA+C;AAAA,EAC/C,+CAA+C;AAAA,EAC/C,0CAA0C;AAAA,EAC1C,iCAAiC;AAAA,EACjC,6BAA6B;AAAA,EAC7B,gCAAgC;AAAA,EAChC,4BAA4B;AAAA,EAC5B,sCAAsC;AAAA,EACtC,qCAAqC;AAAA,EACrC,uCAAuC;AAAA,EACvC,kCACE;AAAA,EACF,kCACE;AAAA,EACF,uCACE;AAAA,EACF,4CACE;AAAA,EACF,iCAAiC;AAAA,EACjC,uCAAuC;AAAA,EACvC,6CACE;AAAA,EACF,6CAA6C;AAAA,EAC7C,iDAAiD;AAAA,EACjD,oDACE;
|
|
4
|
+
"sourcesContent": ["const enUS = {\n 'components.EditableText.requiredMsg': 'Please enter text content',\n 'components.EditableText.placeholder': 'Please input text content, press Enter or click the OK button',\n 'components.EditableText.edit': 'Edit',\n 'components.EditableText.save': 'Save',\n 'components.EditableText.cancel': 'Cancel',\n 'validation.rule.number.message': 'Please enter a number',\n 'validation.rule.floatNumber.message': 'Please enter a number',\n 'validation.rule.email.message': 'Please enter the correct email address',\n 'validation.rule.ip.message': 'Please enter the correct ip address',\n 'validation.rule.chineseMobile.message': 'Please enter the correct mobile number',\n 'validation.rule.password.message': '8-16 digit password, at least two of numbers, letters, and symbols',\n 'validation.rule.buildRule.includeMandatory': 'At least one field in the include option is true',\n 'validation.rule.buildRule.messageFormat': 'Allow {{content}}',\n 'validation.rule.buildRule.startsWithFormat': `, start with {{prefix}}`,\n 'validation.rule.buildRule.minLength': `, at least {{min}} characters`,\n 'validation.rule.buildRule.maxLength': `, up to {{max}} characters`,\n 'validation.rule.buildRule.rangeLength': `, {{min}}~{{max}} characters`,\n 'validation.rule.buildRule.token.separator': ', ',\n 'validation.rule.buildRule.token.chinese': 'Chinese characters',\n 'validation.rule.buildRule.token.chinesePunctuation': 'Chinese punctuation',\n 'validation.rule.buildRule.token.letter': 'letter',\n 'validation.rule.buildRule.token.lowerLetter': 'lowercase letter',\n 'validation.rule.buildRule.token.upperLetter': 'uppercase letter',\n 'validation.rule.buildRule.token.number': 'number',\n 'hooks.useUserMedia.microphone': 'Microphone',\n 'hooks.useUserMedia.camera': 'Camera',\n 'hooks.featureName.microphone': 'Speech',\n 'hooks.featureName.camera': 'Video',\n 'hooks.useUserMedia.disabledWarning': 'User media is disabled',\n 'hooks.useUserMedia.isRecordingNow': 'User media is currently recording and cannot be operated on',\n 'hooks.useUserMedia.devicePermission': '{{deviceType}} Permission',\n 'hooks.useUserMedia.requestTip1':\n 'The browser is requesting {{deviceType}} permission. Please click the \"Allow\" button in the pop-up window that appears later.',\n 'hooks.useUserMedia.requestTip2':\n 'Please do not click the \"Deny\" button, otherwise you will not be able to use the {{featureName}} related feature.',\n 'hooks.useUserMedia.deniedPermission':\n 'You have denied the {{deviceType}} permission, which will prevent you from using the {{featureName}} related feature!',\n 'hooks.useUserMedia.reopenPermissionGuide':\n 'Please click the site settings icon in the address bar, and then follow these steps to reopen {{deviceType}} permission.',\n 'hooks.useUserMedia.notSupport': 'Current browser does not support audio recording',\n 'hooks.useUserMedia.soundDetectTitle': 'Sound Issue Detected',\n 'hooks.useUserMedia.soundDetectDescription':\n 'No sound seems to be detected. Please check if the microphone is working properly or if the correct audio input device is selected.',\n 'hooks.useUserMedia.chooseMicrophoneDevice': 'Please reselect the audio device',\n 'hooks.useUserMedia.rememberDefaultAudioDevice': 'Remember default device?',\n 'hooks.useUserMedia.rememberDefaultAudioDeviceTip':\n 'Please follow these steps to save the default audio device for automatic selection next time.',\n 'hooks.useStompSocket.connectError': 'Connection failed, please try again later',\n 'hooks.useStompSocket.serverDisconnected': 'Server disconnected',\n} as const;\nexport default enUS;\n"],
|
|
5
|
+
"mappings": ";;;;;;;;;;;;;;;;;;;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA,IAAM,OAAO;AAAA,EACX,uCAAuC;AAAA,EACvC,uCAAuC;AAAA,EACvC,gCAAgC;AAAA,EAChC,gCAAgC;AAAA,EAChC,kCAAkC;AAAA,EAClC,kCAAkC;AAAA,EAClC,uCAAuC;AAAA,EACvC,iCAAiC;AAAA,EACjC,8BAA8B;AAAA,EAC9B,yCAAyC;AAAA,EACzC,oCAAoC;AAAA,EACpC,8CAA8C;AAAA,EAC9C,2CAA2C;AAAA,EAC3C,8CAA8C;AAAA,EAC9C,uCAAuC;AAAA,EACvC,uCAAuC;AAAA,EACvC,yCAAyC;AAAA,EACzC,6CAA6C;AAAA,EAC7C,2CAA2C;AAAA,EAC3C,sDAAsD;AAAA,EACtD,0CAA0C;AAAA,EAC1C,+CAA+C;AAAA,EAC/C,+CAA+C;AAAA,EAC/C,0CAA0C;AAAA,EAC1C,iCAAiC;AAAA,EACjC,6BAA6B;AAAA,EAC7B,gCAAgC;AAAA,EAChC,4BAA4B;AAAA,EAC5B,sCAAsC;AAAA,EACtC,qCAAqC;AAAA,EACrC,uCAAuC;AAAA,EACvC,kCACE;AAAA,EACF,kCACE;AAAA,EACF,uCACE;AAAA,EACF,4CACE;AAAA,EACF,iCAAiC;AAAA,EACjC,uCAAuC;AAAA,EACvC,6CACE;AAAA,EACF,6CAA6C;AAAA,EAC7C,iDAAiD;AAAA,EACjD,oDACE;AAAA,EACF,qCAAqC;AAAA,EACrC,2CAA2C;AAC7C;AACA,IAAO,gBAAQ;",
|
|
6
6
|
"names": []
|
|
7
7
|
}
|
|
@@ -40,5 +40,7 @@ declare const zhCN: {
|
|
|
40
40
|
readonly 'hooks.useUserMedia.chooseMicrophoneDevice': "请重新选择音频设备";
|
|
41
41
|
readonly 'hooks.useUserMedia.rememberDefaultAudioDevice': "记住默认设备?";
|
|
42
42
|
readonly 'hooks.useUserMedia.rememberDefaultAudioDeviceTip': "请按照以下步骤保存默认音频设备,以便下次使用时自动选择。";
|
|
43
|
+
readonly 'hooks.useStompSocket.connectError': "连接失败,请稍后重试";
|
|
44
|
+
readonly 'hooks.useStompSocket.serverDisconnected': "服务端已断开连接";
|
|
43
45
|
};
|
|
44
46
|
export default zhCN;
|
|
@@ -63,7 +63,9 @@ var zhCN = {
|
|
|
63
63
|
"hooks.useUserMedia.soundDetectDescription": "似乎没有检测到声音,请检查麦克风是否正常工作,或者可能没有选择正确的音频输入设备。",
|
|
64
64
|
"hooks.useUserMedia.chooseMicrophoneDevice": "请重新选择音频设备",
|
|
65
65
|
"hooks.useUserMedia.rememberDefaultAudioDevice": "记住默认设备?",
|
|
66
|
-
"hooks.useUserMedia.rememberDefaultAudioDeviceTip": "请按照以下步骤保存默认音频设备,以便下次使用时自动选择。"
|
|
66
|
+
"hooks.useUserMedia.rememberDefaultAudioDeviceTip": "请按照以下步骤保存默认音频设备,以便下次使用时自动选择。",
|
|
67
|
+
"hooks.useStompSocket.connectError": "连接失败,请稍后重试",
|
|
68
|
+
"hooks.useStompSocket.serverDisconnected": "服务端已断开连接"
|
|
67
69
|
};
|
|
68
70
|
var zh_CN_default = zhCN;
|
|
69
71
|
//# sourceMappingURL=zh-CN.js.map
|