this is a pain in the ass

This commit is contained in:
Gavin McDonald
2025-06-23 15:33:04 -04:00
parent 59aa904c5a
commit aa938f7258
6 changed files with 377 additions and 20 deletions

95
hooks/useChatGPT.ts Normal file
View File

@@ -0,0 +1,95 @@
import { useEffect, useRef, useState } from 'react';
import { io, Socket } from 'socket.io-client';
interface CursorPosition {
x: number;
y: number;
}
interface PeerMouseHook {
cursors: Record<string, CursorPosition>;
}
export function usePeerMouse(roomId: string): PeerMouseHook {
const [cursors, setCursors] = useState<Record<string, CursorPosition>>({});
const socketRef = useRef<Socket | null>(null);
const peers = useRef<Record<string, RTCPeerConnection>>({});
const channels = useRef<Record<string, RTCDataChannel>>({});
useEffect(() => {
const socket = io();
socketRef.current = socket;
socket.emit('join-room', roomId);
socket.on('new-peer', async (peerId: string) => {
const pc = createPeer(peerId, true);
const offer = await pc.createOffer();
await pc.setLocalDescription(offer);
socket.emit('signal', { to: peerId, data: { description: pc.localDescription } });
});
socket.on('signal', async ({ from, data }) => {
const pc = peers.current[from] || createPeer(from, false);
if (data.description) {
await pc.setRemoteDescription(data.description);
if (data.description.type === 'offer') {
const answer = await pc.createAnswer();
await pc.setLocalDescription(answer);
socket.emit('signal', { to: from, data: { description: pc.localDescription } });
}
}
if (data.candidate) {
await pc.addIceCandidate(data.candidate);
}
});
function createPeer(peerId: string, isInitiator: boolean): RTCPeerConnection {
const pc = new RTCPeerConnection();
if (isInitiator) {
const channel = pc.createDataChannel('mouse');
setupChannel(peerId, channel);
} else {
pc.ondatachannel = (e) => setupChannel(peerId, e.channel);
}
pc.onicecandidate = (e) => {
if (e.candidate) {
socket.emit('signal', { to: peerId, data: { candidate: e.candidate } });
}
};
peers.current[peerId] = pc;
return pc;
}
function setupChannel(peerId: string, channel: RTCDataChannel) {
channels.current[peerId] = channel;
channel.onmessage = (e) => {
const pos = JSON.parse(e.data);
setCursors((prev) => ({ ...prev, [peerId]: pos }));
};
}
function handleMouseMove(e: MouseEvent) {
const pos = JSON.stringify({ x: e.clientX, y: e.clientY });
Object.values(channels.current).forEach((ch) => {
if (ch.readyState === 'open') ch.send(pos);
});
}
window.addEventListener('mousemove', handleMouseMove);
return () => {
window.removeEventListener('mousemove', handleMouseMove);
socket.disconnect();
Object.values(peers.current).forEach((pc) => pc.close());
};
}, [roomId]);
return { cursors };
}

54
hooks/useRTC.ts Normal file
View File

@@ -0,0 +1,54 @@
import { useEffect, useState } from 'react';
import RTCPeer from '@/lib/RTCPeer';
import type { UseSocket } from '@/hooks/useSocket';
import type {} from '@/types';
// interface UseSocketProps {
// gameID: string;
// setGameData: (gameUpdate: GameUpdate) => void;
// setNoGame: (noGame: boolean) => void;
// }
const channelName = 'tilt';
export default function useRTC({
ready,
registerAnsweredReceiver,
registerOfferredReceiver,
rtcAnswer: sendAnswer,
rtcOffer: sendOffer,
}: UseSocket) {
const [peers, setPeers] = useState<RTCPeer[]>([]);
const answerHandler = (answer: RTCSessionDescriptionInit) => {
console.log('[useRTC] answer received', answer);
console.log('[useRTC] peers:', peers.length);
const peer = peers[0];
console.log('peer:', peer);
peer.onAnswer(answer);
};
const offerHandler = (offer: RTCSessionDescriptionInit) => {
console.log('[useRTC] offer received', offer);
setPeers((peers) => {
peers.push(new RTCPeer({ channelName, offer, sendAnswer, sendOffer }));
return peers;
});
};
useEffect(() => {
if (ready) {
console.log('-=-= SETTING THINGS UP =-=-');
registerAnsweredReceiver(answerHandler);
registerOfferredReceiver(offerHandler);
setPeers([new RTCPeer({ channelName, sendAnswer, sendOffer })]);
}
}, [ready]);
return {
count: peers.length,
};
}

