next.js 如何在下一个js 13中处理客户端组件中的firebase和webrc任务

r1zk6ea1  于 2024-01-07  发布在  其他
关注(0)|答案(1)|浏览(90)

我已经创建了一个下一个js 13应用程序与webrc和firestore共享摄像头提要。这是我的page.tsx。我不能让这个服务端,因为我已经使用了react钩子,我不能移动到客户端,因为它包含firestore和服务器数据。

"use client";

import { useEffect, useRef, useState } from "react";
import { firestore } from "@/services/_configs/firestore";

export default function Page({ params }: { params: { userid: string } }) {

    const servers = {
        iceServers: [
            {
                urls: ['stun:stun1.l.google.com:19302',
                    'stun:stun2.l.google.com:19302']
            }
        ],
        iceCandidatePoolSize: 10,
    }

    const pc = new RTCPeerConnection(servers);

    const [viewerId, setViewerId] = useState('');

    const [localStream, setLocalStream] = useState<MediaStream | null>(null);
    const [remoteStream, setRemoteStream] = useState<MediaStream | null>(null);

    const localVideo = useRef<HTMLVideoElement | null>(null);
    const remoteVideo = useRef<HTMLVideoElement | null>(null);

    useEffect(() => {
        navigator.mediaDevices.getUserMedia({ video: true, audio: true })
            .then((currentStream) => {
                setLocalStream(currentStream);
            });
        setRemoteStream(new MediaStream());

        localStream?.getTracks().forEach((track) => {
            pc.addTrack(track, localStream);
        })

        pc.ontrack = event => {
            event.streams[0].getTracks().forEach((track) => {
                remoteStream?.addTrack(track);
            });
        }

        if (localVideo.current != null && remoteVideo.current != null) {
            localVideo.current.srcObject = localStream;
            remoteVideo.current.srcObject = remoteStream;
        }

    }, []);

    const requestLiveStream = async () => {

        const callDoc = firestore.collection('calls').doc();
        const offerCandidate = callDoc.collection('offerCandidates');
        const answerCandidate = callDoc.collection('answerCandidates');

        setViewerId(callDoc.id);

        pc.onicecandidate = event => {
            event.candidate && offerCandidate.add(event.candidate.toJSON());
        }

        const offerDescription = await pc.createOffer();
        await pc.setLocalDescription(offerDescription);

        const offer = {
            sdp: offerDescription.sdp,
            type: offerDescription.type,
        }

        await callDoc.set({ offer });

        callDoc.onSnapshot((snapshot) => {
            const data = snapshot.data();
            if (!pc.currentLocalDescription && data?.answer) {
                const answerDescription = new RTCSessionDescription(data.answer);
                pc.setRemoteDescription(answerDescription);
            }
        })

        answerCandidate.onSnapshot((snapshot) => {
            snapshot.docChanges().forEach((change) => {
                if (change.type === "added") {
                    const candidate = new RTCIceCandidate(change.doc.data());
                    pc.addIceCandidate(candidate);
                }
            })
        });
    }

    const startLiveStream = async () => {
        const callId = viewerId;
        const callDoc = firestore.collection('calls').doc(callId);
        const offerCandidate = callDoc.collection('offerCandidates');
        const answerCandidate = callDoc.collection('answerCandidates');

        pc.onicecandidate = event => {
            event.candidate && answerCandidate.add(event.candidate.toJSON());
        }

        const callData = (await callDoc.get()).data();

        const offerDescription = callData?.offer;
        await pc.setRemoteDescription(new RTCSessionDescription(offerDescription));

        const answerDescription = await pc.createAnswer();
        await pc.setLocalDescription(answerDescription);

        const answer = {
            sdp: answerDescription.sdp,
            type: answerDescription.type,
        }

        await callDoc.update({ answer });

        offerCandidate.onSnapshot((snapshot) => {
            snapshot.docChanges().forEach((change) => {
                console.log(change);
                if (change.type === "added") {
                    let data = change.doc.data();
                    pc.addIceCandidate(new RTCIceCandidate(data));
                }
            })
        });

    }


    return (
        <div>
            <h1>{viewerId}</h1>
            <input
                type="text"
                value={viewerId}
                onChange={(e) => setViewerId(e.target.value)}
            />
            <button onClick={requestLiveStream}>view live</button>

            <button onClick={startLiveStream}>Answer live</button>
            <video playsInline ref={remoteVideo} autoPlay />
            <br />
            <video playsInline ref={localVideo} autoPlay />
        </div>
    );
}

字符串
我得到这个错误,我怎么解决这个问题。

Module not found: Can't resolve 'net'

https://nextjs.org/docs/messages/module-not-found

Import trace for requested module:
./node_modules/request/request.js

./node_modules/request/index.js

./node_modules/retry-request/index.js

./node_modules/google-gax/build/src/streamingCalls/streaming.js

./node_modules/google-gax/build/src/fallback.js

./node_modules/@google-cloud/firestore/build/src/index.js

./src/services/_configs/firestore.ts

./src/app/app/live/page.tsx


我试图将firestore代码分离到另一个,但我不能这样做,因为我有firestore数据更改事件在客户端。

vltsax25

vltsax251#

得到了这个代码的解决方案.我使用"@google-cloud/firestore": "^7.1.0"包.所以它是一个服务器端包,不能发送到客户端.所以我所做的是安装"firebase": "^10.7.1"包和做的请求.它的工作现在.我使用Next.js 13,如果你得到这些错误不要安装net包。只要看看客户端是否使用了服务器专用包。

相关问题