sa2kit 3.2.0 → 3.2.2
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/dist/CollisionBalls-DgKtscU2.d.mts +41 -0
- package/dist/CollisionBalls-DgKtscU2.d.ts +41 -0
- package/dist/ConfigService-Oga_zFRS.d.mts +262 -0
- package/dist/ConfigService-Oga_zFRS.d.ts +262 -0
- package/dist/UniversalFileService-CC4d3wkc.d.ts +139 -0
- package/dist/UniversalFileService-CzAE_G4V.d.mts +139 -0
- package/dist/boothVaultService-lKcnyA-u.d.mts +83 -0
- package/dist/boothVaultService-lKcnyA-u.d.ts +83 -0
- package/dist/business/audioDetection/index.d.mts +2 -0
- package/dist/business/audioDetection/index.d.ts +2 -0
- package/dist/business/audioDetection/index.js +1244 -0
- package/dist/business/audioDetection/index.js.map +1 -0
- package/dist/business/audioDetection/index.mjs +1227 -0
- package/dist/business/audioDetection/index.mjs.map +1 -0
- package/dist/business/calendar/index.d.mts +6 -0
- package/dist/business/calendar/index.d.ts +6 -0
- package/dist/business/calendar/index.js +7433 -0
- package/dist/business/calendar/index.js.map +1 -0
- package/dist/business/calendar/index.mjs +7257 -0
- package/dist/business/calendar/index.mjs.map +1 -0
- package/dist/business/calendar/routes/index.d.mts +191 -0
- package/dist/business/calendar/routes/index.d.ts +191 -0
- package/dist/business/calendar/routes/index.js +844 -0
- package/dist/business/calendar/routes/index.js.map +1 -0
- package/dist/business/calendar/routes/index.mjs +826 -0
- package/dist/business/calendar/routes/index.mjs.map +1 -0
- package/dist/business/festivalCard/index.d.mts +4 -0
- package/dist/business/festivalCard/index.d.ts +4 -0
- package/dist/business/festivalCard/index.js +1492 -0
- package/dist/business/festivalCard/index.js.map +1 -0
- package/dist/business/festivalCard/index.mjs +1475 -0
- package/dist/business/festivalCard/index.mjs.map +1 -0
- package/dist/business/festivalCard/routes/index.d.mts +42 -0
- package/dist/business/festivalCard/routes/index.d.ts +42 -0
- package/dist/business/festivalCard/routes/index.js +361 -0
- package/dist/business/festivalCard/routes/index.js.map +1 -0
- package/dist/business/festivalCard/routes/index.mjs +356 -0
- package/dist/business/festivalCard/routes/index.mjs.map +1 -0
- package/dist/business/festivalCard/server/index.d.mts +120 -0
- package/dist/business/festivalCard/server/index.d.ts +120 -0
- package/dist/business/festivalCard/server/index.js +272 -0
- package/dist/business/festivalCard/server/index.js.map +1 -0
- package/dist/business/festivalCard/server/index.mjs +265 -0
- package/dist/business/festivalCard/server/index.mjs.map +1 -0
- package/dist/business/index.d.mts +34 -0
- package/dist/business/index.d.ts +34 -0
- package/dist/business/index.js +29282 -0
- package/dist/business/index.js.map +1 -0
- package/dist/business/index.mjs +29237 -0
- package/dist/business/index.mjs.map +1 -0
- package/dist/business/mikuContest/ui/web/index.d.mts +2 -0
- package/dist/business/mikuContest/ui/web/index.d.ts +2 -0
- package/dist/business/mikuContest/ui/web/index.js +353 -0
- package/dist/business/mikuContest/ui/web/index.js.map +1 -0
- package/dist/business/mikuContest/ui/web/index.mjs +343 -0
- package/dist/business/mikuContest/ui/web/index.mjs.map +1 -0
- package/dist/business/mikuFireworks3D/index.d.mts +2 -0
- package/dist/business/mikuFireworks3D/index.d.ts +2 -0
- package/dist/business/mikuFireworks3D/index.js +1267 -0
- package/dist/business/mikuFireworks3D/index.js.map +1 -0
- package/dist/business/mikuFireworks3D/index.mjs +1228 -0
- package/dist/business/mikuFireworks3D/index.mjs.map +1 -0
- package/dist/business/mikuFusionGame/index.d.mts +2 -0
- package/dist/business/mikuFusionGame/index.d.ts +2 -0
- package/dist/business/mikuFusionGame/index.js +1208 -0
- package/dist/business/mikuFusionGame/index.js.map +1 -0
- package/dist/business/mikuFusionGame/index.mjs +1195 -0
- package/dist/business/mikuFusionGame/index.mjs.map +1 -0
- package/dist/business/mmd/admin/index.d.mts +487 -0
- package/dist/business/mmd/admin/index.d.ts +487 -0
- package/dist/business/mmd/admin/index.js +1058 -0
- package/dist/business/mmd/admin/index.js.map +1 -0
- package/dist/business/mmd/admin/index.mjs +1027 -0
- package/dist/business/mmd/admin/index.mjs.map +1 -0
- package/dist/business/mmd/index.d.mts +5 -0
- package/dist/business/mmd/index.d.ts +5 -0
- package/dist/business/mmd/index.js +10119 -0
- package/dist/business/mmd/index.js.map +1 -0
- package/dist/business/mmd/index.mjs +10028 -0
- package/dist/business/mmd/index.mjs.map +1 -0
- package/dist/business/mmd/server/index.d.mts +139 -0
- package/dist/business/mmd/server/index.d.ts +139 -0
- package/dist/business/mmd/server/index.js +424 -0
- package/dist/business/mmd/server/index.js.map +1 -0
- package/dist/business/mmd/server/index.mjs +404 -0
- package/dist/business/mmd/server/index.mjs.map +1 -0
- package/dist/business/music/index.d.mts +3 -0
- package/dist/business/music/index.d.ts +3 -0
- package/dist/business/music/index.js +830 -0
- package/dist/business/music/index.js.map +1 -0
- package/dist/business/music/index.mjs +809 -0
- package/dist/business/music/index.mjs.map +1 -0
- package/dist/business/music/server/index.d.mts +1 -0
- package/dist/business/music/server/index.d.ts +1 -0
- package/dist/business/music/server/index.js +194 -0
- package/dist/business/music/server/index.js.map +1 -0
- package/dist/business/music/server/index.mjs +182 -0
- package/dist/business/music/server/index.mjs.map +1 -0
- package/dist/business/navigation/index.d.mts +2 -0
- package/dist/business/navigation/index.d.ts +2 -0
- package/dist/business/navigation/index.js +453 -0
- package/dist/business/navigation/index.js.map +1 -0
- package/dist/business/navigation/index.mjs +443 -0
- package/dist/business/navigation/index.mjs.map +1 -0
- package/dist/business/portfolio/index.d.mts +3 -0
- package/dist/business/portfolio/index.d.ts +3 -0
- package/dist/business/portfolio/index.js +736 -0
- package/dist/business/portfolio/index.js.map +1 -0
- package/dist/business/portfolio/index.mjs +724 -0
- package/dist/business/portfolio/index.mjs.map +1 -0
- package/dist/business/qqbot/server/index.d.mts +167 -0
- package/dist/business/qqbot/server/index.d.ts +167 -0
- package/dist/business/qqbot/server/index.js +394 -0
- package/dist/business/qqbot/server/index.js.map +1 -0
- package/dist/business/qqbot/server/index.mjs +385 -0
- package/dist/business/qqbot/server/index.mjs.map +1 -0
- package/dist/business/qqbot/ui/web/index.d.mts +10 -0
- package/dist/business/qqbot/ui/web/index.d.ts +10 -0
- package/dist/business/qqbot/ui/web/index.js +105 -0
- package/dist/business/qqbot/ui/web/index.js.map +1 -0
- package/dist/business/qqbot/ui/web/index.mjs +99 -0
- package/dist/business/qqbot/ui/web/index.mjs.map +1 -0
- package/dist/business/screenReceiver/index.d.mts +2 -0
- package/dist/business/screenReceiver/index.d.ts +2 -0
- package/dist/business/screenReceiver/index.js +281 -0
- package/dist/business/screenReceiver/index.js.map +1 -0
- package/dist/business/screenReceiver/index.mjs +273 -0
- package/dist/business/screenReceiver/index.mjs.map +1 -0
- package/dist/business/testYourself/admin/index.d.mts +58 -0
- package/dist/business/testYourself/admin/index.d.ts +58 -0
- package/dist/business/testYourself/admin/index.js +1009 -0
- package/dist/business/testYourself/admin/index.js.map +1 -0
- package/dist/business/testYourself/admin/index.mjs +1002 -0
- package/dist/business/testYourself/admin/index.mjs.map +1 -0
- package/dist/business/testYourself/index.d.mts +6 -0
- package/dist/business/testYourself/index.d.ts +6 -0
- package/dist/business/testYourself/index.js +2551 -0
- package/dist/business/testYourself/index.js.map +1 -0
- package/dist/business/testYourself/index.mjs +2531 -0
- package/dist/business/testYourself/index.mjs.map +1 -0
- package/dist/business/testYourself/server/index.d.mts +1029 -0
- package/dist/business/testYourself/server/index.d.ts +1029 -0
- package/dist/business/testYourself/server/index.js +825 -0
- package/dist/business/testYourself/server/index.js.map +1 -0
- package/dist/business/testYourself/server/index.mjs +816 -0
- package/dist/business/testYourself/server/index.mjs.map +1 -0
- package/dist/business/vocaloidBooth/index.d.mts +2 -0
- package/dist/business/vocaloidBooth/index.d.ts +2 -0
- package/dist/business/vocaloidBooth/index.js +172 -0
- package/dist/business/vocaloidBooth/index.js.map +1 -0
- package/dist/business/vocaloidBooth/index.mjs +166 -0
- package/dist/business/vocaloidBooth/index.mjs.map +1 -0
- package/dist/business/vocaloidBooth/server/index.d.mts +111 -0
- package/dist/business/vocaloidBooth/server/index.d.ts +111 -0
- package/dist/business/vocaloidBooth/server/index.js +247 -0
- package/dist/business/vocaloidBooth/server/index.js.map +1 -0
- package/dist/business/vocaloidBooth/server/index.mjs +237 -0
- package/dist/business/vocaloidBooth/server/index.mjs.map +1 -0
- package/dist/business/vocaloidBooth/web/index.d.mts +45 -0
- package/dist/business/vocaloidBooth/web/index.d.ts +45 -0
- package/dist/business/vocaloidBooth/web/index.js +376 -0
- package/dist/business/vocaloidBooth/web/index.js.map +1 -0
- package/dist/business/vocaloidBooth/web/index.mjs +362 -0
- package/dist/business/vocaloidBooth/web/index.mjs.map +1 -0
- package/dist/common/auth/components/index.js +18 -0
- package/dist/common/auth/components/index.js.map +1 -1
- package/dist/common/auth/components/index.mjs +18 -0
- package/dist/common/auth/components/index.mjs.map +1 -1
- package/dist/common/auth/server/index.d.mts +86 -3
- package/dist/common/auth/server/index.d.ts +86 -3
- package/dist/common/auth/server/index.js +614 -1
- package/dist/common/auth/server/index.js.map +1 -1
- package/dist/common/auth/server/index.mjs +599 -3
- package/dist/common/auth/server/index.mjs.map +1 -1
- package/dist/config-BQp3qLAL.d.mts +22 -0
- package/dist/config-BQp3qLAL.d.ts +22 -0
- package/dist/drizzle-schema-BNhqj2AZ.d.mts +1114 -0
- package/dist/drizzle-schema-BNhqj2AZ.d.ts +1114 -0
- package/dist/festivalCardService-D60G-sgr.d.mts +13 -0
- package/dist/festivalCardService-DnLyJpRh.d.ts +13 -0
- package/dist/index-BMgdH5dL.d.mts +1716 -0
- package/dist/index-BO9_Do5y.d.mts +93 -0
- package/dist/index-BO9_Do5y.d.ts +93 -0
- package/dist/index-BSmd4ikf.d.ts +76 -0
- package/dist/index-BSwvWYp2.d.mts +2632 -0
- package/dist/index-Bo_fW3Tl.d.mts +105 -0
- package/dist/index-Bo_fW3Tl.d.ts +105 -0
- package/dist/index-BrKazb8M.d.mts +148 -0
- package/dist/index-BrKazb8M.d.ts +148 -0
- package/dist/index-Bzh6QE4P.d.ts +25 -0
- package/dist/index-C5Ic6eSR.d.mts +25 -0
- package/dist/index-C8i9SIxk.d.ts +2632 -0
- package/dist/index-C_GhVhOT.d.mts +109 -0
- package/dist/index-C_GhVhOT.d.ts +109 -0
- package/dist/index-Cb3UEpG4.d.mts +101 -0
- package/dist/index-CjlkUj01.d.mts +103 -0
- package/dist/index-CucXCBNR.d.mts +302 -0
- package/dist/index-CucXCBNR.d.ts +302 -0
- package/dist/index-DLLPTprx.d.mts +1522 -0
- package/dist/index-DRiZy0dv.d.mts +525 -0
- package/dist/index-DRiZy0dv.d.ts +525 -0
- package/dist/index-Dc_I2t0P.d.mts +103 -0
- package/dist/index-DowAHRIP.d.mts +250 -0
- package/dist/index-DowAHRIP.d.ts +250 -0
- package/dist/index-Dpq_5H2n.d.ts +103 -0
- package/dist/index-Ds2M_9zb.d.ts +101 -0
- package/dist/index-IXMAeTtN.d.ts +1716 -0
- package/dist/index-VFDbZxVM.d.ts +1522 -0
- package/dist/index-jadkp96n.d.ts +103 -0
- package/dist/index-r2-zE3iC.d.mts +76 -0
- package/dist/index.d.mts +10682 -0
- package/dist/index.d.ts +10682 -0
- package/dist/index.js +38233 -0
- package/dist/index.js.map +1 -0
- package/dist/index.mjs +37959 -0
- package/dist/index.mjs.map +1 -0
- package/dist/types-B6B210gX.d.mts +270 -0
- package/dist/types-B6B210gX.d.ts +270 -0
- package/dist/types-B7voqjjA.d.mts +51 -0
- package/dist/types-B7voqjjA.d.ts +51 -0
- package/dist/types-Bdnte5EN.d.mts +292 -0
- package/dist/types-C2z_QQPI.d.mts +70 -0
- package/dist/types-C2z_QQPI.d.ts +70 -0
- package/dist/types-HorDyIRv.d.mts +303 -0
- package/dist/types-HorDyIRv.d.ts +303 -0
- package/dist/types-_rFX1atk.d.ts +292 -0
- package/package.json +10 -2
|
@@ -0,0 +1,273 @@
|
|
|
1
|
+
import React, { useState, useRef, useCallback, useEffect, useMemo } from 'react';
|
|
2
|
+
|
|
3
|
+
// src/business/screenReceiver/useScreenReceiver.ts
|
|
4
|
+
var DEFAULT_STUN_SERVER = { urls: "stun:stun.l.google.com:19302" };
|
|
5
|
+
function parseMessage(payload) {
|
|
6
|
+
try {
|
|
7
|
+
if (typeof payload === "string") return JSON.parse(payload);
|
|
8
|
+
if (payload instanceof ArrayBuffer) return JSON.parse(new TextDecoder().decode(payload));
|
|
9
|
+
return JSON.parse(String(payload));
|
|
10
|
+
} catch {
|
|
11
|
+
return null;
|
|
12
|
+
}
|
|
13
|
+
}
|
|
14
|
+
function useScreenReceiver(options) {
|
|
15
|
+
const { wsUrl, roomId, iceServers = [DEFAULT_STUN_SERVER], maxLogs = 80 } = options;
|
|
16
|
+
const [logs, setLogs] = useState([]);
|
|
17
|
+
const [isConnected, setIsConnected] = useState(false);
|
|
18
|
+
const [connectionState, setConnectionState] = useState("idle");
|
|
19
|
+
const [iceConnectionState, setIceConnectionState] = useState("idle");
|
|
20
|
+
const [stream, setStream] = useState(null);
|
|
21
|
+
const logIdRef = useRef(0);
|
|
22
|
+
const peerRef = useRef({ pendingCandidates: [] });
|
|
23
|
+
const videoRef = useRef(null);
|
|
24
|
+
const appendLog = useCallback(
|
|
25
|
+
(text) => {
|
|
26
|
+
logIdRef.current += 1;
|
|
27
|
+
setLogs((prev) => [...prev, { id: logIdRef.current, text }].slice(-maxLogs));
|
|
28
|
+
},
|
|
29
|
+
[maxLogs]
|
|
30
|
+
);
|
|
31
|
+
const disconnect = useCallback(() => {
|
|
32
|
+
peerRef.current.ws?.close();
|
|
33
|
+
peerRef.current.pc?.close();
|
|
34
|
+
peerRef.current = { pendingCandidates: [] };
|
|
35
|
+
setIsConnected(false);
|
|
36
|
+
setConnectionState("idle");
|
|
37
|
+
setIceConnectionState("idle");
|
|
38
|
+
setStream(null);
|
|
39
|
+
}, []);
|
|
40
|
+
const connect = useCallback(async () => {
|
|
41
|
+
disconnect();
|
|
42
|
+
const normalizedWsUrl = wsUrl.trim();
|
|
43
|
+
const normalizedRoomId = roomId.trim();
|
|
44
|
+
if (!normalizedWsUrl) {
|
|
45
|
+
appendLog("ws url is empty");
|
|
46
|
+
return;
|
|
47
|
+
}
|
|
48
|
+
if (!normalizedRoomId) {
|
|
49
|
+
appendLog("room id is empty");
|
|
50
|
+
return;
|
|
51
|
+
}
|
|
52
|
+
appendLog(`connecting: ${normalizedWsUrl}`);
|
|
53
|
+
let ws;
|
|
54
|
+
try {
|
|
55
|
+
ws = new WebSocket(normalizedWsUrl);
|
|
56
|
+
} catch (error) {
|
|
57
|
+
appendLog(`ws create error: ${String(error)}`);
|
|
58
|
+
return;
|
|
59
|
+
}
|
|
60
|
+
const pc = new RTCPeerConnection({ iceServers });
|
|
61
|
+
pc.addTransceiver("video", { direction: "recvonly" });
|
|
62
|
+
peerRef.current.ws = ws;
|
|
63
|
+
peerRef.current.pc = pc;
|
|
64
|
+
peerRef.current.pendingCandidates = [];
|
|
65
|
+
ws.onopen = () => {
|
|
66
|
+
appendLog("ws connected");
|
|
67
|
+
setIsConnected(true);
|
|
68
|
+
ws.send(JSON.stringify({ type: "join", roomId: normalizedRoomId, role: "viewer" }));
|
|
69
|
+
};
|
|
70
|
+
ws.onclose = () => {
|
|
71
|
+
appendLog("ws closed");
|
|
72
|
+
setIsConnected(false);
|
|
73
|
+
};
|
|
74
|
+
ws.onerror = (event) => {
|
|
75
|
+
appendLog(`ws error: ${event.type}`);
|
|
76
|
+
};
|
|
77
|
+
ws.onmessage = async (event) => {
|
|
78
|
+
const message = parseMessage(event.data);
|
|
79
|
+
if (!message) {
|
|
80
|
+
appendLog("ws message parse error");
|
|
81
|
+
return;
|
|
82
|
+
}
|
|
83
|
+
if (message.type === "joined") {
|
|
84
|
+
const joined = message;
|
|
85
|
+
peerRef.current.selfPeerId = joined.peerId;
|
|
86
|
+
appendLog(`joined room ${joined.roomId} as ${joined.peerId}`);
|
|
87
|
+
return;
|
|
88
|
+
}
|
|
89
|
+
if (message.type === "room_state") {
|
|
90
|
+
const roomState = message;
|
|
91
|
+
const broadcaster = roomState.peers.find((peer) => peer.role === "broadcaster");
|
|
92
|
+
if (broadcaster) {
|
|
93
|
+
peerRef.current.publisherPeerId = broadcaster.peerId;
|
|
94
|
+
appendLog(`broadcaster online: ${broadcaster.peerId}`);
|
|
95
|
+
}
|
|
96
|
+
return;
|
|
97
|
+
}
|
|
98
|
+
if (message.type === "offer" && typeof message.sdp === "string") {
|
|
99
|
+
appendLog("offer received");
|
|
100
|
+
peerRef.current.publisherPeerId = typeof message.fromPeerId === "string" ? message.fromPeerId : void 0;
|
|
101
|
+
await pc.setRemoteDescription({ type: "offer", sdp: message.sdp });
|
|
102
|
+
for (const candidate of peerRef.current.pendingCandidates) {
|
|
103
|
+
try {
|
|
104
|
+
await pc.addIceCandidate(candidate);
|
|
105
|
+
} catch (error) {
|
|
106
|
+
appendLog(`ice err: ${String(error)}`);
|
|
107
|
+
}
|
|
108
|
+
}
|
|
109
|
+
peerRef.current.pendingCandidates = [];
|
|
110
|
+
const answer = await pc.createAnswer();
|
|
111
|
+
await pc.setLocalDescription(answer);
|
|
112
|
+
ws.send(
|
|
113
|
+
JSON.stringify({
|
|
114
|
+
type: "answer",
|
|
115
|
+
sdp: answer.sdp,
|
|
116
|
+
targetPeerId: peerRef.current.publisherPeerId
|
|
117
|
+
})
|
|
118
|
+
);
|
|
119
|
+
appendLog("answer sent");
|
|
120
|
+
return;
|
|
121
|
+
}
|
|
122
|
+
if (message.type === "ice" && message.candidate) {
|
|
123
|
+
if (!pc.remoteDescription) {
|
|
124
|
+
peerRef.current.pendingCandidates.push(message.candidate);
|
|
125
|
+
return;
|
|
126
|
+
}
|
|
127
|
+
try {
|
|
128
|
+
await pc.addIceCandidate(message.candidate);
|
|
129
|
+
} catch (error) {
|
|
130
|
+
appendLog(`ice err: ${String(error)}`);
|
|
131
|
+
}
|
|
132
|
+
return;
|
|
133
|
+
}
|
|
134
|
+
if (message.type === "peer_left") {
|
|
135
|
+
const peerLeft = message;
|
|
136
|
+
if (peerLeft.peerId === peerRef.current.publisherPeerId) {
|
|
137
|
+
peerRef.current.publisherPeerId = void 0;
|
|
138
|
+
appendLog(`publisher left: ${peerLeft.peerId}`);
|
|
139
|
+
}
|
|
140
|
+
return;
|
|
141
|
+
}
|
|
142
|
+
if (message.type === "error") {
|
|
143
|
+
appendLog(`server error: ${String(message.reason ?? "unknown")}`);
|
|
144
|
+
}
|
|
145
|
+
};
|
|
146
|
+
pc.ontrack = (event) => {
|
|
147
|
+
const receivedStream = event.streams[0];
|
|
148
|
+
if (receivedStream) {
|
|
149
|
+
setStream(receivedStream);
|
|
150
|
+
appendLog("track received");
|
|
151
|
+
}
|
|
152
|
+
};
|
|
153
|
+
pc.onicecandidate = (event) => {
|
|
154
|
+
if (!event.candidate) return;
|
|
155
|
+
ws.send(
|
|
156
|
+
JSON.stringify({
|
|
157
|
+
type: "ice",
|
|
158
|
+
candidate: event.candidate,
|
|
159
|
+
targetPeerId: peerRef.current.publisherPeerId
|
|
160
|
+
})
|
|
161
|
+
);
|
|
162
|
+
};
|
|
163
|
+
pc.onconnectionstatechange = () => {
|
|
164
|
+
const state = pc.connectionState || "new";
|
|
165
|
+
setConnectionState(state);
|
|
166
|
+
appendLog(`pc state: ${state}`);
|
|
167
|
+
};
|
|
168
|
+
pc.oniceconnectionstatechange = () => {
|
|
169
|
+
const state = pc.iceConnectionState || "new";
|
|
170
|
+
setIceConnectionState(state);
|
|
171
|
+
appendLog(`ice state: ${state}`);
|
|
172
|
+
};
|
|
173
|
+
}, [appendLog, disconnect, iceServers, roomId, wsUrl]);
|
|
174
|
+
useEffect(() => () => disconnect(), [disconnect]);
|
|
175
|
+
useEffect(() => {
|
|
176
|
+
if (!videoRef.current) return;
|
|
177
|
+
videoRef.current.srcObject = stream;
|
|
178
|
+
}, [stream]);
|
|
179
|
+
const clearLogs = useCallback(() => setLogs([]), []);
|
|
180
|
+
return useMemo(
|
|
181
|
+
() => ({
|
|
182
|
+
connect,
|
|
183
|
+
disconnect,
|
|
184
|
+
clearLogs,
|
|
185
|
+
logs,
|
|
186
|
+
isConnected,
|
|
187
|
+
connectionState,
|
|
188
|
+
iceConnectionState,
|
|
189
|
+
videoRef,
|
|
190
|
+
stream
|
|
191
|
+
}),
|
|
192
|
+
[clearLogs, connect, connectionState, disconnect, iceConnectionState, isConnected, logs, stream]
|
|
193
|
+
);
|
|
194
|
+
}
|
|
195
|
+
|
|
196
|
+
// src/business/screenReceiver/signalUrl.ts
|
|
197
|
+
var DEFAULT_PORT = 8787;
|
|
198
|
+
var DEFAULT_PATH = "/ws";
|
|
199
|
+
function resolveScreenReceiverSignalUrl(options = {}) {
|
|
200
|
+
const { signalUrl, path = DEFAULT_PATH, port = DEFAULT_PORT } = options;
|
|
201
|
+
if (signalUrl && signalUrl.trim()) return signalUrl.trim();
|
|
202
|
+
const normalizedPath = path.startsWith("/") ? path : `/${path}`;
|
|
203
|
+
if (typeof window !== "undefined") {
|
|
204
|
+
const protocol = window.location.protocol === "https:" ? "wss:" : "ws:";
|
|
205
|
+
return `${protocol}//${window.location.host}${normalizedPath}`;
|
|
206
|
+
}
|
|
207
|
+
return `ws://127.0.0.1:${port}${normalizedPath}`;
|
|
208
|
+
}
|
|
209
|
+
var DEFAULT_ROOM_ID = "screen-room-1";
|
|
210
|
+
function ScreenReceiverPanel(props) {
|
|
211
|
+
const { defaultSignalUrl, defaultRoomId = DEFAULT_ROOM_ID, className } = props;
|
|
212
|
+
const initialSignalUrl = useMemo(
|
|
213
|
+
() => resolveScreenReceiverSignalUrl({ signalUrl: defaultSignalUrl }),
|
|
214
|
+
[defaultSignalUrl]
|
|
215
|
+
);
|
|
216
|
+
const [wsUrl, setWsUrl] = useState(initialSignalUrl);
|
|
217
|
+
const [roomId, setRoomId] = useState(defaultRoomId);
|
|
218
|
+
const receiver = useScreenReceiver({ wsUrl, roomId });
|
|
219
|
+
const logs = useMemo(() => receiver.logs.map((entry) => entry.text).join("\n"), [receiver.logs]);
|
|
220
|
+
return /* @__PURE__ */ React.createElement("div", { className }, /* @__PURE__ */ React.createElement("div", { className: "flex flex-col gap-4" }, /* @__PURE__ */ React.createElement("div", { className: "grid gap-3 md:grid-cols-[2fr,1fr,auto]" }, /* @__PURE__ */ React.createElement("label", { className: "flex flex-col gap-1 text-sm" }, /* @__PURE__ */ React.createElement("span", null, "Signaling WS"), /* @__PURE__ */ React.createElement(
|
|
221
|
+
"input",
|
|
222
|
+
{
|
|
223
|
+
className: "rounded-md border px-3 py-2",
|
|
224
|
+
value: wsUrl,
|
|
225
|
+
onChange: (event) => setWsUrl(event.target.value)
|
|
226
|
+
}
|
|
227
|
+
)), /* @__PURE__ */ React.createElement("label", { className: "flex flex-col gap-1 text-sm" }, /* @__PURE__ */ React.createElement("span", null, "Room"), /* @__PURE__ */ React.createElement(
|
|
228
|
+
"input",
|
|
229
|
+
{
|
|
230
|
+
className: "rounded-md border px-3 py-2",
|
|
231
|
+
value: roomId,
|
|
232
|
+
onChange: (event) => setRoomId(event.target.value)
|
|
233
|
+
}
|
|
234
|
+
)), /* @__PURE__ */ React.createElement("div", { className: "flex items-end gap-2" }, /* @__PURE__ */ React.createElement(
|
|
235
|
+
"button",
|
|
236
|
+
{
|
|
237
|
+
onClick: () => void receiver.connect(),
|
|
238
|
+
className: "rounded-md border bg-black px-4 py-2 text-sm text-white"
|
|
239
|
+
},
|
|
240
|
+
receiver.isConnected ? "Reconnect" : "Connect"
|
|
241
|
+
), /* @__PURE__ */ React.createElement(
|
|
242
|
+
"button",
|
|
243
|
+
{
|
|
244
|
+
onClick: receiver.disconnect,
|
|
245
|
+
className: "rounded-md border px-4 py-2 text-sm"
|
|
246
|
+
},
|
|
247
|
+
"Disconnect"
|
|
248
|
+
))), /* @__PURE__ */ React.createElement("div", { className: "grid gap-3 md:grid-cols-3" }, /* @__PURE__ */ React.createElement(StatusItem, { label: "WebSocket", value: receiver.isConnected ? "connected" : "idle" }), /* @__PURE__ */ React.createElement(StatusItem, { label: "PeerConnection", value: receiver.connectionState }), /* @__PURE__ */ React.createElement(StatusItem, { label: "ICE State", value: receiver.iceConnectionState })), /* @__PURE__ */ React.createElement("div", { className: "grid gap-4 lg:grid-cols-[3fr,2fr]" }, /* @__PURE__ */ React.createElement("div", { className: "rounded-xl border bg-black p-3" }, /* @__PURE__ */ React.createElement(
|
|
249
|
+
"video",
|
|
250
|
+
{
|
|
251
|
+
ref: receiver.videoRef,
|
|
252
|
+
autoPlay: true,
|
|
253
|
+
playsInline: true,
|
|
254
|
+
controls: true,
|
|
255
|
+
muted: true,
|
|
256
|
+
className: "h-[360px] w-full rounded-lg bg-black"
|
|
257
|
+
}
|
|
258
|
+
)), /* @__PURE__ */ React.createElement("div", { className: "rounded-xl border p-3" }, /* @__PURE__ */ React.createElement("div", { className: "flex items-center justify-between" }, /* @__PURE__ */ React.createElement("p", { className: "text-sm font-semibold" }, "Session Log"), /* @__PURE__ */ React.createElement(
|
|
259
|
+
"button",
|
|
260
|
+
{
|
|
261
|
+
onClick: receiver.clearLogs,
|
|
262
|
+
className: "text-xs underline-offset-2 hover:underline"
|
|
263
|
+
},
|
|
264
|
+
"Clear"
|
|
265
|
+
)), /* @__PURE__ */ React.createElement("pre", { className: "mt-3 h-[320px] overflow-auto rounded-lg border bg-slate-50 p-3 text-xs" }, logs || "No logs yet.")))));
|
|
266
|
+
}
|
|
267
|
+
function StatusItem({ label, value }) {
|
|
268
|
+
return /* @__PURE__ */ React.createElement("div", { className: "rounded-md border px-3 py-2" }, /* @__PURE__ */ React.createElement("p", { className: "text-xs text-slate-500" }, label), /* @__PURE__ */ React.createElement("p", { className: "text-sm font-medium" }, value));
|
|
269
|
+
}
|
|
270
|
+
|
|
271
|
+
export { ScreenReceiverPanel, resolveScreenReceiverSignalUrl, useScreenReceiver };
|
|
272
|
+
//# sourceMappingURL=index.mjs.map
|
|
273
|
+
//# sourceMappingURL=index.mjs.map
|
|
@@ -0,0 +1 @@
|
|
|
1
|
+
{"version":3,"sources":["../../../src/business/screenReceiver/useScreenReceiver.ts","../../../src/business/screenReceiver/signalUrl.ts","../../../src/business/screenReceiver/ScreenReceiverPanel.tsx"],"names":["useMemo","useState"],"mappings":"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","file":"index.mjs","sourcesContent":["import type { RefObject } from 'react';\nimport { useCallback, useEffect, useMemo, useRef, useState } from 'react';\nimport type {\n ScreenReceiverJoinedMessage,\n ScreenReceiverPeerLeftMessage,\n ScreenReceiverRoomStateMessage,\n} from './types';\n\nconst DEFAULT_STUN_SERVER: RTCIceServer = { urls: 'stun:stun.l.google.com:19302' };\n\nexport interface ScreenReceiverLogEntry {\n id: number;\n text: string;\n}\n\nexport interface UseScreenReceiverOptions {\n wsUrl: string;\n roomId: string;\n iceServers?: RTCIceServer[];\n maxLogs?: number;\n}\n\nexport interface UseScreenReceiverReturn {\n connect: () => Promise<void>;\n disconnect: () => void;\n clearLogs: () => void;\n logs: ScreenReceiverLogEntry[];\n isConnected: boolean;\n connectionState: RTCPeerConnectionState | 'idle';\n iceConnectionState: RTCIceConnectionState | 'idle';\n videoRef: RefObject<HTMLVideoElement>;\n stream: MediaStream | null;\n}\n\ntype PeerContext = {\n ws?: WebSocket;\n pc?: RTCPeerConnection;\n pendingCandidates: RTCIceCandidateInit[];\n selfPeerId?: string;\n publisherPeerId?: string;\n};\n\nfunction parseMessage(payload: MessageEvent['data']): Record<string, any> | null {\n try {\n if (typeof payload === 'string') return JSON.parse(payload);\n if (payload instanceof ArrayBuffer) return JSON.parse(new TextDecoder().decode(payload));\n return JSON.parse(String(payload));\n } catch {\n return null;\n }\n}\n\nexport function useScreenReceiver(options: UseScreenReceiverOptions): UseScreenReceiverReturn {\n const { wsUrl, roomId, iceServers = [DEFAULT_STUN_SERVER], maxLogs = 80 } = options;\n const [logs, setLogs] = useState<ScreenReceiverLogEntry[]>([]);\n const [isConnected, setIsConnected] = useState(false);\n const [connectionState, setConnectionState] = useState<RTCPeerConnectionState | 'idle'>('idle');\n const [iceConnectionState, setIceConnectionState] = useState<RTCIceConnectionState | 'idle'>('idle');\n const [stream, setStream] = useState<MediaStream | null>(null);\n const logIdRef = useRef(0);\n const peerRef = useRef<PeerContext>({ pendingCandidates: [] });\n const videoRef = useRef<HTMLVideoElement>(null);\n\n const appendLog = useCallback(\n (text: string) => {\n logIdRef.current += 1;\n setLogs((prev) => [...prev, { id: logIdRef.current, text }].slice(-maxLogs));\n },\n [maxLogs],\n );\n\n const disconnect = useCallback(() => {\n peerRef.current.ws?.close();\n peerRef.current.pc?.close();\n peerRef.current = { pendingCandidates: [] };\n setIsConnected(false);\n setConnectionState('idle');\n setIceConnectionState('idle');\n setStream(null);\n }, []);\n\n const connect = useCallback(async () => {\n disconnect();\n\n const normalizedWsUrl = wsUrl.trim();\n const normalizedRoomId = roomId.trim();\n if (!normalizedWsUrl) {\n appendLog('ws url is empty');\n return;\n }\n if (!normalizedRoomId) {\n appendLog('room id is empty');\n return;\n }\n\n appendLog(`connecting: ${normalizedWsUrl}`);\n\n let ws: WebSocket;\n try {\n ws = new WebSocket(normalizedWsUrl);\n } catch (error) {\n appendLog(`ws create error: ${String(error)}`);\n return;\n }\n\n const pc = new RTCPeerConnection({ iceServers });\n pc.addTransceiver('video', { direction: 'recvonly' });\n\n peerRef.current.ws = ws;\n peerRef.current.pc = pc;\n peerRef.current.pendingCandidates = [];\n\n ws.onopen = () => {\n appendLog('ws connected');\n setIsConnected(true);\n ws.send(JSON.stringify({ type: 'join', roomId: normalizedRoomId, role: 'viewer' }));\n };\n\n ws.onclose = () => {\n appendLog('ws closed');\n setIsConnected(false);\n };\n\n ws.onerror = (event) => {\n appendLog(`ws error: ${event.type}`);\n };\n\n ws.onmessage = async (event) => {\n const message = parseMessage(event.data);\n if (!message) {\n appendLog('ws message parse error');\n return;\n }\n\n if (message.type === 'joined') {\n const joined = message as ScreenReceiverJoinedMessage;\n peerRef.current.selfPeerId = joined.peerId;\n appendLog(`joined room ${joined.roomId} as ${joined.peerId}`);\n return;\n }\n\n if (message.type === 'room_state') {\n const roomState = message as ScreenReceiverRoomStateMessage;\n const broadcaster = roomState.peers.find((peer) => peer.role === 'broadcaster');\n if (broadcaster) {\n peerRef.current.publisherPeerId = broadcaster.peerId;\n appendLog(`broadcaster online: ${broadcaster.peerId}`);\n }\n return;\n }\n\n if (message.type === 'offer' && typeof message.sdp === 'string') {\n appendLog('offer received');\n peerRef.current.publisherPeerId = typeof message.fromPeerId === 'string' ? message.fromPeerId : undefined;\n await pc.setRemoteDescription({ type: 'offer', sdp: message.sdp });\n for (const candidate of peerRef.current.pendingCandidates) {\n try {\n await pc.addIceCandidate(candidate);\n } catch (error) {\n appendLog(`ice err: ${String(error)}`);\n }\n }\n peerRef.current.pendingCandidates = [];\n\n const answer = await pc.createAnswer();\n await pc.setLocalDescription(answer);\n ws.send(\n JSON.stringify({\n type: 'answer',\n sdp: answer.sdp,\n targetPeerId: peerRef.current.publisherPeerId,\n }),\n );\n appendLog('answer sent');\n return;\n }\n\n if (message.type === 'ice' && message.candidate) {\n if (!pc.remoteDescription) {\n peerRef.current.pendingCandidates.push(message.candidate as RTCIceCandidateInit);\n return;\n }\n try {\n await pc.addIceCandidate(message.candidate as RTCIceCandidateInit);\n } catch (error) {\n appendLog(`ice err: ${String(error)}`);\n }\n return;\n }\n\n if (message.type === 'peer_left') {\n const peerLeft = message as ScreenReceiverPeerLeftMessage;\n if (peerLeft.peerId === peerRef.current.publisherPeerId) {\n peerRef.current.publisherPeerId = undefined;\n appendLog(`publisher left: ${peerLeft.peerId}`);\n }\n return;\n }\n\n if (message.type === 'error') {\n appendLog(`server error: ${String(message.reason ?? 'unknown')}`);\n }\n };\n\n pc.ontrack = (event) => {\n const receivedStream = event.streams[0];\n if (receivedStream) {\n setStream(receivedStream);\n appendLog('track received');\n }\n };\n\n pc.onicecandidate = (event) => {\n if (!event.candidate) return;\n ws.send(\n JSON.stringify({\n type: 'ice',\n candidate: event.candidate,\n targetPeerId: peerRef.current.publisherPeerId,\n }),\n );\n };\n\n pc.onconnectionstatechange = () => {\n const state = pc.connectionState || 'new';\n setConnectionState(state);\n appendLog(`pc state: ${state}`);\n };\n\n pc.oniceconnectionstatechange = () => {\n const state = pc.iceConnectionState || 'new';\n setIceConnectionState(state);\n appendLog(`ice state: ${state}`);\n };\n }, [appendLog, disconnect, iceServers, roomId, wsUrl]);\n\n useEffect(() => () => disconnect(), [disconnect]);\n\n useEffect(() => {\n if (!videoRef.current) return;\n videoRef.current.srcObject = stream;\n }, [stream]);\n\n const clearLogs = useCallback(() => setLogs([]), []);\n\n return useMemo(\n () => ({\n connect,\n disconnect,\n clearLogs,\n logs,\n isConnected,\n connectionState,\n iceConnectionState,\n videoRef,\n stream,\n }),\n [clearLogs, connect, connectionState, disconnect, iceConnectionState, isConnected, logs, stream],\n );\n}\n","const DEFAULT_PORT = 8787;\nconst DEFAULT_PATH = '/ws';\n\nexport interface ResolveScreenReceiverSignalUrlOptions {\n signalUrl?: string;\n path?: string;\n port?: number;\n}\n\nexport function resolveScreenReceiverSignalUrl(\n options: ResolveScreenReceiverSignalUrlOptions = {},\n): string {\n const { signalUrl, path = DEFAULT_PATH, port = DEFAULT_PORT } = options;\n if (signalUrl && signalUrl.trim()) return signalUrl.trim();\n\n const normalizedPath = path.startsWith('/') ? path : `/${path}`;\n if (typeof window !== 'undefined') {\n const protocol = window.location.protocol === 'https:' ? 'wss:' : 'ws:';\n return `${protocol}//${window.location.host}${normalizedPath}`;\n }\n\n return `ws://127.0.0.1:${port}${normalizedPath}`;\n}\n","import React, { useMemo, useState } from 'react';\nimport { resolveScreenReceiverSignalUrl } from './signalUrl';\nimport { useScreenReceiver } from './useScreenReceiver';\n\nexport interface ScreenReceiverPanelProps {\n defaultSignalUrl?: string;\n defaultRoomId?: string;\n className?: string;\n}\n\nconst DEFAULT_ROOM_ID = 'screen-room-1';\n\nexport function ScreenReceiverPanel(props: ScreenReceiverPanelProps) {\n const { defaultSignalUrl, defaultRoomId = DEFAULT_ROOM_ID, className } = props;\n const initialSignalUrl = useMemo(\n () => resolveScreenReceiverSignalUrl({ signalUrl: defaultSignalUrl }),\n [defaultSignalUrl],\n );\n const [wsUrl, setWsUrl] = useState(initialSignalUrl);\n const [roomId, setRoomId] = useState(defaultRoomId);\n const receiver = useScreenReceiver({ wsUrl, roomId });\n const logs = useMemo(() => receiver.logs.map((entry) => entry.text).join('\\n'), [receiver.logs]);\n\n return (\n <div className={className}>\n <div className=\"flex flex-col gap-4\">\n <div className=\"grid gap-3 md:grid-cols-[2fr,1fr,auto]\">\n <label className=\"flex flex-col gap-1 text-sm\">\n <span>Signaling WS</span>\n <input\n className=\"rounded-md border px-3 py-2\"\n value={wsUrl}\n onChange={(event) => setWsUrl(event.target.value)}\n />\n </label>\n <label className=\"flex flex-col gap-1 text-sm\">\n <span>Room</span>\n <input\n className=\"rounded-md border px-3 py-2\"\n value={roomId}\n onChange={(event) => setRoomId(event.target.value)}\n />\n </label>\n <div className=\"flex items-end gap-2\">\n <button\n onClick={() => void receiver.connect()}\n className=\"rounded-md border bg-black px-4 py-2 text-sm text-white\"\n >\n {receiver.isConnected ? 'Reconnect' : 'Connect'}\n </button>\n <button\n onClick={receiver.disconnect}\n className=\"rounded-md border px-4 py-2 text-sm\"\n >\n Disconnect\n </button>\n </div>\n </div>\n\n <div className=\"grid gap-3 md:grid-cols-3\">\n <StatusItem label=\"WebSocket\" value={receiver.isConnected ? 'connected' : 'idle'} />\n <StatusItem label=\"PeerConnection\" value={receiver.connectionState} />\n <StatusItem label=\"ICE State\" value={receiver.iceConnectionState} />\n </div>\n\n <div className=\"grid gap-4 lg:grid-cols-[3fr,2fr]\">\n <div className=\"rounded-xl border bg-black p-3\">\n <video\n ref={receiver.videoRef}\n autoPlay\n playsInline\n controls\n muted\n className=\"h-[360px] w-full rounded-lg bg-black\"\n />\n </div>\n <div className=\"rounded-xl border p-3\">\n <div className=\"flex items-center justify-between\">\n <p className=\"text-sm font-semibold\">Session Log</p>\n <button\n onClick={receiver.clearLogs}\n className=\"text-xs underline-offset-2 hover:underline\"\n >\n Clear\n </button>\n </div>\n <pre className=\"mt-3 h-[320px] overflow-auto rounded-lg border bg-slate-50 p-3 text-xs\">\n {logs || 'No logs yet.'}\n </pre>\n </div>\n </div>\n </div>\n </div>\n );\n}\n\nfunction StatusItem({ label, value }: { label: string; value: string }) {\n return (\n <div className=\"rounded-md border px-3 py-2\">\n <p className=\"text-xs text-slate-500\">{label}</p>\n <p className=\"text-sm font-medium\">{value}</p>\n </div>\n );\n}\n"]}
|
|
@@ -0,0 +1,58 @@
|
|
|
1
|
+
import React__default from 'react';
|
|
2
|
+
import { b as ConfigService, S as SavedConfig } from '../../../ConfigService-Oga_zFRS.mjs';
|
|
3
|
+
|
|
4
|
+
/**
|
|
5
|
+
* 测测你是什么 - 配置管理后台组件
|
|
6
|
+
* Test Yourself - Configuration Manager Component
|
|
7
|
+
*
|
|
8
|
+
* 支持创建、编辑、管理多套测试配置
|
|
9
|
+
*/
|
|
10
|
+
|
|
11
|
+
interface ConfigManagerProps {
|
|
12
|
+
/** 配置服务实例 */
|
|
13
|
+
configService: ConfigService;
|
|
14
|
+
/** 配置变化回调 */
|
|
15
|
+
onConfigChange?: (configs: SavedConfig[]) => void;
|
|
16
|
+
/** 自定义样式 */
|
|
17
|
+
className?: string;
|
|
18
|
+
/** 图片上传处理函数 */
|
|
19
|
+
onImageUpload?: (file: File) => Promise<string>;
|
|
20
|
+
}
|
|
21
|
+
/**
|
|
22
|
+
* 配置管理器组件
|
|
23
|
+
*/
|
|
24
|
+
declare const ConfigManager: React__default.FC<ConfigManagerProps>;
|
|
25
|
+
|
|
26
|
+
/**
|
|
27
|
+
* 测测你是什么 - 配置列表组件
|
|
28
|
+
* Test Yourself - Configuration List Component
|
|
29
|
+
*
|
|
30
|
+
* 用于展示和选择配置
|
|
31
|
+
*/
|
|
32
|
+
|
|
33
|
+
interface ConfigListProps {
|
|
34
|
+
/** 配置服务实例 */
|
|
35
|
+
configService: ConfigService;
|
|
36
|
+
/** 选择配置回调 */
|
|
37
|
+
onSelect?: (id: string) => void;
|
|
38
|
+
/** 编辑配置回调 */
|
|
39
|
+
onEdit?: (id: string) => void;
|
|
40
|
+
/** 删除配置回调 */
|
|
41
|
+
onDelete?: (id: string) => void;
|
|
42
|
+
/** 是否显示操作按钮 */
|
|
43
|
+
showActions?: boolean;
|
|
44
|
+
/** 是否显示预览链接 */
|
|
45
|
+
showPreviewLink?: boolean;
|
|
46
|
+
/** 预览基础URL */
|
|
47
|
+
previewBaseUrl?: string;
|
|
48
|
+
/** 自定义样式 */
|
|
49
|
+
className?: string;
|
|
50
|
+
/** 每页显示数量 */
|
|
51
|
+
pageSize?: number;
|
|
52
|
+
}
|
|
53
|
+
/**
|
|
54
|
+
* 配置列表组件
|
|
55
|
+
*/
|
|
56
|
+
declare const ConfigList: React__default.FC<ConfigListProps>;
|
|
57
|
+
|
|
58
|
+
export { ConfigList, type ConfigListProps, ConfigManager, type ConfigManagerProps };
|
|
@@ -0,0 +1,58 @@
|
|
|
1
|
+
import React__default from 'react';
|
|
2
|
+
import { b as ConfigService, S as SavedConfig } from '../../../ConfigService-Oga_zFRS.js';
|
|
3
|
+
|
|
4
|
+
/**
|
|
5
|
+
* 测测你是什么 - 配置管理后台组件
|
|
6
|
+
* Test Yourself - Configuration Manager Component
|
|
7
|
+
*
|
|
8
|
+
* 支持创建、编辑、管理多套测试配置
|
|
9
|
+
*/
|
|
10
|
+
|
|
11
|
+
interface ConfigManagerProps {
|
|
12
|
+
/** 配置服务实例 */
|
|
13
|
+
configService: ConfigService;
|
|
14
|
+
/** 配置变化回调 */
|
|
15
|
+
onConfigChange?: (configs: SavedConfig[]) => void;
|
|
16
|
+
/** 自定义样式 */
|
|
17
|
+
className?: string;
|
|
18
|
+
/** 图片上传处理函数 */
|
|
19
|
+
onImageUpload?: (file: File) => Promise<string>;
|
|
20
|
+
}
|
|
21
|
+
/**
|
|
22
|
+
* 配置管理器组件
|
|
23
|
+
*/
|
|
24
|
+
declare const ConfigManager: React__default.FC<ConfigManagerProps>;
|
|
25
|
+
|
|
26
|
+
/**
|
|
27
|
+
* 测测你是什么 - 配置列表组件
|
|
28
|
+
* Test Yourself - Configuration List Component
|
|
29
|
+
*
|
|
30
|
+
* 用于展示和选择配置
|
|
31
|
+
*/
|
|
32
|
+
|
|
33
|
+
interface ConfigListProps {
|
|
34
|
+
/** 配置服务实例 */
|
|
35
|
+
configService: ConfigService;
|
|
36
|
+
/** 选择配置回调 */
|
|
37
|
+
onSelect?: (id: string) => void;
|
|
38
|
+
/** 编辑配置回调 */
|
|
39
|
+
onEdit?: (id: string) => void;
|
|
40
|
+
/** 删除配置回调 */
|
|
41
|
+
onDelete?: (id: string) => void;
|
|
42
|
+
/** 是否显示操作按钮 */
|
|
43
|
+
showActions?: boolean;
|
|
44
|
+
/** 是否显示预览链接 */
|
|
45
|
+
showPreviewLink?: boolean;
|
|
46
|
+
/** 预览基础URL */
|
|
47
|
+
previewBaseUrl?: string;
|
|
48
|
+
/** 自定义样式 */
|
|
49
|
+
className?: string;
|
|
50
|
+
/** 每页显示数量 */
|
|
51
|
+
pageSize?: number;
|
|
52
|
+
}
|
|
53
|
+
/**
|
|
54
|
+
* 配置列表组件
|
|
55
|
+
*/
|
|
56
|
+
declare const ConfigList: React__default.FC<ConfigListProps>;
|
|
57
|
+
|
|
58
|
+
export { ConfigList, type ConfigListProps, ConfigManager, type ConfigManagerProps };
|