View File

@@ -1,20 +1,37 @@
import { useEffect } from 'react';
import { useEffect, useRef, useState } from 'react';
import { socket } from '@/socket';
import type { GameUpdate } from '@/types';
interface UseSocketProps {
export interface UseSocketProps {
gameID: string;
setGameData: (gameUpdate: GameUpdate) => void;
setNoGame: (noGame: boolean) => void;
}
export default function useSocket({ gameID, setGameData, setNoGame }: UseSocketProps) {
export interface UseSocket {
ready: boolean;
flipCard: (cardIndex: number) => void;
handleSettings: (cardData: GameUpdate) => void;
redraw: (cardIndex: number) => void;
rtcAnswer: (answer: RTCSessionDescriptionInit) => void;
registerAnsweredReceiver: (receiver: (answer: RTCSessionDescriptionInit) => void) => void;
rtcOffer: (offer: RTCSessionDescriptionInit) => void;
registerOfferredReceiver: (receiver: (offer: RTCSessionDescriptionInit) => void) => void;
select: (cardIndex: number, cardID: string) => void;
}
export default function useSocket({ gameID, setGameData, setNoGame }: UseSocketProps): UseSocket {
const [ready, setReady] = useState(false);
const answerRef = useRef<(answer: RTCSessionDescriptionInit) => void>(null);
const offerRef = useRef<(offer: RTCSessionDescriptionInit) => void>(null);
useEffect(() => {
if (gameID) {
socket.emit('join', gameID);
socket.on('init', (data: GameUpdate) => {
setReady(true);
setGameData(data);
});
@@ -30,6 +47,16 @@ export default function useSocket({ gameID, setGameData, setNoGame }: UseSocketP
socket.on('flip-error', (error) => {
console.error('Error:', error);
});
socket.on('rtc-answered', (answered: RTCSessionDescriptionInit) => {
if (answerRef.current) answerRef.current(answered);
});
socket.on('rtc-offered', (offered: RTCSessionDescriptionInit) => {
if (offerRef.current) {
offerRef.current(offered);
}
});
}
return () => {
@@ -38,27 +65,16 @@ export default function useSocket({ gameID, setGameData, setNoGame }: UseSocketP
}, [gameID]);
const flipCard = (cardIndex: number) => {
console.log('flip-card', {
gameID,
cardIndex,
});
socket.emit('flip-card', {
gameID,
cardIndex,
});
};
const redraw = (cardIndex: number) => {
socket.emit('redraw', {
gameID,
cardIndex,
});
};
const select = (cardIndex: number, cardID: string) => {
socket.emit('select', {
gameID,
cardIndex,
cardID,
});
};
const handleSettings = (gameData: GameUpdate) => {
socket.emit('settings', {
gameID,
@@ -66,10 +82,42 @@ export default function useSocket({ gameID, setGameData, setNoGame }: UseSocketP
});
};
const redraw = (cardIndex: number) => {
socket.emit('redraw', {
gameID,
cardIndex,
});
};
const rtcAnswer = (answer: RTCSessionDescriptionInit) => {
console.log('rtc-answer', { gameID, answer });
socket.emit('rtc-answer', { gameID, answer });
};
const rtcOffer = (offer: RTCSessionDescriptionInit) => {
console.log('rtc-offer', { gameID, offer });
socket.emit('rtc-offer', { gameID, offer });
};
const select = (cardIndex: number, cardID: string) => {
socket.emit('select', {
gameID,
cardIndex,
cardID,
});
};
return {
ready,
flipCard,
redraw,
select,
handleSettings,
redraw,
rtcAnswer,
registerAnsweredReceiver: (receiver: (obj: RTCSessionDescriptionInit) => void[]) =>
(answerRef.current = receiver),
rtcOffer,
registerOfferredReceiver: (receiver: (obj: RTCSessionDescriptionInit) => void[]) =>
(offerRef.current = receiver),
select,
};
}